1 2 /* Parser generator main program */ 3 4 /* This expects a filename containing the grammar as argv[1] (UNIX) 5 or asks the console for such a file name (THINK C). 6 It writes its output on two files in the current directory: 7 - "graminit.c" gets the grammar as a bunch of initialized data 8 - "graminit.h" gets the grammar's non-terminals as #defines. 9 Error messages and status info during the generation process are 10 written to stdout, or sometimes to stderr. */ 11 12 /* XXX TO DO: 13 - check for duplicate definitions of names (instead of fatal err) 14 */ 15 16 #include "Python.h" 17 #include "pgenheaders.h" 18 #include "grammar.h" 19 #include "node.h" 20 #include "parsetok.h" 21 #include "pgen.h" 22 23 int Py_DebugFlag; 24 int Py_VerboseFlag; 25 int Py_IgnoreEnvironmentFlag; 26 27 /* Forward */ 28 grammar *getgrammar(char *filename); 29 30 void 31 Py_Exit(int sts) 32 { 33 exit(sts); 34 } 35 36 int 37 main(int argc, char **argv) 38 { 39 grammar *g; 40 FILE *fp; 41 char *filename, *graminit_h, *graminit_c; 42 43 if (argc != 4) { 44 fprintf(stderr, 45 "usage: %s grammar graminit.h graminit.c\n", argv[0]); 46 Py_Exit(2); 47 } 48 filename = argv[1]; 49 graminit_h = argv[2]; 50 graminit_c = argv[3]; 51 g = getgrammar(filename); 52 fp = fopen(graminit_c, "w"); 53 if (fp == NULL) { 54 perror(graminit_c); 55 Py_Exit(1); 56 } 57 if (Py_DebugFlag) 58 printf("Writing %s ...\n", graminit_c); 59 printgrammar(g, fp); 60 fclose(fp); 61 fp = fopen(graminit_h, "w"); 62 if (fp == NULL) { 63 perror(graminit_h); 64 Py_Exit(1); 65 } 66 if (Py_DebugFlag) 67 printf("Writing %s ...\n", graminit_h); 68 printnonterminals(g, fp); 69 fclose(fp); 70 freegrammar(g); 71 Py_Exit(0); 72 return 0; /* Make gcc -Wall happy */ 73 } 74 75 grammar * 76 getgrammar(char *filename) 77 { 78 FILE *fp; 79 node *n; 80 grammar *g0, *g; 81 perrdetail err; 82 83 fp = fopen(filename, "r"); 84 if (fp == NULL) { 85 perror(filename); 86 Py_Exit(1); 87 } 88 g0 = meta_grammar(); 89 n = PyParser_ParseFile(fp, filename, g0, g0->g_start, 90 (char *)NULL, (char *)NULL, &err); 91 fclose(fp); 92 if (n == NULL) { 93 fprintf(stderr, "Parsing error %d, line %d.\n", 94 err.error, err.lineno); 95 if (err.text != NULL) { 96 size_t i; 97 fprintf(stderr, "%s", err.text); 98 i = strlen(err.text); 99 if (i == 0 || err.text[i-1] != '\n') 100 fprintf(stderr, "\n"); 101 for (i = 0; i < err.offset; i++) { 102 if (err.text[i] == '\t') 103 putc('\t', stderr); 104 else 105 putc(' ', stderr); 106 } 107 fprintf(stderr, "^\n"); 108 PyObject_FREE(err.text); 109 } 110 Py_Exit(1); 111 } 112 g = pgen(n); 113 if (g == NULL) { 114 printf("Bad grammar.\n"); 115 Py_Exit(1); 116 } 117 return g; 118 } 119 120 /* Can't happen in pgen */ 121 PyObject* 122 PyErr_Occurred() 123 { 124 return 0; 125 } 126 127 void 128 Py_FatalError(const char *msg) 129 { 130 fprintf(stderr, "pgen: FATAL ERROR: %s\n", msg); 131 Py_Exit(1); 132 } 133 134 /* No-nonsense my_readline() for tokenizer.c */ 135 136 char * 137 PyOS_Readline(FILE *sys_stdin, FILE *sys_stdout, char *prompt) 138 { 139 size_t n = 1000; 140 char *p = (char *)PyMem_MALLOC(n); 141 char *q; 142 if (p == NULL) 143 return NULL; 144 fprintf(stderr, "%s", prompt); 145 q = fgets(p, n, sys_stdin); 146 if (q == NULL) { 147 *p = '\0'; 148 return p; 149 } 150 n = strlen(p); 151 if (n > 0 && p[n-1] != '\n') 152 p[n-1] = '\n'; 153 return (char *)PyMem_REALLOC(p, n+1); 154 } 155 156 /* No-nonsense fgets */ 157 char * 158 Py_UniversalNewlineFgets(char *buf, int n, FILE *stream, PyObject *fobj) 159 { 160 return fgets(buf, n, stream); 161 } 162 163 164 #include <stdarg.h> 165 166 void 167 PySys_WriteStderr(const char *format, ...) 168 { 169 va_list va; 170 171 va_start(va, format); 172 vfprintf(stderr, format, va); 173 va_end(va); 174 } 175