sub1.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591
  1. # include "ldefs.h"
  2. uchar *
  3. getl(uchar *p) /* return next line of input, throw away trailing '\n' */
  4. /* returns 0 if eof is had immediately */
  5. {
  6. int c;
  7. uchar *s, *t;
  8. t = s = p;
  9. while(((c = gch()) != 0) && c != '\n')
  10. *t++ = c;
  11. *t = 0;
  12. if(c == 0 && s == t) return((uchar *)0);
  13. prev = '\n';
  14. pres = '\n';
  15. return(s);
  16. }
  17. void
  18. printerr(char *type, char *fmt, va_list argl)
  19. {
  20. char buf[1024];
  21. if(!eof)fprint(errorf,"%d: ",yyline);
  22. fprint(errorf,"(%s) ", type);
  23. vseprint(buf, buf+sizeof(buf), fmt, argl);
  24. fprint(errorf, "%s\n", buf);
  25. }
  26. void
  27. error(char *s,...)
  28. {
  29. va_list argl;
  30. va_start(argl, s);
  31. printerr("Error", s, argl);
  32. va_end(argl);
  33. # ifdef DEBUG
  34. if(debug && sect != ENDSECTION) {
  35. sect1dump();
  36. sect2dump();
  37. }
  38. # endif
  39. if(
  40. # ifdef DEBUG
  41. debug ||
  42. # endif
  43. report == 1) statistics();
  44. exits("error"); /* error return code */
  45. }
  46. void
  47. warning(char *s,...)
  48. {
  49. va_list argl;
  50. va_start(argl, s);
  51. printerr("Warning", s, argl);
  52. va_end(argl);
  53. Bflush(&fout);
  54. }
  55. void
  56. lgate(void)
  57. {
  58. int fd;
  59. if (lgatflg) return;
  60. lgatflg=1;
  61. if(foutopen == 0){
  62. fd = create("lex.yy.c", OWRITE, 0666);
  63. if(fd < 0)
  64. error("Can't open lex.yy.c");
  65. Binit(&fout, fd, OWRITE);
  66. foutopen = 1;
  67. }
  68. phead1();
  69. }
  70. void
  71. cclinter(int sw)
  72. {
  73. /* sw = 1 ==> ccl */
  74. int i, j, k;
  75. int m;
  76. if(!sw){ /* is NCCL */
  77. for(i=1;i<NCH;i++)
  78. symbol[i] ^= 1; /* reverse value */
  79. }
  80. for(i=1;i<NCH;i++)
  81. if(symbol[i]) break;
  82. if(i >= NCH) return;
  83. i = cindex[i];
  84. /* see if ccl is already in our table */
  85. j = 0;
  86. if(i){
  87. for(j=1;j<NCH;j++){
  88. if((symbol[j] && cindex[j] != i) ||
  89. (!symbol[j] && cindex[j] == i)) break;
  90. }
  91. }
  92. if(j >= NCH) return; /* already in */
  93. m = 0;
  94. k = 0;
  95. for(i=1;i<NCH;i++)
  96. if(symbol[i]){
  97. if(!cindex[i]){
  98. cindex[i] = ccount;
  99. symbol[i] = 0;
  100. m = 1;
  101. } else k = 1;
  102. }
  103. /* m == 1 implies last value of ccount has been used */
  104. if(m)ccount++;
  105. if(k == 0) return; /* is now in as ccount wholly */
  106. /* intersection must be computed */
  107. for(i=1;i<NCH;i++){
  108. if(symbol[i]){
  109. m = 0;
  110. j = cindex[i]; /* will be non-zero */
  111. for(k=1;k<NCH;k++){
  112. if(cindex[k] == j){
  113. if(symbol[k]) symbol[k] = 0;
  114. else {
  115. cindex[k] = ccount;
  116. m = 1;
  117. }
  118. }
  119. }
  120. if(m)ccount++;
  121. }
  122. }
  123. }
  124. int
  125. usescape(int c)
  126. {
  127. int d;
  128. switch(c){
  129. case 'n': c = '\n'; break;
  130. case 'r': c = '\r'; break;
  131. case 't': c = '\t'; break;
  132. case 'b': c = '\b'; break;
  133. case 'f': c = 014; break; /* form feed for ascii */
  134. case '0': case '1': case '2': case '3':
  135. case '4': case '5': case '6': case '7':
  136. c -= '0';
  137. while('0' <= (d=gch()) && d <= '7'){
  138. c = c * 8 + (d-'0');
  139. if(!('0' <= peek && peek <= '7')) break;
  140. }
  141. break;
  142. }
  143. return(c);
  144. }
  145. int
  146. lookup(uchar *s, uchar **t)
  147. {
  148. int i;
  149. i = 0;
  150. while(*t){
  151. if(strcmp((char *)s, *(char **)t) == 0)
  152. return(i);
  153. i++;
  154. t++;
  155. }
  156. return(-1);
  157. }
  158. int
  159. cpyact(void)
  160. { /* copy C action to the next ; or closing } */
  161. int brac, c, mth;
  162. int savline, sw;
  163. brac = 0;
  164. sw = TRUE;
  165. savline = 0;
  166. while(!eof){
  167. c = gch();
  168. swt:
  169. switch( c ){
  170. case '|': if(brac == 0 && sw == TRUE){
  171. if(peek == '|')gch(); /* eat up an extra '|' */
  172. return(0);
  173. }
  174. break;
  175. case ';':
  176. if( brac == 0 ){
  177. Bputc(&fout, c);
  178. Bputc(&fout, '\n');
  179. return(1);
  180. }
  181. break;
  182. case '{':
  183. brac++;
  184. savline=yyline;
  185. break;
  186. case '}':
  187. brac--;
  188. if( brac == 0 ){
  189. Bputc(&fout, c);
  190. Bputc(&fout, '\n');
  191. return(1);
  192. }
  193. break;
  194. case '/': /* look for comments */
  195. Bputc(&fout, c);
  196. c = gch();
  197. if( c != '*' ) goto swt;
  198. /* it really is a comment */
  199. Bputc(&fout, c);
  200. savline=yyline;
  201. while( c=gch() ){
  202. if( c=='*' ){
  203. Bputc(&fout, c);
  204. if( (c=gch()) == '/' ) goto loop;
  205. }
  206. Bputc(&fout, c);
  207. }
  208. yyline=savline;
  209. error( "EOF inside comment" );
  210. case '\'': /* character constant */
  211. mth = '\'';
  212. goto string;
  213. case '"': /* character string */
  214. mth = '"';
  215. string:
  216. Bputc(&fout, c);
  217. while( c=gch() ){
  218. if( c=='\\' ){
  219. Bputc(&fout, c);
  220. c=gch();
  221. }
  222. else if( c==mth ) goto loop;
  223. Bputc(&fout, c);
  224. if (c == '\n') {
  225. yyline--;
  226. error( "Non-terminated string or character constant");
  227. }
  228. }
  229. error( "EOF in string or character constant" );
  230. case '\0':
  231. yyline = savline;
  232. error("Action does not terminate");
  233. default:
  234. break; /* usual character */
  235. }
  236. loop:
  237. if(c != ' ' && c != '\t' && c != '\n') sw = FALSE;
  238. Bputc(&fout, c);
  239. }
  240. error("Premature EOF");
  241. return(0);
  242. }
  243. int
  244. gch(void){
  245. int c;
  246. prev = pres;
  247. c = pres = peek;
  248. peek = pushptr > pushc ? *--pushptr : Bgetc(fin);
  249. if(peek == Beof && sargc > 1){
  250. Bterm(fin);
  251. fin = Bopen(sargv[fptr++],OREAD);
  252. if(fin == 0)
  253. error("Cannot open file %s",sargv[fptr-1]);
  254. peek = Bgetc(fin);
  255. sargc--;
  256. sargv++;
  257. }
  258. if(c == Beof) {
  259. eof = TRUE;
  260. Bterm(fin);
  261. return(0);
  262. }
  263. if(c == '\n')yyline++;
  264. return(c);
  265. }
  266. int
  267. mn2(int a, int d, int c)
  268. {
  269. name[tptr] = a;
  270. left[tptr] = d;
  271. right[tptr] = c;
  272. parent[tptr] = 0;
  273. nullstr[tptr] = 0;
  274. switch(a){
  275. case RSTR:
  276. parent[d] = tptr;
  277. break;
  278. case BAR:
  279. case RNEWE:
  280. if(nullstr[d] || nullstr[c]) nullstr[tptr] = TRUE;
  281. parent[d] = parent[c] = tptr;
  282. break;
  283. case RCAT:
  284. case DIV:
  285. if(nullstr[d] && nullstr[c])nullstr[tptr] = TRUE;
  286. parent[d] = parent[c] = tptr;
  287. break;
  288. case RSCON:
  289. parent[d] = tptr;
  290. nullstr[tptr] = nullstr[d];
  291. break;
  292. # ifdef DEBUG
  293. default:
  294. warning("bad switch mn2 %d %d",a,d);
  295. break;
  296. # endif
  297. }
  298. if(tptr > treesize)
  299. error("Parse tree too big %s",(treesize == TREESIZE?"\nTry using %e num":""));
  300. return(tptr++);
  301. }
  302. int
  303. mn1(int a, int d)
  304. {
  305. name[tptr] = a;
  306. left[tptr] = d;
  307. parent[tptr] = 0;
  308. nullstr[tptr] = 0;
  309. switch(a){
  310. case RCCL:
  311. case RNCCL:
  312. if(strlen((char *)d) == 0) nullstr[tptr] = TRUE;
  313. break;
  314. case STAR:
  315. case QUEST:
  316. nullstr[tptr] = TRUE;
  317. parent[d] = tptr;
  318. break;
  319. case PLUS:
  320. case CARAT:
  321. nullstr[tptr] = nullstr[d];
  322. parent[d] = tptr;
  323. break;
  324. case S2FINAL:
  325. nullstr[tptr] = TRUE;
  326. break;
  327. # ifdef DEBUG
  328. case FINAL:
  329. case S1FINAL:
  330. break;
  331. default:
  332. warning("bad switch mn1 %d %d",a,d);
  333. break;
  334. # endif
  335. }
  336. if(tptr > treesize)
  337. error("Parse tree too big %s",(treesize == TREESIZE?"\nTry using %e num":""));
  338. return(tptr++);
  339. }
  340. int
  341. mn0(int a)
  342. {
  343. name[tptr] = a;
  344. parent[tptr] = 0;
  345. nullstr[tptr] = 0;
  346. if(a >= NCH) switch(a){
  347. case RNULLS: nullstr[tptr] = TRUE; break;
  348. # ifdef DEBUG
  349. default:
  350. warning("bad switch mn0 %d",a);
  351. break;
  352. # endif
  353. }
  354. if(tptr > treesize)
  355. error("Parse tree too big %s",(treesize == TREESIZE?"\nTry using %e num":""));
  356. return(tptr++);
  357. }
  358. void
  359. munputc(int p)
  360. {
  361. *pushptr++ = peek; /* watch out for this */
  362. peek = p;
  363. if(pushptr >= pushc+TOKENSIZE)
  364. error("Too many characters pushed");
  365. }
  366. void
  367. munputs(uchar *p)
  368. {
  369. int i,j;
  370. *pushptr++ = peek;
  371. peek = p[0];
  372. i = strlen((char*)p);
  373. for(j = i-1; j>=1; j--)
  374. *pushptr++ = p[j];
  375. if(pushptr >= pushc+TOKENSIZE)
  376. error("Too many characters pushed");
  377. }
  378. int
  379. dupl(int n)
  380. {
  381. /* duplicate the subtree whose root is n, return ptr to it */
  382. int i;
  383. i = name[n];
  384. if(i < NCH) return(mn0(i));
  385. switch(i){
  386. case RNULLS:
  387. return(mn0(i));
  388. case RCCL: case RNCCL: case FINAL: case S1FINAL: case S2FINAL:
  389. return(mn1(i,left[n]));
  390. case STAR: case QUEST: case PLUS: case CARAT:
  391. return(mn1(i,dupl(left[n])));
  392. case RSTR: case RSCON:
  393. return(mn2(i,dupl(left[n]),right[n]));
  394. case BAR: case RNEWE: case RCAT: case DIV:
  395. return(mn2(i,dupl(left[n]),dupl(right[n])));
  396. # ifdef DEBUG
  397. default:
  398. warning("bad switch dupl %d",n);
  399. # endif
  400. }
  401. return(0);
  402. }
  403. # ifdef DEBUG
  404. void
  405. allprint(int c)
  406. {
  407. switch(c){
  408. case 014:
  409. print("\\f");
  410. charc++;
  411. break;
  412. case '\n':
  413. print("\\n");
  414. charc++;
  415. break;
  416. case '\t':
  417. print("\\t");
  418. charc++;
  419. break;
  420. case '\b':
  421. print("\\b");
  422. charc++;
  423. break;
  424. case ' ':
  425. print("\\\bb");
  426. break;
  427. default:
  428. if(!isprint(c)){
  429. print("\\%-3o",c);
  430. charc += 3;
  431. } else
  432. print("%c", c);
  433. break;
  434. }
  435. charc++;
  436. }
  437. void
  438. strpt(uchar *s)
  439. {
  440. charc = 0;
  441. while(*s){
  442. allprint(*s++);
  443. if(charc > LINESIZE){
  444. charc = 0;
  445. print("\n\t");
  446. }
  447. }
  448. }
  449. void
  450. sect1dump(void)
  451. {
  452. int i;
  453. print("Sect 1:\n");
  454. if(def[0]){
  455. print("str trans\n");
  456. i = -1;
  457. while(def[++i])
  458. print("%s\t%s\n",def[i],subs[i]);
  459. }
  460. if(sname[0]){
  461. print("start names\n");
  462. i = -1;
  463. while(sname[++i])
  464. print("%s\n",sname[i]);
  465. }
  466. }
  467. void
  468. sect2dump(void)
  469. {
  470. print("Sect 2:\n");
  471. treedump();
  472. }
  473. void
  474. treedump(void)
  475. {
  476. int t;
  477. uchar *p;
  478. print("treedump %d nodes:\n",tptr);
  479. for(t=0;t<tptr;t++){
  480. print("%4d ",t);
  481. parent[t] ? print("p=%4d",parent[t]) : print(" ");
  482. print(" ");
  483. if(name[t] < NCH)
  484. allprint(name[t]);
  485. else switch(name[t]){
  486. case RSTR:
  487. print("%d ",left[t]);
  488. allprint(right[t]);
  489. break;
  490. case RCCL:
  491. print("ccl ");
  492. strpt(left[t]);
  493. break;
  494. case RNCCL:
  495. print("nccl ");
  496. strpt(left[t]);
  497. break;
  498. case DIV:
  499. print("/ %d %d",left[t],right[t]);
  500. break;
  501. case BAR:
  502. print("| %d %d",left[t],right[t]);
  503. break;
  504. case RCAT:
  505. print("cat %d %d",left[t],right[t]);
  506. break;
  507. case PLUS:
  508. print("+ %d",left[t]);
  509. break;
  510. case STAR:
  511. print("* %d",left[t]);
  512. break;
  513. case CARAT:
  514. print("^ %d",left[t]);
  515. break;
  516. case QUEST:
  517. print("? %d",left[t]);
  518. break;
  519. case RNULLS:
  520. print("nullstring");
  521. break;
  522. case FINAL:
  523. print("final %d",left[t]);
  524. break;
  525. case S1FINAL:
  526. print("s1final %d",left[t]);
  527. break;
  528. case S2FINAL:
  529. print("s2final %d",left[t]);
  530. break;
  531. case RNEWE:
  532. print("new %d %d",left[t],right[t]);
  533. break;
  534. case RSCON:
  535. p = (uchar *)right[t];
  536. print("start %s",sname[*p++-1]);
  537. while(*p)
  538. print(", %s",sname[*p++-1]);
  539. print(" %d",left[t]);
  540. break;
  541. default:
  542. print("unknown %d %d %d",name[t],left[t],right[t]);
  543. break;
  544. }
  545. if(nullstr[t])print("\t(null poss.)");
  546. print("\n");
  547. }
  548. }
  549. # endif