⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 getarg.c

📁 在EM85XX
💻 C
📖 第 1 页 / 共 2 页
字号:
/****************************************************************************  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 + -