| /* Parser generator main program */ |
| |
| /* This expects a filename containing the grammar as argv[1] (UNIX) |
| or asks the console for such a file name (THINK C). |
| It writes its output on two files in the current directory: |
| - "graminit.c" gets the grammar as a bunch of initialized data |
| - "graminit.h" gets the grammar's non-terminals as #defines. |
| Error messages and status info during the generation process are |
| written to stdout, or sometimes to stderr. */ |
| |
| #include "pgenheaders.h" |
| #include "grammar.h" |
| #include "node.h" |
| #include "parsetok.h" |
| #include "pgen.h" |
| |
| int debugging; |
| |
| /* Forward */ |
| grammar *getgrammar PROTO((char *filename)); |
| #ifdef THINK_C |
| int main PROTO((int, char **)); |
| char *askfile PROTO((void)); |
| #endif |
| |
| int |
| main(argc, argv) |
| int argc; |
| char **argv; |
| { |
| grammar *g; |
| node *n; |
| FILE *fp; |
| char *filename; |
| |
| #ifdef THINK_C |
| filename = askfile(); |
| #else |
| if (argc != 2) { |
| fprintf(stderr, "usage: %s grammar\n", argv[0]); |
| exit(2); |
| } |
| filename = argv[1]; |
| #endif |
| g = getgrammar(filename); |
| fp = fopen("graminit.c", "w"); |
| if (fp == NULL) { |
| perror("graminit.c"); |
| exit(1); |
| } |
| printf("Writing graminit.c ...\n"); |
| printgrammar(g, fp); |
| fclose(fp); |
| fp = fopen("graminit.h", "w"); |
| if (fp == NULL) { |
| perror("graminit.h"); |
| exit(1); |
| } |
| printf("Writing graminit.h ...\n"); |
| printnonterminals(g, fp); |
| fclose(fp); |
| exit(0); |
| } |
| |
| grammar * |
| getgrammar(filename) |
| char *filename; |
| { |
| FILE *fp; |
| node *n; |
| grammar *g0, *g; |
| |
| fp = fopen(filename, "r"); |
| if (fp == NULL) { |
| perror(filename); |
| exit(1); |
| } |
| g0 = meta_grammar(); |
| n = NULL; |
| parsefile(fp, filename, g0, g0->g_start, (char *)NULL, (char *)NULL, &n); |
| fclose(fp); |
| if (n == NULL) { |
| fprintf(stderr, "Parsing error.\n"); |
| exit(1); |
| } |
| g = pgen(n); |
| if (g == NULL) { |
| printf("Bad grammar.\n"); |
| exit(1); |
| } |
| return g; |
| } |
| |
| #ifdef THINK_C |
| char * |
| askfile() |
| { |
| char buf[256]; |
| static char name[256]; |
| printf("Input file name: "); |
| if (fgets(buf, sizeof buf, stdin) == NULL) { |
| printf("EOF\n"); |
| exit(1); |
| } |
| /* XXX The (unsigned char *) case is needed by THINK C */ |
| if (sscanf((unsigned char *)buf, " %s ", name) != 1) { |
| printf("No file\n"); |
| exit(1); |
| } |
| return name; |
| } |
| #endif |
| |
| void |
| fatal(msg) |
| char *msg; |
| { |
| fprintf(stderr, "pgen: FATAL ERROR: %s\n", msg); |
| exit(1); |
| } |
| |
| /* XXX TO DO: |
| - check for duplicate definitions of names (instead of fatal err) |
| */ |