// clang-format off /* * $Id:$ * * Copyright (c) 2000-2006, Darren Hiebert, Elias Pschernig * * 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 BlitzBasic * (BlitzMax), PureBasic and FreeBasic language files. For now, this is kept * quite simple - but feel free to ask for more things added any time - * patches are of course most welcome. */ /* * INCLUDE FILES */ #include "third_party/ctags/general.h" /* must always come first */ #include "libc/mem/alg.h" #include "libc/str/str.h" #include "third_party/ctags/options.h" #include "third_party/ctags/parse.h" #include "third_party/ctags/read.h" #include "third_party/ctags/routines.h" #include "third_party/ctags/vstring.h" /* * DATA DEFINITIONS */ typedef enum { K_CONST, K_FUNCTION, K_LABEL, K_TYPE, K_VARIABLE, K_ENUM } BasicKind; typedef struct { char const *token; BasicKind kind; int skip; } KeyWord; static kindOption BasicKinds[] = { {TRUE, 'c', "constant", "constants"}, {TRUE, 'f', "function", "functions"}, {TRUE, 'l', "label", "labels"}, {TRUE, 't', "type", "types"}, {TRUE, 'v', "variable", "variables"}, {TRUE, 'g', "enum", "enumerations"} }; static KeyWord blitzbasic_keywords[] = { {"const", K_CONST, 0}, {"global", K_VARIABLE, 0}, {"dim", K_VARIABLE, 0}, {"function", K_FUNCTION, 0}, {"type", K_TYPE, 0}, {NULL, 0, 0} }; static KeyWord purebasic_keywords[] = { {"newlist", K_VARIABLE, 0}, {"global", K_VARIABLE, 0}, {"dim", K_VARIABLE, 0}, {"procedure", K_FUNCTION, 0}, {"interface", K_TYPE, 0}, {"structure", K_TYPE, 0}, {NULL, 0, 0} }; static KeyWord freebasic_keywords[] = { {"const", K_CONST, 0}, {"dim as", K_VARIABLE, 1}, {"dim", K_VARIABLE, 0}, {"common", K_VARIABLE, 0}, {"function", K_FUNCTION, 0}, {"sub", K_FUNCTION, 0}, {"private sub", K_FUNCTION, 0}, {"public sub", K_FUNCTION, 0}, {"private function", K_FUNCTION, 0}, {"public function", K_FUNCTION, 0}, {"type", K_TYPE, 0}, {"enum", K_ENUM, 0}, {NULL, 0, 0} }; /* * FUNCTION DEFINITIONS */ /* Match the name of a tag (function, variable, type, ...) starting at pos. */ static char const *extract_name (char const *pos, vString * name) { while (isspace (*pos)) pos++; vStringClear (name); for (; *pos && !isspace (*pos) && *pos != '(' && *pos != ','; pos++) vStringPut (name, *pos); vStringTerminate (name); return pos; } /* Match a keyword starting at p (case insensitive). */ static int match_keyword (const char *p, KeyWord const *kw) { vString *name; size_t i; int j; for (i = 0; i < strlen (kw->token); i++) { if (tolower (p[i]) != kw->token[i]) return 0; } name = vStringNew (); p += i; for (j = 0; j < 1 + kw->skip; j++) { p = extract_name (p, name); } makeSimpleTag (name, BasicKinds, kw->kind); vStringDelete (name); return 1; } /* Match a "label:" style label. */ static void match_colon_label (char const *p) { char const *end = p + strlen (p) - 1; while (isspace (*end)) end--; if (*end == ':') { vString *name = vStringNew (); vStringNCatS (name, p, end - p); makeSimpleTag (name, BasicKinds, K_LABEL); vStringDelete (name); } } /* Match a ".label" style label. */ static void match_dot_label (char const *p) { if (*p == '.') { vString *name = vStringNew (); extract_name (p + 1, name); makeSimpleTag (name, BasicKinds, K_LABEL); vStringDelete (name); } } static void findBasicTags (void) { const char *line; const char *extension = fileExtension (vStringValue (File.name)); KeyWord *keywords; if (strcmp (extension, "bb") == 0) keywords = blitzbasic_keywords; else if (strcmp (extension, "pb") == 0) keywords = purebasic_keywords; else keywords = freebasic_keywords; while ((line = (const char *) fileReadLine ()) != NULL) { const char *p = line; KeyWord const *kw; while (isspace (*p)) p++; /* Empty line? */ if (!*p) continue; /* In Basic, keywords always are at the start of the line. */ for (kw = keywords; kw->token; kw++) if (match_keyword (p, kw)) break; /* Is it a label? */ if (strcmp (extension, "bb") == 0) match_dot_label (p); else match_colon_label (p); } } parserDefinition *BasicParser (void) { static char const *extensions[] = { "bas", "bi", "bb", "pb", NULL }; parserDefinition *def = parserNew ("Basic"); def->kinds = BasicKinds; def->kindCount = KIND_COUNT (BasicKinds); def->extensions = extensions; def->parser = findBasicTags; return def; } /* vi:set tabstop=4 shiftwidth=4: */