/* @(#)getargs.c 2.76 17/09/25 Copyright 1985, 1988, 1994-2017 J. Schilling */ #include #ifndef lint static UConst char sccsid[] = "@(#)getargs.c 2.76 17/09/25 Copyright 1985, 1988, 1994-2017 J. Schilling"; #endif #define NEW /* * Copyright (c) 1985, 1988, 1994-2017 J. Schilling * * 1.3.88 Start implementation of release 2 */ /* * Parse arguments on a command line. * Format string type specifier (appearing directly after flag name): * '' BOOL size int set to TRUE (1) * '%' Extended format, next char determines the type: * '%0' BOOL with size modifier set to FALSE (0) * '%1' BOOL with size modifier set to TRUE(1) * '*' string * '?' char * '#' number * '&' call function for any type flag * '~' call function for BOOLEAN flag * '+' inctype +++ NEU +++ * * The format string 'f* ' may be used to disallow -ffoo for f* * The same behavior is implemented for 'f# ', 'f? ' and 'f& '. * The ' ' needs to immediately follow the format type specifier. * * The format string 'f*_' may be used to disallow -f foo for f* * The same behavior is implemented for 'f#_', 'f?_' and 'f&_'. * The '_' needs to immediately follow the format type specifier. * This also allows to implement optional arguments to options. * Note: 'f#_' is only implemented for orthogonality, -f will * be converted to an integer value of 0. * * The '#', '+' and '%[01]' types may have size modifiers added: * 'c'/'C' char * 's'/'S' short * 'i'/'I' int (default == no size modifier) * 'l'/'L' long * 'll'/'LL' long long */ /* * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * See the file CDDL.Schily.txt in this distribution for details. * A copy of the CDDL is also available via the Internet at * http://www.opensource.org/licenses/cddl1.txt * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file CDDL.Schily.txt from this distribution. */ /* LINTLIBRARY */ #include #include #include #include #include #include #include #include /* * Various return values */ #define RETMAX 2 /* Max. value for getargerror() */ #define FLAGDELIM 2 /* "--" stopped flag processing */ #define NOTAFLAG 1 /* Not a flag type argument */ #define NOARGS 0 /* No more args */ #define BADFLAG (-1) /* Not a valid flag argument */ #define BADFMT (-2) /* Error in format string */ #define NOTAFILE (-3) /* Seems to be a flag type arg */ #define RETMIN (-3) /* Min. value for getargerror() */ LOCAL char *retnames[] = { "NOTAFILE", "BADFMT", "BADFLAG", "NOARGS", "NOTAFLAG", "FLAGDELIM", }; #define RNAME(a) (retnames[(a)-RETMIN]) #define SCANONLY 0 /* Do not try to set argument values */ #define SETARGS 1 /* Set argument values from cmdline */ #define CHECKARGS 2 /* To not update argv and argc */ #define ARGVECTOR 4 /* Use vector instead of list interface */ #define NOEQUAL 8 /* -opt=val not allowed for -opt val */ #define SINGLEARG 16 /* Last singlechar opt may have arg */ LOCAL struct ga_props *_getprops __PR((struct ga_props *)); int _getargs __PR((int *, char *const **, void *, int, struct ga_props *, va_list)); LOCAL int dofile __PR((int *, char *const **, const char **, struct ga_props *)); LOCAL int doflag __PR((int *, char *const **, const char *, void *, int, va_list)); LOCAL int dosflags __PR((const char *, void *, int *, char *const **, int, va_list)); LOCAL int checkfmt __PR((const char *)); LOCAL int checkeql __PR((const char *)); LOCAL va_list va_dummy; LOCAL char fmtspecs[] = "#?*&~+%"; #define isfmtspec(c) (strchr(fmtspecs, c) != NULL) LOCAL struct ga_props props_default = { 0, 0, sizeof (struct ga_props) }; LOCAL struct ga_props props_posix = { GAF_POSIX_DEFAULT, 0, sizeof (struct ga_props) }; EXPORT int _getarginit(props, size, flags) struct ga_props *props; size_t size; UInt32_t flags; { if (size > sizeof (struct ga_props)) return (-1); /* * GAF_POSIX may be used as an alias for the flags that currently * define POSIX behavior. */ if (flags == GAF_POSIX) flags = GAF_POSIX_DEFAULT; props->ga_flags = flags; props->ga_oflags = 0; props->ga_size = size; return (0); } LOCAL struct ga_props * _getprops(props) struct ga_props *props; { if (props == GA_NO_PROPS) props = &props_default; else if (props == GA_POSIX_PROPS) props = &props_posix; props->ga_oflags = 0; /* * GAF_POSIX may be used as an alias for the flags that currently * define POSIX behavior. */ if (props->ga_flags == GAF_POSIX) props->ga_flags = GAF_POSIX_DEFAULT; return (props); } /* * get flags until a non flag type argument is reached (old version) */ /* VARARGS3 */ #ifdef PROTOTYPES EXPORT int getargs(int *pac, char *const **pav, const char *fmt, ...) #else EXPORT int getargs(pac, pav, fmt, va_alist) int *pac; char **pav[]; char *fmt; va_dcl #endif { va_list args; int ret; #ifdef PROTOTYPES va_start(args, fmt); #else va_start(args); #endif ret = _getargs(pac, pav, (void *)fmt, SETARGS, GA_NO_PROPS, args); va_end(args); return (ret); } /* * get flags until a non flag type argument is reached (list version) */ /* VARARGS4 */ #ifdef PROTOTYPES EXPORT int getlargs(int *pac, char *const **pav, struct ga_props *props, const char *fmt, ...) #else EXPORT int getlargs(pac, pav, props, fmt, va_alist) int *pac; char **pav[]; struct ga_props *props; char *fmt; va_dcl #endif { va_list args; int ret; #ifdef PROTOTYPES va_start(args, fmt); #else va_start(args); #endif ret = _getargs(pac, pav, (void *)fmt, SETARGS, props, args); va_end(args); return (ret); } /* * get flags until a non flag type argument is reached (vector version) */ EXPORT int getvargs(pac, pav, props, vfmt) int *pac; char * const *pav[]; struct ga_props *props; struct ga_flags *vfmt; { return (_getargs(pac, pav, vfmt, SETARGS | ARGVECTOR, props, va_dummy)); } /* * get all flags on the command line, do not stop on files (old version) */ /* VARARGS3 */ #ifdef PROTOTYPES EXPORT int getallargs(int *pac, char *const **pav, const char *fmt, ...) #else EXPORT int getallargs(pac, pav, fmt, va_alist) int *pac; char **pav[]; char *fmt; va_dcl #endif { va_list args; int ret; #ifdef PROTOTYPES va_start(args, fmt); #else va_start(args); #endif for (; ; (*pac)--, (*pav)++) { if ((ret = _getargs(pac, pav, (void *)fmt, SETARGS, GA_NO_PROPS, args)) < NOTAFLAG) break; } va_end(args); return (ret); } /* * get all flags on the command line, do not stop on files (list version) */ /* VARARGS4 */ #ifdef PROTOTYPES EXPORT int getlallargs(int *pac, char *const **pav, struct ga_props *props, const char *fmt, ...) #else EXPORT int getlallargs(pac, pav, props, fmt, va_alist) int *pac; char **pav[]; struct ga_props *props; char *fmt; va_dcl #endif { va_list args; int ret; #ifdef PROTOTYPES va_start(args, fmt); #else va_start(args); #endif props = _getprops(props); for (; ; (*pac)--, (*pav)++) { if ((ret = _getargs(pac, pav, (void *)fmt, SETARGS, props, args)) < NOTAFLAG) break; /* * The default is to parse all options on the command line and * to let "--" only make the next argument a non-option. */ if (ret == FLAGDELIM && (props->ga_flags & GAF_DELIM_DASHDASH)) break; } va_end(args); return (ret); } /* * get all flags on the command line, do not stop on files (vector version) */ EXPORT int getvallargs(pac, pav, props, vfmt) int *pac; char * const *pav[]; struct ga_props *props; struct ga_flags *vfmt; { int ret; props = _getprops(props); for (; ; (*pac)--, (*pav)++) { if ((ret = _getargs(pac, pav, vfmt, SETARGS | ARGVECTOR, props, va_dummy)) < NOTAFLAG) break; /* * The default is to parse all options on the command line and * to let "--" only make the next argument a non-option. */ if (ret == FLAGDELIM && (props->ga_flags & GAF_DELIM_DASHDASH)) break; } return (ret); } /* * get next non flag type argument (i.e. a file) (old version) * getfiles() is a dry run getargs() */ EXPORT int getfiles(pac, pav, fmt) int *pac; char *const *pav[]; const char *fmt; { return (_getargs(pac, pav, (void *)fmt, SCANONLY, GA_NO_PROPS, va_dummy)); } /* * get next non flag type argument (i.e. a file) (list version) * getlfiles() is a dry run getlargs() */ EXPORT int getlfiles(pac, pav, props, fmt) int *pac; char *const *pav[]; struct ga_props *props; const char *fmt; { return (_getargs(pac, pav, (void *)fmt, SCANONLY, props, va_dummy)); } /* * get next non flag type argument (i.e. a file) (vector version) * getvfiles() is a dry run getvargs() */ EXPORT int getvfiles(pac, pav, props, vfmt) int *pac; char *const *pav[]; struct ga_props *props; struct ga_flags *vfmt; { return (_getargs(pac, pav, vfmt, SCANONLY | ARGVECTOR, props, va_dummy)); } /* * check args until the next non flag type argmument is reached * *pac is decremented, *pav is incremented so that the * non flag type argument is at *pav[0] * * return code: * +2 FLAGDELIM "--" stopped flag processing * +1 NOTAFLAG not a flag type argument (is a file) * 0 NOARGS no more args * -1 BADFLAG a non-matching flag type argument * -2 BADFMT bad syntax in format string */ /* LOCAL int */ EXPORT int _getargs(pac, pav, vfmt, flags, props, args) register int *pac; register char *const **pav; void *vfmt; int flags; struct ga_props *props; va_list args; { const char *argp; int ret; props = _getprops(props); if (props->ga_flags & GAF_NO_EQUAL) flags |= NOEQUAL; if (props->ga_flags & GAF_SINGLEARG) flags |= SINGLEARG; for (; *pac > 0; (*pac)--, (*pav)++) { argp = **pav; ret = dofile(pac, pav, &argp, props); if (ret != NOTAFILE) return (ret); ret = doflag(pac, pav, argp, vfmt, flags, args); if (ret != NOTAFLAG) return (ret); } return (NOARGS); } /* * check if *pargp is a file type argument */ LOCAL int dofile(pac, pav, pargp, props) register int *pac; register char *const **pav; const char **pargp; struct ga_props *props; { register const char *argp = *pargp; if (argp[0] == '-') { /* * "-" is a special non flag type argument * that usually means take stdin instead of a named file */ if (argp[1] == '\0') return (NOTAFLAG); /* * "--" is a prefix to take the next argument * as non flag type argument * NOTE: POSIX requires "--" to indicate the end of the * flags on the command line. Programs that like to be * 100% POSIX compliant call only get[lv]args() once * and then process the list of files from cav[0]. */ if (argp[1] == '-' && argp[2] == '\0') { if (--(*pac) > 0) { (*pav)++; return (FLAGDELIM); } else { return (NOARGS); } } } /* * Now check if it may be flag type argument at all. * Flag type arguments begin with a '-', a '+' or contain a '=' * i.e. -flag +flag or flag= * The behavior here may be controlled by props->ga_flags to * allow getargs() to e.g. behave fully POSIX compliant. */ if (argp[0] != '-') { if (argp[0] == '+' && (props->ga_flags & GAF_NO_PLUS) == 0) return (NOTAFILE); /* This is a flag type arg */ /* * If 'flag=value' is not allowed at all, speed things up * and do not call checkeql() to check for '='. */ if (props->ga_flags & GAF_NO_EQUAL) return (NOTAFLAG); if (checkeql(argp) && (props->ga_flags & GAF_NEED_DASH) == 0) return (NOTAFILE); /* This is a flag type arg */ return (NOTAFLAG); } return (NOTAFILE); /* This is a flag type arg */ } /* * compare argp with the format string * if a match is found store the result a la scanf in one of the * arguments pointed to in the va_list * * If (flags & SETARGS) == 0, only check arguments for getfiles() * In case that (flags & SETARGS) == 0 or that (flags & ARGVECTOR) != 0, * va_list may be a dummy argument. */ LOCAL int doflag(pac, pav, argp, vfmt, flags, oargs) int *pac; char *const **pav; register const char *argp; void *vfmt; int flags; va_list oargs; { register const char *fmt = (const char *)vfmt; struct ga_flags *flagp = vfmt; const char *fmtp; long val; Llong llval; int singlecharflag = 0; BOOL isspec; BOOL hasdash = FALSE; BOOL doubledash = FALSE; BOOL haseql = checkeql(argp); const char *sargp; const char *sfmt; va_list args; char *const *spav = *pav; int spac = *pac; void *curarg = (void *)0; sfmt = fmt; /* * flags beginning with '-' don't have to include the '-' in * the format string. * flags beginning with '+' have to include it in the format string. */ if (argp[0] == '-') { argp++; hasdash = TRUE; /* * Implement legacy support for --longopt * If we find a double dash, we do not look for combinations * of boolean single char flags. */ if (argp[0] == '-') { argp++; doubledash = TRUE; /* * Allow -- only for long options. */ if (argp[1] == '\0') { return (BADFLAG); } } } sargp = argp; /* * Initialize 'args' to the start of the argument list. * I don't know any portable way to copy an arbitrary * C object so I use a system-specific routine * (probably a macro) from stdarg.h. (Remember that * if va_list is an array, 'args' will be a pointer * and '&args' won't be what I would need for memcpy.) * It is a system requirement for SVr4 compatibility * to be able to do this assgignement. If your system * defines va_list to be an array but does not define * va_copy() you are lost. * This is needed to make sure, that 'oargs' will not * be clobbered. */ va_copy(args, oargs); if (flags & ARGVECTOR) { sfmt = fmt = flagp->ga_format; if (fmt == NULL) sfmt = fmt = ""; if (flags & SETARGS) curarg = flagp->ga_arg; } else if (flags & SETARGS) { curarg = va_arg(args, void *); } /* * check if the first flag in format string is a singlechar flag */ again: if (fmt[0] != '\0' && (fmt[1] == ',' || fmt[1] == '+' || fmt[1] == '~' || fmt[1] == '%' || fmt[1] == '\0')) singlecharflag++; /* * check the whole format string for a match */ for (;;) { for (; *fmt; fmt++, argp++) { if (*fmt == '\\') { /* * Allow "#?*&+" to appear inside a flag. * NOTE: they must be escaped by '\\' only * inside the the format string. */ fmt++; isspec = FALSE; } else { isspec = isfmtspec(*fmt); } /* * If isspec is TRUE, the arg beeing checked starts * like a valid flag. Argp now points to the rest. */ if (isspec) { /* * If *argp is '+' and we are on the * beginning of the arg that is currently * checked, this cannot be an inc type flag. */ if (*argp == '+' && argp == sargp) continue; /* * skip over to arg of flag */ if (*argp == '=') { if (flags & NOEQUAL) return (BADFLAG); argp++; } else if (*argp != '\0' && haseql) { /* * Flag and arg are not separated by a * space. * Check here for: * xxxxx=yyyyy match on '&' * Checked before: * abc=yyyyy match on 'abc&' * or 'abc*' * or 'abc#' * We come here if 'argp' starts with * the same sequence as a valid flag * and contains an equal sign. * We have tested before if the text * before 'argp' matches exactly. * At this point we have no exact match * and we only allow to match * the special pattern '&'. * We need this e.g. for 'make'. * We allow any flag type argument to * match the format string "&" to set * up a function that handles all odd * stuff that getargs will not grok. * In addition, to allow getargs to be * used for CPP type flags we allow to * match -Dabc=xyz on 'D&'. Note that * Dabc=xyz will not match 'D&'. */ if ((!hasdash && argp != sargp) || *fmt != '&') goto nextarg; } /* * The format string 'f* ' may be used * to disallow -ffoo for f* * * The same behavior is implemented for * 'f# '. 'f? ' and 'f& '. */ if (!haseql && *argp != '\0' && (fmt[0] == '*' || fmt[0] == '#' || fmt[0] == '?' || fmt[0] == '&') && fmt[1] == ' ') { goto nextarg; } /* * *arpp == '\0' || !haseql * We come here if 'argp' starts with * the same sequence as a valid flag. * This will match on the following args: * -farg match on 'f*' * -f12 match on 'f#' * +12 match on '+#' * -12 match on '#' * and all args that are separated from * their flags. * In the switch statement below, we check * if the text after 'argp' (if *argp != 0) or * the next arg is a valid arg for this flag. */ break; } else if (*fmt == *argp) { if (argp[1] == '\0' && (fmt[1] == '\0' || fmt[1] == ',')) { if (flags & SETARGS) *((int *)curarg) = TRUE; return (checkfmt(fmt)); /* XXX */ } } else { /* * skip over to next format identifier * & reset arg pointer */ nextarg: while (*fmt != ',' && *fmt != '\0') { /* function has extra arg on stack */ if ((*fmt == '&' || *fmt == '~') && (flags & (SETARGS|ARGVECTOR)) == SETARGS) { curarg = va_arg(args, void *); } fmt++; } argp = sargp; break; } } switch (*fmt) { case '\0': /* * Boolean type has been tested before. */ if (flags & ARGVECTOR) { if (flagp[1].ga_format != NULL) { flagp++; sfmt = fmt = flagp->ga_format; if (flags & SETARGS) curarg = flagp->ga_arg; argp = sargp; goto again; } } if (singlecharflag && !doubledash && (val = dosflags(sargp, vfmt, pac, pav, CHECKARGS | (flags & ~SETARGS), va_dummy)) == BADFLAG) { return (val); } if (singlecharflag && !doubledash && (val = dosflags(sargp, vfmt, pac, pav, flags, oargs)) != BADFLAG) { return (val); } return (BADFLAG); case ',': fmt++; if (fmt[0] == '\0') /* Should we allow "a,b,c,"? */ return (BADFMT); if (fmt[1] == ',' || fmt[1] == '+' || fmt[1] == '\0') singlecharflag++; if ((flags & (SETARGS|ARGVECTOR)) == SETARGS) curarg = va_arg(args, void *); continue; case '*': if (*argp == '\0' && fmt[1] != '_') { if (*pac > 1) { (*pac)--; (*pav)++; argp = **pav; } else { return (BADFLAG); } } if (fmt[1] == '_') /* To disallow -f foo for f* */ fmt++; else if (fmt[1] == ' ') /* To disallow -ffoo for f* */ fmt++; if (flags & SETARGS) *((const char **)curarg) = argp; return (checkfmt(fmt)); case '?': if (*argp == '\0' && fmt[1] != '_') { if (*pac > 1) { (*pac)--; (*pav)++; argp = **pav; } else { return (BADFLAG); } } if (fmt[1] == '_') /* To disallow -f c for f? */ fmt++; else if (fmt[1] == ' ') /* To disallow -fc for f? */ fmt++; /* * Allow -f '' to specify a nul character. * If more than one char arg, it * cannot be a character argument. */ if (argp[0] != '\0' && argp[1] != '\0') goto nextchance; if (flags & SETARGS) *((char *)curarg) = *argp; return (checkfmt(fmt)); case '+': /* * inc type is similar to boolean, * there is no arg in argp to convert. */ if (*argp != '\0') goto nextchance; /* * If *fmt is '+' and we are on the beginning * of the format desciptor that is currently * checked, this cannot be an inc type flag. */ if (fmt == sfmt || fmt[-1] == ',') goto nextchance; fmtp = fmt; if (fmt[1] == 'l' || fmt[1] == 'L') { if (fmt[2] == 'l' || fmt[2] == 'L') { if (flags & SETARGS) *((Llong *)curarg) += 1; fmt += 2; } else { if (flags & SETARGS) *((long *)curarg) += 1; fmt++; } } else if (fmt[1] == 's' || fmt[1] == 'S') { if (flags & SETARGS) *((short *)curarg) += 1; fmt++; } else if (fmt[1] == 'c' || fmt[1] == 'C') { if (flags & SETARGS) *((char *)curarg) += 1; fmt++; } else { if (fmt[1] == 'i' || fmt[1] == 'I') fmt++; if (flags & SETARGS) *((int *)curarg) += 1; } return (checkfmt(fmt)); case '%': /* * inc type is similar to boolean, * there is no arg in argp to convert. */ if (*argp != '\0') goto nextchance; fmt++; if (*fmt == '1') val = TRUE; else if (*fmt == '0') val = FALSE; else goto nextchance; fmtp = fmt; llval = (Llong)val; if (fmt[1] == 'l' || fmt[1] == 'L') { if (fmt[2] == 'l' || fmt[2] == 'L') { if (flags & SETARGS) *((Llong *)curarg) = llval; fmt += 2; } else { if (flags & SETARGS) *((long *)curarg) = val; fmt++; } } else if (fmt[1] == 's' || fmt[1] == 'S') { if (flags & SETARGS) *((short *)curarg) = val; fmt++; } else if (fmt[1] == 'c' || fmt[1] == 'C') { if (flags & SETARGS) *((char *)curarg) = val; fmt++; } else { if (fmt[1] == 'i' || fmt[1] == 'I') fmt++; if (flags & SETARGS) *((int *)curarg) = val; } return (checkfmt(fmt)); case '#': if (*argp == '\0' && fmt[1] != '_') { if (*pac > 1) { (*pac)--; (*pav)++; argp = **pav; } else { return (BADFLAG); } } if (fmt[1] == '_') /* To disallow -f 123 for f# */ fmt++; else if (fmt[1] == ' ') /* To disallow -f123 for f# */ fmt++; if (*astoll(argp, &llval) != '\0') { /* * arg is not a valid number! * go to next format in the format string * and check if arg matches any other type * in the format specs. */ nextchance: while (*fmt != ',' && *fmt != '\0') { if ((*fmt == '&' || *fmt == '~') && (flags & (SETARGS|ARGVECTOR)) == SETARGS) { curarg = va_arg(args, void *); } fmt++; } argp = sargp; *pac = spac; *pav = spav; continue; } fmtp = fmt; val = (long)llval; if (fmt[1] == 'l' || fmt[1] == 'L') { if (fmt[2] == 'l' || fmt[2] == 'L') { if (flags & SETARGS) *((Llong *)curarg) = llval; fmt += 2; } else { if (flags & SETARGS) *((long *)curarg) = val; fmt++; } } else if (fmt[1] == 's' || fmt[1] == 'S') { if (flags & SETARGS) *((short *)curarg) = (short)val; fmt++; } else if (fmt[1] == 'c' || fmt[1] == 'C') { if (flags & SETARGS) *((char *)curarg) = (char)val; fmt++; } else { if (fmt[1] == 'i' || fmt[1] == 'I') fmt++; if (flags & SETARGS) *((int *)curarg) = (int)val; } return (checkfmt(fmt)); case '~': if (*argp != '\0') goto nextchance; if (haseql) { return (BADFLAG); } goto callfunc; case '&': if (*argp == '\0' && fmt[1] != '_') { if (*pac > 1) { (*pac)--; (*pav)++; argp = **pav; } else { return (BADFLAG); } } callfunc: if (*fmt == '&' && fmt[1] == '_') /* To disallow -f foo for f& */ fmt++; else if (fmt[1] == ' ') /* To disallow -ffoo for f& */ fmt++; if ((val = checkfmt(fmt)) != NOTAFLAG) return (val); fmtp = sargp; if (hasdash) fmtp--; if (doubledash) fmtp--; if ((flags & (SETARGS|ARGVECTOR)) == (SETARGS|ARGVECTOR)) { int ret; if (flagp->ga_funcp == NULL) return (BADFMT); ret = ((*flagp->ga_funcp) (argp, flagp->ga_arg, pac, pav, fmtp)); if (ret != NOTAFILE) return (ret); fmt++; } else if (flags & SETARGS) { int ret; void *funarg = va_arg(args, void *); if (curarg == NULL) return (BADFMT); ret = ((*(getpargfun)curarg) (argp, funarg, pac, pav, fmtp)); if (ret != NOTAFILE) return (ret); fmt++; } else { return (val); } /* * Called function returns NOTAFILE: try next format. */ } } } /* * parse args for combined single char flags */ typedef struct { void *curarg; /* The pointer to the arg to modify */ void *curfun; /* The pointer to the function to call */ char c; /* The single char flag character */ char type; /* The type of the single char flag */ char fmt; /* The format type of the single char flag */ char val; /* The value to assign for BOOL flags */ } sflags; LOCAL int dosflags(argp, vfmt, pac, pav, flags, oargs) register const char *argp; void *vfmt; int *pac; char *const **pav; int flags; va_list oargs; { register const char *fmt = (const char *)vfmt; struct ga_flags *flagp = vfmt; #define MAXSF 64 sflags sf[MAXSF]; char fl[256]; va_list args; register sflags *rsf = sf; register int nsf = 0; register const char *p = argp; register int i; register void *curarg = (void *)0; getpargfun curfun = 0; char type; /* * Initialize 'args' to the start of the argument list. * I don't know any portable way to copy an arbitrary * C object so I use a system-specific routine * (probably a macro) from stdarg.h. (Remember that * if va_list is an array, 'args' will be a pointer * and '&args' won't be what I would need for memcpy.) * It is a system requirement for SVr4 compatibility * to be able to do this assgignement. If your system * defines va_list to be an array but does not define * va_copy() you are lost. * This is needed to make sure, that 'oargs' will not * be clobbered. */ va_copy(args, oargs); if (flags & ARGVECTOR) { fmt = flagp->ga_format; if (fmt == NULL) fmt = ""; if (flags & SETARGS) { curarg = flagp->ga_arg; curfun = flagp->ga_funcp; } } else if (flags & SETARGS) { /* * We set curfun to curarg. We later get the real * curarg in case that we see a function callback * but we need curfun first in this case. */ curarg = va_arg(args, void *); curfun = (getpargfun)curarg; } for (i = 0; i < sizeof (fl); i++) { fl[i] = 0; } while (*p) { for (i = 0; i < nsf; i++) { if (rsf[i].c == *p) break; } if (i >= MAXSF) { va_end(args); return (BADFLAG); } if (i == nsf) { rsf[i].curarg = (void *)0; rsf[i].curfun = (void *)0; rsf[i].c = *p; rsf[i].type = (char)-1; rsf[i].fmt = '\0'; rsf[i].val = (char)TRUE; nsf++; } fl[*p & 0xFF] = i; p++; } again: while (*fmt) { if ((((flags & SINGLEARG) && ((fmt[1] == '*' || fmt[1] == '?' || fmt[1] == '&' || fmt[1] == '#'))) || (!isfmtspec(*fmt) && (fmt[1] == ',' || fmt[1] == '+' || fmt[1] == '~' || fmt[1] == '%' || fmt[1] == '\0'))) && strchr(argp, *fmt)) { for (i = 0; i < nsf; i++) { if (rsf[i].c == *fmt) { if ((fmt[1] == '+') || (fmt[1] == '#' && (flags & SINGLEARG))) { rsf[i].fmt = fmt[1]; fmt++; if (fmt[1] == ',' || fmt[1] == '\0') { rsf[i].type = 'i'; } else if ((fmt[1] == 'l' || fmt[1] == 'L') && (fmt[2] == 'l' || fmt[2] == 'L')) { /* * Type 'Q'uad (ll) */ rsf[i].type = 'Q'; fmt++; } else { /* * Type 'l','i','s','c' */ rsf[i].type = fmt[1]; } } else if (fmt[1] == '%') { fmt++; rsf[i].fmt = '%'; if (fmt[1] == '0') rsf[i].val = (char)FALSE; else if (fmt[1] == '1') rsf[i].val = (char)TRUE; fmt++; if (fmt[1] == ',' || fmt[1] == '\0') { rsf[i].type = 'i'; } else if ((fmt[1] == 'l' || fmt[1] == 'L') && (fmt[2] == 'l' || fmt[2] == 'L')) { /* * Type 'Q'uad (ll) */ rsf[i].type = 'Q'; fmt++; } else { /* * Type 'l','i','s','c' */ rsf[i].type = fmt[1]; } } else if ((fmt[1] == '*' || fmt[1] == '?') && (flags & SINGLEARG)) { fmt++; rsf[i].fmt = fmt[0]; rsf[i].type = fmt[0]; if (fmt[1] == ' ' || fmt[1] == '_') rsf[i].type = fmt[1]; } else if ((fmt[1] == '~') || (fmt[1] == '&' && (flags & SINGLEARG))) { rsf[i].fmt = fmt[1]; rsf[i].type = fmt[1]; /* * Let fmt point to ',' to * prevent to fetch the * func arg twice. */ fmt += 2; if ((rsf[i].fmt == '&') && (fmt[0] == ' ' || fmt[0] == '_')) rsf[i].type = fmt[0]; rsf[i].curfun = (void *)curfun; if ((flags & (SETARGS|ARGVECTOR)) == SETARGS) curarg = va_arg(args, void *); } else { /* * ',' or '\0' for BOOL */ rsf[i].type = fmt[1]; } rsf[i].curarg = curarg; break; } } } while (*fmt != ',' && *fmt != '\0') { /* * function has extra arg on stack. The code above * prevents us from fetching this arg twice. */ if ((*fmt == '&' || *fmt == '~') && (flags & (SETARGS|ARGVECTOR)) == SETARGS) { curarg = va_arg(args, void *); } fmt++; } if (*fmt != '\0') fmt++; else break; if ((flags & (SETARGS|ARGVECTOR)) == SETARGS) { /* * We set curfun to curarg. We later get the real * curarg in case that we see a function callback * but we need curfun first in this case. */ curarg = va_arg(args, void *); curfun = (getpargfun)curarg; } } if ((flags & ARGVECTOR) && flagp[1].ga_format != NULL) { flagp++; fmt = flagp->ga_format; if (flags & SETARGS) { curarg = flagp->ga_arg; curfun = flagp->ga_funcp; } goto again; } for (p = argp; *p; p++) { char tfmt; i = fl[*p & 0xFF]; tfmt = rsf[i].fmt; type = rsf[i].type; if (type == (char)-1) { return (BADFLAG); } if ((flags & SETARGS) == 0 && (tfmt == '*' || tfmt == '?' || tfmt == '&' || tfmt == '#')) { /* * POSIX 12.2 Guideline 5 requires that any number of * options without argument may be followed by a * single option with parameter. So all characters * past this one belong to the argument of this option. */ if (flags & CHECKARGS) break; if (*++p == '\0' && type != '_') { if (*pac > 1) { (*pac)--; (*pav)++; p = **pav; } else { return (BADFLAG); } } break; } if ((flags & SETARGS) && (rsf[i].curfun || rsf[i].curarg)) { Llong llval = 0; long val = 0; if (tfmt == '*' || tfmt == '?' || tfmt == '&' || tfmt == '#') { /* * If type is '_', then -f ... results in an * empty argument. This is because we disallow * -f foo for f* in this case. */ if (*++p == '\0' && type != '_') { if (*pac > 1) { (*pac)--; (*pav)++; p = **pav; } else { return (BADFLAG); } } } if (tfmt == '#' && *astoll(p, &llval) != '\0') { return (BADFLAG); } val = (long)llval; if (type == ',' || type == '\0') { *((int *)rsf[i].curarg) = TRUE; } else if (type == 'i' || type == 'I') { if (tfmt == '+') *((int *)rsf[i].curarg) += 1; else if (tfmt == '#') *((int *)rsf[i].curarg) = (int)val; else *((int *)rsf[i].curarg) = rsf[i].val; } else if (type == 'l' || type == 'L') { if (tfmt == '+') *((long *)rsf[i].curarg) += 1; else if (tfmt == '#') *((long *)rsf[i].curarg) = val; else *((long *)rsf[i].curarg) = rsf[i].val; } else if (type == 'Q') { if (tfmt == '+') *((Llong *)rsf[i].curarg) += 1; else if (tfmt == '#') *((Llong *)rsf[i].curarg) = llval; else *((Llong *)rsf[i].curarg) = rsf[i].val; } else if (type == 's' || type == 'S') { if (tfmt == '+') *((short *)rsf[i].curarg) += 1; else if (tfmt == '#') *((short *)rsf[i].curarg) = (short)val; else *((short *)rsf[i].curarg) = rsf[i].val; } else if (type == 'c' || type == 'C') { if (tfmt == '+') *((char *)rsf[i].curarg) += 1; else if (tfmt == '#') *((char *)rsf[i].curarg) = (char)val; else *((char *)rsf[i].curarg) = rsf[i].val; } else if (tfmt == '*' || tfmt == '?') { if (tfmt == '*') *((const char **)rsf[i].curarg) = p; else *((char *)rsf[i].curarg) = *p; break; } else if (tfmt == '~' || tfmt == '&') { int ret; char cfmt[3]; cfmt[0] = '-'; cfmt[1] = rsf[i].c; cfmt[2] = '\0'; if (rsf[i].curfun == NULL) return (BADFMT); ret = ((*(getpargfun)rsf[i].curfun) (tfmt == '&' ?p:"", rsf[i].curarg, pac, pav, cfmt)); if (ret != NOTAFLAG) return (ret); if (tfmt == '&') break; } else { return (BADFLAG); } if (tfmt == '#') break; } } return (NOTAFLAG); } /* * If the next format character is a comma or the string delimiter, * there are no invalid format specifiers. Return success. * Otherwise raise the getarg_bad_format condition. */ LOCAL int checkfmt(fmt) const char *fmt; { char c; c = *(++fmt); /* non constant expression */ if (c == ',' || c == '\0') { return (NOTAFLAG); } else { raisecond("getarg_bad_format", (long)fmt); return (BADFMT); } } /* * Parse the string as long as valid characters can be found. * Valid flag identifiers are chosen from the set of * alphanumeric characters, '-' and '_'. * If the next character is an equal sign the string * contains a valid flag identifier. */ LOCAL int checkeql(str) register const char *str; { register unsigned char c; for (c = (unsigned char)*str; isalnum(c) || c == '_' || c == '-' || c == '+'; c = *++str) /* LINTED */ ; return (c == '='); } EXPORT char * getargerror(err) int err; { if (err < RETMIN || err > RETMAX) return ("Illegal arg error"); return (RNAME(err)); }