2000-06-14 07:09:10 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* cmdline.c */
|
|
|
|
/* */
|
|
|
|
/* Helper functions for command line parsing */
|
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* (C) 2000 Ullrich von Bassewitz */
|
|
|
|
/* Wacholderweg 14 */
|
|
|
|
/* D-70597 Stuttgart */
|
|
|
|
/* EMail: uz@musoftware.de */
|
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* This software is provided 'as-is', without any expressed or implied */
|
|
|
|
/* warranty. In no event will the authors be held liable for any damages */
|
|
|
|
/* arising from the use of this software. */
|
|
|
|
/* */
|
|
|
|
/* Permission is granted to anyone to use this software for any purpose, */
|
|
|
|
/* including commercial applications, and to alter it and redistribute it */
|
|
|
|
/* freely, subject to the following restrictions: */
|
|
|
|
/* */
|
|
|
|
/* 1. The origin of this software must not be misrepresented; you must not */
|
|
|
|
/* claim that you wrote the original software. If you use this software */
|
|
|
|
/* in a product, an acknowledgment in the product documentation would be */
|
|
|
|
/* appreciated but is not required. */
|
|
|
|
/* 2. Altered source versions must be plainly marked as such, and must not */
|
|
|
|
/* be misrepresented as being the original software. */
|
|
|
|
/* 3. This notice may not be removed or altered from any source */
|
|
|
|
/* distribution. */
|
|
|
|
/* */
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2001-03-09 21:59:23 +00:00
|
|
|
#include <stdio.h>
|
2000-06-14 09:32:22 +00:00
|
|
|
#include <string.h>
|
2001-03-09 21:59:23 +00:00
|
|
|
#include <errno.h>
|
2000-06-14 07:09:10 +00:00
|
|
|
|
2001-03-09 21:59:23 +00:00
|
|
|
/* common */
|
2000-06-14 09:32:22 +00:00
|
|
|
#include "abend.h"
|
2001-03-09 21:59:23 +00:00
|
|
|
#include "chartype.h"
|
2003-06-03 22:19:46 +00:00
|
|
|
#include "fname.h"
|
2001-03-09 21:59:23 +00:00
|
|
|
#include "xmalloc.h"
|
2000-06-14 07:09:10 +00:00
|
|
|
#include "cmdline.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* Data */
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-06-14 09:32:22 +00:00
|
|
|
/* Program name - is set after call to InitCmdLine */
|
|
|
|
const char* ProgName;
|
|
|
|
|
|
|
|
/* The program argument vector */
|
2001-03-09 21:59:23 +00:00
|
|
|
char** ArgVec = 0;
|
|
|
|
unsigned ArgCount = 0;
|
|
|
|
|
|
|
|
/* Struct to pass the command line */
|
|
|
|
typedef struct {
|
2003-06-03 22:19:46 +00:00
|
|
|
char** Vec; /* The argument vector */
|
|
|
|
unsigned Count; /* Actual number of arguments */
|
|
|
|
unsigned Size; /* Number of argument allocated */
|
2001-03-09 21:59:23 +00:00
|
|
|
} CmdLine;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* Helper functions */
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void NewCmdLine (CmdLine* L)
|
|
|
|
/* Initialize a CmdLine struct */
|
|
|
|
{
|
|
|
|
unsigned I;
|
|
|
|
|
|
|
|
/* Initialize the struct */
|
|
|
|
L->Size = 8;
|
|
|
|
L->Count = 0;
|
|
|
|
L->Vec = xmalloc (L->Size * sizeof (L->Vec[0]));
|
|
|
|
|
|
|
|
/* Copy the arguments. We have to allocate them on free store, otherwise
|
|
|
|
* we would have to keep track which one is on free store and which not,
|
|
|
|
* which is a lot more overhead.
|
|
|
|
*/
|
|
|
|
for (I = 0; I < L->Count; ++I) {
|
|
|
|
L->Vec[I] = xstrdup (ArgVec[I]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2003-06-03 22:19:46 +00:00
|
|
|
static void AddArg (CmdLine* L, char* Arg)
|
2001-03-09 21:59:23 +00:00
|
|
|
/* Add one argument to the list */
|
|
|
|
{
|
|
|
|
if (L->Size <= L->Count) {
|
|
|
|
/* No space left, reallocate */
|
|
|
|
unsigned NewSize = L->Size * 2;
|
|
|
|
char** NewVec = xmalloc (NewSize * sizeof (L->Vec[0]));
|
|
|
|
memcpy (NewVec, L->Vec, L->Count * sizeof (L->Vec[0]));
|
|
|
|
xfree (L->Vec);
|
|
|
|
L->Vec = NewVec;
|
|
|
|
L->Size = NewSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have space left, add a copy of the argument */
|
2003-06-03 22:19:46 +00:00
|
|
|
L->Vec[L->Count++] = Arg;
|
2001-03-09 21:59:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void ExpandFile (CmdLine* L, const char* Name)
|
|
|
|
/* Add the contents of a file to the command line. Each line is a separate
|
|
|
|
* argument with leading and trailing whitespace removed.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
char Buf [256];
|
|
|
|
|
|
|
|
/* Try to open the file for reading */
|
|
|
|
FILE* F = fopen (Name, "r");
|
|
|
|
if (F == 0) {
|
|
|
|
AbEnd ("Cannot open \"%s\": %s", Name, strerror (errno));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* File is open, read all lines */
|
|
|
|
while (fgets (Buf, sizeof (Buf), F) != 0) {
|
|
|
|
|
|
|
|
/* Get a pointer to the buffer */
|
|
|
|
const char* B = Buf;
|
|
|
|
|
|
|
|
/* Skip trailing whitespace (this will also kill the newline that is
|
|
|
|
* appended by fgets().
|
|
|
|
*/
|
|
|
|
unsigned Len = strlen (Buf);
|
|
|
|
while (Len > 0 && IsSpace (Buf [Len-1])) {
|
|
|
|
--Len;
|
|
|
|
}
|
|
|
|
Buf [Len] = '\0';
|
|
|
|
|
|
|
|
/* Skip leading spaces */
|
|
|
|
while (IsSpace (*B)) {
|
|
|
|
++B;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip empty lines to work around problems with some editors */
|
|
|
|
if (*B == '\0') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add anything not empty to the command line */
|
2003-06-03 22:19:46 +00:00
|
|
|
AddArg (L, xstrdup (B));
|
2001-03-09 21:59:23 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Close the file, ignore errors here since we had the file open for
|
|
|
|
* reading only.
|
|
|
|
*/
|
|
|
|
(void) fclose (F);
|
|
|
|
}
|
2000-06-14 07:09:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
2000-06-14 09:32:22 +00:00
|
|
|
/* Code */
|
2000-06-14 07:09:10 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2001-10-11 14:45:59 +00:00
|
|
|
void InitCmdLine (int* aArgCount, char** aArgVec[], const char* aProgName)
|
2000-06-14 07:09:10 +00:00
|
|
|
/* Initialize command line parsing. aArgVec is the argument array terminated by
|
|
|
|
* a NULL pointer (as usual), ArgCount is the number of valid arguments in the
|
|
|
|
* array. Both arguments are remembered in static storage.
|
|
|
|
*/
|
|
|
|
{
|
2001-03-09 21:59:23 +00:00
|
|
|
CmdLine L;
|
2001-10-11 14:45:59 +00:00
|
|
|
int I;
|
2000-06-14 09:32:22 +00:00
|
|
|
|
|
|
|
/* Get the program name from argv[0] but strip a path */
|
2001-03-23 17:59:23 +00:00
|
|
|
if (*(aArgVec)[0] == 0) {
|
2001-03-09 21:59:23 +00:00
|
|
|
/* Use the default name given */
|
|
|
|
ProgName = aProgName;
|
2000-06-14 09:32:22 +00:00
|
|
|
} else {
|
2001-03-09 21:59:23 +00:00
|
|
|
/* Strip a path */
|
2003-06-03 22:19:46 +00:00
|
|
|
ProgName = FindName ((*aArgVec)[0]);
|
2001-03-09 21:59:23 +00:00
|
|
|
if (ProgName[0] == '\0') {
|
|
|
|
/* Use the default */
|
|
|
|
ProgName = aProgName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make a CmdLine struct */
|
|
|
|
NewCmdLine (&L);
|
|
|
|
|
|
|
|
/* Walk over the parameters and add them to the CmdLine struct. Add a
|
|
|
|
* special handling for arguments preceeded by the '@' sign - these are
|
|
|
|
* actually files containing arguments.
|
|
|
|
*/
|
|
|
|
for (I = 0; I < *aArgCount; ++I) {
|
|
|
|
|
|
|
|
/* Get the next argument */
|
|
|
|
char* Arg = (*aArgVec)[I];
|
|
|
|
|
|
|
|
/* Is this a file argument? */
|
|
|
|
if (Arg && Arg[0] == '@') {
|
|
|
|
|
|
|
|
/* Expand the file */
|
|
|
|
ExpandFile (&L, Arg+1);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
/* No file, just add a copy */
|
|
|
|
AddArg (&L, Arg);
|
|
|
|
|
2000-06-14 09:32:22 +00:00
|
|
|
}
|
|
|
|
}
|
2001-03-09 21:59:23 +00:00
|
|
|
|
|
|
|
/* Store the new argument list in a safe place... */
|
|
|
|
ArgCount = L.Count;
|
|
|
|
ArgVec = L.Vec;
|
|
|
|
|
|
|
|
/* ...and pass back the changed data also */
|
|
|
|
*aArgCount = L.Count;
|
|
|
|
*aArgVec = L.Vec;
|
2000-06-14 07:09:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void UnknownOption (const char* Opt)
|
2000-08-19 18:16:54 +00:00
|
|
|
/* Print an error about an unknown option and die. */
|
2000-06-14 07:09:10 +00:00
|
|
|
{
|
2002-04-08 18:42:44 +00:00
|
|
|
AbEnd ("Unknown option: %s", Opt);
|
2000-06-14 07:09:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void NeedArg (const char* Opt)
|
|
|
|
/* Print an error about a missing option argument and exit. */
|
|
|
|
{
|
2002-04-08 18:42:44 +00:00
|
|
|
AbEnd ("Option requires an argument: %s", Opt);
|
2000-06-14 07:09:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2000-06-14 07:48:27 +00:00
|
|
|
void InvDef (const char* Def)
|
|
|
|
/* Print an error about an invalid definition and die */
|
2000-06-14 07:09:10 +00:00
|
|
|
{
|
2002-04-08 18:42:44 +00:00
|
|
|
AbEnd ("Invalid definition: `%s'", Def);
|
2000-06-14 07:09:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2001-10-11 14:50:10 +00:00
|
|
|
const char* GetArg (unsigned* ArgNum, unsigned Len)
|
2000-06-14 07:09:10 +00:00
|
|
|
/* Get an argument for a short option. The argument may be appended to the
|
|
|
|
* option itself or may be separate. Len is the length of the option string.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
const char* Arg = ArgVec[*ArgNum];
|
|
|
|
if (Arg[Len] != '\0') {
|
|
|
|
/* Argument appended */
|
|
|
|
return Arg + Len;
|
|
|
|
} else {
|
|
|
|
/* Separate argument */
|
|
|
|
Arg = ArgVec[*ArgNum + 1];
|
|
|
|
if (Arg == 0) {
|
|
|
|
/* End of arguments */
|
|
|
|
NeedArg (ArgVec[*ArgNum]);
|
|
|
|
}
|
|
|
|
++(*ArgNum);
|
|
|
|
return Arg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2001-10-11 14:45:59 +00:00
|
|
|
void LongOption (unsigned* ArgNum, const LongOpt* OptTab, unsigned OptCount)
|
2000-06-14 07:09:10 +00:00
|
|
|
/* Handle a long command line option */
|
|
|
|
{
|
|
|
|
/* Get the option and the argument (which may be zero) */
|
|
|
|
const char* Opt = ArgVec[*ArgNum];
|
|
|
|
|
|
|
|
/* Search the table for a match */
|
|
|
|
while (OptCount) {
|
|
|
|
if (strcmp (Opt, OptTab->Option) == 0) {
|
|
|
|
/* Found, call the function */
|
|
|
|
if (OptTab->ArgCount > 0) {
|
2000-08-23 14:18:03 +00:00
|
|
|
/* We need an argument, check if we have one */
|
|
|
|
const char* Arg = ArgVec[++(*ArgNum)];
|
|
|
|
if (Arg == 0) {
|
|
|
|
NeedArg (Opt);
|
|
|
|
}
|
|
|
|
OptTab->Func (Opt, Arg);
|
2000-06-14 07:09:10 +00:00
|
|
|
} else {
|
2000-06-14 09:32:22 +00:00
|
|
|
OptTab->Func (Opt, 0);
|
2000-06-14 07:09:10 +00:00
|
|
|
}
|
2000-06-14 07:35:00 +00:00
|
|
|
/* Done */
|
|
|
|
return;
|
2000-06-14 07:09:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Next table entry */
|
|
|
|
--OptCount;
|
|
|
|
++OptTab;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Invalid option */
|
|
|
|
UnknownOption (Opt);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2003-06-03 22:19:46 +00:00
|
|
|
|