📄 configfile.c
字号:
error (errortext, 300);
}
}
}
printf ("\n");
PatchInp();
if (params->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 > 100000)
{
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);
printf ("\n\tParsing error in config file: Parameter Name '%s' not recognized.", items[i]);
continue;
}
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 (params, &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[3])
{
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)
{
int cur_qp = * (int *) (Map[i].Place);
int min_qp = (int) (Map[i].min_limit - bitdepth_qp_scale[0]);
int max_qp = (int) Map[i].max_limit;
if (( cur_qp < min_qp ) || ( cur_qp > max_qp ))
{
snprintf(errortext, ET_SIZE, "Error in input parameter %s. Check configuration file. Value should be in [%d, %d] range.", Map[i].TokenName, min_qp, max_qp );
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;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -