Home | History | Annotate | Download | only in Parser
      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