mirror of
https://github.com/aseprite/aseprite.git
synced 2025-01-01 00:23:35 +00:00
698 lines
27 KiB
C
698 lines
27 KiB
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 HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
|
|
#ifdef __MSDOS__
|
|
#include <alloc.h>
|
|
#endif /* __MSDOS__ */
|
|
|
|
#ifdef HAVE_STDARG_H
|
|
#include <stdarg.h>
|
|
#elif defined(HAVE_VARARGS_H)
|
|
#include <varargs.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "getarg.h"
|
|
|
|
#ifndef MYMALLOC
|
|
#define MYMALLOC /* If no "MyMalloc" routine elsewhere define this. */
|
|
#endif
|
|
|
|
#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, ErrorToken is set to point to 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 MYMALLOC
|
|
static 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 HAVE_STDARG_H
|
|
int
|
|
GAGetArgs(int argc,
|
|
char **argv,
|
|
char *CtrlStr, ...) {
|
|
|
|
int i, Error = FALSE, ParamCount = 0;
|
|
int *Parameters[MAX_PARAM]; /* Save here parameter addresses. */
|
|
char *Option, CtrlStrCopy[CTRL_STR_MAX_LEN];
|
|
va_list ap;
|
|
|
|
strcpy(CtrlStrCopy, CtrlStr);
|
|
va_start(ap, CtrlStr);
|
|
for (i = 1; i <= MAX_PARAM; i++)
|
|
Parameters[i - 1] = va_arg(ap, int *);
|
|
va_end(ap);
|
|
|
|
#elif defined(HAVE_VARARGS_H)
|
|
int GAGetArgs(va_alist)
|
|
va_dcl
|
|
{
|
|
va_list ap;
|
|
int argc, i, Error = FALSE, ParamCount = 0;
|
|
int *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 *);
|
|
#endif /* HAVE_STDARG_H */
|
|
|
|
--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 if 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 param. */
|
|
*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 correct 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",
|
|
(unsigned int *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 'o': /* Get octal integers. */
|
|
ScanRes = sscanf(*((*argv)++), "%o",
|
|
(unsigned int *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 'D': /* Get signed long integers. */
|
|
ScanRes = sscanf(*((*argv)++), "%ld",
|
|
(long *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 'U': /* Get unsigned long integers. */
|
|
ScanRes = sscanf(*((*argv)++), "%lu",
|
|
(unsigned long *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 'X': /* Get hex long integers. */
|
|
ScanRes = sscanf(*((*argv)++), "%lx",
|
|
(unsigned long *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 'O': /* Get octal long integers. */
|
|
ScanRes = sscanf(*((*argv)++), "%lo",
|
|
(unsigned long *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 'f': /* Get float number. */
|
|
ScanRes = sscanf(*((*argv)++), "%f",
|
|
(float *)Parameters[(*ParamCount)++]);
|
|
case 'F': /* Get double float number. */
|
|
ScanRes = sscanf(*((*argv)++), "%lf",
|
|
(double *)Parameters[(*ParamCount)++]);
|
|
break;
|
|
case 's': /* It as a string. */
|
|
ScanRes = 1; /* Allways O.K. */
|
|
GAByteCopy((char *)Parameters[(*ParamCount)++],
|
|
(char *)((*argv)++), sizeof(char *));
|
|
break;
|
|
case '*': /* Get few parameters into one: */
|
|
ScanRes = GAGetMultiParmeters(Parameters, ParamCount,
|
|
&CtrlStrCopy[i], argc, argv);
|
|
if ((ScanRes == 0) && (CtrlStrCopy[i] == '!')) {
|
|
GAErrorToken = Option;
|
|
return CMD_ERR_WildEmpty;
|
|
}
|
|
break;
|
|
default:
|
|
ScanRes = 0; /* Make optimizer warning silent. */
|
|
}
|
|
/* If reading fails and this number is a must (!) then error: */
|
|
if ((ScanRes == 0) && (CtrlStrCopy[i] == '!')) {
|
|
GAErrorToken = Option;
|
|
return CMD_ERR_NumRead;
|
|
}
|
|
if (CtrlStrCopy[i + 1] != '*') {
|
|
(*argc)--; /* Everything is OK - update to next parameter: */
|
|
i += 2; /* Skip to next parameter (if any). */
|
|
} else
|
|
i += 3; /* Skip the '*' also! */
|
|
}
|
|
|
|
return ARG_OK;
|
|
}
|
|
|
|
/***************************************************************************
|
|
* Routine to get few parameters into one pointer such that the returned
|
|
* pointer actually points on a block of pointers to the parameters...
|
|
* For example *F means a pointer to pointers on floats.
|
|
* Returns number of parameters actually read.
|
|
* This routine assumes that all pointers (on any kind of scalar) has the
|
|
* same size (and the union below is totally ovelapped bteween dif. arrays)
|
|
***************************************************************************/
|
|
static int
|
|
GAGetMultiParmeters(int *Parameters[],
|
|
int *ParamCount,
|
|
char *CtrlStrCopy,
|
|
int *argc,
|
|
char ***argv) {
|
|
|
|
int i = 0, ScanRes, NumOfPrm = 0, **Pmain, **Ptemp;
|
|
union TmpArray { /* Save here the temporary data before copying it to */
|
|
int *IntArray[MAX_PARAM]; /* the returned pointer block. */
|
|
long *LngArray[MAX_PARAM];
|
|
float *FltArray[MAX_PARAM];
|
|
double *DblArray[MAX_PARAM];
|
|
char *ChrArray[MAX_PARAM];
|
|
} TmpArray;
|
|
|
|
do {
|
|
switch (CtrlStrCopy[2]) { /* CtrlStr == '!*?' or '%*?' where ? is. */
|
|
case 'd': /* Format to read the parameters: */
|
|
TmpArray.IntArray[NumOfPrm] = (int *)MyMalloc(sizeof(int));
|
|
ScanRes = sscanf(*((*argv)++), "%d",
|
|
(int *)TmpArray.IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'u':
|
|
TmpArray.IntArray[NumOfPrm] = (int *)MyMalloc(sizeof(int));
|
|
ScanRes = sscanf(*((*argv)++), "%u",
|
|
(unsigned int *)TmpArray.IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'o':
|
|
TmpArray.IntArray[NumOfPrm] = (int *)MyMalloc(sizeof(int));
|
|
ScanRes = sscanf(*((*argv)++), "%o",
|
|
(unsigned int *)TmpArray.IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'x':
|
|
TmpArray.IntArray[NumOfPrm] = (int *)MyMalloc(sizeof(int));
|
|
ScanRes = sscanf(*((*argv)++), "%x",
|
|
(unsigned int *)TmpArray.IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'D':
|
|
TmpArray.LngArray[NumOfPrm] = (long *)MyMalloc(sizeof(long));
|
|
ScanRes = sscanf(*((*argv)++), "%ld",
|
|
(long *)TmpArray.IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'U':
|
|
TmpArray.LngArray[NumOfPrm] = (long *)MyMalloc(sizeof(long));
|
|
ScanRes = sscanf(*((*argv)++), "%lu",
|
|
(unsigned long *)TmpArray.
|
|
IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'O':
|
|
TmpArray.LngArray[NumOfPrm] = (long *)MyMalloc(sizeof(long));
|
|
ScanRes = sscanf(*((*argv)++), "%lo",
|
|
(unsigned long *)TmpArray.
|
|
IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'X':
|
|
TmpArray.LngArray[NumOfPrm] = (long *)MyMalloc(sizeof(long));
|
|
ScanRes = sscanf(*((*argv)++), "%lx",
|
|
(unsigned long *)TmpArray.
|
|
IntArray[NumOfPrm++]);
|
|
break;
|
|
case 'f':
|
|
TmpArray.FltArray[NumOfPrm] = (float *)MyMalloc(sizeof(float));
|
|
ScanRes = sscanf(*((*argv)++), "%f",
|
|
(float *)TmpArray.LngArray[NumOfPrm++]);
|
|
break;
|
|
case 'F':
|
|
TmpArray.DblArray[NumOfPrm] =
|
|
(double *)MyMalloc(sizeof(double));
|
|
ScanRes = sscanf(*((*argv)++), "%lf",
|
|
(double *)TmpArray.LngArray[NumOfPrm++]);
|
|
break;
|
|
case 's':
|
|
while ((*argc) && ((**argv)[0] != '-')) {
|
|
TmpArray.ChrArray[NumOfPrm++] = *((*argv)++);
|
|
(*argc)--;
|
|
}
|
|
ScanRes = 0; /* Force quit from do - loop. */
|
|
NumOfPrm++; /* Updated again immediately after loop! */
|
|
(*argv)++; /* "" */
|
|
break;
|
|
default:
|
|
ScanRes = 0; /* Make optimizer warning silent. */
|
|
}
|
|
(*argc)--;
|
|
}
|
|
while (ScanRes == 1); /* Exactly one parameter was read. */
|
|
(*argv)--;
|
|
NumOfPrm--;
|
|
(*argc)++;
|
|
|
|
/* Now allocate the block with the exact size, and set it: */
|
|
Ptemp = Pmain =
|
|
(int **)MyMalloc((unsigned)(NumOfPrm + 1) * sizeof(int *));
|
|
/* And here we use the assumption that all pointers are the same: */
|
|
for (i = 0; i < NumOfPrm; i++)
|
|
*Ptemp++ = TmpArray.IntArray[i];
|
|
*Ptemp = NULL; /* Close the block with NULL pointer. */
|
|
|
|
/* That it save the number of parameters read as first parameter to
|
|
* return and the pointer to the block as second, and return: */
|
|
*Parameters[(*ParamCount)++] = NumOfPrm;
|
|
GAByteCopy((char *)Parameters[(*ParamCount)++], (char *)&Pmain,
|
|
sizeof(char *));
|
|
return NumOfPrm;
|
|
}
|
|
|
|
/***************************************************************************
|
|
* Routine to scan the CtrlStr, upto Max and count the number of parameters
|
|
* to that point:
|
|
* 1. Each option is counted as one parameter - boolean variable (int)
|
|
* 2. Within an option, each %? or !? is counted once - pointer to something
|
|
* 3. Within an option, %*? or !*? is counted twice - one for item count
|
|
* and one for pointer to block pointers.
|
|
* Note ALL variables are passed by address and so of fixed size (address).
|
|
**************************************************************************/
|
|
static void
|
|
GASetParamCount(char *CtrlStr,
|
|
int Max,
|
|
int *ParamCount) {
|
|
int i;
|
|
|
|
*ParamCount = 0;
|
|
for (i = 0; i < Max; i++)
|
|
if (ISCTRLCHAR(CtrlStr[i])) {
|
|
if (CtrlStr[i + 1] == '*')
|
|
*ParamCount += 2;
|
|
else
|
|
(*ParamCount)++;
|
|
}
|
|
}
|
|
|
|
/***************************************************************************
|
|
* Routine to copy exactly n bytes from Src to Dst. Note system library
|
|
* routine strncpy should do the same, but it stops on NULL char !
|
|
**************************************************************************/
|
|
static void
|
|
GAByteCopy(char *Dst,
|
|
char *Src,
|
|
unsigned n) {
|
|
|
|
while (n--)
|
|
*(Dst++) = *(Src++);
|
|
}
|
|
|
|
/***************************************************************************
|
|
* Routine to check if more option (i.e. first char == '-') exists in the
|
|
* given list argc, argv:
|
|
**************************************************************************/
|
|
static int
|
|
GAOptionExists(int argc,
|
|
char **argv) {
|
|
|
|
while (argc--)
|
|
if ((*argv++)[0] == '-')
|
|
return TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
/***************************************************************************
|
|
* Routine to print some error messages, for this module:
|
|
**************************************************************************/
|
|
void
|
|
GAPrintErrMsg(int Error) {
|
|
|
|
fprintf(stderr, "Error in command line parsing - ");
|
|
switch (Error) {
|
|
case 0:;
|
|
fprintf(stderr, "Undefined error");
|
|
break;
|
|
case CMD_ERR_NotAnOpt:
|
|
fprintf(stderr, "None option Found");
|
|
break;
|
|
case CMD_ERR_NoSuchOpt:
|
|
fprintf(stderr, "Undefined option Found");
|
|
break;
|
|
case CMD_ERR_WildEmpty:
|
|
fprintf(stderr, "Empty input for '!*?' seq.");
|
|
break;
|
|
case CMD_ERR_NumRead:
|
|
fprintf(stderr, "Failed on reading number");
|
|
break;
|
|
case CMD_ERR_AllSatis:
|
|
fprintf(stderr, "Fail to satisfy");
|
|
break;
|
|
}
|
|
fprintf(stderr, " - '%s'.\n", GAErrorToken);
|
|
}
|
|
|
|
/***************************************************************************
|
|
* Routine to print correct format of command line allowed:
|
|
**************************************************************************/
|
|
void
|
|
GAPrintHowTo(char *CtrlStr) {
|
|
|
|
int i = 0, SpaceFlag;
|
|
|
|
fprintf(stderr, "Usage: ");
|
|
/* Print program name - first word in ctrl. str. (optional): */
|
|
while (!(ISSPACE(CtrlStr[i])) && (!ISCTRLCHAR(CtrlStr[i + 1])))
|
|
fprintf(stderr, "%c", CtrlStr[i++]);
|
|
|
|
while (i < (int)strlen(CtrlStr)) {
|
|
while ((ISSPACE(CtrlStr[i])) && (i < (int)strlen(CtrlStr)))
|
|
i++;
|
|
switch (CtrlStr[i + 1]) {
|
|
case '%':
|
|
fprintf(stderr, " [-%c", CtrlStr[i++]);
|
|
i += 2; /* Skip the '%-' or '!- after the char! */
|
|
SpaceFlag = TRUE;
|
|
while (!ISCTRLCHAR(CtrlStr[i]) && (i < (int)strlen(CtrlStr)) &&
|
|
(!ISSPACE(CtrlStr[i])))
|
|
if (SpaceFlag) {
|
|
if (CtrlStr[i++] == SPACE_CHAR)
|
|
fprintf(stderr, " ");
|
|
else
|
|
fprintf(stderr, " %c", CtrlStr[i - 1]);
|
|
SpaceFlag = FALSE;
|
|
} else if (CtrlStr[i++] == SPACE_CHAR)
|
|
fprintf(stderr, " ");
|
|
else
|
|
fprintf(stderr, "%c", CtrlStr[i - 1]);
|
|
while (!ISSPACE(CtrlStr[i]) && (i < (int)strlen(CtrlStr))) {
|
|
if (CtrlStr[i] == '*')
|
|
fprintf(stderr, "...");
|
|
i++; /* Skip the rest of it. */
|
|
}
|
|
fprintf(stderr, "]");
|
|
break;
|
|
case '!':
|
|
fprintf(stderr, " -%c", CtrlStr[i++]);
|
|
i += 2; /* Skip the '%-' or '!- after the char! */
|
|
SpaceFlag = TRUE;
|
|
while (!ISCTRLCHAR(CtrlStr[i]) && (i < (int)strlen(CtrlStr)) &&
|
|
(!ISSPACE(CtrlStr[i])))
|
|
if (SpaceFlag) {
|
|
if (CtrlStr[i++] == SPACE_CHAR)
|
|
fprintf(stderr, " ");
|
|
else
|
|
fprintf(stderr, " %c", CtrlStr[i - 1]);
|
|
SpaceFlag = FALSE;
|
|
} else if (CtrlStr[i++] == SPACE_CHAR)
|
|
fprintf(stderr, " ");
|
|
else
|
|
fprintf(stderr, "%c", CtrlStr[i - 1]);
|
|
while (!ISSPACE(CtrlStr[i]) && (i < (int)strlen(CtrlStr))) {
|
|
if (CtrlStr[i] == '*')
|
|
fprintf(stderr, "...");
|
|
i++; /* Skip the rest of it. */
|
|
}
|
|
break;
|
|
default: /* Not checked, but must be last one! */
|
|
fprintf(stderr, " ");
|
|
while (!ISSPACE(CtrlStr[i]) && (i < (int)strlen(CtrlStr)) &&
|
|
!ISCTRLCHAR(CtrlStr[i]))
|
|
fprintf(stderr, "%c", CtrlStr[i++]);
|
|
fprintf(stderr, "\n");
|
|
return;
|
|
}
|
|
}
|
|
fprintf(stderr, "\n");
|
|
}
|
|
|
|
#ifdef MYMALLOC
|
|
|
|
/***************************************************************************
|
|
* My Routine to allocate dynamic memory. All program requests must call
|
|
* this routine (no direct call to malloc). Dies if no memory.
|
|
**************************************************************************/
|
|
static char *
|
|
MyMalloc(unsigned size) {
|
|
|
|
char *p;
|
|
|
|
if ((p = (char *)malloc(size)) != NULL)
|
|
return p;
|
|
|
|
fprintf(stderr, "Not enough memory, exit.\n");
|
|
exit(2);
|
|
|
|
return NULL; /* Makes warning silent. */
|
|
}
|
|
|
|
#endif /* MYMALLOC */
|