/* * * Copyright (c) 1996-2001, Darren Hiebert * * This source code is released for free distribution under the terms of the * GNU General Public License. * * This module contains functions to process command line options. */ /* * INCLUDE FILES */ #include "general.h" /* must always come first */ #include #include #include #include /* to declare isspace () */ #include "ctags.h" #include "main.h" #define OPTION_WRITE #include "options.h" #include "parse.h" /* * MACROS */ #define INVOCATION "Usage: %s [options] [file(s)]\n" #define CTAGS_ENVIRONMENT "CTAGS" #define ETAGS_ENVIRONMENT "ETAGS" #define CTAGS_FILE "tags" #define ETAGS_FILE "TAGS" #ifndef ETAGS # define ETAGS "etags" /* name which causes default use of to -e */ #endif /* The following separators are permitted for list options. */ #define EXTENSION_SEPARATOR '.' #define PATTERN_START '(' #define PATTERN_STOP ')' #define IGNORE_SEPARATORS ", \t\n" #ifndef DEFAULT_FILE_FORMAT # define DEFAULT_FILE_FORMAT 2 #endif #if defined (MSDOS) || defined (WIN32) || defined (OS2) || defined (AMIGA) || defined (HAVE_OPENDIR) # define RECURSE_SUPPORTED #endif #define isCompoundOption(c) (boolean) (strchr ("fohiILpDb", (c)) != NULL) /* * Data declarations */ enum eOptionLimits { MaxHeaderExtensions = 100, /* maximum number of extensions in -h option */ MaxSupportedTagFormat = 2 }; typedef struct sOptionDescription { int usedByEtags; const char *const description; } optionDescription; typedef void (*parametricOptionHandler) (const char *const option, const char *const parameter); typedef const struct { const char* name; /* name of option as specified by user */ parametricOptionHandler handler; /* routine to handle option */ boolean initOnly; /* option must be specified before any files */ } parametricOption; typedef const struct { const char* name; /* name of option as specified by user */ boolean* pValue; /* pointer to option value */ boolean initOnly; /* option must be specified before any files */ } booleanOption; /* * DATA DEFINITIONS */ static boolean ParsedLeadingOptions = FALSE; static const char *const HeaderExtensions [] = { "h", "H", "hh", "hpp", "hxx", "h++", "inc", "def", NULL }; optionValues Option = { { FALSE, /* --extra=f */ FALSE, /* --extra=q */ TRUE, /* --file-scope */ }, { TRUE, /* -fields=a */ TRUE, /* -fields=f */ FALSE, /* -fields=m */ TRUE, /* -fields=i */ FALSE, /* -fields=k */ TRUE, /* -fields=z */ TRUE, /* -fields=K */ FALSE, /* -fields=l */ TRUE, /* -fields=n */ TRUE, /* -fields=s */ TRUE, /* -fields=P */ TRUE /* -fields=A */ }, NULL, /* -I */ FALSE, /* -a */ FALSE, /* -B */ FALSE, /* -e */ #ifdef MACROS_USE_PATTERNS EX_PATTERN, /* -n, --excmd */ #else EX_MIX, /* -n, --excmd */ #endif FALSE, /* -R */ TRUE, /* -u, --sort */ FALSE, /* -V */ FALSE, /* -x */ NULL, /* -L */ NULL, /* -o */ NULL, /* -h */ NULL, /* --etags-include */ DEFAULT_FILE_FORMAT,/* --format */ FALSE, /* --if0 */ FALSE, /* --kind-long */ LANG_AUTO, /* --lang */ TRUE, /* --links */ FALSE, /* --filter */ NULL, /* --filter-terminator */ FALSE, /* --qualified-tags */ FALSE, /* --tag-relative */ FALSE, /* --totals */ FALSE, /* --line-directives */ FALSE, #ifdef TM_DEBUG 0, 0 /* -D, -b */ #endif }; /* - Locally used only */ static optionDescription LongOptionDescription [] = { {1," --version"}, {1," Print version identifier to standard output."}, {1, NULL} }; static const char* const License = "This program is free software; you can redistribute it and/or\n\ modify it under the terms of the GNU General Public License\n\ as published by the Free Software Foundation; either version 2\n\ of the License, or (at your option) any later version.\n\ \n\ This program is distributed in the hope that it will be useful,\n\ but WITHOUT ANY WARRANTY; without even the implied warranty of\n\ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n\ GNU General Public License for more details.\n\ \n\ You should have received a copy of the GNU General Public License\n\ along with this program; if not, write to the Free Software\n\ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.\n"; /* Contains a set of strings describing the set of "features" compiled into * the code. */ static const char *const Features [] = { #ifdef WIN32 "win32", #endif #ifdef DJGPP "msdos_32", #else # ifdef MSDOS "msdos_16", # endif #endif #ifdef OS2 "os2", #endif #ifdef AMIGA "amiga", #endif #ifdef VMS "vms", #endif #ifdef HAVE_FNMATCH "wildcards", #endif #ifdef HAVE_REGEX "regex", #endif #ifndef EXTERNAL_SORT "internal-sort", #endif #ifdef CUSTOM_CONFIGURATION_FILE "custom-conf", #endif #if (defined (MSDOS) || defined (WIN32) || defined (OS2)) && defined (UNIX_PATH_SEPARATOR) "unix-path-separator", #endif #ifdef TM_DEBUG "debug", #endif NULL }; /* * FUNCTION PROTOTYPES */ static boolean parseFileOptions (const char *const fileName); /* * FUNCTION DEFINITIONS */ extern void verbose (const char *const format, ...) { if (Option.verbose) { va_list ap; va_start (ap, format); vprintf (format, ap); va_end (ap); } } static char *stringCopy (const char *const string) { char* result = NULL; if (string != NULL) result = eStrdup (string); return result; } static void freeString (char **const pString) { if (*pString != NULL) { eFree (*pString); *pString = NULL; } } extern void freeList (stringList** const pList) { if (*pList != NULL) { stringListDelete (*pList); *pList = NULL; } } extern void setDefaultTagFileName (void) { if (Option.tagFileName != NULL) ; /* accept given name */ else if (Option.etags) Option.tagFileName = stringCopy (ETAGS_FILE); else Option.tagFileName = stringCopy (CTAGS_FILE); } extern void checkOptions (void) { const char* notice; if (Option.xref) { notice = "xref output"; if (Option.include.fileNames) { error (WARNING, "%s disables file name tags", notice); Option.include.fileNames = FALSE; } } if (Option.append) { notice = "append mode is not compatible with"; if (isDestinationStdout ()) error (FATAL, "%s tags to stdout", notice); } if (Option.filter) { notice = "filter mode"; if (Option.printTotals) { error (WARNING, "%s disables totals", notice); Option.printTotals = FALSE; } if (Option.tagFileName != NULL) error (WARNING, "%s ignores output tag file name", notice); } #ifdef UPDATE_ENABLED if (Option.update) { notice = "update option is not compatible with"; if (Option.etags) error (FATAL, "%s emacs-style tags", notice); if (Option.filter) error (FATAL, "%s filter option", notice); if (isDestinationStdout ()) error (FATAL, "%s tags to stdout", notice); Option.append = TRUE; } #endif } static void setEtagsMode (void) { Option.etags = TRUE; Option.sorted = FALSE; Option.lineDirectives = FALSE; Option.tagRelative = TRUE; } extern void testEtagsInvocation (void) { char* const execName = eStrdup (getExecutableName ()); char* const etags = eStrdup (ETAGS); #ifdef CASE_INSENSITIVE_FILENAMES toLowerString (execName); toLowerString (etags); #endif if (strstr (execName, etags) != NULL) { verbose ("Running in etags mode\n"); setEtagsMode (); } } /* * Cooked argument parsing */ static void parseShortOption (cookedArgs *const args) { args->simple [0] = *args->shortOptions++; args->simple [1] = '\0'; args->item = args->simple; if (! isCompoundOption (*args->simple)) args->parameter = ""; else if (*args->shortOptions == '\0') { argForth (args->args); if (argOff (args->args)) args->parameter = NULL; else args->parameter = argItem (args->args); args->shortOptions = NULL; } else { args->parameter = args->shortOptions; args->shortOptions = NULL; } } static void parseLongOption (cookedArgs *const args, const char *item) { const char* const equal = strchr (item, '='); if (equal == NULL) { args->item = eStrdup (item); args->parameter = ""; } else { const size_t length = equal - item; args->item = xMalloc (length + 1, char); strncpy (args->item, item, length); args->item [length] = '\0'; args->parameter = equal + 1; } } static void cArgRead (cookedArgs *const current) { char* item; Assert (current != NULL); if (! argOff (current->args)) { item = argItem (current->args); current->shortOptions = NULL; Assert (item != NULL); if (strncmp (item, "--", (size_t) 2) == 0) { current->isOption = TRUE; current->longOption = TRUE; parseLongOption (current, item + 2); } else if (*item == '-') { current->isOption = TRUE; current->longOption = FALSE; current->shortOptions = item + 1; parseShortOption (current); } else { current->isOption = FALSE; current->longOption = FALSE; current->item = item; current->parameter = ""; } } } extern cookedArgs* cArgNewFromString (const char* string) { cookedArgs* const result = xMalloc (1, cookedArgs); memset (result, 0, sizeof (cookedArgs)); result->args = argNewFromString (string); cArgRead (result); return result; } extern cookedArgs* cArgNewFromArgv (char* const* const argv) { cookedArgs* const result = xMalloc (1, cookedArgs); memset (result, 0, sizeof (cookedArgs)); result->args = argNewFromArgv (argv); cArgRead (result); return result; } extern cookedArgs* cArgNewFromFile (FILE* const fp) { cookedArgs* const result = xMalloc (1, cookedArgs); memset (result, 0, sizeof (cookedArgs)); result->args = argNewFromFile (fp); cArgRead (result); return result; } extern cookedArgs* cArgNewFromLineFile (FILE* const fp) { cookedArgs* const result = xMalloc (1, cookedArgs); memset (result, 0, sizeof (cookedArgs)); result->args = argNewFromLineFile (fp); cArgRead (result); return result; } extern void cArgDelete (cookedArgs* const current) { Assert (current != NULL); argDelete (current->args); memset (current, 0, sizeof (cookedArgs)); eFree (current); } static boolean cArgOptionPending (cookedArgs* const current) { boolean result = FALSE; if (current->shortOptions != NULL) if (*current->shortOptions != '\0') result = TRUE; return result; } extern boolean cArgOff (cookedArgs* const current) { Assert (current != NULL); return (boolean) (argOff (current->args) && ! cArgOptionPending (current)); } extern boolean cArgIsOption (cookedArgs* const current) { Assert (current != NULL); return current->isOption; } extern const char* cArgItem (cookedArgs* const current) { Assert (current != NULL); return current->item; } extern void cArgForth (cookedArgs* const current) { Assert (current != NULL); Assert (! cArgOff (current)); if (cArgOptionPending (current)) parseShortOption (current); else { Assert (! argOff (current->args)); argForth (current->args); if (! argOff (current->args)) cArgRead (current); else { current->isOption = FALSE; current->longOption = FALSE; current->shortOptions = NULL; current->item = NULL; current->parameter = NULL; } } } /* * File extension and language mapping */ static void addExtensionList (stringList *const slist, const char *const elist, const boolean clear) { char *const extensionList = eStrdup (elist); const char *extension = NULL; boolean first = TRUE; if (clear) { verbose (" clearing\n"); stringListClear (slist); } verbose (" adding: "); if (elist != NULL && *elist != '\0') { extension = extensionList; if (elist [0] == EXTENSION_SEPARATOR) ++extension; } while (extension != NULL) { char *separator = strchr (extension, EXTENSION_SEPARATOR); if (separator != NULL) *separator = '\0'; verbose ("%s%s", first ? "" : ", ", *extension == '\0' ? "(NONE)" : extension); stringListAdd (slist, vStringNewInit (extension)); first = FALSE; if (separator == NULL) extension = NULL; else extension = separator + 1; } if (Option.verbose) { printf ("\n now: "); stringListPrint (slist); putchar ('\n'); } eFree (extensionList); } extern const char *fileExtension (const char *const fileName) { const char *extension; const char *pDelimiter = NULL; #ifdef QDOS pDelimiter = strrchr (fileName, '_'); #endif if (pDelimiter == NULL) pDelimiter = strrchr (fileName, '.'); if (pDelimiter == NULL) extension = ""; else extension = pDelimiter + 1; /* skip to first char of extension */ return extension; } /* Determines whether the specified file name is considered to be a header * file for the purposes of determining whether enclosed tags are global or * static. */ extern boolean isIncludeFile (const char *const fileName) { boolean result = FALSE; const char *const extension = fileExtension (fileName); if (Option.headerExt != NULL) result = stringListExtensionMatched (Option.headerExt, extension); return result; } static void processLanguageForceOption (const char *const option, const char *const parameter) { langType language; if (stricmp (parameter, "auto") == 0) language = LANG_AUTO; else language = getNamedLanguage (parameter); if (strcmp (option, "lang") == 0 || strcmp (option, "language") == 0) error (WARNING, "\"--%s\" option is obsolete; use \"--language-force\" instead", option); if (language == LANG_IGNORE) error (FATAL, "Uknown language specified in \"%s\" option", option); else Option.language = language; } static char* skipPastMap (char* p) { while (*p != EXTENSION_SEPARATOR && *p != PATTERN_START && *p != ',' && *p != '\0') ++p; return p; } /* Parses the mapping beginning at `map', adds it to the language map, and * returns first character past the map. */ static char* addLanguageMap (const langType language, char* map) { char* p = NULL; const char first = *map; if (first == EXTENSION_SEPARATOR) /* extension map */ { ++map; p = skipPastMap (map); if (*p == '\0') { verbose (" .%s", map); addLanguageExtensionMap (language, map); p = map + strlen (map); } else { const char separator = *p; *p = '\0'; verbose (" .%s", map); addLanguageExtensionMap (language, map); *p = separator; } } else if (first == PATTERN_START) /* pattern map */ { ++map; for (p = map ; *p != PATTERN_STOP && *p != '\0' ; ++p) { if (*p == '\\' && *(p + 1) == PATTERN_STOP) ++p; } if (*p == '\0') error (FATAL, "Unterminated file name pattern for %s language", getLanguageName (language)); else { *p++ = '\0'; verbose (" (%s)", map); addLanguagePatternMap (language, map); } } else error (FATAL, "Badly formed language map for %s language", getLanguageName (language)); return p; } static char* processLanguageMap (char* map) { char* const separator = strchr (map, ':'); char* result = NULL; if (separator != NULL) { langType language; char *list = separator + 1; boolean clear = FALSE; *separator = '\0'; language = getNamedLanguage (map); if (language != LANG_IGNORE) { char* p; if (*list == '+') ++list; else clear = TRUE; for (p = list ; *p != ',' && *p != '\0' ; ++p) /*no-op*/ ; if (strnicmp (list, "default", p - list) == 0) { verbose (" Restoring default %s language map: ", getLanguageName (language)); installLanguageMapDefault (language); list = p; } else { if (clear) { verbose (" Setting %s language map:", getLanguageName (language)); clearLanguageMap (language); } else verbose (" Adding to %s language map:", getLanguageName (language)); while (list != NULL && *list != '\0' && *list != ',') list = addLanguageMap (language, list); verbose ("\n"); } if (list != NULL && *list == ',') ++list; result = list; } } return result; } static void processLanguageMapOption (const char *const option, const char *const parameter) { char *const maps = eStrdup (parameter); char *map = maps; if (strcmp (parameter, "default") == 0) { verbose (" Restoring default language maps:\n"); installLanguageMapDefaults (); } else while (map != NULL && *map != '\0') { char* const next = processLanguageMap (map); if (next == NULL) error (WARNING, "Unknown language specified in \"%s\" option", option); map = next; } eFree (maps); } static void processLanguageListOption (const char *const option, const char *const parameter) { char *const langs = eStrdup (parameter); enum { Add, Remove, Replace } mode = Replace; boolean first = TRUE; char *lang = langs; const char* prefix = ""; verbose (" Enabled languages: "); while (lang != NULL) { char *const end = strchr (lang, ','); if (lang [0] == '+') { ++lang; mode = Add; prefix = "+ "; } else if (lang [0] == '-') { ++lang; mode = Remove; prefix = "- "; } if (mode == Replace) enableLanguages (FALSE); if (end != NULL) *end = '\0'; if (lang [0] != '\0') { if (strcmp (lang, "all") == 0) enableLanguages ((boolean) (mode != Remove)); else { const langType language = getNamedLanguage (lang); if (language == LANG_IGNORE) error (WARNING, "Unknown language specified in \"%s\" option", option); else enableLanguage (language, (boolean) (mode != Remove)); } verbose ("%s%s%s", (first ? "" : ", "), prefix, lang); prefix = ""; first = FALSE; if (mode == Replace) mode = Add; } lang = (end != NULL ? end + 1 : NULL); } verbose ("\n"); eFree (langs); } static void processOptionFile (const char *const option, const char *const parameter) { if (parameter == NULL || parameter [0] == '\0') error (WARNING, "no option file supplied for \"%s\"", option); else if (! parseFileOptions (parameter)) error (FATAL | PERROR, "cannot open option file \"%s\"", parameter); } static void installHeaderListDefaults (void) { Option.headerExt = stringListNewFromArgv (HeaderExtensions); if (Option.verbose) { printf (" Setting default header extensions: "); stringListPrint (Option.headerExt); putchar ('\n'); } } static void processHeaderListOption (const int option, const char *parameter) { /* Check to make sure that the user did not enter "ctags -h *.c" * by testing to see if the list is a filename that exists. */ if (doesFileExist (parameter)) error (FATAL, "-%c: Invalid list", option); if (strcmp (parameter, "default") == 0) installHeaderListDefaults (); else { boolean clear = TRUE; if (parameter [0] == '+') { ++parameter; clear = FALSE; } if (Option.headerExt == NULL) Option.headerExt = stringListNew (); verbose (" Header Extensions:\n"); addExtensionList (Option.headerExt, parameter, clear); } } /* * Token ignore processing */ /* Determines whether or not "name" should be ignored, per the ignore list. */ extern boolean isIgnoreToken (const char *const name, boolean *const pIgnoreParens, const char **const replacement) { boolean result = FALSE; if (Option.ignore != NULL) { const size_t nameLen = strlen (name); unsigned int i; if (pIgnoreParens != NULL) *pIgnoreParens = FALSE; for (i = 0 ; i < stringListCount (Option.ignore) ; ++i) { vString *token = stringListItem (Option.ignore, i); if (strncmp (vStringValue (token), name, nameLen) == 0) { const size_t tokenLen = vStringLength (token); if (nameLen == tokenLen) { result = TRUE; break; } else if (tokenLen == nameLen + 1 && vStringChar (token, tokenLen - 1) == '+') { result = TRUE; if (pIgnoreParens != NULL) *pIgnoreParens = TRUE; break; } else if (vStringChar (token, nameLen) == '=') { if (replacement != NULL) *replacement = vStringValue (token) + nameLen + 1; break; } } } } return result; } static void saveIgnoreToken (vString *const ignoreToken) { if (Option.ignore == NULL) Option.ignore = stringListNew (); stringListAdd (Option.ignore, ignoreToken); verbose (" ignore token: %s\n", vStringValue (ignoreToken)); } static void readIgnoreList (const char *const list) { char* newList = stringCopy (list); const char *token = strtok (newList, IGNORE_SEPARATORS); while (token != NULL) { vString *const entry = vStringNewInit (token); saveIgnoreToken (entry); token = strtok (NULL, IGNORE_SEPARATORS); } eFree (newList); } void addIgnoreListFromFile (const char *const fileName) { stringList* tokens = stringListNewFromFile (fileName); if (Option.ignore == NULL) Option.ignore = tokens; else stringListCombine (Option.ignore, tokens); } static void processIgnoreOption (const char *const list) { if (strchr ("@./\\", list [0]) != NULL) { const char* fileName = (*list == '@') ? list + 1 : list; addIgnoreListFromFile (fileName); } #if defined (MSDOS) || defined (WIN32) || defined (OS2) else if (isalpha (list [0]) && list [1] == ':') addIgnoreListFromFile (list); #endif else if (strcmp (list, "-") == 0) { freeList (&Option.ignore); verbose (" clearing list\n"); } else readIgnoreList (list); } /* * Specific option processing */ static void printfFeatureList (void) { int i; for (i = 0 ; Features [i] != NULL ; ++i) { if (i == 0) printf (" Optional compiled features: "); printf ("%s+%s", (i>0 ? ", " : ""), Features [i]); #ifdef CUSTOM_CONFIGURATION_FILE if (strcmp (Features [i], "custom-conf") == 0) printf ("=%s", CUSTOM_CONFIGURATION_FILE); #endif } if (i > 0) putchar ('\n'); } static void printProgramIdentification (void) { printf ("%s %s, Copyright (C) 1996-2001 %s\n", PROGRAM_NAME, PROGRAM_VERSION, AUTHOR_NAME); printf (" Addresses: <%s>, %s\n", AUTHOR_EMAIL, PROGRAM_URL); printfFeatureList (); } static void printInvocationDescription (void) { printf (INVOCATION, getExecutableName ()); } static void printOptionDescriptions (const optionDescription *const optDesc) { int i; for (i = 0 ; optDesc [i].description != NULL ; ++i) { if (! Option.etags || optDesc [i].usedByEtags) puts (optDesc [i].description); } } static void printHelp (const optionDescription *const optDesc) { printProgramIdentification (); putchar ('\n'); printInvocationDescription (); putchar ('\n'); printOptionDescriptions (optDesc); } static void processExcmdOption (const char *const option, const char *const parameter) { switch (*parameter) { case 'm': Option.locate = EX_MIX; break; case 'n': Option.locate = EX_LINENUM; break; case 'p': Option.locate = EX_PATTERN; break; default: error (FATAL, "Invalid value for \"%s\" option", option); break; } } static void processFormatOption (const char *const option, const char *const parameter) { unsigned int format; if (sscanf (parameter, "%u", &format) < 1) error (FATAL, "Invalid value for \"%s\" option",option); else if (format <= (unsigned int) MaxSupportedTagFormat) Option.tagFileFormat = format; else error (FATAL, "Unsupported value for \"%s\" option", option); } static void processEtagsInclude (const char *const __unused__ option, const char *const parameter) { vString *const file = vStringNewInit (parameter); if (Option.etagsInclude == NULL) Option.etagsInclude = stringListNew (); stringListAdd (Option.etagsInclude, file); } static void processFilterTerminatorOption (const char *const __unused__ option, const char *const parameter) { freeString (&Option.filterTerminator); Option.filterTerminator = stringCopy (parameter); } static void processExtensionFieldsOption (const char *const option, const char *const parameter) { struct sExtFields *const field = &Option.extensionFields; const char *p = parameter; boolean mode = TRUE; int c; if (*p != '+' && *p != '-') { field->access = FALSE; field->fileScope = FALSE; field->implementation = FALSE; field->inheritance = FALSE; field->kind = FALSE; field->kindKey = FALSE; field->kindLong = FALSE; field->language = FALSE; field->scope = FALSE; field->argList = FALSE; } while ((c = *p++) != '\0') switch (c) { case '+': mode = TRUE; break; case '-': mode = FALSE; break; case 'a': field->access = mode; break; case 'f': field->fileScope = mode; break; case 'm': field->implementation = mode; break; case 'i': field->inheritance = mode; break; case 'k': field->kind = mode; break; case 'K': field->kindLong = mode; break; case 'l': field->language = mode; break; case 'n': field->lineNumber = mode; break; case 's': field->scope = mode; break; case 'z': field->kindKey = mode; break; case 'P': field->filePosition = mode; break; case 'A': field->argList = mode; break; default: error(WARNING, "Unsupported parameter '%c' for \"%s\" option", c, option); break; } } static void processExtraTagsOption (const char *const option, const char *const parameter) { struct sInclude *const inc = &Option.include; const char *p = parameter; boolean mode = TRUE; int c; if (*p != '+' && *p != '-') { inc->fileNames = FALSE; inc->qualifiedTags = FALSE; #if 0 inc->fileScope = FALSE; #endif } while ((c = *p++) != '\0') switch (c) { case '+': mode = TRUE; break; case '-': mode = FALSE; break; case 'f': inc->fileNames = mode; break; case 'q': inc->qualifiedTags = mode; break; #if 0 case 'F': inc->fileScope = mode; break; #endif default: error(WARNING, "Unsupported parameter '%c' for \"%s\" option", c, option); break; } } /* * Option tables */ static parametricOption ParametricOptions [] = { { "etags-include", processEtagsInclude, FALSE }, { "exclude", processExcludeOption, FALSE }, { "excmd", processExcmdOption, FALSE }, { "extra", processExtraTagsOption, FALSE }, { "fields", processExtensionFieldsOption, FALSE }, { "filter-terminator", processFilterTerminatorOption, TRUE }, { "format", processFormatOption, TRUE }, { "lang", processLanguageForceOption, FALSE }, { "language", processLanguageForceOption, FALSE }, { "language-force", processLanguageForceOption, FALSE }, { "languages", processLanguageListOption, FALSE }, { "langdef", processLanguageDefineOption, FALSE }, { "langmap", processLanguageMapOption, FALSE }, { "options", processOptionFile, FALSE }, }; static booleanOption BooleanOptions [] = { { "append", &Option.append, TRUE }, { "file-scope", &Option.include.fileScope, FALSE }, { "file-tags", &Option.include.fileNames, FALSE }, { "filter", &Option.filter, TRUE }, { "if0", &Option.if0, FALSE }, { "kind-long", &Option.kindLong, TRUE }, { "line-directives",&Option.lineDirectives, FALSE }, { "nest",&Option.nestFunction, FALSE }, { "links", &Option.followLinks, FALSE }, #ifdef RECURSE_SUPPORTED { "recurse", &Option.recurse, FALSE }, #endif { "sort", &Option.sorted, TRUE }, { "tag-relative", &Option.tagRelative, TRUE }, { "totals", &Option.printTotals, TRUE }, { "verbose", &Option.verbose, FALSE }, }; /* * Generic option parsing */ static void checkOptionOrder (const char* const option) { if (ParsedLeadingOptions) error (FATAL, "-%s option may not follow a file name", option); } static boolean processParametricOption (const char *const option, const char *const parameter) { const int count = sizeof (ParametricOptions) / sizeof (parametricOption); boolean found = FALSE; int i; for (i = 0 ; i < count && ! found ; ++i) { parametricOption* const entry = &ParametricOptions [i]; if (strcmp (option, entry->name) == 0) { found = TRUE; if (entry->initOnly) checkOptionOrder (option); (entry->handler) (option, parameter); } } return found; } static boolean getBooleanOption (const char *const option, const char *const parameter) { boolean selection = TRUE; if (parameter [0] == '\0') selection = TRUE; else if (strcmp (parameter, "0" ) == 0 || strcmp (parameter, "no" ) == 0 || strcmp (parameter, "off") == 0) selection = FALSE; else if (strcmp (parameter, "1" ) == 0 || strcmp (parameter, "yes") == 0 || strcmp (parameter, "on" ) == 0) selection = TRUE; else error (FATAL, "Invalid value for \"%s\" option", option); return selection; } static boolean processBooleanOption (const char *const option, const char *const parameter) { const int count = sizeof (BooleanOptions) / sizeof (booleanOption); boolean found = FALSE; int i; for (i = 0 ; i < count && ! found ; ++i) { booleanOption* const entry = &BooleanOptions [i]; if (strcmp (option, entry->name) == 0) { found = TRUE; if (entry->initOnly) checkOptionOrder (option); *entry->pValue = getBooleanOption (option, parameter); } } return found; } static void processLongOption (const char *const option, const char *const parameter) { if (parameter == NULL || *parameter == '\0') verbose (" Option: --%s\n", option); else verbose (" Option: --%s=%s\n", option, parameter); if (processBooleanOption (option, parameter)) ; else if (processParametricOption (option, parameter)) ; else if (processKindOption (option, parameter)) ; #ifdef HAVE_REGEX else if (processRegexOption (option, parameter)) ; #endif #define isOption(item) (boolean) (strcmp (item,option) == 0) else if (isOption ("help")) { printHelp (LongOptionDescription); printKindOptions (); exit (0); } else if (isOption ("license")) { printProgramIdentification (); puts (""); puts (License); exit (0); } else if (isOption ("version")) { printProgramIdentification (); exit (0); } #ifndef RECURSE_SUPPORTED else if (isOption ("recurse")) error (WARNING, "%s option not supported on this host", option); #endif else error (FATAL, "Unknown option: --%s", option); #undef isOption } static void processShortOption (const char *const option, const char *const parameter) { if (parameter == NULL || *parameter == '\0') verbose (" Option: -%s\n", option); else verbose (" Option: -%s %s\n", option, parameter); if (isCompoundOption (*option) && *parameter == '\0') error (FATAL, "Missing parameter for \"%s\" option", option); else switch (*option) { case '?': printHelp (LongOptionDescription); exit (0); break; case 'a': checkOptionOrder (option); Option.append = TRUE; break; case 'B': Option.backward = TRUE; break; case 'e': checkOptionOrder (option); setEtagsMode (); break; case 'f': case 'o': checkOptionOrder (option); if (Option.tagFileName != NULL) { error (WARNING, "-%s option specified more than once, last value used", option); freeString (&Option.tagFileName); } else if (parameter [0] == '-' && parameter [1] != '\0') error (FATAL, "output file name may not begin with a '-'"); Option.tagFileName = stringCopy (parameter); break; case 'F': Option.backward = FALSE; break; case 'h': processHeaderListOption (*option, parameter); break; case 'i': processLegacyKindOption (parameter); break; case 'I': processIgnoreOption (parameter); break; case 'L': if (Option.fileList != NULL) { error (WARNING, "-%s option specified more than once, last value used", option); freeString (&Option.fileList); } Option.fileList = stringCopy (parameter); break; case 'n': Option.locate = EX_LINENUM; break; case 'N': Option.locate = EX_PATTERN; break; case 'R': #ifdef RECURSE_SUPPORTED Option.recurse = TRUE; #else error (WARNING, "-%s option not supported on this host", option); #endif break; case 'u': checkOptionOrder (option); Option.sorted = FALSE; break; case 'V': Option.verbose = TRUE; break; case 'w': /* silently ignored */ break; case 'x': checkOptionOrder (option); Option.xref = TRUE; break; default: error (FATAL, "Unknown option: -%s", option); break; } } extern void parseOption (cookedArgs* const args) { Assert (! cArgOff (args)); if (args->isOption) { if (args->longOption) processLongOption (args->item, args->parameter); else { const char *parameter = args->parameter; while (*parameter == ' ') ++parameter; processShortOption (args->item, parameter); } cArgForth (args); } } extern void parseOptions (cookedArgs* const args) { while (! cArgOff (args) && cArgIsOption (args)) parseOption (args); if (! cArgOff (args) && ! cArgIsOption (args)) ParsedLeadingOptions = TRUE; } static boolean parseFileOptions (const char* const fileName) { boolean fileFound = FALSE; FILE* const fp = fopen (fileName, "r"); if (fp != NULL) { cookedArgs* const args = cArgNewFromLineFile (fp); verbose ("Reading options from %s\n", fileName); parseOptions (args); cArgDelete (args); fclose (fp); fileFound = TRUE; } return fileFound; } extern void previewFirstOption (cookedArgs* const args) { if (cArgIsOption (args) && (strcmp (args->item, "V") == 0 || strcmp (args->item, "verbose") == 0)) { parseOption (args); } } static void parseConfigurationFileOptions (void) { const char* const home = getenv ("HOME"); parseFileOptions ("/etc/ctags.conf"); parseFileOptions ("/usr/local/etc/ctags.conf"); #ifdef CUSTOM_CONFIGURATION_FILE parseFileOptions (CUSTOM_CONFIGURATION_FILE); #endif if (home != NULL) { vString* const dotFile = combinePathAndFile (home, ".ctags"); parseFileOptions (vStringValue (dotFile)); vStringDelete (dotFile); } parseFileOptions (".ctags"); } static void parseEnvironmentOptions (void) { const char *envOptions = NULL; if (Option.etags) envOptions = getenv (ETAGS_ENVIRONMENT); if (envOptions == NULL) envOptions = getenv (CTAGS_ENVIRONMENT); if (envOptions != NULL && envOptions [0] != '\0') { cookedArgs* const args = cArgNewFromString (envOptions); verbose ("Reading options from $CTAGS\n"); parseOptions (args); cArgDelete (args); } } extern void readOptionConfiguration (void) { parseConfigurationFileOptions (); parseEnvironmentOptions (); } /* * Option initialization */ extern void initOptions (void) { verbose ("Setting option defaults\n"); installHeaderListDefaults (); verbose (" Installing default language mappings:\n"); installLanguageMapDefaults (); } extern void freeOptionResources (void) { freeString (&Option.tagFileName); freeString (&Option.fileList); freeString (&Option.filterTerminator); freeList (&Option.ignore); freeList (&Option.headerExt); freeList (&Option.etagsInclude); } /* vi:set tabstop=8 shiftwidth=4: */