lex.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. #include "e.h"
  2. #include "y.tab.h"
  3. #include <ctype.h>
  4. #define SSIZE 1000
  5. char token[SSIZE];
  6. int sp;
  7. void space(void);
  8. void dodef(tbl *);
  9. void define(int);
  10. void ifdef(void);
  11. void include(void);
  12. void delim(void);
  13. yylex(void)
  14. {
  15. register int c;
  16. tbl *tp;
  17. begin:
  18. while ((c = input()) == ' ' || c == '\n' || c == '\t')
  19. ;
  20. yylval = c;
  21. switch (c) {
  22. case EOF:
  23. ERROR "unexpected end of input inside equation" WARNING;
  24. return(EOF);
  25. case '~':
  26. return(SPACE);
  27. case '^':
  28. return(THIN);
  29. /* case '\t':
  30. return(TAB);
  31. */
  32. case '{':
  33. return('{');
  34. case '}':
  35. return('}');
  36. case '"':
  37. for (sp = 0; (c=input())!='"' && c != '\n'; ) {
  38. if (c == '\\')
  39. if ((c = input()) != '"')
  40. token[sp++] = '\\';
  41. token[sp++] = c;
  42. if (sp >= SSIZE)
  43. ERROR "quoted string %.20s... too long", token FATAL;
  44. }
  45. token[sp] = '\0';
  46. yylval = (int) &token[0];
  47. if (c == '\n')
  48. ERROR "missing \" in %.20s", token WARNING;
  49. return(QTEXT);
  50. }
  51. if (!display && c == righteq)
  52. return(EOF);
  53. unput(c);
  54. getstr(token, SSIZE);
  55. dprintf(".\tlex token = |%s|\n", token);
  56. if ((tp = lookup(deftbl, token)) != NULL) { /* defined term */
  57. c = input();
  58. unput(c);
  59. if (c == '(') /* macro with args */
  60. dodef(tp);
  61. else { /* no args */
  62. unput(' ');
  63. pbstr(tp->cval);
  64. dprintf(".\tfound %s|=%s|\n", token, tp->cval);
  65. }
  66. goto begin;
  67. }
  68. if ((tp = lookup(keytbl, token)) == NULL) /* not a keyword */
  69. return CONTIG;
  70. switch (tp->ival) { /* some kind of keyword */
  71. case DEFINE: case TDEFINE: case NDEFINE:
  72. define(tp->ival);
  73. break;
  74. case IFDEF:
  75. ifdef();
  76. break;
  77. case DELIM:
  78. delim();
  79. break;
  80. case GSIZE:
  81. globsize();
  82. break;
  83. case GFONT:
  84. globfont();
  85. break;
  86. case INCLUDE:
  87. include();
  88. break;
  89. case SPACE:
  90. space();
  91. break;
  92. case DOTEQ:
  93. /* .EQ inside equation -- should warn if at bottom level */
  94. break;
  95. case DOTEN:
  96. if (curfile == infile)
  97. return EOF;
  98. /* else ignore nested .EN */
  99. break;
  100. default:
  101. return tp->ival;
  102. }
  103. goto begin;
  104. }
  105. void getstr(char *s, int n)
  106. {
  107. register int c;
  108. register char *p;
  109. p = s;
  110. while ((c = input()) == ' ' || c == '\n')
  111. ;
  112. if (c == EOF) {
  113. *s = 0;
  114. return;
  115. }
  116. while (c != ' ' && c != '\t' && c != '\n' && c != '{' && c != '}'
  117. && c != '"' && c != '~' && c != '^') {
  118. if (!display && c == righteq)
  119. break;
  120. if (c == '(' && p > s) { /* might be defined(...) */
  121. *p = '\0';
  122. if (lookup(deftbl, s) != NULL)
  123. break;
  124. }
  125. if (c == '\\')
  126. if ((c = input()) != '"')
  127. *p++ = '\\';
  128. *p++ = c;
  129. if (--n <= 0)
  130. ERROR "token %.20s... too long", s FATAL;
  131. c = input();
  132. }
  133. unput(c);
  134. *p = '\0';
  135. yylval = (int) s;
  136. }
  137. cstr(char *s, int quote, int maxs)
  138. {
  139. int del, c, i;
  140. s[0] = 0;
  141. while ((del=input()) == ' ' || del == '\t')
  142. ;
  143. if (quote)
  144. for (i=0; (c=input()) != del && c != EOF;) {
  145. s[i++] = c;
  146. if (i >= maxs)
  147. return(1); /* disaster */
  148. }
  149. else {
  150. if (del == '\n')
  151. return(1);
  152. s[0] = del;
  153. for (i=1; (c=input())!=' ' && c!= '\t' && c!='\n' && c!=EOF;) {
  154. s[i++] = c;
  155. if (i >= maxs)
  156. return(1); /* disaster */
  157. }
  158. }
  159. s[i] = '\0';
  160. if (c == EOF)
  161. ERROR "Unexpected end of input at %.20s", s FATAL;
  162. return(0);
  163. }
  164. void define(int type)
  165. {
  166. char *p1, *p2;
  167. extern int ftune(char *, char *);
  168. getstr(token, SSIZE); /* get name */
  169. if (type != DEFINE) {
  170. cstr(token, 1, SSIZE); /* skip the definition too */
  171. return;
  172. }
  173. p1 = strsave(token);
  174. if (cstr(token, 1, SSIZE))
  175. ERROR "Unterminated definition at %.20s", token FATAL;
  176. if (lookup(ftunetbl, p1) != NULL) { /* double tuning param */
  177. dprintf(".\ttune %s %s\n", p1, token);
  178. ftune(p1, token);
  179. } else {
  180. p2 = strsave(token);
  181. install(deftbl, p1, p2, 0);
  182. dprintf(".\tname %s defined as %s\n", p1, p2);
  183. }
  184. }
  185. void ifdef(void) /* do body if name is defined */
  186. {
  187. char name[100], *p;
  188. getstr(name, sizeof(name)); /* get name */
  189. cstr(token, 1, SSIZE); /* and body */
  190. if (lookup(deftbl, name) != NULL) { /* found it */
  191. p = strsave(token);
  192. pushsrc(Free, p);
  193. pushsrc(String, p);
  194. }
  195. }
  196. char *spaceval = NULL;
  197. void space(void) /* collect line of form "space amt" to replace \x in output */
  198. {
  199. getstr(token, SSIZE);
  200. spaceval = strsave(token);
  201. dprintf(".\tsetting spaceval to %s\n", token);
  202. }
  203. char *strsave(char *s)
  204. {
  205. register char *q;
  206. q = malloc(strlen(s)+1);
  207. if (q == NULL)
  208. ERROR "out of space in strsave on %s", s FATAL;
  209. strcpy(q, s);
  210. return(q);
  211. }
  212. void include(void)
  213. {
  214. char name[100];
  215. FILE *fin;
  216. int c;
  217. extern int errno;
  218. while ((c = input()) == ' ')
  219. ;
  220. unput(c);
  221. cstr(name, c == '"', sizeof(name)); /* gets it quoted or not */
  222. if ((fin = fopen(name, "r")) == NULL)
  223. ERROR "can't open file %s", name FATAL;
  224. errno = 0;
  225. curfile++;
  226. curfile->fin = fin;
  227. curfile->fname = strsave(name);
  228. curfile->lineno = 0;
  229. printf(".lf 1 %s\n", curfile->fname);
  230. pushsrc(File, curfile->fname);
  231. }
  232. void delim(void)
  233. {
  234. yyval = eqnreg = 0;
  235. if (cstr(token, 0, SSIZE))
  236. ERROR "Bizarre delimiters" FATAL;
  237. lefteq = token[0];
  238. righteq = token[1];
  239. if (!isprint(lefteq) || !isprint(righteq))
  240. ERROR "Bizarre delimiters" FATAL;
  241. if (lefteq == 'o' && righteq == 'f')
  242. lefteq = righteq = '\0';
  243. }