1990-10-14 09:07:46 -03:00
|
|
|
/* Parser generator main program */
|
|
|
|
|
1990-12-20 11:06:42 -04:00
|
|
|
/* 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. */
|
1990-10-14 09:07:46 -03:00
|
|
|
|
1990-12-20 11:06:42 -04:00
|
|
|
#include "pgenheaders.h"
|
1990-10-14 09:07:46 -03:00
|
|
|
#include "grammar.h"
|
|
|
|
#include "node.h"
|
|
|
|
#include "parsetok.h"
|
|
|
|
#include "pgen.h"
|
|
|
|
|
|
|
|
int debugging;
|
|
|
|
|
1990-12-20 11:06:42 -04:00
|
|
|
/* Forward */
|
|
|
|
grammar *getgrammar PROTO((char *filename));
|
1990-10-14 09:07:46 -03:00
|
|
|
#ifdef THINK_C
|
1990-12-20 11:06:42 -04:00
|
|
|
int main PROTO((int, char **));
|
|
|
|
char *askfile PROTO((void));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int
|
|
|
|
main(argc, argv)
|
|
|
|
int argc;
|
|
|
|
char **argv;
|
1990-10-14 09:07:46 -03:00
|
|
|
{
|
1990-12-20 11:06:42 -04:00
|
|
|
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");
|
1990-10-14 09:07:46 -03:00
|
|
|
exit(1);
|
|
|
|
}
|
1990-12-20 11:06:42 -04:00
|
|
|
printf("Writing graminit.c ...\n");
|
|
|
|
printgrammar(g, fp);
|
|
|
|
fclose(fp);
|
|
|
|
fp = fopen("graminit.h", "w");
|
|
|
|
if (fp == NULL) {
|
|
|
|
perror("graminit.h");
|
1990-10-14 09:07:46 -03:00
|
|
|
exit(1);
|
|
|
|
}
|
1990-12-20 11:06:42 -04:00
|
|
|
printf("Writing graminit.h ...\n");
|
|
|
|
printnonterminals(g, fp);
|
|
|
|
fclose(fp);
|
|
|
|
exit(0);
|
1990-10-14 09:07:46 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
1990-12-20 11:06:42 -04:00
|
|
|
parsefile(fp, filename, g0, g0->g_start, (char *)NULL, (char *)NULL, &n);
|
1990-10-14 09:07:46 -03:00
|
|
|
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
|
1990-12-20 11:06:42 -04:00
|
|
|
char *
|
|
|
|
askfile()
|
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
static char name[256];
|
|
|
|
printf("Input file name: ");
|
|
|
|
if (fgets(buf, sizeof buf, stdin) == NULL) {
|
|
|
|
printf("EOF\n");
|
1990-10-14 09:07:46 -03:00
|
|
|
exit(1);
|
|
|
|
}
|
1990-12-20 19:11:02 -04:00
|
|
|
/* XXX The (unsigned char *) case is needed by THINK C 3.0 */
|
|
|
|
if (sscanf(/*(unsigned char *)*/buf, " %s ", name) != 1) {
|
1990-12-20 11:06:42 -04:00
|
|
|
printf("No file\n");
|
1990-10-14 09:07:46 -03:00
|
|
|
exit(1);
|
|
|
|
}
|
1990-12-20 11:06:42 -04:00
|
|
|
return name;
|
1990-10-14 09:07:46 -03:00
|
|
|
}
|
1990-12-20 11:06:42 -04:00
|
|
|
#endif
|
1990-10-14 09:07:46 -03:00
|
|
|
|
|
|
|
void
|
|
|
|
fatal(msg)
|
|
|
|
char *msg;
|
|
|
|
{
|
|
|
|
fprintf(stderr, "pgen: FATAL ERROR: %s\n", msg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
1990-12-20 11:06:42 -04:00
|
|
|
/* XXX TO DO:
|
|
|
|
- check for duplicate definitions of names (instead of fatal err)
|
1990-10-14 09:07:46 -03:00
|
|
|
*/
|