summaryrefslogtreecommitdiffstats
path: root/libSYS/src/cmdl_parser.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'libSYS/src/cmdl_parser.cpp')
-rw-r--r--libSYS/src/cmdl_parser.cpp584
1 files changed, 584 insertions, 0 deletions
diff --git a/libSYS/src/cmdl_parser.cpp b/libSYS/src/cmdl_parser.cpp
new file mode 100644
index 0000000..a93a2f4
--- /dev/null
+++ b/libSYS/src/cmdl_parser.cpp
@@ -0,0 +1,584 @@
+
+/* -----------------------------------------------------------------------------------------------------------
+Software License for The Fraunhofer FDK AAC Codec Library for Android
+
+© Copyright 1995 - 2012 Fraunhofer-Gesellschaft zur Förderung der angewandten Forschung e.V.
+ All rights reserved.
+
+ 1. INTRODUCTION
+The Fraunhofer FDK AAC Codec Library for Android ("FDK AAC Codec") is software that implements
+the MPEG Advanced Audio Coding ("AAC") encoding and decoding scheme for digital audio.
+This FDK AAC Codec software is intended to be used on a wide variety of Android devices.
+
+AAC's HE-AAC and HE-AAC v2 versions are regarded as today's most efficient general perceptual
+audio codecs. AAC-ELD is considered the best-performing full-bandwidth communications codec by
+independent studies and is widely deployed. AAC has been standardized by ISO and IEC as part
+of the MPEG specifications.
+
+Patent licenses for necessary patent claims for the FDK AAC Codec (including those of Fraunhofer)
+may be obtained through Via Licensing (www.vialicensing.com) or through the respective patent owners
+individually for the purpose of encoding or decoding bit streams in products that are compliant with
+the ISO/IEC MPEG audio standards. Please note that most manufacturers of Android devices already license
+these patent claims through Via Licensing or directly from the patent owners, and therefore FDK AAC Codec
+software may already be covered under those patent licenses when it is used for those licensed purposes only.
+
+Commercially-licensed AAC software libraries, including floating-point versions with enhanced sound quality,
+are also available from Fraunhofer. Users are encouraged to check the Fraunhofer website for additional
+applications information and documentation.
+
+2. COPYRIGHT LICENSE
+
+Redistribution and use in source and binary forms, with or without modification, are permitted without
+payment of copyright license fees provided that you satisfy the following conditions:
+
+You must retain the complete text of this software license in redistributions of the FDK AAC Codec or
+your modifications thereto in source code form.
+
+You must retain the complete text of this software license in the documentation and/or other materials
+provided with redistributions of the FDK AAC Codec or your modifications thereto in binary form.
+You must make available free of charge copies of the complete source code of the FDK AAC Codec and your
+modifications thereto to recipients of copies in binary form.
+
+The name of Fraunhofer may not be used to endorse or promote products derived from this library without
+prior written permission.
+
+You may not charge copyright license fees for anyone to use, copy or distribute the FDK AAC Codec
+software or your modifications thereto.
+
+Your modified versions of the FDK AAC Codec must carry prominent notices stating that you changed the software
+and the date of any change. For modified versions of the FDK AAC Codec, the term
+"Fraunhofer FDK AAC Codec Library for Android" must be replaced by the term
+"Third-Party Modified Version of the Fraunhofer FDK AAC Codec Library for Android."
+
+3. NO PATENT LICENSE
+
+NO EXPRESS OR IMPLIED LICENSES TO ANY PATENT CLAIMS, including without limitation the patents of Fraunhofer,
+ARE GRANTED BY THIS SOFTWARE LICENSE. Fraunhofer provides no warranty of patent non-infringement with
+respect to this software.
+
+You may use this FDK AAC Codec software or modifications thereto only for purposes that are authorized
+by appropriate patent licenses.
+
+4. DISCLAIMER
+
+This FDK AAC Codec software is provided by Fraunhofer on behalf of the copyright holders and contributors
+"AS IS" and WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, including but not limited to the implied warranties
+of merchantability and fitness for a particular purpose. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
+CONTRIBUTORS BE LIABLE for any direct, indirect, incidental, special, exemplary, or consequential damages,
+including but not limited to procurement of substitute goods or services; loss of use, data, or profits,
+or business interruption, however caused and on any theory of liability, whether in contract, strict
+liability, or tort (including negligence), arising in any way out of the use of this software, even if
+advised of the possibility of such damage.
+
+5. CONTACT INFORMATION
+
+Fraunhofer Institute for Integrated Circuits IIS
+Attention: Audio and Multimedia Departments - FDK AAC LL
+Am Wolfsmantel 33
+91058 Erlangen, Germany
+
+www.iis.fraunhofer.de/amm
+amm-info@iis.fraunhofer.de
+----------------------------------------------------------------------------------------------------------- */
+
+/************************** Fraunhofer IIS FDK SysLib **********************
+
+ Author(s):
+ Description: command line parser
+
+******************************************************************************/
+
+
+
+#define _CRT_SECURE_NO_WARNINGS
+
+/* Work around for broken android toolchain: sys/types.h:137: error: 'uint64_t' does not name a type */
+#define _SYS_TYPES_H_
+
+#include <stdarg.h>
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <ctype.h>
+
+#include "cmdl_parser.h"
+#include "genericStds.h"
+
+
+
+/************************ interface ************************/
+
+static INT GetArgFromString(INT argc, TEXTCHAR* argv[], TEXTCHAR* search_string, TEXTCHAR type, TEXTCHAR* found_string , INT* switches_used);
+static void RemoveWhiteSpace(const TEXTCHAR* pReqArgs, TEXTCHAR* Removed);
+static INT CheckArg(TEXTCHAR* arg, TEXTCHAR* str, UINT numArgs, TEXTCHAR type, TEXTCHAR* current_str);
+static INT CheckForUnusedSwitches(INT argc, /*TEXTCHAR* argv[],*/ INT* switches_used);
+static INT ParseString(TEXTCHAR* str, INT*, TEXTCHAR*, TEXTCHAR*);
+static void GetNumberOfArgs(TEXTCHAR* str, INT* nArgs);
+
+static
+void removeQuotes(char *str)
+{
+ if (str[0] == '"') {
+ FDKstrcpy(str, str+1);
+ str[FDKstrlen(str)-1] = 0;
+ }
+}
+
+
+/********************** implementation *********************/
+
+INT IIS_ScanCmdl(INT argc, TEXTCHAR* argv[], const TEXTCHAR* str, ...)
+{
+ INT i = 0;
+ INT found_and_set = 0;
+ INT nArgs = 0;
+ INT* switches_used = 0;
+ INT* b_str_opt = 0;
+ TEXTCHAR* s_str = 0;
+ TEXTCHAR* c_str_type = 0;
+ TEXTCHAR* str_clean = 0;
+
+ va_list ap;
+
+ if (argc == 0 || argc == 1)
+ {
+ FDKprintf("No command line arguments\n");
+ goto bail;
+ }
+
+ str_clean = (TEXTCHAR*) FDKcalloc((unsigned int)_tcslen(str), sizeof(TEXTCHAR));
+ if (str_clean == NULL) {
+ FDKprintf("Error allocating memory line %d, file %s\n", __LINE__, __FILE__);
+ return 0;
+ }
+
+ RemoveWhiteSpace(str, str_clean );
+ GetNumberOfArgs(str_clean, &nArgs);
+
+ b_str_opt = (INT*) FDKcalloc(nArgs, sizeof(INT));
+ s_str = (TEXTCHAR*) FDKcalloc(nArgs*CMDL_MAX_ARGC, sizeof(TEXTCHAR) );
+ c_str_type = (TEXTCHAR*) FDKcalloc(nArgs, sizeof(TEXTCHAR));
+ switches_used = (INT*) FDKcalloc(argc, sizeof(INT));
+
+ if (b_str_opt == NULL || s_str == NULL || c_str_type == NULL || switches_used == NULL) {
+ FDKprintf("Error allocating memory line %d, file %s\n", __LINE__, __FILE__);
+ goto bail;
+ }
+
+ if ( ParseString( str_clean, b_str_opt, s_str, c_str_type )) {
+ goto bail;
+ }
+
+ va_start(ap, str);
+
+ for ( i = 0; i < nArgs; i++ )
+ {
+ TEXTCHAR arg[CMDL_MAX_STRLEN] = {L'\0'};
+ TEXTCHAR* p_arg = arg;
+ TEXTCHAR* current_str = &(s_str[i*CMDL_MAX_ARGC]);
+
+ if (GetArgFromString(argc, argv, current_str, c_str_type[i], arg, switches_used )
+ && !b_str_opt[i] )
+ {
+#ifdef _UNICODE
+ _ftprintf(stderr, _TEXT("\n\nError: Parsing argument for required switch '%ls'.\n" ), current_str);
+#else
+ _ftprintf(stderr, _TEXT("\n\nError: Parsing argument for required switch '%s'.\n" ), current_str);
+#endif
+ found_and_set = 0;
+ goto bail;
+ }
+ if (CheckArg(p_arg, s_str, nArgs, c_str_type[i], current_str))
+ {
+ goto bail;
+ }
+
+ switch (c_str_type[i] )
+ {
+ case 's':
+ {
+ TEXTCHAR* tmp;
+ tmp = va_arg(ap, TEXTCHAR*);
+
+ if ( arg[0] == '\0' )
+ break;
+
+ _tcsncpy( tmp, arg, CMDL_MAX_STRLEN );
+ /* Remove quotes. Windows Mobile Workaround. */
+ removeQuotes(tmp);
+ found_and_set++;
+ break;
+ }
+ case 'd':
+ {
+ INT* tmp = va_arg(ap, INT*);
+
+ if ( arg[0] == '\0' )
+ break;
+
+ *tmp = _tcstol(arg, NULL, 0);
+ found_and_set++;
+ break;
+ }
+ case 'c':
+ {
+ char* tmp = va_arg(ap, char*);
+
+ if ( arg[0] == '\0' )
+ break;
+
+ *tmp = *arg;
+ found_and_set++;
+ break;
+ }
+ case 'u':
+ {
+ UCHAR* tmp = va_arg(ap, UCHAR*);
+
+ if ( arg[0] == '\0' )
+ break;
+
+ *tmp = _tstoi(arg);
+ found_and_set++;
+ break;
+ }
+ case 'f':
+ {
+ float* tmp = (float*) va_arg( ap,double*);
+
+ if ( arg[0] == '\0' )
+ break;
+
+ *tmp = (float) _tstof(arg);
+ found_and_set++;
+ break;
+ }
+ case 'y': // support 'data type double'
+ {
+ double* tmp = (double*) va_arg( ap,double*);
+ // use sscanf instead _tstof because of gcc
+ //_tstof(arg,"%lf",tmp); // '%lf' reads as double
+ *tmp = _tstof(arg); // '%lf' reads as double
+ found_and_set++;
+ break;
+ }
+ case '1':
+ {
+
+ INT* tmp = va_arg( ap, INT*);
+
+ if ( arg[0] == '\0' )
+ break;
+
+ *tmp = 1;
+ found_and_set++;
+ break;
+ }
+
+ default:
+ FDKprintfErr("Bug: unsupported data identifier \"%c\"\n", c_str_type[i]);
+ break;
+
+ }
+
+ }
+
+ va_end(ap);
+
+ CheckForUnusedSwitches(argc, /*argv,*/ switches_used);
+
+bail:
+ if (b_str_opt) FDKfree(b_str_opt);
+ if (s_str) FDKfree(s_str);
+ if (c_str_type) FDKfree(c_str_type);
+ if (str_clean) FDKfree(str_clean);
+ if (switches_used) FDKfree(switches_used);
+
+ return found_and_set;
+}
+
+
+void GetNumberOfArgs(TEXTCHAR* str, INT* nArgs)
+{
+ UINT i = 0;
+ for ( i = 0; i < _tcslen(str); ++i )
+ {
+ if ( str[i] == '%')
+ *nArgs+= 1;
+ }
+
+}
+
+INT ParseString(TEXTCHAR* str, INT* b_str_opt, TEXTCHAR* s_str, TEXTCHAR* c_str_type )
+{
+ UINT i = 0;
+ INT argCounter = 0;
+
+ TEXTCHAR* str_start = 0;
+ TEXTCHAR* str_stop = 0;
+
+
+ str_start = str;
+ str_stop = str_start;
+
+ for ( i = 0; i < _tcslen(str) - 1; ++i )
+ {
+ if ( str[i] == '%' ) /* We have an Argument */
+ {
+ if ( argCounter )
+ {
+ if ( b_str_opt[argCounter-1] )
+ str_start = str_stop + 3;
+
+ else
+ str_start = str_stop + 2;
+ }
+
+ /* Save Argument type */
+ c_str_type[argCounter] = str[i+1];
+
+ if ( *str_start == '(' ) /* Optional Argument */
+ {
+ b_str_opt[argCounter] = 1;
+ str_start++;
+ }
+
+ /* Save Argument */
+ str[i] = '\0';
+
+ _tcsncpy(&(s_str[argCounter*CMDL_MAX_ARGC]), str_start, CMDL_MAX_ARGC );
+
+ str[i] = '%';
+
+ str_stop = &(str[i]);
+
+ if ( b_str_opt[argCounter] )
+ {
+ if ( i+2 > ( _tcslen(str) -1 ))
+ {
+ _ftprintf(stderr,_TEXT("\n\nInternal Parser Error: Strlen Problem\n") );
+ return 1;
+ }
+ if ( str[i+2] != ')' )
+ {
+ _ftprintf(stderr,_TEXT("\n\nInternal Parser Error: Missing bracket ')'\n") );
+ return 1;
+ }
+
+ }
+
+
+ argCounter++;
+ }
+
+
+ }
+
+ return 0;
+ }
+
+
+
+
+void RemoveWhiteSpace(const TEXTCHAR* pReqArgs, TEXTCHAR* pRemoved)
+{
+ UINT i = 0;
+ INT k = 0;
+ UINT len = (UINT)_tcslen(pReqArgs);
+
+
+ for ( i = 0; i < len; ++i )
+ {
+
+ if ( pReqArgs[i] != ' ' )
+ {
+ pRemoved[k] = pReqArgs[i];
+ k++;
+ }
+ }
+}
+
+
+INT GetArgFromString(INT argc, TEXTCHAR* argv[], TEXTCHAR* search_string, TEXTCHAR type, TEXTCHAR* found_string, INT* sw_used )
+{
+ INT i = 0;
+
+ for (i = 1; i < argc; ++i ) {
+ if ( !_tcscmp(search_string, argv[i]) ) /* Strings are equal */
+ {
+ if ( type == '1' ) /* Switch without argument */
+ {
+ _tcsncpy( found_string, _TEXT("1"), 1);
+ sw_used[i] = 1;
+ return 0;
+
+ }
+
+ if ( i == (argc - 1)) /* We have %s or %d but are finished*/
+ return 1;
+
+ if ( _tcslen(argv[i+1]) > CMDL_MAX_STRLEN )
+ {
+#ifdef _UNICODE
+ _ftprintf (stderr,_TEXT("Warning: Ignoring argument for switch '%ls'. "), search_string );
+#else
+ _ftprintf (stderr,_TEXT("Warning: Ignoring argument for switch '%s'. "), search_string );
+#endif
+ _ftprintf (stderr,_TEXT("Argument is too LONG.\n") );
+ return 1;
+ }
+ else
+ {
+ _tcsncpy( found_string, argv[i+1], CMDL_MAX_STRLEN);
+ sw_used[i] = 1;
+ sw_used[i+1] = 1;
+ return 0;
+ }
+ }
+ }
+ return 1;
+}
+
+
+
+INT CheckArg(TEXTCHAR* arg, TEXTCHAR* str, UINT numArgs, TEXTCHAR type, TEXTCHAR* cur_str)
+{
+ UINT i = 0;
+
+ /* No argument given-> return */
+ if (arg[0] == '\0')
+ return 0;
+
+
+ /* Check if arg is switch */
+ for ( i = 0; i < numArgs; ++i )
+ {
+ if (!_tcscmp(arg, &(str[i*CMDL_MAX_ARGC])))
+ {
+#ifdef _UNICODE
+ _ftprintf(stderr, _TEXT("\n\nError: Argument '%ls' for switch '%ls' is not valid \n" ), arg, cur_str );
+#else
+ _ftprintf(stderr, _TEXT("\n\nError: Argument '%s' for switch '%s' is not valid \n" ), arg, cur_str );
+#endif
+ return 1;
+ }
+
+ }
+ /* Check if type is %d but a string is given */
+
+ for ( i = 0; i < _tcslen(arg); ++i )
+ {
+ if ( (type == 'd') && !_istdigit(arg[i]) && arg[i] != 'x' )
+ {
+#ifdef _UNICODE
+ _ftprintf(stderr, _TEXT("\n\nError: Argument '%ls' for switch '%ls' is not a valid number.\n" ), arg, cur_str);
+#else
+ _ftprintf(stderr, _TEXT("\n\nError: Argument '%s' for switch '%s' is not a valid number.\n" ), arg, cur_str);
+#endif
+ return 1;
+ }
+ }
+
+
+ return 0;
+}
+
+
+INT CheckForUnusedSwitches(INT argc, /*TEXTCHAR* argv[],*/ INT* switches_used)
+{
+ INT i = 0;
+
+ for( i = 1; i < argc; ++i )
+ {
+ if ( !switches_used[i] )
+ {
+ ++i;
+ }
+ }
+
+ return 0;
+}
+
+
+
+static char line[CMDL_MAX_STRLEN*CMDL_MAX_ARGC];
+static char *argv_ptr[CMDL_MAX_ARGC];
+#ifdef CMDFILE_PREFIX
+static char tmp[256]; /* this array is used to store the prefix and the filepath/name */
+#endif
+
+int IIS_ProcessCmdlList(const char* param_filename, int (*pFunction)(int, TEXTCHAR**))
+{
+ /* static to reduce required stack size */
+
+ FDKFILE *config_fp;
+ int argc;
+ char *line_ptr;
+
+#ifdef CMDFILE_PREFIX
+ FDKstrcpy(tmp, CMDFILE_PREFIX);
+ FDKstrcpy(tmp+FDKstrlen(CMDFILE_PREFIX), param_filename);
+ /* Open the file with command lines */
+ config_fp = FDKfopen(tmp, "r");
+#else
+ /* Open the file with command lines */
+ config_fp = FDKfopen(param_filename, "r");
+#endif
+
+ if(config_fp == NULL)
+ {
+#ifdef CMDFILE_PREFIX
+ FDKprintf("\ncould not open config file %s", tmp);
+#else
+ FDKprintf("\ncould not open config file %s", param_filename);
+#endif
+ return 1;
+ }
+
+ /* Obtain a command line from config file */
+ while (FDKfgets(line, CMDL_MAX_STRLEN*CMDL_MAX_ARGC, config_fp) != NULL)
+ {
+ argc = 1;
+
+ /* Eat \n */
+ line_ptr = (char*)FDKstrchr(line, '\n');
+ if (line_ptr != NULL)
+ *line_ptr = ' ';
+
+ line_ptr = line;
+
+ /* Scan the line and put the command line params into argv */
+ do {
+ /* Skip consecutive blanks. */
+ while (*line_ptr == ' ' && line_ptr < line+CMDL_MAX_STRLEN)
+ line_ptr++;
+ /* Assign argument. TODO: maybe handle quotes */
+ argv_ptr[argc] = line_ptr;
+ /* Get pointer to next blank. */
+ line_ptr = (char*)FDKstrchr(line_ptr, ' ');
+ /* */
+ if (line_ptr != NULL) {
+ /* Null terminate */
+ *line_ptr = 0;
+ /* Skip former blank (now null character) */
+ line_ptr++;
+ /* Advance argument counter */
+ }
+ argc++;
+ } while ( line_ptr != NULL && argc < CMDL_MAX_ARGC);
+
+ /* call "would be main()" */
+ if (argc > 2 && *argv_ptr[1] != '#' && FDKstrlen(argv_ptr[1])>1)
+ {
+ int retval;
+
+ retval = (*pFunction)(argc, argv_ptr);
+
+ FDKprintf("main returned %d\n", retval);
+ }
+ }
+
+ FDKfclose(config_fp);
+ return 0;
+}
+