You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

178 lines
3.8 KiB

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