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.

189 lines
4.0 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(const char *filename);
  24. void Py_Exit(int) _Py_NO_RETURN;
  25. void
  26. Py_Exit(int sts)
  27. {
  28. exit(sts);
  29. }
  30. #ifdef WITH_THREAD
  31. /* Needed by obmalloc.c */
  32. int PyGILState_Check(void)
  33. { return 1; }
  34. #endif
  35. void _PyMem_DumpTraceback(int fd, const void *ptr)
  36. {}
  37. int
  38. main(int argc, char **argv)
  39. {
  40. grammar *g;
  41. FILE *fp;
  42. char *filename, *graminit_h, *graminit_c;
  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. grammar *
  75. getgrammar(const char *filename)
  76. {
  77. FILE *fp;
  78. node *n;
  79. grammar *g0, *g;
  80. perrdetail err;
  81. fp = fopen(filename, "r");
  82. if (fp == NULL) {
  83. perror(filename);
  84. Py_Exit(1);
  85. }
  86. g0 = meta_grammar();
  87. n = PyParser_ParseFile(fp, filename, g0, g0->g_start,
  88. (char *)NULL, (char *)NULL, &err);
  89. fclose(fp);
  90. if (n == NULL) {
  91. fprintf(stderr, "Parsing error %d, line %d.\n",
  92. err.error, err.lineno);
  93. if (err.text != NULL) {
  94. size_t len;
  95. int i;
  96. fprintf(stderr, "%s", err.text);
  97. len = strlen(err.text);
  98. if (len == 0 || err.text[len-1] != '\n')
  99. fprintf(stderr, "\n");
  100. for (i = 0; i < err.offset; i++) {
  101. if (err.text[i] == '\t')
  102. putc('\t', stderr);
  103. else
  104. putc(' ', stderr);
  105. }
  106. fprintf(stderr, "^\n");
  107. PyObject_FREE(err.text);
  108. }
  109. Py_Exit(1);
  110. }
  111. g = pgen(n);
  112. PyNode_Free(n);
  113. if (g == NULL) {
  114. printf("Bad grammar.\n");
  115. Py_Exit(1);
  116. }
  117. return g;
  118. }
  119. /* Can't happen in pgen */
  120. PyObject*
  121. PyErr_Occurred()
  122. {
  123. return 0;
  124. }
  125. void
  126. Py_FatalError(const char *msg)
  127. {
  128. fprintf(stderr, "pgen: FATAL ERROR: %s\n", msg);
  129. Py_Exit(1);
  130. }
  131. /* No-nonsense my_readline() for tokenizer.c */
  132. char *
  133. PyOS_Readline(FILE *sys_stdin, FILE *sys_stdout, const char *prompt)
  134. {
  135. size_t n = 1000;
  136. char *p = (char *)PyMem_MALLOC(n);
  137. char *q;
  138. if (p == NULL)
  139. return NULL;
  140. fprintf(stderr, "%s", prompt);
  141. q = fgets(p, n, sys_stdin);
  142. if (q == NULL) {
  143. *p = '\0';
  144. return p;
  145. }
  146. n = strlen(p);
  147. if (n > 0 && p[n-1] != '\n')
  148. p[n-1] = '\n';
  149. return (char *)PyMem_REALLOC(p, n+1);
  150. }
  151. /* No-nonsense fgets */
  152. char *
  153. Py_UniversalNewlineFgets(char *buf, int n, FILE *stream, PyObject *fobj)
  154. {
  155. return fgets(buf, n, stream);
  156. }
  157. #include <stdarg.h>
  158. void
  159. PySys_WriteStderr(const char *format, ...)
  160. {
  161. va_list va;
  162. va_start(va, format);
  163. vfprintf(stderr, format, va);
  164. va_end(va);
  165. }