From: Dale Weiler Date: Sun, 15 Apr 2012 11:59:22 +0000 (-0400) Subject: Removed primitive AST tree generator ... I'm planning a rewrite as we speak. X-Git-Tag: 0.1-rc1~665 X-Git-Url: https://git.rm.cloudns.org/?a=commitdiff_plain;h=29a3c02c354a4f6056c0e311a42293f4dc639fd8;p=xonotic%2Fgmqcc.git Removed primitive AST tree generator ... I'm planning a rewrite as we speak. --- diff --git a/gmqcc.h b/gmqcc.h index 409762b..d201c78 100644 --- a/gmqcc.h +++ b/gmqcc.h @@ -95,11 +95,7 @@ int error(int, const char *, ...); //=================================================================== //========================== parse.c ================================ //=================================================================== -int parse_tree(struct lex_file *); -struct parsenode { - struct parsenode *next; - int type; /* some token */ -}; +int parse_gen(struct lex_file *); //=================================================================== //========================== typedef.c ============================== @@ -261,6 +257,5 @@ enum { INSTR_BITOR }; - void code_write(); #endif diff --git a/main.c b/main.c index 754a165..0313df5 100644 --- a/main.c +++ b/main.c @@ -41,9 +41,9 @@ int main(int argc, char **argv) { fclose(fp); return 0; } - parse_tree(lex); + parse_gen(lex); mem_d(lex->name); - lex_close (lex); + lex_close(lex); } code_write(); diff --git a/parse.c b/parse.c index 0f2efe8..cc228ed 100644 --- a/parse.c +++ b/parse.c @@ -26,7 +26,7 @@ #include #include "gmqcc.h" -/* compile-time constant for constants */ +/* compile-time constant for type constants */ typedef struct { char *name; int type; @@ -35,170 +35,11 @@ typedef struct { } constant; VECTOR_MAKE(constant, compile_constants); -/* - * These are not lexical tokens: These are parse tree types. Most people - * perform tokenizing on language punctuation which is wrong. That stuff - * is technically already tokenized, it just needs to be parsed into a tree - */ -#define PARSE_TYPE_DO 0 -#define PARSE_TYPE_ELSE 1 -#define PARSE_TYPE_IF 2 -#define PARSE_TYPE_WHILE 3 -#define PARSE_TYPE_BREAK 4 -#define PARSE_TYPE_CONTINUE 5 -#define PARSE_TYPE_RETURN 6 -#define PARSE_TYPE_GOTO 7 -#define PARSE_TYPE_FOR 8 -#define PARSE_TYPE_VOID 9 -#define PARSE_TYPE_STRING 10 -#define PARSE_TYPE_FLOAT 11 -#define PARSE_TYPE_VECTOR 12 -#define PARSE_TYPE_ENTITY 13 -#define PARSE_TYPE_LAND 14 -#define PARSE_TYPE_LOR 15 -#define PARSE_TYPE_LTEQ 16 -#define PARSE_TYPE_GTEQ 17 -#define PARSE_TYPE_EQEQ 18 -#define PARSE_TYPE_LNEQ 19 -#define PARSE_TYPE_COMMA 20 -#define PARSE_TYPE_LNOT 21 -#define PARSE_TYPE_STAR 22 -#define PARSE_TYPE_DIVIDE 23 -#define PARSE_TYPE_LPARTH 24 -#define PARSE_TYPE_RPARTH 25 -#define PARSE_TYPE_MINUS 26 -#define PARSE_TYPE_ADD 27 -#define PARSE_TYPE_EQUAL 28 -#define PARSE_TYPE_LBS 29 -#define PARSE_TYPE_RBS 30 -#define PARSE_TYPE_ELIP 31 -#define PARSE_TYPE_DOT 32 -#define PARSE_TYPE_LT 33 -#define PARSE_TYPE_GT 34 -#define PARSE_TYPE_BAND 35 -#define PARSE_TYPE_BOR 36 -#define PARSE_TYPE_DONE 37 -#define PARSE_TYPE_IDENT 38 - -/* - * Adds a parse type to the parse tree, this is where all the hard - * work actually begins. - */ -#define PARSE_TREE_ADD(X) \ - do { \ - parsetree->next = mem_a(sizeof(struct parsenode)); \ - parsetree->next->next = NULL; \ - parsetree->next->type = (X); \ - parsetree = parsetree->next; \ - } while (0) -#define STORE(X) { \ - printf(X); \ - break; \ -} - -void parse_debug(struct parsenode *tree) { - long fill = 0; - while (tree) { - switch (tree->type) { - case PARSE_TYPE_ADD: STORE("OPERATOR: ADD \n"); - case PARSE_TYPE_BAND: STORE("OPERATOR: BITAND \n"); - case PARSE_TYPE_BOR: STORE("OPERATOR: BITOR \n"); - case PARSE_TYPE_COMMA: STORE("OPERATOR: SEPERATOR\n"); - case PARSE_TYPE_DOT: STORE("OPERATOR: DOT\n"); - case PARSE_TYPE_DIVIDE: STORE("OPERATOR: DIVIDE\n"); - case PARSE_TYPE_EQUAL: STORE("OPERATOR: ASSIGNMENT\n"); - - case PARSE_TYPE_BREAK: STORE("STATEMENT: BREAK \n"); - case PARSE_TYPE_CONTINUE: STORE("STATEMENT: CONTINUE\n"); - case PARSE_TYPE_GOTO: STORE("STATEMENT: GOTO\n"); - case PARSE_TYPE_RETURN: STORE("STATEMENT: RETURN\n"); - case PARSE_TYPE_DONE: STORE("STATEMENT: DONE\n"); - - case PARSE_TYPE_VOID: STORE("DECLTYPE: VOID\n"); - case PARSE_TYPE_STRING: STORE("DECLTYPE: STRING\n"); - case PARSE_TYPE_ELIP: STORE("DECLTYPE: VALIST\n"); - case PARSE_TYPE_ENTITY: STORE("DECLTYPE: ENTITY\n"); - case PARSE_TYPE_FLOAT: STORE("DECLTYPE: FLOAT\n"); - case PARSE_TYPE_VECTOR: STORE("DECLTYPE: VECTOR\n"); - - case PARSE_TYPE_GT: STORE("TEST: GREATER THAN\n"); - case PARSE_TYPE_LT: STORE("TEST: LESS THAN\n"); - case PARSE_TYPE_GTEQ: STORE("TEST: GREATER THAN OR EQUAL\n"); - case PARSE_TYPE_LTEQ: STORE("TEST: LESS THAN OR EQUAL\n"); - case PARSE_TYPE_LNEQ: STORE("TEST: NOT EQUAL\n"); - case PARSE_TYPE_EQEQ: STORE("TEST: EQUAL-EQUAL\n"); - - case PARSE_TYPE_LBS: STORE("BLOCK: BEG\n"); - case PARSE_TYPE_RBS: STORE("BLOCK: END\n"); - case PARSE_TYPE_ELSE: STORE("BLOCK: ELSE\n"); - case PARSE_TYPE_IF: STORE("BLOCK: IF\n"); - - case PARSE_TYPE_LAND: STORE("LOGICAL: AND\n"); - case PARSE_TYPE_LNOT: STORE("LOGICAL: NOT\n"); - case PARSE_TYPE_LOR: STORE("LOGICAL: OR\n"); - - case PARSE_TYPE_LPARTH: STORE("PARTH: BEG\n"); - case PARSE_TYPE_RPARTH: STORE("PARTH: END\n"); - - case PARSE_TYPE_WHILE: STORE("LOOP: WHILE\n"); - case PARSE_TYPE_FOR: STORE("LOOP: FOR\n"); - case PARSE_TYPE_DO: STORE("LOOP: DO\n"); - } - tree = tree->next; - } -} - -/* - * Performs a parse operation: This is a macro to prevent bugs, if the - * calls to lex_token are'nt exactly enough to feed to the end of the - * actual lexees for the current thing that is being parsed, the state - * of the next iteration in the creation of the parse tree will be wrong - * and everything will fail. - */ -#define PARSE_PERFORM(X,C) { \ - token = lex_token(file); \ - { C } \ - while (token != '\n') { \ - token = lex_token(file); \ - } \ - PARSE_TREE_ADD(X); \ - break; \ -} - -void parse_clear(struct parsenode *tree) { - if (!tree) return; - struct parsenode *temp = NULL; - while (tree != NULL) { - temp = tree; - tree = tree->next; - mem_d (temp); - } - - /* free any potential typedefs */ - typedef_clear(); -} - /* * Generates a parse tree out of the lexees generated by the lexer. This * is where the tree is built. This is where valid check is performed. */ -int parse_tree(struct lex_file *file) { - struct parsenode *parsetree = NULL; - struct parsenode *parseroot = NULL; - - /* - * Allocate memory for our parse tree: - * the parse tree is just a singly linked list which will contain - * all the data for code generation. - */ - if (!parseroot) { - parseroot = mem_a(sizeof(struct parsenode)); - if (!parseroot) - return error(ERROR_INTERNAL, "Ran out of memory", " "); - parsetree = parseroot; - parsetree->type = -1; /* not a valid type -- root element */ - } - +int parse_gen(struct lex_file *file) { int token = 0; while ((token = lex_token(file)) != ERROR_LEX && \ token != ERROR_COMPILER && \ @@ -230,11 +71,11 @@ int parse_tree(struct lex_file *file) { break; } - case TOKEN_VOID: PARSE_TREE_ADD(PARSE_TYPE_VOID); goto fall; - case TOKEN_STRING: PARSE_TREE_ADD(PARSE_TYPE_STRING); goto fall; - case TOKEN_VECTOR: PARSE_TREE_ADD(PARSE_TYPE_VECTOR); goto fall; - case TOKEN_ENTITY: PARSE_TREE_ADD(PARSE_TYPE_ENTITY); goto fall; - case TOKEN_FLOAT: PARSE_TREE_ADD(PARSE_TYPE_FLOAT); goto fall; + case TOKEN_VOID: goto fall; + case TOKEN_STRING: goto fall; + case TOKEN_VECTOR: goto fall; + case TOKEN_ENTITY: goto fall; + case TOKEN_FLOAT: goto fall; { fall:; char *name = NULL; @@ -423,9 +264,9 @@ int parse_tree(struct lex_file *file) { error(ERROR_INTERNAL, "Include subsystem failure\n"); exit (-1); } - parse_tree(next); - mem_d (copy); - lex_close (next); + parse_gen(next); + mem_d (copy); + lex_close(next); } /* skip all tokens to end of directive */ while (token != '\n') @@ -434,12 +275,9 @@ int parse_tree(struct lex_file *file) { case LEX_IDENT: token = lex_token(file); - PARSE_TREE_ADD(PARSE_TYPE_IDENT); break; } } - parse_debug(parseroot); lex_reset(file); - parse_clear(parseroot); return 1; } diff --git a/test/vector.qc b/test/vector.qc index 6235525..6fe0418 100644 --- a/test/vector.qc +++ b/test/vector.qc @@ -1,9 +1,9 @@ -vector vec1 = { -0, +0, 0 }; -vector vec2 = { .0, .0, .0 }; -vector vec3 = { -.0, +.0, +0.1 }; -vector vec4 = { 1.1, 2.2, 3.3 }; -vector vec5 = { 2., 3., 4. }; -vector vec6 = { -2., +3., -4. }; +vector vec1 = {-0 +0 0 }; +vector vec2 = {.0 .0 .0 }; +vector vec3 = {-.0 +.0 +0.1 }; +vector vec4 = {1.1 2.2 3.3 }; +vector vec5 = {2. 3. 4. }; +vector vec6 = {-2. +3. -4. }; /* * These are just comments: Ideally there is still some broken things * for the vector yet. which sort of sucks. diff --git a/util.c b/util.c index a155907..76bd448 100644 --- a/util.c +++ b/util.c @@ -39,7 +39,7 @@ void *util_memory_a(unsigned int byte, unsigned int line, const char *file) { data->byte = byte; data->file = file; - printf("[MEM] allocation: %08u (bytes) at %s:%u\n", byte, file, line); + util_debug("MEM", "allocation: %08u (bytes) at %s:%u\n", byte, file, line); return (void*)((uintptr_t)data+sizeof(struct memblock_t)); } @@ -48,7 +48,7 @@ void util_memory_d(void *ptrn, unsigned int line, const char *file) { void *data = (void*)((uintptr_t)ptrn-sizeof(struct memblock_t)); struct memblock_t *info = (struct memblock_t*)data; - printf("[MEM] released: %08u (bytes) at %s:%u\n", info->byte, file, line); + util_debug("MEM", "released: %08u (bytes) at %s:%u\n", info->byte, file, line); free(data); } @@ -81,9 +81,13 @@ char *util_strdup(const char *s) { return ptr; } -void util_debug(const char *ms, ...) { +void util_debug(const char *area, const char *ms, ...) { va_list va; va_start(va, ms); - vprintf (ms, va); + fprintf (stdout, "DEBUG: "); + fputc ('[', stdout); + fprintf (stdout, area); + fputs ("] ", stdout); + vfprintf(stdout, ms, va); va_end (va); }