| |
| /*! |
| *********************************************************************** |
| * \file |
| * configfile.c |
| * \brief |
| * Configuration handling. |
| * \author |
| * Main contributors (see contributors.h for copyright, address and affiliation details) |
| * - Stephan Wenger <stewe@cs.tu-berlin.de> |
| * \note |
| * In the future this module should hide the Parameters and offer only |
| * Functions for their access. Modules which make frequent use of some parameters |
| * (e.g. picture size in macroblocks) are free to buffer them on local variables. |
| * This will not only avoid global variable and make the code more readable, but also |
| * speed it up. It will also greatly facilitate future enhancements such as the |
| * handling of different picture sizes in the same sequence. \n |
| * \n |
| * For now, everything is just copied to the inp_par structure (gulp) |
| * |
| ************************************************************************************** |
| * \par Configuration File Format |
| ************************************************************************************** |
| * Format is line oriented, maximum of one parameter per line \n |
| * \n |
| * Lines have the following format: \n |
| * \<ParameterName\> = \<ParameterValue\> # Comments \\n \n |
| * Whitespace is space and \\t |
| * \par |
| * \<ParameterName\> are the predefined names for Parameters and are case sensitive. |
| * See configfile.h for the definition of those names and their mapping to |
| * configinput->values. |
| * \par |
| * \<ParameterValue\> are either integers [0..9]* or strings. |
| * Integers must fit into the wordlengths, signed values are generally assumed. |
| * Strings containing no whitespace characters can be used directly. Strings containing |
| * whitespace characters are to be inclosed in double quotes ("string with whitespace") |
| * The double quote character is forbidden (may want to implement something smarter here). |
| * \par |
| * Any Parameters whose ParameterName is undefined lead to the termination of the program |
| * with an error message. |
| * |
| * \par Known bug/Shortcoming: |
| * zero-length strings (i.e. to signal an non-existing file |
| * have to be coded as "". |
| * |
| * \par Rules for using command files |
| * \n |
| * All Parameters are initially taken from DEFAULTCONFIGFILENAME, defined in configfile.h. |
| * If an -f \<config\> parameter is present in the command line then this file is used to |
| * update the defaults of DEFAULTCONFIGFILENAME. There can be more than one -f parameters |
| * present. If -p <ParameterName = ParameterValue> parameters are present then these |
| * override the default and the additional config file's settings, and are themselves |
| * overridden by future -p parameters. There must be whitespace between -f and -p commands |
| * and their respective parameters |
| *********************************************************************** |
| */ |
| |
| #define INCLUDED_BY_CONFIGFILE_C |
| |
| #include <stdlib.h> |
| #include <string.h> |
| #include <assert.h> |
| #include <fcntl.h> |
| #include <sys/stat.h> |
| |
| #include "global.h" |
| #include "configfile.h" |
| #include "fmo.h" |
| |
| static const char *mybasename(const char *str) { |
| const char *base = strrchr(str, '/'); |
| return base ? base+1 : str; |
| } |
| |
| char *GetConfigFileContent (char *Filename); |
| static void ParseContent (char *buf, int bufsize); |
| static int ParameterNameToMapIndex (char *s); |
| static int InitEncoderParams(void); |
| static int TestEncoderParams(int bitdepth_qp_scale); |
| static int DisplayEncoderParams(void); |
| static void PatchInp (void); |
| static void ProfileCheck(void); |
| static void LevelCheck(void); |
| |
| |
| #define MAX_ITEMS_TO_PARSE 10000 |
| |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * print help message and exit |
| *********************************************************************** |
| */ |
| void JMHelpExit (void) |
| { |
| fprintf( stderr, "\n lencod [-h] [-d defenc.cfg] {[-f curenc1.cfg]...[-f curencN.cfg]}" |
| " {[-p EncParam1=EncValue1]..[-p EncParamM=EncValueM]}\n\n" |
| "## Parameters\n\n" |
| |
| "## Options\n" |
| " -h : prints function usage\n" |
| " -d : use <defenc.cfg> as default file for parameter initializations.\n" |
| " If not used then file defaults to encoder.cfg in local directory.\n" |
| " -f : read <curencM.cfg> for reseting selected encoder parameters.\n" |
| " Multiple files could be used that set different parameters\n" |
| " -p : Set parameter <EncParamM> to <EncValueM>.\n" |
| " See default encoder.cfg file for description of all parameters.\n\n" |
| |
| "## Supported video file formats\n" |
| " RAW: .yuv -> YUV 4:2:0\n\n" |
| |
| "## Examples of usage:\n" |
| " lencod\n" |
| " lencod -h\n" |
| " lencod -d default.cfg\n" |
| " lencod -f curenc1.cfg\n" |
| " lencod -f curenc1.cfg -p InputFile=\"e:\\data\\container_qcif_30.yuv\" -p SourceWidth=176 -p SourceHeight=144\n" |
| " lencod -f curenc1.cfg -p FramesToBeEncoded=30 -p QPISlice=28 -p QPPSlice=28 -p QPBSlice=30\n"); |
| |
| exit(-1); |
| } |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Parse the command line parameters and read the config files. |
| * \param ac |
| * number of command line parameters |
| * \param av |
| * command line parameters |
| *********************************************************************** |
| */ |
| void Configure (int ac, char *av[]) |
| { |
| char *content; |
| int CLcount, ContentLen, NumberParams; |
| char *filename=DEFAULTCONFIGFILENAME; |
| |
| memset (&configinput, 0, sizeof (InputParameters)); |
| //Set default parameters. |
| printf ("Setting Default Parameters...\n"); |
| InitEncoderParams(); |
| |
| // Process default config file |
| CLcount = 1; |
| |
| if (ac==2) |
| { |
| if (0 == strncmp (av[1], "-h", 2)) |
| { |
| JMHelpExit(); |
| } |
| } |
| |
| if (ac>=3) |
| { |
| if (0 == strncmp (av[1], "-d", 2)) |
| { |
| filename=av[2]; |
| CLcount = 3; |
| } |
| if (0 == strncmp (av[1], "-h", 2)) |
| { |
| JMHelpExit(); |
| } |
| } |
| printf ("Parsing Configfile %s", mybasename(filename)); |
| content = GetConfigFileContent (filename); |
| if (NULL==content) |
| error (errortext, 300); |
| ParseContent (content, strlen(content)); |
| printf ("\n"); |
| free (content); |
| |
| // Parse the command line |
| |
| while (CLcount < ac) |
| { |
| if (0 == strncmp (av[CLcount], "-h", 2)) |
| { |
| JMHelpExit(); |
| } |
| |
| if (0 == strncmp (av[CLcount], "-f", 2)) // A file parameter? |
| { |
| content = GetConfigFileContent (av[CLcount+1]); |
| if (NULL==content) |
| error (errortext, 300); |
| printf ("Parsing Configfile %s", av[CLcount+1]); |
| ParseContent (content, strlen (content)); |
| printf ("\n"); |
| free (content); |
| CLcount += 2; |
| } else |
| { |
| if (0 == strncmp (av[CLcount], "-p", 2)) // A config change? |
| { |
| // Collect all data until next parameter (starting with -<x> (x is any character)), |
| // put it into content, and parse content. |
| |
| CLcount++; |
| ContentLen = 0; |
| NumberParams = CLcount; |
| |
| // determine the necessary size for content |
| while (NumberParams < ac && av[NumberParams][0] != '-') |
| ContentLen += strlen (av[NumberParams++]); // Space for all the strings |
| ContentLen += 1000; // Additional 1000 bytes for spaces and \0s |
| |
| |
| if ((content = malloc (ContentLen))==NULL) no_mem_exit("Configure: content");; |
| content[0] = '\0'; |
| |
| // concatenate all parameters identified before |
| |
| while (CLcount < NumberParams) |
| { |
| char *source = &av[CLcount][0]; |
| char *destin = &content[strlen (content)]; |
| |
| while (*source != '\0') |
| { |
| if (*source == '=') // The Parser expects whitespace before and after '=' |
| { |
| *destin++=' '; *destin++='='; *destin++=' '; // Hence make sure we add it |
| } else |
| *destin++=*source; |
| source++; |
| } |
| *destin = '\0'; |
| CLcount++; |
| } |
| printf ("Parsing command line string '%s'", "IGNORED"); |
| ParseContent (content, strlen(content)); |
| free (content); |
| printf ("\n"); |
| } |
| else |
| { |
| snprintf (errortext, ET_SIZE, "Error in command line, ac %d, around string '%s', missing -f or -p parameters?", CLcount, av[CLcount]); |
| error (errortext, 300); |
| } |
| } |
| } |
| printf ("\n"); |
| PatchInp(); |
| if (input->DisplayEncParams) |
| DisplayEncoderParams(); |
| } |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * allocates memory buf, opens file Filename in f, reads contents into |
| * buf and returns buf |
| * \param Filename |
| * name of config file |
| * \return |
| * if successfull, content of config file |
| * NULL in case of error. Error message will be set in errortext |
| *********************************************************************** |
| */ |
| char *GetConfigFileContent (char *Filename) |
| { |
| long FileSize; |
| FILE *f; |
| char *buf; |
| |
| if (NULL == (f = fopen (Filename, "r"))) |
| { |
| snprintf (errortext, ET_SIZE, "Cannot open configuration file %s.", Filename); |
| return NULL; |
| } |
| |
| if (0 != fseek (f, 0, SEEK_END)) |
| { |
| snprintf (errortext, ET_SIZE, "Cannot fseek in configuration file %s.", Filename); |
| return NULL; |
| } |
| |
| FileSize = ftell (f); |
| if (FileSize < 0 || FileSize > 60000) |
| { |
| snprintf (errortext, ET_SIZE, "Unreasonable Filesize %ld reported by ftell for configuration file %s.", FileSize, Filename); |
| return NULL; |
| } |
| if (0 != fseek (f, 0, SEEK_SET)) |
| { |
| snprintf (errortext, ET_SIZE, "Cannot fseek in configuration file %s.", Filename); |
| return NULL; |
| } |
| |
| if ((buf = malloc (FileSize + 1))==NULL) no_mem_exit("GetConfigFileContent: buf"); |
| |
| // Note that ftell() gives us the file size as the file system sees it. The actual file size, |
| // as reported by fread() below will be often smaller due to CR/LF to CR conversion and/or |
| // control characters after the dos EOF marker in the file. |
| |
| FileSize = fread (buf, 1, FileSize, f); |
| buf[FileSize] = '\0'; |
| |
| |
| fclose (f); |
| return buf; |
| } |
| |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Parses the character array buf and writes global variable input, which is defined in |
| * configfile.h. This hack will continue to be necessary to facilitate the addition of |
| * new parameters through the Map[] mechanism (Need compiler-generated addresses in map[]). |
| * \param buf |
| * buffer to be parsed |
| * \param bufsize |
| * buffer size of buffer |
| *********************************************************************** |
| */ |
| void ParseContent (char *buf, int bufsize) |
| { |
| |
| char *items[MAX_ITEMS_TO_PARSE]; |
| int MapIdx; |
| int item = 0; |
| int InString = 0, InItem = 0; |
| char *p = buf; |
| char *bufend = &buf[bufsize]; |
| int IntContent; |
| double DoubleContent; |
| int i; |
| |
| // Stage one: Generate an argc/argv-type list in items[], without comments and whitespace. |
| // This is context insensitive and could be done most easily with lex(1). |
| |
| while (p < bufend) |
| { |
| switch (*p) |
| { |
| case 13: |
| p++; |
| break; |
| case '#': // Found comment |
| *p = '\0'; // Replace '#' with '\0' in case of comment immediately following integer or string |
| while (*p != '\n' && p < bufend) // Skip till EOL or EOF, whichever comes first |
| p++; |
| InString = 0; |
| InItem = 0; |
| break; |
| case '\n': |
| InItem = 0; |
| InString = 0; |
| *p++='\0'; |
| break; |
| case ' ': |
| case '\t': // Skip whitespace, leave state unchanged |
| if (InString) |
| p++; |
| else |
| { // Terminate non-strings once whitespace is found |
| *p++ = '\0'; |
| InItem = 0; |
| } |
| break; |
| |
| case '"': // Begin/End of String |
| *p++ = '\0'; |
| if (!InString) |
| { |
| items[item++] = p; |
| InItem = ~InItem; |
| } |
| else |
| InItem = 0; |
| InString = ~InString; // Toggle |
| break; |
| |
| default: |
| if (!InItem) |
| { |
| items[item++] = p; |
| InItem = ~InItem; |
| } |
| p++; |
| } |
| } |
| |
| item--; |
| |
| for (i=0; i<item; i+= 3) |
| { |
| if (0 > (MapIdx = ParameterNameToMapIndex (items[i]))) |
| { |
| snprintf (errortext, ET_SIZE, " Parsing error in config file: Parameter Name '%s' not recognized.", items[i]); |
| error (errortext, 300); |
| } |
| if (strcasecmp ("=", items[i+1])) |
| { |
| snprintf (errortext, ET_SIZE, " Parsing error in config file: '=' expected as the second token in each line."); |
| error (errortext, 300); |
| } |
| |
| // Now interpret the Value, context sensitive... |
| |
| switch (Map[MapIdx].Type) |
| { |
| case 0: // Numerical |
| if (1 != sscanf (items[i+2], "%d", &IntContent)) |
| { |
| snprintf (errortext, ET_SIZE, " Parsing error: Expected numerical value for Parameter of %s, found '%s'.", items[i], items[i+2]); |
| error (errortext, 300); |
| } |
| * (int *) (Map[MapIdx].Place) = IntContent; |
| printf ("."); |
| break; |
| case 1: |
| strncpy ((char *) Map[MapIdx].Place, items [i+2], FILE_NAME_SIZE); |
| printf ("."); |
| break; |
| case 2: // Numerical double |
| if (1 != sscanf (items[i+2], "%lf", &DoubleContent)) |
| { |
| snprintf (errortext, ET_SIZE, " Parsing error: Expected numerical value for Parameter of %s, found '%s'.", items[i], items[i+2]); |
| error (errortext, 300); |
| } |
| * (double *) (Map[MapIdx].Place) = DoubleContent; |
| printf ("."); |
| break; |
| default: |
| error ("Unknown value type in the map definition of configfile.h",-1); |
| } |
| } |
| memcpy (input, &configinput, sizeof (InputParameters)); |
| } |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Returns the index number from Map[] for a given parameter name. |
| * \param s |
| * parameter name string |
| * \return |
| * the index number if the string is a valid parameter name, \n |
| * -1 for error |
| *********************************************************************** |
| */ |
| static int ParameterNameToMapIndex (char *s) |
| { |
| int i = 0; |
| |
| while (Map[i].TokenName != NULL) |
| if (0==strcasecmp (Map[i].TokenName, s)) |
| return i; |
| else |
| i++; |
| return -1; |
| } |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Sets initial values for encoding parameters. |
| * \return |
| * -1 for error |
| *********************************************************************** |
| */ |
| static int InitEncoderParams(void) |
| { |
| int i = 0; |
| |
| while (Map[i].TokenName != NULL) |
| { |
| if (Map[i].Type == 0) |
| * (int *) (Map[i].Place) = (int) Map[i].Default; |
| else if (Map[i].Type == 2) |
| * (double *) (Map[i].Place) = Map[i].Default; |
| i++; |
| } |
| return -1; |
| } |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Validates encoding parameters. |
| * \return |
| * -1 for error |
| *********************************************************************** |
| */ |
| static int TestEncoderParams(int bitdepth_qp_scale) |
| { |
| int i = 0; |
| |
| while (Map[i].TokenName != NULL) |
| { |
| if (Map[i].param_limits == 1) |
| { |
| if (Map[i].Type == 0) |
| { |
| if ( * (int *) (Map[i].Place) < (int) Map[i].min_limit || * (int *) (Map[i].Place) > (int) Map[i].max_limit ) |
| { |
| snprintf(errortext, ET_SIZE, "Error in input parameter %s. Check configuration file. Value should be in [%d, %d] range.", Map[i].TokenName, (int) Map[i].min_limit,(int)Map[i].max_limit ); |
| error (errortext, 400); |
| } |
| |
| } |
| else if (Map[i].Type == 2) |
| { |
| if ( * (double *) (Map[i].Place) < Map[i].min_limit || * (double *) (Map[i].Place) > Map[i].max_limit ) |
| { |
| snprintf(errortext, ET_SIZE, "Error in input parameter %s. Check configuration file. Value should be in [%.2f, %.2f] range.", Map[i].TokenName,Map[i].min_limit ,Map[i].max_limit ); |
| error (errortext, 400); |
| } |
| } |
| } |
| else if (Map[i].param_limits == 2) |
| { |
| if (Map[i].Type == 0) |
| { |
| if ( * (int *) (Map[i].Place) < (int) Map[i].min_limit ) |
| { |
| snprintf(errortext, ET_SIZE, "Error in input parameter %s. Check configuration file. Value should not be smaller than %d.", Map[i].TokenName, (int) Map[i].min_limit); |
| error (errortext, 400); |
| } |
| } |
| else if (Map[i].Type == 2) |
| { |
| if ( * (double *) (Map[i].Place) < Map[i].min_limit ) |
| { |
| snprintf(errortext, ET_SIZE, "Error in input parameter %s. Check configuration file. Value should not be smaller than %2.f.", Map[i].TokenName,Map[i].min_limit); |
| error (errortext, 400); |
| } |
| } |
| } |
| else if (Map[i].param_limits == 3) // Only used for QPs |
| { |
| if (Map[i].Type == 0) |
| { |
| if ( * (int *) (Map[i].Place) < (int) (Map[i].min_limit - bitdepth_qp_scale) || * (int *) (Map[i].Place) > (int) Map[i].max_limit ) |
| { |
| snprintf(errortext, ET_SIZE, "Error in input parameter %s. Check configuration file. Value should be in [%d, %d] range.", Map[i].TokenName, (int) (Map[i].min_limit - bitdepth_qp_scale),(int)Map[i].max_limit ); |
| error (errortext, 400); |
| } |
| } |
| } |
| |
| i++; |
| } |
| return -1; |
| } |
| |
| |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Outputs encoding parameters. |
| * \return |
| * -1 for error |
| *********************************************************************** |
| */ |
| static int DisplayEncoderParams(void) |
| { |
| int i = 0; |
| |
| printf("******************************************************\n"); |
| printf("* Encoder Parameters *\n"); |
| printf("******************************************************\n"); |
| while (Map[i].TokenName != NULL) |
| { |
| if (Map[i].Type == 0) |
| printf("Parameter %s = %d\n",Map[i].TokenName,* (int *) (Map[i].Place)); |
| else if (Map[i].Type == 1) |
| printf("Parameter %s = ""%s""\n",Map[i].TokenName,(char *) (Map[i].Place)); |
| else if (Map[i].Type == 2) |
| printf("Parameter %s = %.2f\n",Map[i].TokenName,* (double *) (Map[i].Place)); |
| i++; |
| } |
| printf("******************************************************\n"); |
| return -1; |
| } |
| |
| /*! |
| ************************************************************************ |
| * \brief |
| * calculate Ceil(Log2(uiVal)) |
| ************************************************************************ |
| */ |
| unsigned CeilLog2( unsigned uiVal) |
| { |
| unsigned uiTmp = uiVal-1; |
| unsigned uiRet = 0; |
| |
| while( uiTmp != 0 ) |
| { |
| uiTmp >>= 1; |
| uiRet++; |
| } |
| return uiRet; |
| } |
| |
| |
| /*! |
| *********************************************************************** |
| * \brief |
| * Checks the input parameters for consistency. |
| *********************************************************************** |
| */ |
| static void PatchInp (void) |
| { |
| int bitdepth_qp_scale = 6*(input->BitDepthLuma - 8); |
| |
| // These variables are added for FMO |
| FILE * sgfile=NULL; |
| int i,j; |
| int frame_mb_only; |
| int mb_width, mb_height, mapunit_height; |
| int storedBplus1; |
| |
| TestEncoderParams(bitdepth_qp_scale); |
| |
| if (input->FrameRate == 0.0) |
| input->FrameRate = INIT_FRAME_RATE; |
| |
| // Set block sizes |
| |
| // Skip/Direct16x16 |
| input->part_size[0][0] = 4; |
| input->part_size[0][1] = 4; |
| // 16x16 |
| input->part_size[1][0] = 4; |
| input->part_size[1][1] = 4; |
| // 16x8 |
| input->part_size[2][0] = 4; |
| input->part_size[2][1] = 2; |
| // 8x16 |
| input->part_size[3][0] = 2; |
| input->part_size[3][1] = 4; |
| // 8x8 |
| input->part_size[4][0] = 2; |
| input->part_size[4][1] = 2; |
| // 8x4 |
| input->part_size[5][0] = 2; |
| input->part_size[5][1] = 1; |
| // 4x8 |
| input->part_size[6][0] = 1; |
| input->part_size[6][1] = 2; |
| // 4x4 |
| input->part_size[7][0] = 1; |
| input->part_size[7][1] = 1; |
| |
| input->blocktype_lut[0][0] = 7; // 4x4 |
| input->blocktype_lut[0][1] = 6; // 4x8 |
| input->blocktype_lut[1][0] = 5; // 8x4 |
| input->blocktype_lut[1][1] = 4; // 8x8 |
| input->blocktype_lut[1][3] = 3; // 8x16 |
| input->blocktype_lut[3][1] = 2; // 16x8 |
| input->blocktype_lut[3][3] = 1; // 16x16 |
| |
| for (j = 0; j<8;j++) |
| { |
| for (i = 0; i<2; i++) |
| { |
| input->blc_size[j][i] = input->part_size[j][i] * BLOCK_SIZE; |
| } |
| } |
| |
| // set proper log2_max_frame_num_minus4. |
| storedBplus1 = (input->BRefPictures ) ? input->successive_Bframe + 1: 1; |
| |
| if (input->Log2MaxFNumMinus4 == -1) |
| log2_max_frame_num_minus4 = iClip3(0,12, (int) (CeilLog2(input->no_frames * storedBplus1) - 4)); |
| else |
| log2_max_frame_num_minus4 = input->Log2MaxFNumMinus4; |
| |
| if (log2_max_frame_num_minus4 == 0 && input->num_ref_frames == 16) |
| { |
| snprintf(errortext, ET_SIZE, " NumberReferenceFrames=%d and Log2MaxFNumMinus4=%d may lead to an invalid value of frame_num.", input->num_ref_frames, input-> Log2MaxFNumMinus4); |
| error (errortext, 500); |
| } |
| |
| // set proper log2_max_pic_order_cnt_lsb_minus4. |
| if (input->Log2MaxPOCLsbMinus4 == - 1) |
| log2_max_pic_order_cnt_lsb_minus4 = iClip3(0,12, (int) (CeilLog2( 2*input->no_frames * (input->jumpd + 1)) - 4)); |
| else |
| log2_max_pic_order_cnt_lsb_minus4 = input->Log2MaxPOCLsbMinus4; |
| |
| if (((1<<(log2_max_pic_order_cnt_lsb_minus4 + 3)) < input->jumpd * 4) && input->Log2MaxPOCLsbMinus4 != -1) |
| error("log2_max_pic_order_cnt_lsb_minus4 might not be sufficient for encoding. Increase value.",400); |
| |
| // B picture consistency check |
| if(input->successive_Bframe > input->jumpd) |
| { |
| snprintf(errortext, ET_SIZE, "Number of B-frames %d can not exceed the number of frames skipped", input->successive_Bframe); |
| error (errortext, 400); |
| } |
| |
| // Direct Mode consistency check |
| if(input->successive_Bframe && input->direct_spatial_mv_pred_flag != DIR_SPATIAL && input->direct_spatial_mv_pred_flag != DIR_TEMPORAL) |
| { |
| snprintf(errortext, ET_SIZE, "Unsupported direct mode=%d, use TEMPORAL=0 or SPATIAL=1", input->direct_spatial_mv_pred_flag); |
| error (errortext, 400); |
| } |
| |
| if (input->PicInterlace>0 || input->MbInterlace>0) |
| { |
| if (input->directInferenceFlag==0) |
| printf("\nDirectInferenceFlag set to 1 due to interlace coding."); |
| input->directInferenceFlag = 1; |
| } |
| |
| if (input->PicInterlace>0) |
| { |
| if (input->IntraBottom!=0 && input->IntraBottom!=1) |
| { |
| snprintf(errortext, ET_SIZE, "Incorrect value %d for IntraBottom. Use 0 (disable) or 1 (enable).", input->IntraBottom); |
| error (errortext, 400); |
| } |
| } |
| // Cabac/UVLC consistency check |
| if (input->symbol_mode != UVLC && input->symbol_mode != CABAC) |
| { |
| snprintf (errortext, ET_SIZE, "Unsupported symbol mode=%d, use UVLC=0 or CABAC=1",input->symbol_mode); |
| error (errortext, 400); |
| } |
| |
| // Open Files |
| if ((p_in=open(input->infile, OPENFLAGS_READ))==-1) |
| { |
| snprintf(errortext, ET_SIZE, "Input file %s does not exist",input->infile); |
| error (errortext, 500); |
| } |
| |
| if (strlen (input->ReconFile) > 0 && (p_dec=open(input->ReconFile, OPENFLAGS_WRITE, OPEN_PERMISSIONS))==-1) |
| { |
| snprintf(errortext, ET_SIZE, "Error open file %s", input->ReconFile); |
| error (errortext, 500); |
| } |
| |
| #if TRACE |
| if (strlen (input->TraceFile) > 0 && (p_trace=fopen(input->TraceFile,"w"))==NULL) |
| { |
| snprintf(errortext, ET_SIZE, "Error open file %s", input->TraceFile); |
| error (errortext, 500); |
| } |
| #endif |
| |
| if (input->img_width % 16 != 0) |
| { |
| img->auto_crop_right = 16-(input->img_width % 16); |
| } |
| else |
| { |
| img->auto_crop_right=0; |
| } |
| if (input->PicInterlace || input->MbInterlace) |
| { |
| if (input->img_height % 2 != 0) |
| { |
| error ("even number of lines required for interlaced coding", 500); |
| } |
| if (input->img_height % 32 != 0) |
| { |
| img->auto_crop_bottom = 32-(input->img_height % 32); |
| } |
| else |
| { |
| img->auto_crop_bottom=0; |
| } |
| } |
| else |
| { |
| if (input->img_height % 16 != 0) |
| { |
| img->auto_crop_bottom = 16-(input->img_height % 16); |
| } |
| else |
| { |
| img->auto_crop_bottom=0; |
| } |
| } |
| if (img->auto_crop_bottom || img->auto_crop_right) |
| { |
| fprintf (stderr, "Warning: Automatic cropping activated: Coded frame Size: %dx%d\n", input->img_width+img->auto_crop_right, input->img_height+img->auto_crop_bottom); |
| } |
| |
| if ((input->slice_mode==1)&&(input->MbInterlace!=0)) |
| { |
| if ((input->slice_argument%2)!=0) |
| { |
| fprintf ( stderr, "Warning: slice border within macroblock pair. "); |
| if (input->slice_argument > 1) |
| { |
| input->slice_argument--; |
| } |
| else |
| { |
| input->slice_argument++; |
| } |
| fprintf ( stderr, "Using %d MBs per slice.\n", input->slice_argument); |
| } |
| } |
| /* |
| // add check for MAXSLICEGROUPIDS |
| if(input->num_slice_groups_minus1>=MAXSLICEGROUPIDS) |
| { |
| snprintf(errortext, ET_SIZE, "num_slice_groups_minus1 exceeds MAXSLICEGROUPIDS"); |
| error (errortext, 500); |
| } |
| */ |
| |
| // Following codes are to read slice group configuration from SliceGroupConfigFileName for slice group type 0,2 or 6 |
| if( (input->num_slice_groups_minus1!=0)&& |
| ((input->slice_group_map_type == 0) || (input->slice_group_map_type == 2) || (input->slice_group_map_type == 6)) ) |
| { |
| if (strlen (input->SliceGroupConfigFileName) > 0 && (sgfile=fopen(input->SliceGroupConfigFileName,"r"))==NULL) |
| { |
| snprintf(errortext, ET_SIZE, "Error open file %s", input->SliceGroupConfigFileName); |
| error (errortext, 500); |
| } |
| else |
| { |
| if (input->slice_group_map_type == 0) |
| { |
| input->run_length_minus1=(int *)malloc(sizeof(int)*(input->num_slice_groups_minus1+1)); |
| if (NULL==input->run_length_minus1) |
| no_mem_exit("PatchInp: input->run_length_minus1"); |
| |
| // each line contains one 'run_length_minus1' value |
| for(i=0;i<=input->num_slice_groups_minus1;i++) |
| { |
| fscanf(sgfile,"%d",(input->run_length_minus1+i)); |
| fscanf(sgfile,"%*[^\n]"); |
| } |
| } |
| else if (input->slice_group_map_type == 2) |
| { |
| input->top_left=(int *)malloc(sizeof(int)*input->num_slice_groups_minus1); |
| input->bottom_right=(int *)malloc(sizeof(int)*input->num_slice_groups_minus1); |
| if (NULL==input->top_left) |
| no_mem_exit("PatchInp: input->top_left"); |
| if (NULL==input->bottom_right) |
| no_mem_exit("PatchInp: input->bottom_right"); |
| |
| // every two lines contain 'top_left' and 'bottom_right' value |
| for(i=0;i<input->num_slice_groups_minus1;i++) |
| { |
| fscanf(sgfile,"%d",(input->top_left+i)); |
| fscanf(sgfile,"%*[^\n]"); |
| fscanf(sgfile,"%d",(input->bottom_right+i)); |
| fscanf(sgfile,"%*[^\n]"); |
| } |
| } |
| else if (input->slice_group_map_type == 6) |
| { |
| int tmp; |
| |
| frame_mb_only = !(input->PicInterlace || input->MbInterlace); |
| mb_width= (input->img_width+img->auto_crop_right)>>4; |
| mb_height= (input->img_height+img->auto_crop_bottom)>>4; |
| mapunit_height=mb_height/(2-frame_mb_only); |
| |
| input->slice_group_id=(byte * ) malloc(sizeof(byte)*mapunit_height*mb_width); |
| if (NULL==input->slice_group_id) |
| no_mem_exit("PatchInp: input->slice_group_id"); |
| |
| // each line contains slice_group_id for one Macroblock |
| for (i=0;i<mapunit_height*mb_width;i++) |
| { |
| fscanf(sgfile,"%d", &tmp); |
| input->slice_group_id[i]= (byte) tmp; |
| if ( *(input->slice_group_id+i) > input->num_slice_groups_minus1 ) |
| { |
| snprintf(errortext, ET_SIZE, "Error read slice group information from file %s", input->SliceGroupConfigFileName); |
| error (errortext, 500); |
| } |
| fscanf(sgfile,"%*[^\n]"); |
| } |
| } |
| fclose(sgfile); |
| } |
| } |
| |
| |
| if (input->ReferenceReorder && (input->PicInterlace || input->MbInterlace)) |
| { |
| snprintf(errortext, ET_SIZE, "ReferenceReorder Not supported with Interlace encoding methods\n"); |
| error (errortext, 400); |
| } |
| |
| if (input->PocMemoryManagement && (input->PicInterlace || input->MbInterlace)) |
| { |
| snprintf(errortext, ET_SIZE, "PocMemoryManagement not supported with Interlace encoding methods\n"); |
| error (errortext, 400); |
| } |
| |
| // frame/field consistency check |
| if (input->PicInterlace != FRAME_CODING && input->PicInterlace != ADAPTIVE_CODING && input->PicInterlace != FIELD_CODING) |
| { |
| snprintf (errortext, ET_SIZE, "Unsupported PicInterlace=%d, use frame based coding=0 or field based coding=1 or adaptive=2",input->PicInterlace); |
| error (errortext, 400); |
| } |
| |
| // frame/field consistency check |
| if (input->MbInterlace != FRAME_CODING && input->MbInterlace != ADAPTIVE_CODING && input->MbInterlace != FIELD_CODING && input->MbInterlace != FRAME_MB_PAIR_CODING) |
| { |
| snprintf (errortext, ET_SIZE, "Unsupported MbInterlace=%d, use frame based coding=0 or field based coding=1 or adaptive=2 or frame MB pair only=3",input->MbInterlace); |
| error (errortext, 400); |
| } |
| |
| |
| if ((!input->rdopt)&&(input->MbInterlace)) |
| { |
| snprintf(errortext, ET_SIZE, "MB AFF is not compatible with non-rd-optimized coding."); |
| error (errortext, 500); |
| } |
| |
| /*if (input->rdopt>2) |
| { |
| snprintf(errortext, ET_SIZE, "RDOptimization=3 mode has been deactivated do to diverging of real and simulated decoders."); |
| error (errortext, 500); |
| }*/ |
| |
| // check RDoptimization mode and profile. FMD does not support Frex Profiles. |
| if (input->rdopt==2 && input->ProfileIDC>=FREXT_HP) |
| { |
| snprintf(errortext, ET_SIZE, "Fast Mode Decision methods does not support FREX Profiles"); |
| error (errortext, 500); |
| } |
| |
| if ( (input->MEErrorMetric[Q_PEL] == ERROR_SATD && input->MEErrorMetric[H_PEL] == ERROR_SAD && input->MEErrorMetric[F_PEL] == ERROR_SAD) |
| && input->SearchMode > FAST_FULL_SEARCH && input->SearchMode < EPZS) |
| { |
| snprintf(errortext, ET_SIZE, "MEDistortionQPel=2, MEDistortionHPel=0, MEDistortionFPel=0 is not allowed when SearchMode is set to 1 or 2."); |
| error (errortext, 500); |
| } |
| |
| // Tian Dong: May 31, 2002 |
| // The number of frames in one sub-seq in enhanced layer should not exceed |
| // the number of reference frame number. |
| if ( input->NumFramesInELSubSeq >= input->num_ref_frames || input->NumFramesInELSubSeq < 0 ) |
| { |
| snprintf(errortext, ET_SIZE, "NumFramesInELSubSeq (%d) is out of range [0,%d).", input->NumFramesInELSubSeq, input->num_ref_frames); |
| error (errortext, 500); |
| } |
| // Tian Dong: Enhanced GOP is not supported in bitstream mode. September, 2002 |
| if ( input->NumFramesInELSubSeq > 0 && input->of_mode == PAR_OF_ANNEXB ) |
| { |
| snprintf(errortext, ET_SIZE, "Enhanced GOP is not supported in bitstream mode and RTP mode yet."); |
| error (errortext, 500); |
| } |
| // Tian Dong (Sept 2002) |
| // The AFF is not compatible with spare picture for the time being. |
| if ((input->PicInterlace || input->MbInterlace) && input->SparePictureOption == TRUE) |
| { |
| snprintf(errortext, ET_SIZE, "AFF is not compatible with spare picture."); |
| error (errortext, 500); |
| } |
| |
| // Only the RTP mode is compatible with spare picture for the time being. |
| if (input->of_mode != PAR_OF_RTP && input->SparePictureOption == TRUE) |
| { |
| snprintf(errortext, ET_SIZE, "Only RTP output mode is compatible with spare picture features."); |
| error (errortext, 500); |
| } |
| |
| if( (input->WeightedPrediction > 0 || input->WeightedBiprediction > 0) && (input->MbInterlace)) |
| { |
| snprintf(errortext, ET_SIZE, "Weighted prediction coding is not supported for MB AFF currently."); |
| error (errortext, 500); |
| } |
| if ( input->NumFramesInELSubSeq > 0 && input->WeightedPrediction > 0) |
| { |
| snprintf(errortext, ET_SIZE, "Enhanced GOP is not supported in weighted prediction coding mode yet."); |
| error (errortext, 500); |
| } |
| |
| //! the number of slice groups is forced to be 1 for slice group type 3-5 |
| if(input->num_slice_groups_minus1 > 0) |
| { |
| if( (input->slice_group_map_type >= 3) && (input->slice_group_map_type<=5) ) |
| input->num_slice_groups_minus1 = 1; |
| } |
| |
| // Rate control |
| if(input->RCEnable) |
| { |
| if ( ((input->img_height+img->auto_crop_bottom)*(input->img_width+img->auto_crop_right)/256)%input->basicunit!=0) |
| { |
| snprintf(errortext, ET_SIZE, "Frame size in macroblocks must be a multiple of BasicUnit."); |
| error (errortext, 500); |
| } |
| |
| if ( (input->successive_Bframe || input->jumpd) && input->RCUpdateMode == RC_MODE_1 ) |
| { |
| snprintf(errortext, ET_SIZE, "Use RC_MODE_1 only for all-intra coding."); |
| error (errortext, 500); |
| } |
| } |
| |
| if ((input->successive_Bframe)&&(input->BRefPictures)&&(input->idr_enable)&&(input->intra_period)&&(input->pic_order_cnt_type!=0)) |
| { |
| error("Stored B pictures combined with IDR pictures only supported in Picture Order Count type 0\n",-1000); |
| } |
| |
| if( !input->direct_spatial_mv_pred_flag && input->num_ref_frames<2 && input->successive_Bframe >0) |
| error("temporal direct needs at least 2 ref frames\n",-1000); |
| |
| // frext |
| if(input->Transform8x8Mode && input->sp_periodicity /*SP-frames*/) |
| { |
| snprintf(errortext, ET_SIZE, "\nThe new 8x8 mode is not implemented for sp-frames."); |
| error (errortext, 500); |
| } |
| |
| if(input->Transform8x8Mode && (input->ProfileIDC<FREXT_HP || input->ProfileIDC>FREXT_Hi444)) |
| { |
| snprintf(errortext, ET_SIZE, "\nTransform8x8Mode may be used only with ProfileIDC %d to %d.", FREXT_HP, FREXT_Hi444); |
| error (errortext, 500); |
| } |
| if(input->ScalingMatrixPresentFlag && (input->ProfileIDC<FREXT_HP || input->ProfileIDC>FREXT_Hi444)) |
| { |
| snprintf(errortext, ET_SIZE, "\nScalingMatrixPresentFlag may be used only with ProfileIDC %d to %d.", FREXT_HP, FREXT_Hi444); |
| error (errortext, 500); |
| } |
| |
| if(input->yuv_format==YUV422 && input->ProfileIDC < FREXT_Hi422) |
| { |
| snprintf(errortext, ET_SIZE, "\nFRExt Profile(YUV Format) Error!\nYUV422 can be used only with ProfileIDC %d or %d\n",FREXT_Hi422, FREXT_Hi444); |
| error (errortext, 500); |
| } |
| if(input->yuv_format==YUV444 && input->ProfileIDC < FREXT_Hi444) |
| { |
| snprintf(errortext, ET_SIZE, "\nFRExt Profile(YUV Format) Error!\nYUV444 can be used only with ProfileIDC %d.\n",FREXT_Hi444); |
| error (errortext, 500); |
| } |
| |
| if (input->successive_Bframe && ((input->BiPredMotionEstimation) && (input->search_range < input->BiPredMESearchRange))) |
| { |
| snprintf(errortext, ET_SIZE, "\nBiPredMESearchRange must be smaller or equal SearchRange."); |
| error (errortext, 500); |
| } |
| |
| // check consistency |
| if ( input->ChromaMEEnable && !(input->ChromaMCBuffer) ) { |
| snprintf(errortext, ET_SIZE, "\nChromaMCBuffer must be set to 1 if ChromaMEEnable is set."); |
| error (errortext, 500); |
| } |
| |
| if ( input->ChromaMEEnable && input->yuv_format == YUV400) { |
| snprintf(errortext, ET_SIZE, "\nChromaMEEnable cannot be used with YUV400 color format."); |
| input->ChromaMEEnable = 0; |
| } |
| |
| if (input->EnableOpenGOP && input->PicInterlace) |
| { |
| snprintf(errortext, ET_SIZE, "Open GOP currently not supported for Field coded pictures."); |
| error (errortext, 500); |
| } |
| |
| if (input->EnableOpenGOP) |
| input->ReferenceReorder = 1; |
| |
| if (input->redundant_pic_flag) |
| { |
| if (input->PicInterlace || input->MbInterlace) |
| { |
| snprintf(errortext, ET_SIZE, "Redundant pictures cannot be used with interlaced tools."); |
| error (errortext, 500); |
| } |
| if (input->RDPictureDecision) |
| { |
| snprintf(errortext, ET_SIZE, "Redundant pictures cannot be used with RDPictureDecision."); |
| error (errortext, 500); |
| } |
| if (input->successive_Bframe) |
| { |
| snprintf(errortext, ET_SIZE, "Redundant pictures cannot be used with B frames."); |
| error (errortext, 500); |
| } |
| if (input->PrimaryGOPLength < (1 << input->NumRedundantHierarchy)) |
| { |
| snprintf(errortext, ET_SIZE, "PrimaryGOPLength must be equal or greater than 2^NumRedundantHierarchy."); |
| error (errortext, 500); |
| } |
| if (input->num_ref_frames < input->PrimaryGOPLength) |
| { |
| snprintf(errortext, ET_SIZE, "NumberReferenceFrames must be greater than or equal to PrimaryGOPLength."); |
| error (errortext, 500); |
| } |
| } |
| |
| if (input->num_ref_frames == 1 && input->successive_Bframe) |
| { |
| fprintf( stderr, "\nWarning: B slices used but only one reference allocated within reference buffer.\n"); |
| fprintf( stderr, " Performance may be considerably compromised! \n"); |
| fprintf( stderr, " 2 or more references recommended for use with B slices.\n"); |
| } |
| if ((input->HierarchicalCoding || input->BRefPictures) && input->successive_Bframe) |
| { |
| fprintf( stderr, "\nWarning: Hierarchical coding or Referenced B slices used.\n"); |
| fprintf( stderr, " Make sure that you have allocated enough references\n"); |
| fprintf( stderr, " in reference buffer to achieve best performance.\n"); |
| } |
| |
| ProfileCheck(); |
| LevelCheck(); |
| } |
| |
| void PatchInputNoFrames(void) |
| { |
| // Tian Dong: May 31, 2002 |
| // If the frames are grouped into two layers, "FramesToBeEncoded" in the config file |
| // will give the number of frames which are in the base layer. Here we let input->no_frames |
| // be the total frame numbers. |
| input->no_frames = 1+ (input->no_frames-1) * (input->NumFramesInELSubSeq+1); |
| if ( input->NumFrameIn2ndIGOP ) |
| input->NumFrameIn2ndIGOP = 1+(input->NumFrameIn2ndIGOP-1) * (input->NumFramesInELSubSeq+1); |
| FirstFrameIn2ndIGOP = input->no_frames; |
| } |
| |
| static void ProfileCheck(void) |
| { |
| if((input->ProfileIDC != 66 ) && |
| (input->ProfileIDC != 77 ) && |
| (input->ProfileIDC != 88 ) && |
| (input->ProfileIDC != FREXT_HP ) && |
| (input->ProfileIDC != FREXT_Hi10P ) && |
| (input->ProfileIDC != FREXT_Hi422 ) && |
| (input->ProfileIDC != FREXT_Hi444 )) |
| { |
| snprintf(errortext, ET_SIZE, "Profile must be baseline(66)/main(77)/extended(88) or FRExt (%d to %d).", FREXT_HP,FREXT_Hi444); |
| error (errortext, 500); |
| } |
| |
| if ((input->partition_mode) && (input->symbol_mode==CABAC)) |
| { |
| snprintf(errortext, ET_SIZE, "Data partitioning and CABAC is not supported in any profile."); |
| error (errortext, 500); |
| } |
| |
| if (input->redundant_pic_flag) |
| { |
| if (input->ProfileIDC != 66) |
| { |
| snprintf(errortext, ET_SIZE, "Redundant pictures are only allowed in Baseline profile."); |
| error (errortext, 500); |
| } |
| } |
| |
| if (input->partition_mode) |
| { |
| if (input->ProfileIDC != 88) |
| { |
| snprintf(errortext, ET_SIZE, "Data partitioning is only allowed in extended profile."); |
| error (errortext, 500); |
| } |
| } |
| |
| if (input->ChromaIntraDisable && input->FastCrIntraDecision) |
| { |
| fprintf( stderr, "\n----------------------------------------------------------------------------------\n"); |
| fprintf( stderr, "\n Warning: ChromaIntraDisable and FastCrIntraDecision cannot be combined together.\n Using only Chroma Intra DC mode.\n"); |
| fprintf( stderr, "\n----------------------------------------------------------------------------------\n"); |
| input->FastCrIntraDecision=0; |
| } |
| |
| // baseline |
| if (input->ProfileIDC == 66 ) |
| { |
| if ((input->successive_Bframe || input->BRefPictures==2) && input->PReplaceBSlice == 0) |
| { |
| snprintf(errortext, ET_SIZE, "B slices are not allowed in baseline."); |
| error (errortext, 500); |
| } |
| if (input->sp_periodicity) |
| { |
| snprintf(errortext, ET_SIZE, "SP pictures are not allowed in baseline."); |
| error (errortext, 500); |
| } |
| if (input->WeightedPrediction) |
| { |
| snprintf(errortext, ET_SIZE, "Weighted prediction is not allowed in baseline."); |
| error (errortext, 500); |
| } |
| if (input->WeightedBiprediction) |
| { |
| snprintf(errortext, ET_SIZE, "Weighted prediction is not allowed in baseline."); |
| error (errortext, 500); |
| } |
| if (input->symbol_mode == CABAC) |
| { |
| snprintf(errortext, ET_SIZE, "CABAC is not allowed in baseline."); |
| error (errortext, 500); |
| } |
| } |
| |
| // main |
| if (input->ProfileIDC == 77 ) |
| { |
| if (input->sp_periodicity) |
| { |
| snprintf(errortext, ET_SIZE, "SP pictures are not allowed in main."); |
| error (errortext, 500); |
| } |
| if (input->num_slice_groups_minus1) |
| { |
| snprintf(errortext, ET_SIZE, "num_slice_groups_minus1>0 (FMO) is not allowed in main."); |
| error (errortext, 500); |
| } |
| } |
| |
| // extended |
| if (input->ProfileIDC == 88 ) |
| { |
| if (!input->directInferenceFlag) |
| { |
| snprintf(errortext, ET_SIZE, "direct_8x8_inference flag must be equal to 1 in extended."); |
| error (errortext, 500); |
| } |
| |
| if (input->symbol_mode == CABAC) |
| { |
| snprintf(errortext, ET_SIZE, "CABAC is not allowed in extended."); |
| error (errortext, 500); |
| } |
| } |
| } |
| |
| static void LevelCheck(void) |
| { |
| if ( (input->LevelIDC>=30) && (input->directInferenceFlag==0)) |
| { |
| fprintf( stderr, "\nWarning: LevelIDC 3.0 and above require direct_8x8_inference to be set to 1. Please check your settings.\n"); |
| input->directInferenceFlag=1; |
| } |
| if ( ((input->LevelIDC<21) || (input->LevelIDC>41)) && (input->PicInterlace > 0 || input->MbInterlace > 0) ) |
| { |
| snprintf(errortext, ET_SIZE, "\nInterlace modes only supported for LevelIDC in the range of 2.1 and 4.1. Please check your settings.\n"); |
| error (errortext, 500); |
| } |
| |
| } |
| |