/* * Copyright (c) 1998-2003, Darren Hiebert * * This source code is released for free distribution under the terms of the * GNU General Public License. * * This module contains functions for generating tags for Fortran language * files. */ /* * INCLUDE FILES */ #include "general.h" /* must always come first */ #include #include #include /* to define tolower () */ #include #include #include "entry.h" #include "keyword.h" #include "main.h" #include "options.h" #include "parse.h" #include "read.h" #include "vstring.h" /* * MACROS */ #define isident(c) (isalnum(c) || (c) == '_') #define isBlank(c) (boolean) (c == ' ' || c == '\t') #define isType(token,t) (boolean) ((token)->type == (t)) #define isKeyword(token,k) (boolean) ((token)->keyword == (k)) #define isSecondaryKeyword(token,k) (boolean) ((token)->secondary == NULL ? \ FALSE : (token)->secondary->keyword == (k)) /* * DATA DECLARATIONS */ typedef enum eException { ExceptionNone, ExceptionEOF, ExceptionFixedFormat, ExceptionLoop } exception_t; /* Used to designate type of line read in fixed source form. */ typedef enum eFortranLineType { LTYPE_UNDETERMINED, LTYPE_INVALID, LTYPE_COMMENT, LTYPE_CONTINUATION, LTYPE_EOF, LTYPE_INITIAL, LTYPE_SHORT } lineType; /* Used to specify type of keyword. */ typedef enum eKeywordId { KEYWORD_NONE = -1, KEYWORD_allocatable, KEYWORD_assignment, KEYWORD_automatic, KEYWORD_block, KEYWORD_byte, KEYWORD_cexternal, KEYWORD_cglobal, KEYWORD_character, KEYWORD_common, KEYWORD_complex, KEYWORD_contains, KEYWORD_data, KEYWORD_dimension, KEYWORD_dllexport, KEYWORD_dllimport, KEYWORD_do, KEYWORD_double, KEYWORD_elemental, KEYWORD_end, KEYWORD_entry, KEYWORD_equivalence, KEYWORD_extends, KEYWORD_external, KEYWORD_format, KEYWORD_function, KEYWORD_if, KEYWORD_implicit, KEYWORD_include, KEYWORD_inline, KEYWORD_integer, KEYWORD_intent, KEYWORD_interface, KEYWORD_intrinsic, KEYWORD_logical, KEYWORD_map, KEYWORD_module, KEYWORD_namelist, KEYWORD_operator, KEYWORD_optional, KEYWORD_parameter, KEYWORD_pascal, KEYWORD_pexternal, KEYWORD_pglobal, KEYWORD_pointer, KEYWORD_precision, KEYWORD_private, KEYWORD_program, KEYWORD_public, KEYWORD_pure, KEYWORD_real, KEYWORD_record, KEYWORD_recursive, KEYWORD_save, KEYWORD_select, KEYWORD_sequence, KEYWORD_static, KEYWORD_stdcall, KEYWORD_structure, KEYWORD_subroutine, KEYWORD_target, KEYWORD_then, KEYWORD_type, KEYWORD_union, KEYWORD_use, KEYWORD_value, KEYWORD_virtual, KEYWORD_volatile, KEYWORD_where, KEYWORD_while } keywordId; /* Used to determine whether keyword is valid for the token language and * what its ID is. */ typedef struct sKeywordDesc { const char *name; keywordId id; } keywordDesc; typedef enum eTokenType { TOKEN_UNDEFINED, TOKEN_COMMA, TOKEN_DOUBLE_COLON, TOKEN_IDENTIFIER, TOKEN_KEYWORD, TOKEN_LABEL, TOKEN_NUMERIC, TOKEN_OPERATOR, TOKEN_PAREN_CLOSE, TOKEN_PAREN_OPEN, TOKEN_PERCENT, TOKEN_STATEMENT_END, TOKEN_STRING } tokenType; typedef enum eTagType { TAG_UNDEFINED = -1, TAG_BLOCK_DATA, TAG_COMMON_BLOCK, TAG_ENTRY_POINT, TAG_FUNCTION, TAG_INTERFACE, TAG_COMPONENT, TAG_LABEL, TAG_LOCAL, TAG_MODULE, TAG_NAMELIST, TAG_PROGRAM, TAG_SUBROUTINE, TAG_DERIVED_TYPE, TAG_VARIABLE, TAG_COUNT /* must be last */ } tagType; typedef struct sTokenInfo { tokenType type; keywordId keyword; tagType tag; vString* string; struct sTokenInfo *secondary; unsigned long lineNumber; MIOPos filePosition; } tokenInfo; /* * DATA DEFINITIONS */ static langType Lang_fortran; static langType Lang_f77; static jmp_buf Exception; static int Ungetc = '\0'; static unsigned int Column = 0; static boolean FreeSourceForm = FALSE; static boolean ParsingString; static tokenInfo *Parent = NULL; /* indexed by tagType */ static kindOption FortranKinds [] = { { TRUE, 'b', "block data", "block data"}, { TRUE, 'c', "macro", "common blocks"}, { TRUE, 'e', "entry", "entry points"}, { TRUE, 'f', "function", "functions"}, { FALSE, 'i', "interface", "interface contents, generic names, and operators"}, { TRUE, 'k', "component", "type and structure components"}, { TRUE, 'l', "label", "labels"}, { FALSE, 'L', "local", "local, common block, and namelist variables"}, { TRUE, 'm', "namespace", "modules"}, { TRUE, 'n', "namelist", "namelists"}, { TRUE, 'p', "package", "programs"}, { TRUE, 's', "member", "subroutines"}, { TRUE, 't', "typedef", "derived types and structures"}, { TRUE, 'v', "variable", "program (global) and module variables"} }; /* For efinitions of Fortran 77 with extensions: * http://www.fortran.com/fortran/F77_std/rjcnf0001.html * http://scienide.uwaterloo.ca/MIPSpro7/007-2362-004/sgi_html/index.html * * For the Compaq Fortran Reference Manual: * http://h18009.www1.hp.com/fortran/docs/lrm/dflrm.htm */ static const keywordDesc FortranKeywordTable [] = { /* keyword keyword ID */ { "allocatable", KEYWORD_allocatable }, { "assignment", KEYWORD_assignment }, { "automatic", KEYWORD_automatic }, { "block", KEYWORD_block }, { "byte", KEYWORD_byte }, { "cexternal", KEYWORD_cexternal }, { "cglobal", KEYWORD_cglobal }, { "character", KEYWORD_character }, { "common", KEYWORD_common }, { "complex", KEYWORD_complex }, { "contains", KEYWORD_contains }, { "data", KEYWORD_data }, { "dimension", KEYWORD_dimension }, { "dll_export", KEYWORD_dllexport }, { "dll_import", KEYWORD_dllimport }, { "do", KEYWORD_do }, { "double", KEYWORD_double }, { "elemental", KEYWORD_elemental }, { "end", KEYWORD_end }, { "entry", KEYWORD_entry }, { "equivalence", KEYWORD_equivalence }, { "extends", KEYWORD_extends }, { "external", KEYWORD_external }, { "format", KEYWORD_format }, { "function", KEYWORD_function }, { "if", KEYWORD_if }, { "implicit", KEYWORD_implicit }, { "include", KEYWORD_include }, { "inline", KEYWORD_inline }, { "integer", KEYWORD_integer }, { "intent", KEYWORD_intent }, { "interface", KEYWORD_interface }, { "intrinsic", KEYWORD_intrinsic }, { "logical", KEYWORD_logical }, { "map", KEYWORD_map }, { "module", KEYWORD_module }, { "namelist", KEYWORD_namelist }, { "operator", KEYWORD_operator }, { "optional", KEYWORD_optional }, { "parameter", KEYWORD_parameter }, { "pascal", KEYWORD_pascal }, { "pexternal", KEYWORD_pexternal }, { "pglobal", KEYWORD_pglobal }, { "pointer", KEYWORD_pointer }, { "precision", KEYWORD_precision }, { "private", KEYWORD_private }, { "program", KEYWORD_program }, { "public", KEYWORD_public }, { "pure", KEYWORD_pure }, { "real", KEYWORD_real }, { "record", KEYWORD_record }, { "recursive", KEYWORD_recursive }, { "save", KEYWORD_save }, { "select", KEYWORD_select }, { "sequence", KEYWORD_sequence }, { "static", KEYWORD_static }, { "stdcall", KEYWORD_stdcall }, { "structure", KEYWORD_structure }, { "subroutine", KEYWORD_subroutine }, { "target", KEYWORD_target }, { "then", KEYWORD_then }, { "type", KEYWORD_type }, { "union", KEYWORD_union }, { "use", KEYWORD_use }, { "value", KEYWORD_value }, { "virtual", KEYWORD_virtual }, { "volatile", KEYWORD_volatile }, { "where", KEYWORD_where }, { "while", KEYWORD_while } }; static struct { unsigned int count; unsigned int max; tokenInfo* list; } Ancestors = { 0, 0, NULL }; /* * FUNCTION PROTOTYPES */ static void parseStructureStmt (tokenInfo *const token); static void parseUnionStmt (tokenInfo *const token); static void parseDerivedTypeDef (tokenInfo *const token); static void parseFunctionSubprogram (tokenInfo *const token); static void parseSubroutineSubprogram (tokenInfo *const token); /* * FUNCTION DEFINITIONS */ static void ancestorPush (tokenInfo *const token) { enum { incrementalIncrease = 10 }; if (Ancestors.list == NULL) { Assert (Ancestors.max == 0); Ancestors.count = 0; Ancestors.max = incrementalIncrease; Ancestors.list = xMalloc (Ancestors.max, tokenInfo); } else if (Ancestors.count == Ancestors.max) { Ancestors.max += incrementalIncrease; Ancestors.list = xRealloc (Ancestors.list, Ancestors.max, tokenInfo); } Ancestors.list [Ancestors.count] = *token; Ancestors.list [Ancestors.count].string = vStringNewCopy (token->string); Ancestors.count++; } static void ancestorPop (void) { Assert (Ancestors.count > 0); --Ancestors.count; vStringDelete (Ancestors.list [Ancestors.count].string); Ancestors.list [Ancestors.count].type = TOKEN_UNDEFINED; Ancestors.list [Ancestors.count].keyword = KEYWORD_NONE; Ancestors.list [Ancestors.count].secondary = NULL; Ancestors.list [Ancestors.count].tag = TAG_UNDEFINED; Ancestors.list [Ancestors.count].string = NULL; Ancestors.list [Ancestors.count].lineNumber = 0L; } static const tokenInfo* ancestorScope (void) { tokenInfo *result = NULL; unsigned int i; for (i = Ancestors.count ; i > 0 && result == NULL ; --i) { tokenInfo *const token = Ancestors.list + i - 1; if (token->type == TOKEN_IDENTIFIER && token->tag != TAG_UNDEFINED && token->tag != TAG_INTERFACE) result = token; } return result; } static const tokenInfo* ancestorTop (void) { Assert (Ancestors.count > 0); return &Ancestors.list [Ancestors.count - 1]; } #define ancestorCount() (Ancestors.count) static void ancestorClear (void) { while (Ancestors.count > 0) ancestorPop (); if (Ancestors.list != NULL) eFree (Ancestors.list); Ancestors.list = NULL; Ancestors.count = 0; Ancestors.max = 0; } static boolean insideInterface (void) { boolean result = FALSE; unsigned int i; for (i = 0 ; i < Ancestors.count && !result ; ++i) { if (Ancestors.list [i].tag == TAG_INTERFACE) result = TRUE; } return result; } static void buildFortranKeywordHash (const langType language) { const size_t count = sizeof (FortranKeywordTable) / sizeof (FortranKeywordTable [0]); size_t i; for (i = 0 ; i < count ; ++i) { const keywordDesc* const p = &FortranKeywordTable [i]; addKeyword (p->name, language, (int) p->id); } } /* * Tag generation functions */ static tokenInfo *newToken (void) { tokenInfo *const token = xMalloc (1, tokenInfo); token->type = TOKEN_UNDEFINED; token->keyword = KEYWORD_NONE; token->tag = TAG_UNDEFINED; token->string = vStringNew (); token->secondary = NULL; token->lineNumber = getSourceLineNumber (); token->filePosition = getInputFilePosition (); return token; } static tokenInfo *newTokenFrom (tokenInfo *const token) { tokenInfo *result = newToken (); *result = *token; result->string = vStringNewCopy (token->string); token->secondary = NULL; return result; } static void deleteToken (tokenInfo *const token) { if (token != NULL) { vStringDelete (token->string); deleteToken (token->secondary); token->secondary = NULL; eFree (token); } } static boolean isFileScope (const tagType type) { return (boolean) (type == TAG_LABEL || type == TAG_LOCAL); } static boolean includeTag (const tagType type) { boolean include; Assert (type != TAG_UNDEFINED); include = FortranKinds [(int) type].enabled; if (include && isFileScope (type)) include = Option.include.fileScope; return include; } static void makeFortranTag (tokenInfo *const token, tagType tag) { token->tag = tag; if (includeTag (token->tag)) { const char *const name = vStringValue (token->string); tagEntryInfo e; initTagEntry (&e, name); if (token->tag == TAG_COMMON_BLOCK) e.lineNumberEntry = (boolean) (Option.locate != EX_PATTERN); e.lineNumber = token->lineNumber; e.filePosition = token->filePosition; e.isFileScope = isFileScope (token->tag); e.kindName = FortranKinds [token->tag].name; e.kind = FortranKinds [token->tag].letter; e.truncateLine = (boolean) (token->tag != TAG_LABEL); if (ancestorCount () > 0) { const tokenInfo* const scope = ancestorScope (); if (scope != NULL) { e.extensionFields.scope [0] = FortranKinds [scope->tag].name; e.extensionFields.scope [1] = vStringValue (scope->string); } } if (! insideInterface () || includeTag (TAG_INTERFACE)) makeTagEntry (&e); } } /* * Parsing functions */ static int skipLine (void) { int c; do c = fileGetc (); while (c != EOF && c != '\n'); return c; } static void makeLabelTag (vString *const label) { tokenInfo *token = newToken (); token->type = TOKEN_LABEL; vStringCopy (token->string, label); makeFortranTag (token, TAG_LABEL); deleteToken (token); } static lineType getLineType (void) { vString *label = vStringNew (); int column = 0; lineType type = LTYPE_UNDETERMINED; do /* read in first 6 "margin" characters */ { int c = fileGetc (); /* 3.2.1 Comment_Line. A comment line is any line that contains * a C or an asterisk in column 1, or contains only blank characters * in columns 1 through 72. A comment line that contains a C or * an asterisk in column 1 may contain any character capable of * representation in the processor in columns 2 through 72. */ /* EXCEPTION! Some compilers permit '!' as a commment character here. * * Treat # and $ in column 1 as comment to permit preprocessor directives. * Treat D and d in column 1 as comment for HP debug statements. */ if (column == 0 && strchr ("*Cc!#$Dd", c) != NULL) type = LTYPE_COMMENT; else if (c == '\t') /* EXCEPTION! Some compilers permit a tab here */ { column = 8; type = LTYPE_INITIAL; } else if (column == 5) { /* 3.2.2 Initial_Line. An initial line is any line that is not * a comment line and contains the character blank or the digit 0 * in column 6. Columns 1 through 5 may contain a statement label * (3.4), or each of the columns 1 through 5 must contain the * character blank. */ if (c == ' ' || c == '0') type = LTYPE_INITIAL; /* 3.2.3 Continuation_Line. A continuation line is any line that * contains any character of the FORTRAN character set other than * the character blank or the digit 0 in column 6 and contains * only blank characters in columns 1 through 5. */ else if (vStringLength (label) == 0) type = LTYPE_CONTINUATION; else type = LTYPE_INVALID; } else if (c == ' ') ; else if (c == EOF) type = LTYPE_EOF; else if (c == '\n') type = LTYPE_SHORT; else if (isdigit (c)) vStringPut (label, c); else type = LTYPE_INVALID; ++column; } while (column < 6 && type == LTYPE_UNDETERMINED); Assert (type != LTYPE_UNDETERMINED); if (vStringLength (label) > 0) { vStringTerminate (label); makeLabelTag (label); } vStringDelete (label); return type; } static int getFixedFormChar (void) { boolean newline = FALSE; lineType type; int c = '\0'; if (Column > 0) { #ifdef STRICT_FIXED_FORM /* EXCEPTION! Some compilers permit more than 72 characters per line. */ if (Column > 71) c = skipLine (); else #endif { c = fileGetc (); ++Column; } if (c == '\n') { newline = TRUE; /* need to check for continuation line */ Column = 0; } else if (c == '!' && ! ParsingString) { c = skipLine (); newline = TRUE; /* need to check for continuation line */ Column = 0; } else if (c == '&') /* check for free source form */ { const int c2 = fileGetc (); if (c2 == '\n') longjmp (Exception, (int) ExceptionFixedFormat); else fileUngetc (c2); } } while (Column == 0) { type = getLineType (); switch (type) { case LTYPE_UNDETERMINED: case LTYPE_INVALID: longjmp (Exception, (int) ExceptionFixedFormat); break; case LTYPE_SHORT: break; case LTYPE_COMMENT: skipLine (); break; case LTYPE_EOF: Column = 6; if (newline) c = '\n'; else c = EOF; break; case LTYPE_INITIAL: if (newline) { c = '\n'; Column = 6; break; } /* fall through to next case */ case LTYPE_CONTINUATION: Column = 5; do { c = fileGetc (); ++Column; } while (isBlank (c)); if (c == '\n') Column = 0; else if (Column > 6) { fileUngetc (c); c = ' '; } break; default: Assert ("Unexpected line type" == NULL); } } return c; } static int skipToNextLine (void) { int c = skipLine (); if (c != EOF) c = fileGetc (); return c; } static int getFreeFormChar (void) { static boolean newline = TRUE; boolean advanceLine = FALSE; int c = fileGetc (); /* If the last nonblank, non-comment character of a FORTRAN 90 * free-format text line is an ampersand then the next non-comment * line is a continuation line. */ if (c == '&') { do c = fileGetc (); while (isspace (c) && c != '\n'); if (c == '\n') { newline = TRUE; advanceLine = TRUE; } else if (c == '!') advanceLine = TRUE; else { fileUngetc (c); c = '&'; } } else if (newline && (c == '!' || c == '#')) advanceLine = TRUE; while (advanceLine) { while (isspace (c)) c = fileGetc (); if (c == '!' || (newline && c == '#')) { c = skipToNextLine (); newline = TRUE; continue; } if (c == '&') c = fileGetc (); else advanceLine = FALSE; } newline = (boolean) (c == '\n'); return c; } static int getChar (void) { int c; if (Ungetc != '\0') { c = Ungetc; Ungetc = '\0'; } else if (FreeSourceForm) c = getFreeFormChar (); else c = getFixedFormChar (); return c; } static void ungetChar (const int c) { Ungetc = c; } /* If a numeric is passed in 'c', this is used as the first digit of the * numeric being parsed. */ static vString *parseInteger (int c) { vString *string = vStringNew (); if (c == '-') { vStringPut (string, c); c = getChar (); } else if (! isdigit (c)) c = getChar (); while (c != EOF && isdigit (c)) { vStringPut (string, c); c = getChar (); } vStringTerminate (string); if (c == '_') { do c = getChar (); while (c != EOF && isalpha (c)); } ungetChar (c); return string; } static vString *parseNumeric (int c) { vString *string = vStringNew (); vString *integer = parseInteger (c); vStringCopy (string, integer); vStringDelete (integer); c = getChar (); if (c == '.') { integer = parseInteger ('\0'); vStringPut (string, c); vStringCat (string, integer); vStringDelete (integer); c = getChar (); } if (tolower (c) == 'e') { integer = parseInteger ('\0'); vStringPut (string, c); vStringCat (string, integer); vStringDelete (integer); } else ungetChar (c); vStringTerminate (string); return string; } static void parseString (vString *const string, const int delimiter) { const unsigned long inputLineNumber = getInputLineNumber (); int c; ParsingString = TRUE; c = getChar (); while (c != delimiter && c != '\n' && c != EOF) { vStringPut (string, c); c = getChar (); } if (c == '\n' || c == EOF) { verbose ("%s: unterminated character string at line %lu\n", getInputFileName (), inputLineNumber); if (c == EOF) longjmp (Exception, (int) ExceptionEOF); else if (! FreeSourceForm) longjmp (Exception, (int) ExceptionFixedFormat); } vStringTerminate (string); ParsingString = FALSE; } /* Read a C identifier beginning with "firstChar" and places it into "name". */ static void parseIdentifier (vString *const string, const int firstChar) { int c = firstChar; do { vStringPut (string, c); c = getChar (); } while (isident (c)); vStringTerminate (string); ungetChar (c); /* unget non-identifier character */ } static void checkForLabel (void) { tokenInfo* token = NULL; int length; int c; do c = getChar (); while (isBlank (c)); for (length = 0 ; isdigit (c) && length < 5 ; ++length) { if (token == NULL) { token = newToken (); token->type = TOKEN_LABEL; } vStringPut (token->string, c); c = getChar (); } if (length > 0 && token != NULL) { vStringTerminate (token->string); makeFortranTag (token, TAG_LABEL); deleteToken (token); } ungetChar (c); } /* Analyzes the identifier contained in a statement described by the * statement structure and adjusts the structure according the significance * of the identifier. */ static keywordId analyzeToken (vString *const name, langType language) { static vString *keyword = NULL; keywordId id; if (keyword == NULL) keyword = vStringNew (); vStringCopyToLower (keyword, name); id = (keywordId) lookupKeyword (vStringValue (keyword), language); return id; } static void readIdentifier (tokenInfo *const token, const int c) { parseIdentifier (token->string, c); token->keyword = analyzeToken (token->string, Lang_fortran); if (! isKeyword (token, KEYWORD_NONE)) token->type = TOKEN_KEYWORD; else { token->type = TOKEN_IDENTIFIER; if (strncmp (vStringValue (token->string), "end", 3) == 0) { vString *const sub = vStringNewInit (vStringValue (token->string) + 3); const keywordId kw = analyzeToken (sub, Lang_fortran); vStringDelete (sub); if (kw != KEYWORD_NONE) { token->secondary = newToken (); token->secondary->type = TOKEN_KEYWORD; token->secondary->keyword = kw; token->keyword = KEYWORD_end; } } } } static void readToken (tokenInfo *const token) { int c; deleteToken (token->secondary); token->type = TOKEN_UNDEFINED; token->tag = TAG_UNDEFINED; token->keyword = KEYWORD_NONE; token->secondary = NULL; vStringClear (token->string); getNextChar: c = getChar (); token->lineNumber = getSourceLineNumber (); token->filePosition = getInputFilePosition (); switch (c) { case EOF: longjmp (Exception, (int) ExceptionEOF); break; case ' ': goto getNextChar; case '\t': goto getNextChar; case ',': token->type = TOKEN_COMMA; break; case '(': token->type = TOKEN_PAREN_OPEN; break; case ')': token->type = TOKEN_PAREN_CLOSE; break; case '%': token->type = TOKEN_PERCENT; break; case '*': case '/': case '+': case '-': case '=': case '<': case '>': { const char *const operatorChars = "*/+=<>"; do { vStringPut (token->string, c); c = getChar (); } while (strchr (operatorChars, c) != NULL); ungetChar (c); vStringTerminate (token->string); token->type = TOKEN_OPERATOR; break; } case '!': if (FreeSourceForm) { do c = getChar (); while (c != '\n' && c != EOF); } else { skipLine (); Column = 0; } /* fall through to newline case */ case '\n': token->type = TOKEN_STATEMENT_END; if (FreeSourceForm) checkForLabel (); break; case '.': parseIdentifier (token->string, c); c = getChar (); if (c == '.') { vStringPut (token->string, c); vStringTerminate (token->string); token->type = TOKEN_OPERATOR; } else { ungetChar (c); token->type = TOKEN_UNDEFINED; } break; case '"': case '\'': parseString (token->string, c); token->type = TOKEN_STRING; break; case ';': token->type = TOKEN_STATEMENT_END; break; case ':': c = getChar (); if (c == ':') token->type = TOKEN_DOUBLE_COLON; else { ungetChar (c); token->type = TOKEN_UNDEFINED; } break; default: if (isalpha (c)) readIdentifier (token, c); else if (isdigit (c)) { vString *numeric = parseNumeric (c); vStringCat (token->string, numeric); vStringDelete (numeric); token->type = TOKEN_NUMERIC; } else token->type = TOKEN_UNDEFINED; break; } } static void readSubToken (tokenInfo *const token) { if (token->secondary == NULL) { token->secondary = newToken (); readToken (token->secondary); } } /* * Scanning functions */ static void skipToToken (tokenInfo *const token, tokenType type) { while (! isType (token, type) && ! isType (token, TOKEN_STATEMENT_END) && !(token->secondary != NULL && isType (token->secondary, TOKEN_STATEMENT_END))) readToken (token); } static void skipPast (tokenInfo *const token, tokenType type) { skipToToken (token, type); if (! isType (token, TOKEN_STATEMENT_END)) readToken (token); } static void skipToNextStatement (tokenInfo *const token) { do { skipToToken (token, TOKEN_STATEMENT_END); readToken (token); } while (isType (token, TOKEN_STATEMENT_END)); } /* skip over parenthesis enclosed contents starting at next token. * Token is left at the first token following closing parenthesis. If an * opening parenthesis is not found, `token' is moved to the end of the * statement. */ static void skipOverParens (tokenInfo *const token) { int level = 0; do { if (isType (token, TOKEN_STATEMENT_END)) break; else if (isType (token, TOKEN_PAREN_OPEN)) ++level; else if (isType (token, TOKEN_PAREN_CLOSE)) --level; readToken (token); } while (level > 0); } static boolean isTypeSpec (tokenInfo *const token) { boolean result; switch (token->keyword) { case KEYWORD_byte: case KEYWORD_integer: case KEYWORD_real: case KEYWORD_double: case KEYWORD_complex: case KEYWORD_character: case KEYWORD_logical: case KEYWORD_record: case KEYWORD_type: result = TRUE; break; default: result = FALSE; break; } return result; } static boolean isSubprogramPrefix (tokenInfo *const token) { boolean result; switch (token->keyword) { case KEYWORD_elemental: case KEYWORD_pure: case KEYWORD_recursive: case KEYWORD_stdcall: result = TRUE; break; default: result = FALSE; break; } return result; } /* type-spec * is INTEGER [kind-selector] * or REAL [kind-selector] is ( etc. ) * or DOUBLE PRECISION * or COMPLEX [kind-selector] * or CHARACTER [kind-selector] * or LOGICAL [kind-selector] * or TYPE ( type-name ) * * Note that INTEGER and REAL may be followed by "*N" where "N" is an integer */ static void parseTypeSpec (tokenInfo *const token) { /* parse type-spec, leaving `token' at first token following type-spec */ Assert (isTypeSpec (token)); switch (token->keyword) { case KEYWORD_character: /* skip char-selector */ readToken (token); if (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "*") == 0) readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); else if (isType (token, TOKEN_NUMERIC)) readToken (token); break; case KEYWORD_byte: case KEYWORD_complex: case KEYWORD_integer: case KEYWORD_logical: case KEYWORD_real: readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); /* skip kind-selector */ if (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "*") == 0) { readToken (token); readToken (token); } break; case KEYWORD_double: readToken (token); if (isKeyword (token, KEYWORD_complex) || isKeyword (token, KEYWORD_precision)) readToken (token); else skipToToken (token, TOKEN_STATEMENT_END); break; case KEYWORD_record: readToken (token); if (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "/") == 0) { readToken (token); /* skip to structure name */ readToken (token); /* skip to '/' */ readToken (token); /* skip to variable name */ } break; case KEYWORD_type: readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); /* skip type-name */ else parseDerivedTypeDef (token); break; default: skipToToken (token, TOKEN_STATEMENT_END); break; } } static boolean skipStatementIfKeyword (tokenInfo *const token, keywordId keyword) { boolean result = FALSE; if (isKeyword (token, keyword)) { result = TRUE; skipToNextStatement (token); } return result; } /* parse a list of qualifying specifiers, leaving `token' at first token * following list. Examples of such specifiers are: * [[, attr-spec] ::] * [[, component-attr-spec-list] ::] * * attr-spec * is PARAMETER * or access-spec (is PUBLIC or PRIVATE) * or ALLOCATABLE * or DIMENSION ( array-spec ) * or EXTERNAL * or INTENT ( intent-spec ) * or INTRINSIC * or OPTIONAL * or POINTER * or SAVE * or TARGET * * component-attr-spec * is POINTER * or DIMENSION ( component-array-spec ) * or EXTENDS ( type name ) */ static void parseQualifierSpecList (tokenInfo *const token) { do { readToken (token); /* should be an attr-spec */ switch (token->keyword) { case KEYWORD_parameter: case KEYWORD_allocatable: case KEYWORD_external: case KEYWORD_intrinsic: case KEYWORD_optional: case KEYWORD_private: case KEYWORD_pointer: case KEYWORD_public: case KEYWORD_save: case KEYWORD_target: readToken (token); break; case KEYWORD_dimension: case KEYWORD_extends: case KEYWORD_intent: readToken (token); skipOverParens (token); break; default: skipToToken (token, TOKEN_STATEMENT_END); break; } } while (isType (token, TOKEN_COMMA)); if (! isType (token, TOKEN_DOUBLE_COLON)) skipToToken (token, TOKEN_STATEMENT_END); } static tagType variableTagType (void) { tagType result = TAG_VARIABLE; if (ancestorCount () > 0) { const tokenInfo* const parent = ancestorTop (); switch (parent->tag) { case TAG_MODULE: result = TAG_VARIABLE; break; case TAG_DERIVED_TYPE: result = TAG_COMPONENT; break; case TAG_FUNCTION: result = TAG_LOCAL; break; case TAG_SUBROUTINE: result = TAG_LOCAL; break; default: result = TAG_VARIABLE; break; } } return result; } static void parseEntityDecl (tokenInfo *const token) { Assert (isType (token, TOKEN_IDENTIFIER)); makeFortranTag (token, variableTagType ()); readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); if (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "*") == 0) { readToken (token); /* read char-length */ if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); else readToken (token); } if (isType (token, TOKEN_OPERATOR)) { if (strcmp (vStringValue (token->string), "/") == 0) { /* skip over initializations of structure field */ readToken (token); skipPast (token, TOKEN_OPERATOR); } else if (strcmp (vStringValue (token->string), "=") == 0) { while (! isType (token, TOKEN_COMMA) && ! isType (token, TOKEN_STATEMENT_END)) { readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); } } } /* token left at either comma or statement end */ } static void parseEntityDeclList (tokenInfo *const token) { if (isType (token, TOKEN_PERCENT)) skipToNextStatement (token); else while (isType (token, TOKEN_IDENTIFIER) || (isType (token, TOKEN_KEYWORD) && !isKeyword (token, KEYWORD_function) && !isKeyword (token, KEYWORD_subroutine))) { /* compilers accept keywoeds as identifiers */ if (isType (token, TOKEN_KEYWORD)) token->type = TOKEN_IDENTIFIER; parseEntityDecl (token); if (isType (token, TOKEN_COMMA)) readToken (token); else if (isType (token, TOKEN_STATEMENT_END)) { skipToNextStatement (token); break; } } } /* type-declaration-stmt is * type-spec [[, attr-spec] ... ::] entity-decl-list */ static void parseTypeDeclarationStmt (tokenInfo *const token) { Assert (isTypeSpec (token)); parseTypeSpec (token); if (!isType (token, TOKEN_STATEMENT_END)) /* if not end of derived type... */ { if (isType (token, TOKEN_COMMA)) parseQualifierSpecList (token); if (isType (token, TOKEN_DOUBLE_COLON)) readToken (token); parseEntityDeclList (token); } if (isType (token, TOKEN_STATEMENT_END)) skipToNextStatement (token); } /* namelist-stmt is * NAMELIST /namelist-group-name/ namelist-group-object-list * [[,]/[namelist-group-name]/ namelist-block-object-list] ... * * namelist-group-object is * variable-name * * common-stmt is * COMMON [/[common-block-name]/] common-block-object-list * [[,]/[common-block-name]/ common-block-object-list] ... * * common-block-object is * variable-name [ ( explicit-shape-spec-list ) ] */ static void parseCommonNamelistStmt (tokenInfo *const token, tagType type) { Assert (isKeyword (token, KEYWORD_common) || isKeyword (token, KEYWORD_namelist)); readToken (token); do { if (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "/") == 0) { readToken (token); if (isType (token, TOKEN_IDENTIFIER)) { makeFortranTag (token, type); readToken (token); } skipPast (token, TOKEN_OPERATOR); } if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, TAG_LOCAL); readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) skipOverParens (token); /* skip explicit-shape-spec-list */ if (isType (token, TOKEN_COMMA)) readToken (token); } while (! isType (token, TOKEN_STATEMENT_END)); skipToNextStatement (token); } static void parseFieldDefinition (tokenInfo *const token) { if (isTypeSpec (token)) parseTypeDeclarationStmt (token); else if (isKeyword (token, KEYWORD_structure)) parseStructureStmt (token); else if (isKeyword (token, KEYWORD_union)) parseUnionStmt (token); else skipToNextStatement (token); } static void parseMap (tokenInfo *const token) { Assert (isKeyword (token, KEYWORD_map)); skipToNextStatement (token); while (! isKeyword (token, KEYWORD_end)) parseFieldDefinition (token); readSubToken (token); /* should be at KEYWORD_map token */ skipToNextStatement (token); } /* UNION * MAP * [field-definition] [field-definition] ... * END MAP * MAP * [field-definition] [field-definition] ... * END MAP * [MAP * [field-definition] * [field-definition] ... * END MAP] ... * END UNION * * * * Typed data declarations (variables or arrays) in structure declarations * have the form of normal Fortran typed data declarations. Data items with * different types can be freely intermixed within a structure declaration. * * Unnamed fields can be declared in a structure by specifying the pseudo * name %FILL in place of an actual field name. You can use this mechanism to * generate empty space in a record for purposes such as alignment. * * All mapped field declarations that are made within a UNION declaration * share a common location within the containing structure. When initializing * the fields within a UNION, the final initialization value assigned * overlays any value previously assigned to a field definition that shares * that field. */ static void parseUnionStmt (tokenInfo *const token) { Assert (isKeyword (token, KEYWORD_union)); skipToNextStatement (token); while (isKeyword (token, KEYWORD_map)) parseMap (token); /* should be at KEYWORD_end token */ readSubToken (token); /* secondary token should be KEYWORD_end token */ skipToNextStatement (token); } /* STRUCTURE [/structure-name/] [field-names] * [field-definition] * [field-definition] ... * END STRUCTURE * * structure-name * identifies the structure in a subsequent RECORD statement. * Substructures can be established within a structure by means of either * a nested STRUCTURE declaration or a RECORD statement. * * field-names * (for substructure declarations only) one or more names having the * structure of the substructure being defined. * * field-definition * can be one or more of the following: * * Typed data declarations, which can optionally include one or more * data initialization values. * * Substructure declarations (defined by either RECORD statements or * subsequent STRUCTURE statements). * * UNION declarations, which are mapped fields defined by a block of * statements. The syntax of a UNION declaration is described below. * * PARAMETER statements, which do not affect the form of the * structure. */ static void parseStructureStmt (tokenInfo *const token) { tokenInfo *name; Assert (isKeyword (token, KEYWORD_structure)); readToken (token); if (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "/") == 0) { /* read structure name */ readToken (token); if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, TAG_DERIVED_TYPE); name = newTokenFrom (token); skipPast (token, TOKEN_OPERATOR); } else { /* fake out anonymous structure */ name = newToken (); name->type = TOKEN_IDENTIFIER; name->tag = TAG_DERIVED_TYPE; vStringCopyS (name->string, "anonymous"); } while (isType (token, TOKEN_IDENTIFIER)) { /* read field names */ makeFortranTag (token, TAG_COMPONENT); readToken (token); if (isType (token, TOKEN_COMMA)) readToken (token); } skipToNextStatement (token); ancestorPush (name); while (! isKeyword (token, KEYWORD_end)) parseFieldDefinition (token); readSubToken (token); /* secondary token should be KEYWORD_structure token */ skipToNextStatement (token); ancestorPop (); deleteToken (name); } /* specification-stmt * is access-stmt (is access-spec [[::] access-id-list) * or allocatable-stmt (is ALLOCATABLE [::] array-name etc.) * or common-stmt (is COMMON [ / [common-block-name] /] etc.) * or data-stmt (is DATA data-stmt-list [[,] data-stmt-set] ...) * or dimension-stmt (is DIMENSION [::] array-name etc.) * or equivalence-stmt (is EQUIVALENCE equivalence-set-list) * or external-stmt (is EXTERNAL etc.) * or intent-stmt (is INTENT ( intent-spec ) [::] etc.) * or instrinsic-stmt (is INTRINSIC etc.) * or namelist-stmt (is NAMELIST / namelist-group-name / etc.) * or optional-stmt (is OPTIONAL [::] etc.) * or pointer-stmt (is POINTER [::] object-name etc.) * or save-stmt (is SAVE etc.) * or target-stmt (is TARGET [::] object-name etc.) * * access-spec is PUBLIC or PRIVATE */ static boolean parseSpecificationStmt (tokenInfo *const token) { boolean result = TRUE; switch (token->keyword) { case KEYWORD_common: parseCommonNamelistStmt (token, TAG_COMMON_BLOCK); break; case KEYWORD_namelist: parseCommonNamelistStmt (token, TAG_NAMELIST); break; case KEYWORD_structure: parseStructureStmt (token); break; case KEYWORD_allocatable: case KEYWORD_data: case KEYWORD_dimension: case KEYWORD_equivalence: case KEYWORD_extends: case KEYWORD_external: case KEYWORD_intent: case KEYWORD_intrinsic: case KEYWORD_optional: case KEYWORD_pointer: case KEYWORD_private: case KEYWORD_public: case KEYWORD_save: case KEYWORD_target: skipToNextStatement (token); break; default: result = FALSE; break; } return result; } /* component-def-stmt is * type-spec [[, component-attr-spec-list] ::] component-decl-list * * component-decl is * component-name [ ( component-array-spec ) ] [ * char-length ] */ static void parseComponentDefStmt (tokenInfo *const token) { Assert (isTypeSpec (token)); parseTypeSpec (token); if (isType (token, TOKEN_COMMA)) parseQualifierSpecList (token); if (isType (token, TOKEN_DOUBLE_COLON)) readToken (token); parseEntityDeclList (token); } /* derived-type-def is * derived-type-stmt is (TYPE [[, access-spec] ::] type-name * [private-sequence-stmt] ... (is PRIVATE or SEQUENCE) * component-def-stmt * [component-def-stmt] ... * end-type-stmt */ static void parseDerivedTypeDef (tokenInfo *const token) { if (isType (token, TOKEN_COMMA)) parseQualifierSpecList (token); if (isType (token, TOKEN_DOUBLE_COLON)) readToken (token); if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, TAG_DERIVED_TYPE); ancestorPush (token); skipToNextStatement (token); if (isKeyword (token, KEYWORD_private) || isKeyword (token, KEYWORD_sequence)) { skipToNextStatement (token); } while (! isKeyword (token, KEYWORD_end)) { if (isTypeSpec (token)) parseComponentDefStmt (token); else skipToNextStatement (token); } readSubToken (token); /* secondary token should be KEYWORD_type token */ skipToToken (token, TOKEN_STATEMENT_END); ancestorPop (); } /* interface-block * interface-stmt (is INTERFACE [generic-spec]) * [interface-body] * [module-procedure-stmt] ... * end-interface-stmt (is END INTERFACE) * * generic-spec * is generic-name * or OPERATOR ( defined-operator ) * or ASSIGNMENT ( = ) * * interface-body * is function-stmt * [specification-part] * end-function-stmt * or subroutine-stmt * [specification-part] * end-subroutine-stmt * * module-procedure-stmt is * MODULE PROCEDURE procedure-name-list */ static void parseInterfaceBlock (tokenInfo *const token) { tokenInfo *name = NULL; Assert (isKeyword (token, KEYWORD_interface)); readToken (token); if (isType (token, TOKEN_IDENTIFIER)) { makeFortranTag (token, TAG_INTERFACE); name = newTokenFrom (token); } else if (isKeyword (token, KEYWORD_assignment) || isKeyword (token, KEYWORD_operator)) { readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) readToken (token); if (isType (token, TOKEN_OPERATOR)) { makeFortranTag (token, TAG_INTERFACE); name = newTokenFrom (token); } } if (name == NULL) { name = newToken (); name->type = TOKEN_IDENTIFIER; name->tag = TAG_INTERFACE; } ancestorPush (name); while (! isKeyword (token, KEYWORD_end)) { switch (token->keyword) { case KEYWORD_function: parseFunctionSubprogram (token); break; case KEYWORD_subroutine: parseSubroutineSubprogram (token); break; default: if (isSubprogramPrefix (token)) readToken (token); else if (isTypeSpec (token)) parseTypeSpec (token); else skipToNextStatement (token); break; } } readSubToken (token); /* secondary token should be KEYWORD_interface token */ skipToNextStatement (token); ancestorPop (); deleteToken (name); } /* entry-stmt is * ENTRY entry-name [ ( dummy-arg-list ) ] */ static void parseEntryStmt (tokenInfo *const token) { Assert (isKeyword (token, KEYWORD_entry)); readToken (token); if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, TAG_ENTRY_POINT); skipToNextStatement (token); } /* stmt-function-stmt is * function-name ([dummy-arg-name-list]) = scalar-expr */ static boolean parseStmtFunctionStmt (tokenInfo *const token) { boolean result = FALSE; Assert (isType (token, TOKEN_IDENTIFIER)); #if 0 /* cannot reliably parse this yet */ makeFortranTag (token, TAG_FUNCTION); #endif readToken (token); if (isType (token, TOKEN_PAREN_OPEN)) { skipOverParens (token); result = (boolean) (isType (token, TOKEN_OPERATOR) && strcmp (vStringValue (token->string), "=") == 0); } skipToNextStatement (token); return result; } static boolean isIgnoredDeclaration (tokenInfo *const token) { boolean result; switch (token->keyword) { case KEYWORD_cexternal: case KEYWORD_cglobal: case KEYWORD_dllexport: case KEYWORD_dllimport: case KEYWORD_external: case KEYWORD_format: case KEYWORD_include: case KEYWORD_inline: case KEYWORD_parameter: case KEYWORD_pascal: case KEYWORD_pexternal: case KEYWORD_pglobal: case KEYWORD_static: case KEYWORD_value: case KEYWORD_virtual: case KEYWORD_volatile: result = TRUE; break; default: result = FALSE; break; } return result; } /* declaration-construct * [derived-type-def] * [interface-block] * [type-declaration-stmt] * [specification-stmt] * [parameter-stmt] (is PARAMETER ( named-constant-def-list ) * [format-stmt] (is FORMAT format-specification) * [entry-stmt] * [stmt-function-stmt] */ static boolean parseDeclarationConstruct (tokenInfo *const token) { boolean result = TRUE; switch (token->keyword) { case KEYWORD_entry: parseEntryStmt (token); break; case KEYWORD_interface: parseInterfaceBlock (token); break; case KEYWORD_stdcall: readToken (token); break; /* derived type handled by parseTypeDeclarationStmt(); */ case KEYWORD_automatic: readToken (token); if (isTypeSpec (token)) parseTypeDeclarationStmt (token); else skipToNextStatement (token); result = TRUE; break; default: if (isIgnoredDeclaration (token)) skipToNextStatement (token); else if (isTypeSpec (token)) { parseTypeDeclarationStmt (token); result = TRUE; } else if (isType (token, TOKEN_IDENTIFIER)) result = parseStmtFunctionStmt (token); else result = parseSpecificationStmt (token); break; } return result; } /* implicit-part-stmt * is [implicit-stmt] (is IMPLICIT etc.) * or [parameter-stmt] (is PARAMETER etc.) * or [format-stmt] (is FORMAT etc.) * or [entry-stmt] (is ENTRY entry-name etc.) */ static boolean parseImplicitPartStmt (tokenInfo *const token) { boolean result = TRUE; switch (token->keyword) { case KEYWORD_entry: parseEntryStmt (token); break; case KEYWORD_implicit: case KEYWORD_include: case KEYWORD_parameter: case KEYWORD_format: skipToNextStatement (token); break; default: result = FALSE; break; } return result; } /* specification-part is * [use-stmt] ... (is USE module-name etc.) * [implicit-part] (is [implicit-part-stmt] ... [implicit-stmt]) * [declaration-construct] ... */ static boolean parseSpecificationPart (tokenInfo *const token) { boolean result = FALSE; while (skipStatementIfKeyword (token, KEYWORD_use)) result = TRUE; while (parseImplicitPartStmt (token)) result = TRUE; while (parseDeclarationConstruct (token)) result = TRUE; return result; } /* block-data is * block-data-stmt (is BLOCK DATA [block-data-name] * [specification-part] * end-block-data-stmt (is END [BLOCK DATA [block-data-name]]) */ static void parseBlockData (tokenInfo *const token) { Assert (isKeyword (token, KEYWORD_block)); readToken (token); if (isKeyword (token, KEYWORD_data)) { readToken (token); if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, TAG_BLOCK_DATA); } ancestorPush (token); skipToNextStatement (token); parseSpecificationPart (token); while (! isKeyword (token, KEYWORD_end)) skipToNextStatement (token); readSubToken (token); /* secondary token should be KEYWORD_NONE or KEYWORD_block token */ skipToNextStatement (token); ancestorPop (); } /* internal-subprogram-part is * contains-stmt (is CONTAINS) * internal-subprogram * [internal-subprogram] ... * * internal-subprogram * is function-subprogram * or subroutine-subprogram */ static void parseInternalSubprogramPart (tokenInfo *const token) { boolean done = FALSE; if (isKeyword (token, KEYWORD_contains)) skipToNextStatement (token); do { switch (token->keyword) { case KEYWORD_function: parseFunctionSubprogram (token); break; case KEYWORD_subroutine: parseSubroutineSubprogram (token); break; case KEYWORD_end: done = TRUE; break; default: if (isSubprogramPrefix (token)) readToken (token); else if (isTypeSpec (token)) parseTypeSpec (token); else readToken (token); break; } } while (! done); } /* module is * module-stmt (is MODULE module-name) * [specification-part] * [module-subprogram-part] * end-module-stmt (is END [MODULE [module-name]]) * * module-subprogram-part * contains-stmt (is CONTAINS) * module-subprogram * [module-subprogram] ... * * module-subprogram * is function-subprogram * or subroutine-subprogram */ static void parseModule (tokenInfo *const token) { Assert (isKeyword (token, KEYWORD_module)); readToken (token); if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, TAG_MODULE); ancestorPush (token); skipToNextStatement (token); parseSpecificationPart (token); if (isKeyword (token, KEYWORD_contains)) parseInternalSubprogramPart (token); while (! isKeyword (token, KEYWORD_end)) skipToNextStatement (token); readSubToken (token); /* secondary token should be KEYWORD_NONE or KEYWORD_module token */ skipToNextStatement (token); ancestorPop (); } /* execution-part * executable-construct * * executable-contstruct is * execution-part-construct [execution-part-construct] * * execution-part-construct * is executable-construct * or format-stmt * or data-stmt * or entry-stmt */ static boolean parseExecutionPart (tokenInfo *const token) { boolean result = FALSE; boolean done = FALSE; while (! done) { switch (token->keyword) { default: if (isSubprogramPrefix (token)) readToken (token); else skipToNextStatement (token); result = TRUE; break; case KEYWORD_entry: parseEntryStmt (token); result = TRUE; break; case KEYWORD_contains: case KEYWORD_function: case KEYWORD_subroutine: done = TRUE; break; case KEYWORD_end: readSubToken (token); if (isSecondaryKeyword (token, KEYWORD_do) || isSecondaryKeyword (token, KEYWORD_if) || isSecondaryKeyword (token, KEYWORD_select) || isSecondaryKeyword (token, KEYWORD_where)) { skipToNextStatement (token); result = TRUE; } else done = TRUE; break; } } return result; } static void parseSubprogram (tokenInfo *const token, const tagType tag) { Assert (isKeyword (token, KEYWORD_program) || isKeyword (token, KEYWORD_function) || isKeyword (token, KEYWORD_subroutine)); readToken (token); if (isType (token, TOKEN_IDENTIFIER)) makeFortranTag (token, tag); ancestorPush (token); skipToNextStatement (token); parseSpecificationPart (token); parseExecutionPart (token); if (isKeyword (token, KEYWORD_contains)) parseInternalSubprogramPart (token); /* should be at KEYWORD_end token */ readSubToken (token); /* secondary token should be one of KEYWORD_NONE, KEYWORD_program, * KEYWORD_function, KEYWORD_function */ skipToNextStatement (token); ancestorPop (); } /* function-subprogram is * function-stmt (is [prefix] FUNCTION function-name etc.) * [specification-part] * [execution-part] * [internal-subprogram-part] * end-function-stmt (is END [FUNCTION [function-name]]) * * prefix * is type-spec [RECURSIVE] * or [RECURSIVE] type-spec */ static void parseFunctionSubprogram (tokenInfo *const token) { parseSubprogram (token, TAG_FUNCTION); } /* subroutine-subprogram is * subroutine-stmt (is [RECURSIVE] SUBROUTINE subroutine-name etc.) * [specification-part] * [execution-part] * [internal-subprogram-part] * end-subroutine-stmt (is END [SUBROUTINE [function-name]]) */ static void parseSubroutineSubprogram (tokenInfo *const token) { parseSubprogram (token, TAG_SUBROUTINE); } /* main-program is * [program-stmt] (is PROGRAM program-name) * [specification-part] * [execution-part] * [internal-subprogram-part ] * end-program-stmt */ static void parseMainProgram (tokenInfo *const token) { parseSubprogram (token, TAG_PROGRAM); } /* program-unit * is main-program * or external-subprogram (is function-subprogram or subroutine-subprogram) * or module * or block-data */ static void parseProgramUnit (tokenInfo *const token) { readToken (token); do { if (isType (token, TOKEN_STATEMENT_END)) readToken (token); else switch (token->keyword) { case KEYWORD_block: parseBlockData (token); break; case KEYWORD_end: skipToNextStatement (token); break; case KEYWORD_function: parseFunctionSubprogram (token); break; case KEYWORD_module: parseModule (token); break; case KEYWORD_program: parseMainProgram (token); break; case KEYWORD_subroutine: parseSubroutineSubprogram (token); break; default: if (isSubprogramPrefix (token)) readToken (token); else { boolean one = parseSpecificationPart (token); boolean two = parseExecutionPart (token); if (! (one || two)) readToken (token); } break; } } while (TRUE); } static boolean findFortranTags (const unsigned int passCount) { tokenInfo *token; exception_t exception; boolean retry; Assert (passCount < 3); Parent = newToken (); token = newToken (); FreeSourceForm = (boolean) (passCount > 1); Column = 0; exception = (exception_t) setjmp (Exception); if (exception == ExceptionEOF) retry = FALSE; else if (exception == ExceptionFixedFormat && ! FreeSourceForm) { verbose ("%s: not fixed source form; retry as free source form\n", getInputFileName ()); retry = TRUE; } else { parseProgramUnit (token); retry = FALSE; } ancestorClear (); deleteToken (token); deleteToken (Parent); return retry; } static void initializeFortran (const langType language) { Lang_fortran = language; buildFortranKeywordHash (language); } static void initializeF77 (const langType language) { Lang_f77 = language; buildFortranKeywordHash (language); } extern parserDefinition* FortranParser (void) { static const char *const extensions [] = { "f90", "f95", "f03", #ifndef CASE_INSENSITIVE_FILENAMES "F90", "F95", "F03", #endif NULL }; parserDefinition* def = parserNew ("Fortran"); def->kinds = FortranKinds; def->kindCount = KIND_COUNT (FortranKinds); def->extensions = extensions; def->parser2 = findFortranTags; def->initialize = initializeFortran; return def; } extern parserDefinition* F77Parser (void) { static const char *const extensions [] = { "f", "for", "ftn", "f77", #ifndef CASE_INSENSITIVE_FILENAMES "F", "FOR", "FTN", "F77", #endif NULL }; parserDefinition* def = parserNew ("F77"); def->kinds = FortranKinds; def->kindCount = KIND_COUNT (FortranKinds); def->extensions = extensions; def->parser2 = findFortranTags; def->initialize = initializeF77; return def; } /* vi:set tabstop=4 shiftwidth=4: */