📄 getarg.c
字号:
/**************************************************************************** Routines to grab the parameters from the command line : ** All the routines except the main one, starts with GA (Get Arguments) to ** prevent from names conflicts. ** It is assumed in these routine that any pointer, for any type has the ** same length (i.e. length of int pointer is equal to char pointer etc.) ** ** The following routines are available in this module: ** 1. int GAGetArgs(argc, argv, CtrlStr, Variables...) ** where argc, argv as received on entry. ** CtrlStr is the contrl string (see below) ** Variables are all the variables to be set according to CtrlStr. ** Note that all the variables MUST be transfered by address. ** return 0 on correct parsing, otherwise error number (see GetArg.h). ** 2. GAPrintHowTo(CtrlStr) ** Print the control string to stderr, in the correct format needed. ** This feature is very useful in case of error during GetArgs parsing. ** Chars equal to SPACE_CHAR are not printed (regular spaces are NOT ** allowed, and so using SPACE_CHAR you can create space in PrintHowTo). ** 3. GAPrintErrMsg(Error) ** Print the error to stderr, according to Error (usually returned by ** GAGetArgs). ** ** CtrlStr format: ** The control string passed to GetArgs controls the way argv (argc) are ** parsed. Each entry in this string must not have any spaces in it. The ** First Entry is the name of the program which is usually ignored except ** when GAPrintHowTo is called. All the other entries (except the last one ** which we will come back to it later) must have the following format: ** 1. One letter which sets the option letter. ** 2. '!' or '%' to determines if this option is really optional ('%') or ** it must exists ('!')... ** 3. '-' allways. ** 4. Alpha numeric string, usually ignored, but used by GAPrintHowTo to ** print the meaning of this input. ** 5. Sequences starts with '!' or '%'. Again if '!' then this sequence ** must exists (only if its option flag is given of course), and if '%' ** it is optional. Each sequence will continue with one or two ** characters which defines the kind of the input: ** a. d, x, o, u - integer is expected (decimal, hex, octal base or ** unsigned). ** b. D, X, O, U - long integer is expected (same as above). ** c. f - float number is expected. ** d. F - double number is expected. ** e. s - string is expected. ** f. *? - any number of '?' kind (d, x, o, u, D, X, O, U, f, F, s) ** will match this one. If '?' is numeric, it scans until ** none numeric input is given. If '?' is 's' then it scans ** up to the next option or end of argv. ** ** If the last parameter given in the CtrlStr, is not an option (i.e. the ** second char is not in ['!', '%'] and the third one is not '-'), all what ** remained from argv is linked to it. ** ** The variables passed to GAGetArgs (starting from 4th parameter) MUST ** match the order of the CtrlStr: ** For each option, one integer address must be passed. This integer must ** initialized by 0. If that option is given in the command line, it will ** be set to one. ** In addition, the sequences that might follow an option require the ** following parameters to pass: ** 1. d, x, o, u - pointer to integer (int *). ** 2. D, X, O, U - pointer to long (long *). ** 3. f - pointer to float (float *). ** 4. F - pointer to double (double *). ** 5. s - pointer to char (char *). NO allocation is needed! ** 6. *? - TWO variables are passed for each wild request. the first ** one is (address of) integer, and it will return number of ** parameters actually matched this sequence, and the second ** one is a pointer to pointer to ? (? **), and will return an ** address to a block of pointers to ? kind, terminated with ** NULL pointer. NO pre-allocation is needed. ** note that these two variables are pretty like the argv/argc ** pair... ** ** Examples: ** ** "Example1 i%-OneInteger!d s%-Strings!*s j%- k!-Float!f Files" ** Will match: Example1 -i 77 -s String1 String2 String3 -k 88.2 File1 File2** or match: Example1 -s String1 -k 88.3 -i 999 -j ** but not: Example1 -i 77 78 (option i expects one integer, k must be). ** Note the option k must exists, and that the order of the options is not ** not important. In the first examples File1 & File2 will match the Files ** in the command line. ** A call to GAPrintHowTo with this CtrlStr will print to stderr: ** Example1 [-i OneIngeter] [-s Strings...] [-j] -k Float Files... ** ** Notes: ** ** 1. This module assumes that all the pointers to all kind of data types ** have the same length and format, i.e. sizeof(int *) == sizeof(char *).** ** Gershon Elber Ver 0.2 Mar 88 ****************************************************************************** History: ** 11 Mar 88 - Version 1.0 by Gershon Elber. ****************************************************************************/#ifdef __MSDOS__#include <stdlib.h>#include <alloc.h>#endif /* __MSDOS__ */#ifdef USE_VARARGS#include <varargs.h>#endif /* USE_VARARGS */#ifndef __MSDOS__#include <stdlib.h>#endif#include <stdio.h>#include <string.h>#include "getarg.h"#define MYMALLOC /* If no "MyAlloc" routine elsewhere define this. */#define MAX_PARAM 100 /* maximum number of parameters allowed. */#define CTRL_STR_MAX_LEN 1024#define SPACE_CHAR '|' /* The character not to print using HowTo. */#ifndef TRUE#define TRUE -1#define FALSE 0#endif /* TRUE */#define ARG_OK 0#define ISSPACE(x) ((x) <= ' ') /* Not conventional - but works fine! *//* The two characters '%' and '!' are used in the control string: */#define ISCTRLCHAR(x) (((x) == '%') || ((x) == '!'))static char *GAErrorToken;/* On error code, ErrorToken is set to point on it.*/static int GATestAllSatis(char *CtrlStrCopy, char *CtrlStr, int *argc, char ***argv, int *Parameters[MAX_PARAM], int *ParamCount);static int GAUpdateParameters(int *Parameters[], int *ParamCount, char *Option, char *CtrlStrCopy, char *CtrlStr, int *argc, char ***argv);static int GAGetParmeters(int *Parameters[], int *ParamCount, char *CtrlStrCopy , char *Option, int *argc, char ***argv);static int GAGetMultiParmeters(int *Parameters[], int *ParamCount, char *CtrlStrCopy, int *argc, char ***argv);static void GASetParamCount(char *CtrlStr, int Max, int *ParamCount);static void GAByteCopy(char *Dst, char *Src, unsigned n);static int GAOptionExists(int argc, char **argv);#ifdef MYMALLOCstatic char *MyMalloc(unsigned size);#endif /* MYMALLOC *//**************************************************************************** Routine to access the command line argument and interpret them: ** Return ARG_OK (0) is case of succesfull parsing, error code else... ****************************************************************************/#ifdef USE_VARARGSint GAGetArgs(int va_alist, ...){ va_list ap; int argc, i, Error = FALSE, ParamCount = 0, *Parameters[MAX_PARAM]; /* Save here parameter addresses. */ char **argv, *CtrlStr, *Option, CtrlStrCopy[CTRL_STR_MAX_LEN]; va_start(ap); argc = va_arg(ap, int); argv = va_arg(ap, char **); CtrlStr = va_arg(ap, char *); va_end(ap); strcpy(CtrlStrCopy, CtrlStr); /* Using base address of parameters we access other parameters addr: */ /* Note that me (for sure!) samples data beyond the current function */ /* frame, but we accesson these set address only by demand. */ for (i = 1; i <= MAX_PARAM; i++) Parameters[i-1] = va_arg(ap, int *);#elseint GAGetArgs(int argc, char **argv, char *CtrlStr, ...){ int i, Error = FALSE, ParamCount = 0, *Parameters[MAX_PARAM]; /* Save here parameter addresses. */ char *Option, CtrlStrCopy[CTRL_STR_MAX_LEN]; strcpy(CtrlStrCopy, CtrlStr); /* Using base address of parameters we access other parameters addr: */ /* Note that me (for sure!) samples data beyond the current function */ /* frame, but we accesson these set address only by demand. */ for (i = 1; i <= MAX_PARAM; i++) Parameters[i-1] = (int *) *(i + &CtrlStr);#endif /* USE_VARARG */ --argc; argv++; /* Skip the program name (first in argv/c list). */ while (argc >= 0) { if (!GAOptionExists(argc, argv)) break; /* The loop. */ argc--; Option = *argv++; if ((Error = GAUpdateParameters(Parameters, &ParamCount, Option, CtrlStrCopy, CtrlStr, &argc, &argv)) != FALSE) return Error; } /* Check for results and update trail of command line: */ return GATestAllSatis(CtrlStrCopy, CtrlStr, &argc, &argv, Parameters, &ParamCount);}/**************************************************************************** Routine to search for unsatisfied flags - simply scan the list for !- ** sequence. Before this scan, this routine updates the rest of the command ** line into the last two parameters if it is requested by the CtrlStr ** (last item in CtrlStr is NOT an option). ** Return ARG_OK if all satisfied, CMD_ERR_AllSatis error else. ****************************************************************************/static int GATestAllSatis(char *CtrlStrCopy, char *CtrlStr, int *argc, char ***argv, int *Parameters[MAX_PARAM], int *ParamCount){ int i; static char *LocalToken = NULL; /* If LocalToken is not initialized - do it now. Note that this string */ /* should be writable as well so we can not assign it directly. */ if (LocalToken == NULL) { LocalToken = (char *) malloc(3); strcpy(LocalToken, "-?"); } /* Check is last item is an option. If not then copy rest of command */ /* line into it as 1. NumOfprm, 2. pointer to block of pointers. */ for (i = strlen(CtrlStr) - 1; i > 0 && !ISSPACE(CtrlStr[i]); i--); if (!ISCTRLCHAR(CtrlStr[i + 2])) { GASetParamCount(CtrlStr, i, ParamCount); /* Point in correct prm.. */ *Parameters[(*ParamCount)++] = *argc; GAByteCopy((char *) Parameters[(*ParamCount)++], (char *) argv, sizeof(char *)); } i = 0; while (++i < (int)strlen(CtrlStrCopy)) if ((CtrlStrCopy[i] == '-') && (CtrlStrCopy[i-1] == '!')) { GAErrorToken = LocalToken; LocalToken[1] = CtrlStrCopy[i-2]; /* Set the corrent flag. */ return CMD_ERR_AllSatis; } return ARG_OK;}/**************************************************************************** Routine to update the parameters according to the given Option: ****************************************************************************/static int GAUpdateParameters(int *Parameters[], int *ParamCount, char *Option, char *CtrlStrCopy, char *CtrlStr, int *argc, char ***argv){ int i, BooleanTrue = Option[2] != '-'; if (Option[0] != '-') { GAErrorToken = Option; return CMD_ERR_NotAnOpt; } i = 0; /* Scan the CtrlStrCopy for that option: */ while (i + 2 < (int)strlen(CtrlStrCopy)) { if ((CtrlStrCopy[i] == Option[1]) && (ISCTRLCHAR(CtrlStrCopy[i + 1])) && (CtrlStrCopy[i+2] == '-')) { /* We found that option! */ break; } i++; } if (i + 2 >= (int)strlen(CtrlStrCopy)) { GAErrorToken = Option; return CMD_ERR_NoSuchOpt; } /* If we are here, then we found that option in CtrlStr - Strip it off: */ CtrlStrCopy[i] = CtrlStrCopy[i + 1] = CtrlStrCopy[i + 2] = (char) ' '; GASetParamCount(CtrlStr, i, ParamCount);/*Set it to point in correct prm.*/ i += 3; /* Set boolean flag for that option. */ *Parameters[(*ParamCount)++] = BooleanTrue; if (ISSPACE(CtrlStrCopy[i])) return ARG_OK; /* Only a boolean flag is needed. */ /* Skip the text between the boolean option and data follows: */ while (!ISCTRLCHAR(CtrlStrCopy[i])) i++; /* Get the parameters and return the appropriete return code: */ return GAGetParmeters(Parameters, ParamCount, &CtrlStrCopy[i], Option, argc, argv);}/**************************************************************************** Routine to get parameters according to the CtrlStr given from argv/c : ****************************************************************************/static int GAGetParmeters(int *Parameters[], int *ParamCount, char *CtrlStrCopy , char *Option, int *argc, char ***argv){ int i = 0, ScanRes; while (!(ISSPACE(CtrlStrCopy[i]))) { switch (CtrlStrCopy[i+1]) { case 'd': /* Get signed integers. */ ScanRes = sscanf(*((*argv)++), "%d", (int *) Parameters[(*ParamCount)++]); break; case 'u': /* Get unsigned integers. */ ScanRes = sscanf(*((*argv)++), "%u", (unsigned *) Parameters[(*ParamCount)++]); break; case 'x': /* Get hex integers. */ ScanRes = sscanf(*((*argv)++), "%x", (int *) Parameters[(*ParamCount)++]); break; case 'o': /* Get octal integers. */ ScanRes = sscanf(*((*argv)++), "%o", (int *) Parameters[(*ParamCount)++]); break; case 'D': /* Get signed long integers. */ ScanRes = sscanf(*((*argv)++), "%ld", (long *) Parameters[(*ParamCount)++]); break; case 'U': /* Get unsigned long integers. */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -