blob: 63be88f8ed267c2583996a1ac6dfce3ba861d13c [file] [log] [blame]
Guido van Rossumf70e43a1991-02-19 12:39:46 +00001
Guido van Rossum85a5fbb1990-10-14 12:07:46 +00002/* Parser generator main program */
3
Guido van Rossum3f5da241990-12-20 15:06:42 +00004/* 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. */
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000011
Guido van Rossum1d5735e1994-08-30 08:27:36 +000012/* XXX TO DO:
13 - check for duplicate definitions of names (instead of fatal err)
14*/
15
Jack Jansen7b8c7542002-04-14 20:12:41 +000016#include "Python.h"
Guido van Rossum3f5da241990-12-20 15:06:42 +000017#include "pgenheaders.h"
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000018#include "grammar.h"
19#include "node.h"
20#include "parsetok.h"
21#include "pgen.h"
22
Guido van Rossum86bea461997-04-29 21:03:06 +000023int Py_DebugFlag;
Guido van Rossumc0952001998-04-10 19:34:15 +000024int Py_VerboseFlag;
Guido van Rossum79e39bd2002-05-31 14:32:07 +000025int Py_IgnoreEnvironmentFlag;
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000026
Guido van Rossum3f5da241990-12-20 15:06:42 +000027/* Forward */
Tim Petersdbd9ba62000-07-09 03:09:57 +000028grammar *getgrammar(char *filename);
Guido van Rossum1d5735e1994-08-30 08:27:36 +000029#ifdef THINK_C
Tim Petersdbd9ba62000-07-09 03:09:57 +000030int main(int, char **);
31char *askfile(void);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000032#endif
33
Guido van Rossumf1dc5661993-07-05 10:31:29 +000034void
Thomas Wouters23c9e002000-07-22 19:20:54 +000035Py_Exit(int sts)
Guido van Rossumf1dc5661993-07-05 10:31:29 +000036{
37 exit(sts);
38}
39
Guido van Rossum3f5da241990-12-20 15:06:42 +000040int
Thomas Wouters23c9e002000-07-22 19:20:54 +000041main(int argc, char **argv)
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000042{
43 grammar *g;
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000044 FILE *fp;
Neil Schemenauer84521c02001-02-16 03:57:53 +000045 char *filename, *graminit_h, *graminit_c;
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000046
Guido van Rossum1d5735e1994-08-30 08:27:36 +000047#ifdef THINK_C
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000048 filename = askfile();
Neil Schemenauer84521c02001-02-16 03:57:53 +000049 graminit_h = askfile();
50 graminit_c = askfile();
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000051#else
Neil Schemenauer84521c02001-02-16 03:57:53 +000052 if (argc != 4) {
53 fprintf(stderr,
54 "usage: %s grammar graminit.h graminit.c\n", argv[0]);
Guido van Rossum86bea461997-04-29 21:03:06 +000055 Py_Exit(2);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000056 }
57 filename = argv[1];
Neil Schemenauer84521c02001-02-16 03:57:53 +000058 graminit_h = argv[2];
59 graminit_c = argv[3];
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000060#endif
61 g = getgrammar(filename);
Neil Schemenauer84521c02001-02-16 03:57:53 +000062 fp = fopen(graminit_c, "w");
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000063 if (fp == NULL) {
Neil Schemenauer84521c02001-02-16 03:57:53 +000064 perror(graminit_c);
Guido van Rossum86bea461997-04-29 21:03:06 +000065 Py_Exit(1);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000066 }
Guido van Rossum25dfe2c2001-09-11 16:43:16 +000067 if (Py_DebugFlag)
68 printf("Writing %s ...\n", graminit_c);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000069 printgrammar(g, fp);
70 fclose(fp);
Neil Schemenauer84521c02001-02-16 03:57:53 +000071 fp = fopen(graminit_h, "w");
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000072 if (fp == NULL) {
Neil Schemenauer84521c02001-02-16 03:57:53 +000073 perror(graminit_h);
Guido van Rossum86bea461997-04-29 21:03:06 +000074 Py_Exit(1);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000075 }
Guido van Rossum25dfe2c2001-09-11 16:43:16 +000076 if (Py_DebugFlag)
77 printf("Writing %s ...\n", graminit_h);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000078 printnonterminals(g, fp);
79 fclose(fp);
Guido van Rossum86bea461997-04-29 21:03:06 +000080 Py_Exit(0);
Guido van Rossumfd8a3931996-12-02 18:27:33 +000081 return 0; /* Make gcc -Wall happy */
Guido van Rossum85a5fbb1990-10-14 12:07:46 +000082}
83
Guido van Rossum3f5da241990-12-20 15:06:42 +000084grammar *
Thomas Wouters23c9e002000-07-22 19:20:54 +000085getgrammar(char *filename)
Guido van Rossum3f5da241990-12-20 15:06:42 +000086{
87 FILE *fp;
88 node *n;
89 grammar *g0, *g;
Guido van Rossum1d5735e1994-08-30 08:27:36 +000090 perrdetail err;
Guido van Rossum3f5da241990-12-20 15:06:42 +000091
92 fp = fopen(filename, "r");
93 if (fp == NULL) {
94 perror(filename);
Guido van Rossum86bea461997-04-29 21:03:06 +000095 Py_Exit(1);
Guido van Rossum3f5da241990-12-20 15:06:42 +000096 }
97 g0 = meta_grammar();
Guido van Rossum86bea461997-04-29 21:03:06 +000098 n = PyParser_ParseFile(fp, filename, g0, g0->g_start,
Guido van Rossum1d5735e1994-08-30 08:27:36 +000099 (char *)NULL, (char *)NULL, &err);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000100 fclose(fp);
101 if (n == NULL) {
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000102 fprintf(stderr, "Parsing error %d, line %d.\n",
103 err.error, err.lineno);
104 if (err.text != NULL) {
Guido van Rossum6da34342000-06-28 22:00:02 +0000105 size_t i;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000106 fprintf(stderr, "%s", err.text);
107 i = strlen(err.text);
108 if (i == 0 || err.text[i-1] != '\n')
109 fprintf(stderr, "\n");
110 for (i = 0; i < err.offset; i++) {
111 if (err.text[i] == '\t')
112 putc('\t', stderr);
113 else
114 putc(' ', stderr);
115 }
116 fprintf(stderr, "^\n");
Guido van Rossumb18618d2000-05-03 23:44:39 +0000117 PyMem_DEL(err.text);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000118 }
Guido van Rossum86bea461997-04-29 21:03:06 +0000119 Py_Exit(1);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000120 }
121 g = pgen(n);
122 if (g == NULL) {
123 printf("Bad grammar.\n");
Guido van Rossum86bea461997-04-29 21:03:06 +0000124 Py_Exit(1);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000125 }
126 return g;
127}
128
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000129#ifdef THINK_C
Guido van Rossum3f5da241990-12-20 15:06:42 +0000130char *
Thomas Wouters23c9e002000-07-22 19:20:54 +0000131askfile(void)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000132{
133 char buf[256];
134 static char name[256];
135 printf("Input file name: ");
136 if (fgets(buf, sizeof buf, stdin) == NULL) {
137 printf("EOF\n");
Guido van Rossum86bea461997-04-29 21:03:06 +0000138 Py_Exit(1);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000139 }
Guido van Rossum706eea81990-12-20 23:11:02 +0000140 /* XXX The (unsigned char *) case is needed by THINK C 3.0 */
141 if (sscanf(/*(unsigned char *)*/buf, " %s ", name) != 1) {
Guido van Rossum3f5da241990-12-20 15:06:42 +0000142 printf("No file\n");
Guido van Rossum86bea461997-04-29 21:03:06 +0000143 Py_Exit(1);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000144 }
145 return name;
146}
147#endif
148
Guido van Rossum85a5fbb1990-10-14 12:07:46 +0000149void
Tim Peters7c321a82002-07-09 02:57:01 +0000150Py_FatalError(const char *msg)
Guido van Rossum85a5fbb1990-10-14 12:07:46 +0000151{
152 fprintf(stderr, "pgen: FATAL ERROR: %s\n", msg);
Guido van Rossum86bea461997-04-29 21:03:06 +0000153 Py_Exit(1);
Guido van Rossum85a5fbb1990-10-14 12:07:46 +0000154}
155
Guido van Rossum7ebb23c1992-03-27 17:31:35 +0000156#ifdef macintosh
157/* ARGSUSED */
158int
Thomas Wouters23c9e002000-07-22 19:20:54 +0000159guesstabsize(char *path)
Guido van Rossum7ebb23c1992-03-27 17:31:35 +0000160{
161 return 4;
162}
163#endif
164
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000165/* No-nonsense my_readline() for tokenizer.c */
166
167char *
Thomas Wouters23c9e002000-07-22 19:20:54 +0000168PyOS_Readline(char *prompt)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000169{
Guido van Rossum6da34342000-06-28 22:00:02 +0000170 size_t n = 1000;
Guido van Rossumb18618d2000-05-03 23:44:39 +0000171 char *p = PyMem_MALLOC(n);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000172 char *q;
173 if (p == NULL)
174 return NULL;
175 fprintf(stderr, "%s", prompt);
176 q = fgets(p, n, stdin);
177 if (q == NULL) {
178 *p = '\0';
179 return p;
180 }
181 n = strlen(p);
182 if (n > 0 && p[n-1] != '\n')
183 p[n-1] = '\n';
Guido van Rossumb18618d2000-05-03 23:44:39 +0000184 return PyMem_REALLOC(p, n+1);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000185}
Guido van Rossum47fbc401998-08-25 18:12:36 +0000186
Jack Jansen7b8c7542002-04-14 20:12:41 +0000187#ifdef WITH_UNIVERSAL_NEWLINES
188/* No-nonsense fgets */
189char *
190Py_UniversalNewlineFgets(char *buf, int n, FILE *stream, PyObject *fobj)
191{
192 return fgets(buf, n, stream);
193}
194#endif
195
196
Guido van Rossum47fbc401998-08-25 18:12:36 +0000197#include <stdarg.h>
Guido van Rossum47fbc401998-08-25 18:12:36 +0000198
199void
Guido van Rossum47fbc401998-08-25 18:12:36 +0000200PySys_WriteStderr(const char *format, ...)
Guido van Rossum47fbc401998-08-25 18:12:36 +0000201{
202 va_list va;
203
Guido van Rossum47fbc401998-08-25 18:12:36 +0000204 va_start(va, format);
Guido van Rossum47fbc401998-08-25 18:12:36 +0000205 vfprintf(stderr, format, va);
206 va_end(va);
207}