dmake: do not set MAKEFLAGS=k
[unleashed/tickless.git] / usr / src / lib / libpp / common / ppproto.c
blob147aecf87c788b19c54821d2e51ff210258ff0c9
1 /***********************************************************************
2 * *
3 * This software is part of the ast package *
4 * Copyright (c) 1986-2009 AT&T Intellectual Property *
5 * and is licensed under the *
6 * Common Public License, Version 1.0 *
7 * by AT&T Intellectual Property *
8 * *
9 * A copy of the License is available at *
10 * http://www.opensource.org/licenses/cpl1.0.txt *
11 * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
12 * *
13 * Information and Software Systems Research *
14 * AT&T Research *
15 * Florham Park NJ *
16 * *
17 * Glenn Fowler <gsf@research.att.com> *
18 * *
19 ***********************************************************************/
20 #pragma prototyped
22 * Glenn Fowler
23 * AT&T Research
25 * convert C prototypes to ANSI, K&R and C++ styles or K&R to ANSI
26 * slips into the pp block read
28 * define PROTOMAIN for standalone proto
29 * PROTOMAIN is coded for minimal library support
32 static const char id[] = "\n@(#)$Id: proto (AT&T Research) 2008-05-11 $\0\n";
34 #if PROTOMAIN
36 #include "ppfsm.c"
38 #include <hashkey.h>
40 #if PROTO_STANDALONE
41 #undef O_RDONLY
42 #endif
44 #else
46 #include "pplib.h"
47 #include "ppfsm.h"
49 #endif
51 #define MAGICGEN "/* : : generated by proto : : */\n"
53 #define MAGICDIR "pragma" /* proto magic directive */
54 #define MAGICARG "prototyped" /* proto magic directive arg */
55 #define MAGICOFF "noticed" /* no notice if found in pragma */
56 #define MAGICTOP 64 /* must be in these top lines */
57 #define NOTICED "Copyright" /* no notice if found in magic */
58 #define PUBLICDOMAIN "Public Domain" /* no notice if found in magic */
60 struct proto /* proto buffer state */
62 int brace; /* {..} level */
63 int call; /* call level */
64 int fd; /* input file descriptor */
65 char* file; /* input file name */
66 long flags; /* coupled flags */
67 long options; /* uncoupled flags */
68 char* package; /* header package */
69 int line; /* input line count */
70 int test; /* testing */
72 char* tp; /* input token base */
74 int iz; /* input buffer size */
75 char* ib; /* input buffer base */
76 char* ip; /* input buffer pointer */
78 int oz; /* output buffer size */
79 char* ob; /* output buffer base */
80 char* op; /* output buffer pointer */
81 char* ox; /* output buffer externalize */
83 char cc[3]; /* beg mid end comment char */
84 char pushback[4]; /* pushback area for caller */
86 char variadic[256]; /* variadic args buffer */
88 /* output buffer */
89 /* slide buffer */
90 /* input buffer */
94 * proto is separate from pp so these undef's are ok
97 #undef CLASSIC
98 #define CLASSIC (1L<<0)
99 #undef DECLARE
100 #define DECLARE (1L<<1)
101 #undef DEFINE
102 #define DEFINE (1L<<2)
103 #undef DIRECTIVE
104 #define DIRECTIVE (1L<<3)
105 #undef ERROR
106 #define ERROR (1L<<4)
107 #undef EXTERN
108 #define EXTERN (1L<<5)
109 #undef EXTERNALIZE
110 #define EXTERNALIZE (1L<<6)
111 #undef IDID
112 #define IDID (1L<<7)
113 #undef INDIRECT
114 #define INDIRECT (1L<<8)
115 #undef INIT
116 #define INIT (1L<<9)
117 #undef INIT_DEFINE
118 #define INIT_DEFINE (1L<<10)
119 #undef INIT_INCLUDE
120 #define INIT_INCLUDE (1L<<11)
121 #undef JUNK
122 #define JUNK (1L<<12)
123 #undef LINESYNC
124 #define LINESYNC (1L<<13)
125 #undef MANGLE
126 #define MANGLE (1L<<14)
127 #undef MATCH
128 #define MATCH (1L<<15)
129 #undef MORE
130 #define MORE (1L<<16)
131 #undef OTHER
132 #define OTHER (1L<<17)
133 #undef PASS
134 #define PASS (1L<<18)
135 #undef PLUSONLY
136 #define PLUSONLY (1L<<19)
137 #undef PLUSPLUS
138 #define PLUSPLUS (1L<<20)
139 #undef RECURSIVE
140 #define RECURSIVE (1L<<21)
141 #undef SHARP
142 #define SHARP (1L<<22)
143 #undef SKIP
144 #define SKIP (1L<<23)
145 #undef SLIDE
146 #define SLIDE (1L<<24)
147 #undef TOKENS
148 #define TOKENS (1L<<25)
149 #undef TYPEDEF
150 #define TYPEDEF (1L<<26)
151 #undef VARIADIC
152 #define VARIADIC (1L<<27)
153 #undef VARIADIC2
154 #define VARIADIC2 (1L<<28)
155 #undef YACC
156 #define YACC (1L<<29)
157 #undef YACCSPLIT
158 #define YACCSPLIT (1L<<30)
159 #undef YACC2
160 #define YACC2 (1L<<31)
162 #undef GLOBAL
163 #define GLOBAL (MORE)
165 #undef REGULAR
166 #define REGULAR (1L<<0)
168 #ifndef CHUNK
169 #define CHUNK 1024
170 #endif
171 #define BLOCK (8*CHUNK)
173 #define T_VA_START (N_TOKEN+1)
175 #define RESERVED(b,e,n) ((((long)(b))<<16)|(((long)(e))<<8)|((long)(n)))
178 * generate integer
179 * pointer to end returned
182 static char*
183 number(register char* p, register long n)
185 register long d;
187 for (d = 1000000; d > 1; d /= 10)
188 if (n >= d) *p++ = '0' + (n / d) % 10;
189 *p++ = '0' + n % 10;
190 return p;
193 #if PROTOMAIN
195 static int errors;
197 #if PROTO_STANDALONE
200 * namespace pollution forces us to claim parts of libc
203 #undef memcpy
204 #define memcpy(t,f,n) memcopy(t,f,n)
205 #undef strcpy
206 #define strcpy(t,f) strcopy(t,f)
207 #undef strlen
208 #define strlen(s) sstrlen(s)
209 #undef strncmp
210 #define strncmp(s,t,n) sstrncmp(s,t,n)
213 * environmentally safe strlen()
216 static int
217 sstrlen(register const char* s)
219 register const char* b;
221 for (b = s; *s; s++);
222 return s - b;
226 * environmentally safe strncmp()
229 static int
230 sstrncmp(register const char* s, register char* t, register int n)
232 register const char* e = s + n;
234 while (s < e)
236 if (*s != *t || !*s)
237 return *s - *t;
238 s++;
239 t++;
241 return 0;
245 * strcpy() except pointer to end returned
248 static char*
249 strcopy(register char* s, register const char* t)
251 while (*s++ = *t++);
252 return s - 1;
255 #endif
257 static void
258 proto_error(char* iob, int level, char* msg, char* arg)
260 register char* p;
261 char buf[1024];
263 p = strcopy(buf, "proto: ");
264 if (iob)
266 register struct proto* proto = (struct proto*)(iob - sizeof(struct proto));
268 if (proto->line)
270 if (proto->file)
272 *p++ = '"';
273 p = strcopy(p, proto->file);
274 *p++ = '"';
275 *p++ = ',';
276 *p++ = ' ';
278 p = strcopy(p, "line ");
279 p = number(p, proto->line);
281 else if (proto->file)
282 p = strcopy(p, proto->file);
284 else
286 p = strcopy(p, msg);
287 msg = arg;
288 arg = 0;
290 if (*(p - 1) != ' ')
292 *p++ = ':';
293 *p++ = ' ';
295 if (level == 1)
296 p = strcopy(p, "warning: ");
297 p = strcopy(p, msg);
298 if (arg)
300 *p++ = ' ';
301 p = strcopy(p, arg);
303 *p++ = '\n';
304 write(2, buf, p - buf);
305 if (level >= 3)
306 exit(level - 2);
307 if (level >= 2)
308 errors++;
312 * memcpy() but pointer to end returned
315 static char*
316 memcopy(register char* s, register char* t, int n)
318 register char* e = t + n;
320 while (t < e) *s++ = *t++;
321 return s;
324 #include "../libast/port/astlicense.c"
326 #else
328 #define memcopy(s,t,n) (((char*)memcpy(s,t,n))+(n))
330 #endif
333 * generate line sync
334 * pointer to end returned
337 static char*
338 linesync(register struct proto* proto, register char* p, register long n)
340 #if PROTOMAIN
341 if (proto->flags & LINESYNC)
342 #endif
344 #if PROTOMAIN
345 p = strcopy(p, "\n#line ");
346 #else
347 p = strcopy(p, "\n# ");
348 #endif
349 p = number(p, n);
350 *p++ = '\n';
352 return p;
356 * output init header
357 * pointer to end returned
360 static char*
361 init(struct proto* proto, char* op, int flags)
363 register char* s;
365 if (flags & INIT_DEFINE)
367 op = strcopy(op, "\
369 #if !defined(__PROTO__)\n\
370 # if defined(__STDC__) || defined(__cplusplus) || defined(_proto) || defined(c_plusplus)\n\
371 # if defined(__cplusplus)\n\
372 # define __LINKAGE__ \"C\"\n\
373 # else\n\
374 # define __LINKAGE__\n\
375 # endif\n\
376 # define __STDARG__\n\
377 # define __PROTO__(x) x\n\
378 # define __OTORP__(x)\n\
379 # define __PARAM__(n,o) n\n\
380 # if !defined(__STDC__) && !defined(__cplusplus)\n\
381 # if !defined(c_plusplus)\n\
382 # define const\n\
383 # endif\n\
384 # define signed\n\
385 # define void int\n\
386 # define volatile\n\
387 # define __V_ char\n\
388 # else\n\
389 # define __V_ void\n\
390 # endif\n\
391 # else\n\
392 # define __PROTO__(x) ()\n\
393 # define __OTORP__(x) x\n\
394 # define __PARAM__(n,o) o\n\
395 # define __LINKAGE__\n\
396 # define __V_ char\n\
397 # define const\n\
398 # define signed\n\
399 # define void int\n\
400 # define volatile\n\
401 # endif\n\
402 # define __MANGLE__ __LINKAGE__\n\
403 # if defined(__cplusplus) || defined(c_plusplus)\n\
404 # define __VARARG__ ...\n\
405 # else\n\
406 # define __VARARG__\n\
407 # endif\n\
408 # if defined(__STDARG__)\n\
409 # define __VA_START__(p,a) va_start(p,a)\n\
410 # else\n\
411 # define __VA_START__(p,a) va_start(p)\n\
412 # endif\n\
413 # if !defined(__INLINE__)\n\
414 # if defined(__cplusplus)\n\
415 # define __INLINE__ extern __MANGLE__ inline\n\
416 # else\n\
417 # if defined(_WIN32) && !defined(__GNUC__)\n\
418 # define __INLINE__ __inline\n\
419 # endif\n\
420 # endif\n\
421 # endif\n\
422 #endif\n\
423 #if !defined(__LINKAGE__)\n\
424 #define __LINKAGE__ /* 2004-08-11 transition */\n\
425 #endif\n\
428 else
429 op = strcopy(op, "\
431 #if !defined(__PROTO__)\n\
432 #include <prototyped.h>\n\
433 #endif\n\
434 #if !defined(__LINKAGE__)\n\
435 #define __LINKAGE__ /* 2004-08-11 transition */\n\
436 #endif\n\
438 if (proto->package)
440 s = "\
441 #ifndef __MANGLE_%_DATA__\n\
442 # ifdef _BLD_%\n\
443 # ifdef __EXPORT__\n\
444 # define __MANGLE_%_DATA__ __MANGLE__ __EXPORT__\n\
445 # else\n\
446 # define __MANGLE_%_DATA__ __MANGLE__\n\
447 # endif\n\
448 # define __MANGLE_%_FUNC__ __MANGLE__\n\
449 # else\n\
450 # ifdef __IMPORT__\n\
451 # define __MANGLE_%_DATA__ __MANGLE__ __IMPORT__\n\
452 # else\n\
453 # define __MANGLE_%_DATA__ __MANGLE__\n\
454 # endif\n\
455 # define __MANGLE_%_FUNC__ __MANGLE__\n\
456 # endif\n\
457 #endif\n\
459 for (;;)
461 switch (*op++ = *s++)
463 case 0:
464 op--;
465 break;
466 case '%':
467 op = strcopy(op - 1, proto->package);
468 continue;
469 default:
470 continue;
472 break;
475 return op;
478 #define BACKOUT() (op=ko)
479 #define CACHE() do{CACHEIN();CACHEOUT();call=proto->call;}while(0)
480 #define CACHEIN() (ip=proto->ip)
481 #define CACHEOUT() (op=proto->op)
482 #define GETCHR() (*(unsigned char*)ip++)
483 #define KEEPOUT() (ko=op)
484 #define LASTOUT() (*(op-1))
485 #define PUTCHR(c) (*op++=(c))
486 #define SYNC() do{SYNCIN();SYNCOUT();proto->flags&=~(EXTERN|INIT|OTHER|VARIADIC|VARIADIC2);proto->flags|=flags&(EXTERN|INIT|OTHER|VARIADIC|VARIADIC2);proto->call=call;}while(0)
487 #define SYNCIN() (proto->ip=ip)
488 #define SYNCOUT() (proto->op=op)
489 #define UNGETCHR() (ip--)
490 #define UNPUTCHR() (op--)
493 * advance to the next non-space character
496 static char*
497 nns(register char* s)
499 while (*s == ' ' || *s == '\t' || *s == '\n')
500 s++;
501 return s;
504 #define DIR_if 01
505 #define DIR_el 02
506 #define DIR_en 03
507 #define DIR 03
510 * update directive mask
513 static int
514 directive(register char* s, int dir)
516 switch (*(s = nns(s)))
518 case 'e':
519 case 'i':
520 dir <<= 2;
521 switch (*++s)
523 case 'f':
524 dir |= DIR_if;
525 break;
526 case 'l':
527 dir |= DIR_el;
528 break;
529 case 'n':
530 dir |= DIR_en;
531 break;
533 break;
535 return dir;
539 * the tokenizer
540 * top level calls loop until EOB
541 * recursive calls just return the next token
544 static int
545 lex(register struct proto* proto, register long flags)
547 register char* ip;
548 register char* op;
549 register int c;
550 register int state;
551 register short* rp;
552 char* m;
553 char* e;
554 char* t;
555 char* bp;
556 char* v;
557 char* im;
558 char* ko;
559 char* aom;
560 int n;
561 int line;
562 int quot;
563 int brack;
564 int sub;
565 int x;
566 int vc;
568 char* ie = 0;
569 char* om = 0;
570 char* aim = 0;
571 char* aie = 0;
572 char* func = 0;
573 int call = 0;
574 int dir = 0;
575 int group = 0;
576 int last = 0;
577 int paren = 0;
578 #if PROTOMAIN
579 char* qe = 0;
580 int qn = 0;
581 int args = 0;
582 #endif
584 CACHE();
585 #if PROTOMAIN
586 if (flags & EXTERN) KEEPOUT();
587 #endif
588 fsm_start:
589 proto->tp = ip;
590 state = PROTO;
591 bp = ip;
594 rp = fsm[state];
595 fsm_get:
596 while (!(state = rp[c = GETCHR()]));
597 fsm_next:
599 } while (state > 0);
600 if ((n = ip - bp - 1) > 0)
602 ip = bp;
603 MEMCPY(op, ip, n);
604 ip++;
606 state = ~state;
607 fsm_terminal:
608 switch (TERM(state))
610 case S_CHR:
611 if (op > proto->ob && *(op - 1) == '=' && (op == proto->ob + 1 || *(op - 2) != '=')) switch (c)
613 case '+':
614 case '-':
615 case '*':
616 case '&':
617 PUTCHR(' ');
618 break;
620 PUTCHR(c);
621 break;
623 case S_CHRB:
624 UNGETCHR();
625 c = LASTOUT();
626 break;
628 case S_COMMENT:
629 switch (c)
631 case '\n':
632 if (INCOMMENTXX(rp)) goto fsm_newline;
633 PUTCHR(c);
634 proto->line++;
635 rp = fsm[COM2];
636 break;
637 case '/':
638 #if PROTOMAIN
639 if ((flags & (EXTERN|MATCH)) == EXTERN) BACKOUT();
640 else
641 #endif
642 PUTCHR(c);
643 if (INCOMMENTXX(rp))
645 rp = fsm[COM5];
646 break;
648 goto fsm_start;
649 case EOF:
650 break;
651 default:
652 #if PROTOMAIN
653 if ((flags & (EXTERN|MATCH)) == EXTERN) BACKOUT();
654 else
655 #endif
656 PUTCHR(c);
657 rp = fsm[INCOMMENTXX(rp) ? COM5 : COM3];
658 break;
660 bp = ip;
661 goto fsm_get;
663 case S_EOB:
664 if (c)
666 if (state = fsm[TERMINAL][INDEX(rp)+1])
667 goto fsm_terminal;
668 SYNC();
669 return 0;
671 UNGETCHR();
672 fsm_eob:
673 if ((flags & (DECLARE|GLOBAL|RECURSIVE)) == GLOBAL && (proto->flags & MORE))
675 #if PROTOMAIN
676 if (!(flags & EXTERN)) /* XXX */
677 #endif
678 flags |= SLIDE;
679 c = ip - proto->ib;
680 if (!(flags & MATCH))
681 im = proto->tp;
682 if (ip > proto->ib)
684 n = ip - im;
685 if (ip - n < proto->ib)
686 proto->flags |= ERROR;
687 memcopy(proto->ib - n, ip - n, n);
688 ip = proto->ib;
690 proto->tp -= c;
691 if (flags & MATCH)
693 im -= c;
694 ie -= c;
696 if (aim)
697 aim -= c;
698 if (aie)
699 aie -= c;
700 if ((n = read(proto->fd, ip, proto->iz)) > 0)
702 if ((proto->options & REGULAR) && n < proto->iz)
704 proto->flags &= ~MORE;
705 close(proto->fd);
707 *(ip + n) = 0;
708 if (state & SPLICE)
709 goto fsm_splice;
710 bp = ip;
711 goto fsm_get;
713 *ip = 0;
714 proto->flags &= ~MORE;
715 close(proto->fd);
717 if (state & SPLICE)
718 goto fsm_splice;
719 /* NOTE: RECURSIVE lex() should really SLIDE too */
720 if (!(flags & RECURSIVE) && (state = rp[c = EOF]))
722 bp = ip;
723 goto fsm_next;
725 SYNC();
726 return 0;
728 case S_LITBEG:
729 quot = c;
730 #if PROTOMAIN
731 if (c == '"' && qe)
733 for (n = 0, t = qe + 1; t < op && (*t == ' ' || *t == '\t' || *t == '\n' && ++n || *t >= 'A' && *t <= 'Z' || *t == '_'); t++);
734 if (t == op)
736 op = qe;
737 qe = 0;
738 qn = n;
740 else PUTCHR(c);
742 else
743 #endif
744 PUTCHR(c);
745 rp = fsm[LIT1];
746 bp = ip;
747 goto fsm_get;
749 case S_LITEND:
750 if (c == quot)
752 #if PROTOMAIN
753 if (!(flags & DIRECTIVE))
754 qe = (c == '"') ? op : (char*)0;
755 #endif
756 PUTCHR(c);
757 #if PROTOMAIN
758 while (qn > 0)
760 qn--;
761 PUTCHR('\n');
763 #endif
765 else if (c != '\n' && c != EOF)
767 PUTCHR(c);
768 bp = ip;
769 goto fsm_get;
771 else
773 #if PROTOMAIN
774 while (qn > 0)
776 qn--;
777 PUTCHR('\n');
779 #endif
780 UNGETCHR();
782 c = T_INVALID;
783 break;
785 case S_LITESC:
786 #if PROTOMAIN
787 if (flags & CLASSIC) PUTCHR(c);
788 else
789 #endif
790 switch (c)
792 case 'a':
793 n = CC_bel;
794 goto fsm_oct;
795 case 'E':
796 n = CC_esc;
797 goto fsm_oct;
798 case 'v':
799 n = CC_vt;
800 goto fsm_oct;
801 case 'x':
802 SYNC();
803 lex(proto, (flags & GLOBAL) | RECURSIVE);
804 for (n = x = 0; (c = GETCHR()), x < 3; x++) switch (c)
806 case '0': case '1': case '2': case '3':
807 case '4': case '5': case '6': case '7':
808 case '8': case '9':
809 n = (n << 4) + c - '0';
810 break;
811 case 'a': case 'b': case 'c': case 'd':
812 case 'e': case 'f':
813 n = (n << 4) + c - 'a' + 10;
814 break;
815 case 'A': case 'B': case 'C': case 'D':
816 case 'E': case 'F':
817 n = (n << 4) + c - 'A' + 10;
818 break;
819 default:
820 goto fsm_hex;
822 fsm_hex:
823 UNGETCHR();
824 fsm_oct:
825 PUTCHR(((n >> 6) & 07) + '0');
826 PUTCHR(((n >> 3) & 07) + '0');
827 PUTCHR((n & 07) + '0');
828 break;
829 default:
830 PUTCHR(c);
831 break;
833 rp = fsm[LIT1];
834 bp = ip;
835 goto fsm_get;
837 case S_MACRO:
838 UNGETCHR();
839 #if PROTOMAIN
840 if ((flags & EXTERN) && *proto->tp == 's' && !strncmp(proto->tp, "static", 6))
842 c = T_EXTERN;
843 break;
845 #endif
846 if (*proto->tp == '_' && !strncmp(proto->tp, "__STDPP__directive", 6)) c = '#';
847 else c = T_ID;
849 break;
851 case S_NL:
852 fsm_newline:
853 proto->line++;
854 #if PROTOMAIN
855 if (flags & EXTERN)
857 if (op != proto->ob && LASTOUT() != ' ' && LASTOUT() != '\n')
858 PUTCHR(' ');
860 else
861 #endif
862 PUTCHR(c);
863 if (flags & DIRECTIVE)
865 #if PROTOMAIN
866 if (flags & CLASSIC)
868 if (flags & EXTERN) BACKOUT();
869 if (flags & JUNK)
871 *(ip - 1) = 0;
872 op = strcopy(om, "/* ");
873 op = strcopy(op, im);
874 op = strcopy(op, " */\n");
876 flags &= ~(DEFINE|DIRECTIVE|IDID|INDIRECT|JUNK|MATCH|SHARP|TYPEDEF);
878 else
879 #endif
881 if ((flags & (DEFINE|SHARP)) == (DEFINE|SHARP))
883 *(ip - 1) = 0;
884 op = strcopy(om, "#if defined(__STDC__) || defined(__STDPP__)\n");
885 op = strcopy(op, im);
886 op = strcopy(op, "\n#else\n");
887 bp = ip;
888 ip = im;
889 *op++ = *ip++;
890 while (*op = *ip++)
891 if (*op++ == '#' && *ip != '(')
893 op--;
894 while (*--op == ' ' || *op == '\t');
895 if (*ip == '#')
897 op = strcopy(op + 1, "/**/");
898 while (*++ip == ' ' || *ip == '\t');
900 else
902 if (*op != '"') *++op = '"';
903 op++;
904 while (*ip == ' ' || *ip == '\t') ip++;
905 while ((c = *ip) >= 'A' && c <= 'Z' || c >= 'a' && c <= 'z' || c >= '0' && c <= '9' || c == '_') *op++ = *ip++;
906 while (*ip == ' ' || *ip == '\t') ip++;
907 if (*ip == '"') ip++;
908 else *op++ = '"';
911 ip = bp;
912 op = strcopy(op, "\n#endif\n");
913 op = linesync(proto, op, proto->line);
915 flags &= ~(DEFINE|DIRECTIVE|IDID|INDIRECT|MATCH|OTHER|SHARP|SKIP|TOKENS|TYPEDEF);
917 call = 0;
918 group = 0;
919 paren = 0;
920 last = '\n';
922 if (paren == 0 && (flags & (MATCH|RECURSIVE|SKIP|SLIDE)) == SLIDE)
924 #if PROTOMAIN
925 if (flags & EXTERN) BACKOUT();
926 #endif
927 SYNC();
928 return 0;
930 goto fsm_start;
932 case S_QUAL:
933 PUTCHR(c);
934 rp = fsm[NEXT(state)];
935 bp = ip;
936 goto fsm_get;
938 case S_TOK:
939 PUTCHR(c);
940 c = TYPE(state);
941 break;
943 case S_TOKB:
944 UNGETCHR();
945 c = TYPE(state);
946 break;
948 case S_RESERVED:
949 UNGETCHR();
950 c = T_ID;
951 if (!(flags & DECLARE)) switch (RESERVED(*proto->tp, *(ip - 1), ip - proto->tp))
953 case RESERVED('N', 'N', 3):
954 if (proto->tp[1] == 'o')
955 c = T_DO;
956 break;
957 case RESERVED('d', 'o', 2):
958 c = T_DO;
959 break;
960 case RESERVED('e', 'e', 4):
961 if (!(flags & RECURSIVE) && (flags & (DIRECTIVE|TOKENS)) != DIRECTIVE && !strncmp(proto->tp, "else", 4))
963 c = T_ELSE;
964 goto fsm_id;
966 break;
967 case RESERVED('e', 'n', 6):
968 if (!strncmp(proto->tp, "extern", 6))
969 c = T_EXTERN;
970 break;
971 case RESERVED('f', 'r', 3):
972 if (!(flags & RECURSIVE) && !strncmp(proto->tp, "for", 3))
974 c = T_FOR;
975 goto fsm_id;
977 break;
978 case RESERVED('i', 'f', 2):
979 c = T_IF;
980 break;
981 case RESERVED('i', 'e', 6):
982 if (!strncmp(proto->tp, "inline", 6) && !(flags & (MATCH|SKIP|TOKENS|TYPEDEF)) && proto->brace == 0 && paren == 0 && group == 0 && (last == ';' || last == '}' || last == '\n' || last == 0))
984 flags |= SKIP;
985 SYNC();
986 line = proto->line;
987 op = strcopy(op - 6, "__INLINE__");
988 SYNC();
990 break;
991 case RESERVED('r', 'n', 6):
992 if (!(flags & RECURSIVE) && !strncmp(proto->tp, "return", 6))
994 c = T_RETURN;
995 goto fsm_id;
997 break;
998 case RESERVED('s', 'c', 6):
999 if ((proto->options & EXTERNALIZE) && !strncmp(proto->tp, "static", 6))
1001 proto->ox = op - 6;
1002 flags |= EXTERNALIZE;
1004 break;
1005 case RESERVED('t', 'f', 7):
1006 if (!(flags & RECURSIVE) && !strncmp(proto->tp, "typedef", 7))
1008 flags |= TYPEDEF;
1009 c = T_EXTERN;
1011 break;
1012 case RESERVED('v', 't', 8):
1013 if (*ip == '(' && !strncmp(proto->tp, "va_start", 8)) c = T_VA_START;
1014 break;
1015 case RESERVED('v', 'd', 4):
1016 if (!strncmp(proto->tp, "void", 4))
1018 if (flags & (CLASSIC|PLUSONLY|INIT_DEFINE|INIT_INCLUDE)) c = T_VOID;
1019 else
1021 SYNC();
1022 line = proto->line;
1023 if (lex(proto, (flags & GLOBAL) | RECURSIVE) == '*')
1025 memcopy(op - 4, "__V_", 4);
1026 memcopy(ip - 4, "__V_", 4);
1028 else c = T_VOID;
1029 proto->line = line;
1030 SYNC();
1031 bp = ip;
1034 break;
1035 case RESERVED('w', 'e', 5):
1036 if (!(flags & RECURSIVE) && !strncmp(proto->tp, "while", 5))
1038 c = T_WHILE;
1039 goto fsm_id;
1041 break;
1043 #if PROTOMAIN
1044 if ((flags & CLASSIC) && c != T_EXTERN)
1045 c = T_ID;
1046 #endif
1047 break;
1049 case S_VS:
1050 goto fsm_start;
1052 case S_WS:
1053 UNGETCHR();
1054 #if PROTOMAIN
1055 if ((flags & (EXTERN|MATCH)) == EXTERN)
1057 while (op > proto->ob && (*(op - 1) == ' ' || *(op - 1) == '\t'))
1058 op--;
1059 if (op > proto->ob && *(op - 1) != '\n') *op++ = ' ';
1061 #endif
1062 goto fsm_start;
1064 default:
1065 if (state & SPLICE)
1067 if (c == '\\')
1069 if (!(n = GETCHR()))
1071 goto fsm_eob;
1072 fsm_splice:
1073 c = '\\';
1074 n = GETCHR();
1076 if (n == '\n')
1078 proto->line++;
1079 PUTCHR('\\');
1080 PUTCHR('\n');
1081 bp = ip;
1082 goto fsm_get;
1084 UNGETCHR();
1086 state &= ~SPLICE;
1087 if (state >= TERMINAL)
1088 goto fsm_terminal;
1089 rp = fsm[state];
1091 PUTCHR(c);
1092 bp = ip;
1093 goto fsm_get;
1095 if (!(flags & (INIT_DEFINE|INIT_INCLUDE|RECURSIVE)))
1097 if (!(flags & DIRECTIVE)) switch (c)
1099 case '(':
1100 #if PROTOMAIN
1101 if (!(flags & CLASSIC) || proto->brace == 0)
1102 #endif
1104 if (paren++ == 0)
1106 #if PROTOMAIN
1107 if (!(flags & CLASSIC) || group <= 1)
1108 #endif
1110 #if PROTOMAIN
1111 args = 0;
1112 #endif
1113 if (group++ == 0) group++;
1114 else if (flags & INDIRECT) call++;
1115 flags |= MATCH;
1116 im = ip - 1;
1117 om = op - 1;
1119 sub = 0;
1121 else if (paren == 2 && !aim)
1123 sub++;
1124 if (last == '(')
1126 flags &= ~MATCH;
1127 om = 0;
1129 else if (flags & INDIRECT)
1131 aim = ip - 1;
1132 aom = op - 1;
1134 else if ((flags & (MATCH|TOKENS)) == MATCH)
1136 for (m = ip - 2; m > im && (*m == ' ' || *m == '\t'); m--);
1137 if (m != im && sub == 1)
1139 m = im + (*nns(ip) == '*');
1141 if (m == im)
1143 flags &= ~MATCH;
1144 om = 0;
1147 else if ((flags & MATCH) && sub == 1 && *nns(ip) != '*')
1149 flags &= ~MATCH;
1150 om = 0;
1153 flags &= ~TOKENS;
1155 break;
1156 case ')':
1157 #if PROTOMAIN
1158 if (!(flags & CLASSIC) || proto->brace == 0)
1159 #endif
1160 if (--paren == 0)
1162 #if PROTOMAIN
1163 if (flags & CLASSIC)
1165 if (group != 2)
1167 c = T_ID;
1168 break;
1170 group++;
1172 #endif
1173 ie = ip;
1175 else if (paren == 1 && (flags & INDIRECT) && !aie)
1176 aie = ip;
1177 break;
1178 case '*':
1179 if (last == '(' && group == 2)
1181 group--;
1182 if (paren == 1)
1184 flags |= INDIRECT;
1185 aim = aie = 0;
1188 break;
1189 case '#':
1190 dir = directive(ip, dir);
1191 if (proto->brace == 0 && paren == 0 && last != '=' && (flags & (CLASSIC|DECLARE|DIRECTIVE|MATCH|PLUSONLY|SKIP|TOKENS)) == (MATCH|TOKENS) && ((dir & DIR) != DIR_en || ((dir>>2) & DIR) != DIR_if))
1192 flags |= DIRECTIVE;
1193 else if (!(flags & (DECLARE|DIRECTIVE)))
1195 flags |= DIRECTIVE;
1196 if (!(flags & PLUSONLY))
1198 bp = ip;
1199 while (*ip == ' ' || *ip == '\t') ip++;
1200 if (*ip == 'l' && *++ip == 'i' && *++ip == 'n' && *++ip == 'e')
1202 if (*++ip == ' ' || *ip == '\t')
1204 proto->line = 0;
1205 while (*++ip >= '0' && *ip <= '9')
1206 proto->line = proto->line * 10 + *ip - '0';
1207 proto->line--;
1210 #if PROTOMAIN
1211 else if ((flags & (CLASSIC|EXTERN)) == CLASSIC)
1213 n = 0;
1214 t = ip + 6;
1215 while (ip < t && *ip >= 'a' && *ip <= 'z')
1216 n = HASHKEYPART(n, *ip++);
1217 switch (n)
1219 case HASHKEY4('e','l','s','e'):
1220 case HASHKEY5('e','n','d','i','f'):
1221 while (*ip == ' ' || *ip == '\t') ip++;
1222 if (*ip != '\n' && *ip != '/' && *(ip + 1) != '*')
1224 flags |= JUNK|MATCH;
1225 im = ip;
1226 om = op + (ip - bp);
1228 break;
1229 case HASHKEY4('e','l','i','f'):
1230 case HASHKEY5('e','r','r','o','r'):
1231 case HASHKEY2('i','f'):
1232 case HASHKEY5('i','f','d','e','f'):
1233 case HASHKEY6('i','f','n','d','e','f'):
1234 case HASHKEY5('u','n','d','e','f'):
1235 break;
1236 case HASHKEY6('i','n','c','l','u','d'):
1237 if (*ip == 'e') ip++;
1238 /*FALLTHROUGH*/
1239 case HASHKEY6('d','e','f','i','n','e'):
1240 case HASHKEY6('p','r','a','g','m','a'):
1241 if (*ip < 'a' || *ip > 'z') break;
1242 /*FALLTHROUGH*/
1243 default:
1244 flags |= JUNK|MATCH;
1245 im = bp - 1;
1246 om = op - 1;
1247 break;
1250 else
1251 #endif
1253 if (*ip == 'i' && *++ip == 'n' && *++ip == 'c' && *++ip == 'l' && *++ip == 'u' && *++ip == 'd' && *++ip == 'e')
1255 while (*++ip == ' ' || *ip == '\t');
1256 if (*ip++ == '<' && *ip++ == 's' && *ip++ == 't' && *ip++ == 'd' && *ip++ == 'a' && *ip++ == 'r' && *ip++ == 'g' && *ip++ == '.' && *ip++ == 'h' && *ip++ == '>')
1258 op = strcopy(op, "\
1259 if !defined(va_start)\n\
1260 #if defined(__STDARG__)\n\
1261 #include <stdarg.h>\n\
1262 #else\n\
1263 #include <varargs.h>\n\
1264 #endif\n\
1265 #endif\n\
1267 op = linesync(proto, op, proto->line);
1268 break;
1271 else if (*ip == 'd' && *++ip == 'e' && *++ ip == 'f' && *++ip == 'i' && *++ip == 'n' && *++ip == 'e' && (*++ip == ' ' || *ip == '\t'))
1273 while (*++ip == ' ' || *ip == '\t');
1274 if (*ip == 'e' && *++ip == 'x' && *++ ip == 't' && *++ip == 'e' && *++ip == 'r' && *++ip == 'n' && (*++ip == ' ' || *ip == '\t'))
1276 t = ip;
1277 while (*++t == ' ' || *t == '\t');
1278 if (*t == 'e' && *++t == 'x' && *++ t == 't' && *++t == 'e' && *++t == 'r' && *++t == 'n' && (*++t == ' ' || *t == '\t' || *t == '\n' || *t == '\r'))
1279 ip = t;
1280 t = ip;
1281 while (*++t == ' ' || *t == '\t');
1282 if (*t == '_' && *(t + 1) == '_')
1284 op = strcopy(op, "undef __MANGLE__\n");
1285 op = linesync(proto, op, proto->line);
1286 op = strcopy(op, "#define __MANGLE__ __LINKAGE__");
1287 break;
1290 flags |= DEFINE|MATCH;
1291 im = bp - 1;
1292 om = op - 1;
1294 else if (*ip == 'u' && *++ip == 'n' && *++ ip == 'd' && *++ip == 'e' && *++ip == 'f' && (*++ip == ' ' || *ip == '\t'))
1296 while (*++ip == ' ' || *ip == '\t');
1297 if (*ip == 'e' && *++ip == 'x' && *++ ip == 't' && *++ip == 'e' && *++ip == 'r' && *++ip == 'n' && (*++ip == ' ' || *ip == '\t' || *ip == '\n' || *ip == '\r'))
1299 op = strcopy(op, "undef __MANGLE__\n");
1300 op = linesync(proto, op, proto->line);
1301 op = strcopy(op, "#define __MANGLE__ __LINKAGE__");
1302 break;
1304 flags |= DEFINE|MATCH;
1305 im = bp - 1;
1306 om = op - 1;
1309 ip = bp;
1311 break;
1313 else
1314 break;
1315 /*FALLTHROUGH*/
1316 case '{':
1317 if (proto->brace++ == 0 && paren == 0)
1319 if (last == '=') flags |= INIT;
1320 #if PROTOMAIN
1321 else if (flags & CLASSIC)
1323 if ((flags & (MATCH|OTHER|SKIP)) == MATCH)
1325 if (args)
1327 v = number(op, args < 0 ? -args : args);
1328 v = strcopy(v, " argument actual/formal mismatch");
1329 *v++ = ' ';
1330 v = memcopy(v, im, ie - im);
1331 *v = 0;
1332 proto_error((char*)proto + sizeof(struct proto), 2, op, NiL);
1334 ip--;
1335 /*UNDENT...*/
1336 v = ie;
1337 while (ie < ip)
1338 if (*ie++ == '/' && *ie == '*')
1340 e = ie - 1;
1341 while (++ie < ip)
1343 if (*ie == '*')
1345 while (ie < ip && *ie == '*') ie++;
1346 if (ie < ip && *ie == '/')
1348 while (++ie < ip && (*ie == ' ' || *ie == '\t'));
1349 while (e > v && (*(e - 1) == ' ' || *(e - 1) == '\t')) e--;
1350 if (e > v && *e != '\n') *e++ = ' ';
1351 t = ie;
1352 while (--e >= v)
1353 *--t = *e;
1354 v = t;
1355 break;
1360 ie = v;
1361 /*...INDENT*/
1362 op = om++;
1363 if (flags & EXTERN)
1365 v = op;
1366 while (v > ko && *--v != ' ');
1367 if (*v != ' ')
1369 om = (v = (op += 4)) + 1;
1370 while (v >= ko + 4)
1372 *v = *(v - 4);
1373 v--;
1375 memcopy(ko, "int ", 4);
1377 if (*v == ' ')
1379 while (*(v + 1) == '*')
1380 *v++ = '*';
1381 *v = '\t';
1382 if ((v - ko) <= 8)
1384 om = (e = ++op) + 1;
1385 while (e > v)
1387 *e = *(e - 1);
1388 e--;
1392 om = (v = (op += 7)) + 1;
1393 while (v >= ko + 7)
1395 *v = *(v - 7);
1396 v--;
1398 memcopy(ko, "extern ", 7);
1400 PUTCHR('(');
1401 t = op;
1402 e = 0;
1403 /*UNDENT...*/
1404 while (ie < ip)
1406 if ((c = *ie) == ' ' || c == '\t' || c == '\n')
1408 while ((c = *++ie) == ' ' || c == '\t' || c == '\n');
1409 if (ie >= ip) break;
1410 if (c != '*' && op > om) PUTCHR(' ');
1412 if ((n = ((c = *ie) == ',')) || c == ';')
1414 if (flags & EXTERN)
1416 m = op;
1417 while (op > om && ((c = *(op - 1)) == '(' || c == ')' || c == '[' || c == ']'))
1418 op--;
1419 v = op;
1420 while (op > om && (c = *(op - 1)) != ' ' && c != '*')
1421 op--;
1422 while (*(op - 1) == ' ')
1423 op--;
1424 if (!e)
1426 e = op;
1427 while (e > om && *(e - 1) == '*')
1428 e--;
1430 #if _s5r4_386_compiler_bug_fixed_
1431 if (op <= om || *(op - 1) == ',' && (*op++ = ' '))
1432 op = strcopy(op, "int");
1433 #else
1434 if (op <= om)
1435 op = strcopy(op, "int");
1436 else if (*(op - 1) == ',')
1437 op = strcopy(op, " int");
1438 #endif
1439 while (v < m)
1440 PUTCHR(*v++);
1442 PUTCHR(',');
1443 if (n)
1445 if (x = !e) e = op - 1;
1446 PUTCHR(' ');
1447 m = t;
1448 while (m < e)
1449 PUTCHR(*m++);
1450 if (x)
1452 m = e;
1453 while (*--e != ' ');
1454 while (*(e - 1) == '*') e--;
1455 op -= m - e;
1458 while ((c = *++ie) == ' ' || c == '\t' || c == '\n');
1459 if (ie >= ip) UNPUTCHR();
1460 else PUTCHR(' ');
1461 if (!n)
1463 t = op;
1464 e = 0;
1467 else if (*ie == '*')
1469 if (op > om && (c = *(op - 1)) == ' ') op--;
1470 while (*ie == '*') PUTCHR(*ie++);
1471 while (*ie == ' ' || *ie == '\t' || *ie == '\n') ie++;
1472 if (c != '(') PUTCHR(' ');
1474 else if (*ie == '(')
1476 if (op > om && *(op - 1) == ' ') op--;
1477 PUTCHR(*ie++);
1478 while (*ie == ' ' || *ie == '\t' || *ie == '\n') ie++;
1480 else if (*ie == ')')
1482 if (op > om && *(op - 1) == '(')
1483 proto_error((char*)proto + sizeof(struct proto), 1, "function pointer argument prototype omitted", NiL);
1484 PUTCHR(*ie++);
1485 while (*ie == ' ' || *ie == '\t' || *ie == '\n') ie++;
1487 else if ((flags & EXTERN) && (op == om || *(op - 1) == ' ') && *ie == 'r' && !strncmp(ie, "register", 8) && (*(ie + 8) == ' ' || *(ie + 8) == '\t' || *(ie + 8) == '\n'))
1489 ie += 8;
1490 if (op > om) UNPUTCHR();
1492 else PUTCHR(*ie++);
1494 /*...INDENT*/
1495 if (op <= om) op = strcopy(op, "void");
1496 PUTCHR(')');
1497 if (flags & EXTERN)
1499 PUTCHR(';');
1500 PUTCHR('\n');
1501 SYNCOUT();
1502 KEEPOUT();
1504 else
1506 PUTCHR('\n');
1507 PUTCHR(*ip);
1509 ip++;
1510 flags &= ~(MATCH|SKIP);
1513 #endif
1514 else if ((flags & (MATCH|PLUSONLY|SKIP|TOKENS)) == (MATCH|TOKENS))
1516 line = proto->line;
1517 op = strcopy(om, " __PARAM__(");
1518 op = memcopy(op, im, ie - im);
1519 PUTCHR(',');
1520 PUTCHR(' ');
1521 PUTCHR('(');
1522 flags &= ~(MATCH|SKIP);
1523 if (flags & VARIADIC)
1525 if ((vc = ie - im + 1) > sizeof(proto->variadic)) vc = sizeof(proto->variadic);
1526 memcopy(proto->variadic, im, vc);
1527 op = strcopy(op, "va_alist)) __OTORP__(va_dcl)\n{");
1529 else
1531 flags |= SKIP;
1532 proto->ip = im;
1533 proto->op = op;
1534 group = 0;
1535 brack = 0;
1536 for (;;)
1538 switch (lex(proto, (flags & GLOBAL) | RECURSIVE))
1540 case '[':
1541 brack++;
1542 continue;
1543 case ']':
1544 brack--;
1545 continue;
1546 case '(':
1547 if (paren++) group++;
1548 continue;
1549 case ')':
1550 if (--paren == 0)
1552 group = 0;
1553 if (flags & MATCH)
1555 flags &= ~(MATCH|SKIP);
1556 op = memcopy(op, m, e - m);
1558 break;
1560 continue;
1561 case ',':
1562 if (paren == 1)
1564 group = 0;
1565 if (flags & MATCH)
1567 flags &= ~(MATCH|SKIP);
1568 op = memcopy(op, m, e - m);
1570 PUTCHR(',');
1571 PUTCHR(' ');
1572 proto->op = op;
1574 continue;
1575 case T_ID:
1576 if (group <= 1 && !brack)
1578 flags |= MATCH;
1579 m = proto->tp;
1580 e = proto->ip;
1582 continue;
1583 default:
1584 continue;
1586 break;
1588 PUTCHR(')');
1589 PUTCHR(')');
1591 if (!(flags & SKIP))
1593 flags |= SKIP;
1594 proto->op = strcopy(op, " __OTORP__(");
1595 proto->ip = im + 1;
1596 n = *(ie - 1);
1597 *(ie - 1) = ';';
1598 c = *ie;
1599 *ie = 0;
1600 lex(proto, (flags & GLOBAL) | DECLARE);
1601 *(ie - 1) = n;
1602 *ie = c;
1603 proto->ip = ie;
1604 op = proto->op;
1605 PUTCHR(')');
1607 if (flags & EXTERNALIZE) memcpy(proto->ox, "extern", 6);
1608 op = linesync(proto, op, proto->line = line);
1609 if (flags & DIRECTIVE)
1611 proto->brace = 0;
1612 PUTCHR('\n');
1613 PUTCHR('#');
1615 else if (!(flags & VARIADIC)) PUTCHR('{');
1618 flags &= ~(IDID|INDIRECT|MATCH|OTHER|SKIP);
1619 call = 0;
1620 group = 0;
1621 break;
1622 case '}':
1623 flags &= ~(IDID|INDIRECT|MATCH|OTHER|SKIP|TOKENS);
1624 if (--proto->brace == 0)
1626 flags &= ~(INIT|VARIADIC|VARIADIC2);
1627 #if PROTOMAIN
1628 if (flags & EXTERN) BACKOUT();
1629 #endif
1631 call = 0;
1632 group = 0;
1633 paren = 0;
1634 break;
1635 case '=':
1636 if (last == '?') flags |= DIRECTIVE;
1637 else if (paren == 0 && (flags & (INIT|MATCH|SKIP)) == MATCH)
1639 if (last == ')' && proto->brace && (group != 2 || call != 2)) flags |= SKIP;
1640 else goto fsm_statement;
1642 goto fsm_other;
1643 case ',':
1644 #if PROTOMAIN
1645 if (flags & CLASSIC)
1647 if (paren == 1) args++;
1648 else
1650 args--;
1651 flags &= ~MATCH;
1653 break;
1655 #endif
1656 if (paren == 0 && (flags & DECLARE)) *(op - 1) = c = ';';
1657 /*FALLTHROUGH*/
1658 case ';':
1659 fsm_statement:
1660 if (flags & INIT) /* ignore */;
1661 #if PROTOMAIN
1662 else if (flags & CLASSIC)
1664 if (paren == 0)
1666 if ((flags & MATCH) && last == ')')
1667 flags &= ~MATCH;
1668 if (!(flags & MATCH))
1670 call = 0;
1671 group = 0;
1672 flags &= ~SKIP;
1673 if (flags & EXTERN) BACKOUT();
1674 if (flags & SLIDE)
1676 SYNC();
1677 return 0;
1680 else
1682 args--;
1683 if ((flags & (EXTERN|SKIP)) == (EXTERN|SKIP))
1684 BACKOUT();
1688 #endif
1689 else if (paren == 0)
1691 if ((flags & (MATCH|OTHER|SKIP)) == MATCH && call > 1)
1693 if ((flags & MANGLE) && func)
1695 func[0] = 'F';
1696 func[1] = 'U';
1697 func[2] = 'N';
1698 func[3] = 'C';
1699 func = 0;
1701 if ((flags & (DECLARE|INDIRECT)) == INDIRECT && aim && aie < im)
1703 while (aie < ip && (*aie == ' ' || *aie == '\t' || *aie == '\n')) aie++;
1704 v = aim;
1705 while (v < aie)
1706 if (*v++ == ')') break;
1707 while (v < aie && (*v == ' ' || *v == '\t' || *v == '\n')) v++;
1708 if (v == aie || !(flags & PLUSPLUS))
1710 if (flags & PLUSPLUS) n = 3;
1711 else if (v == aie && *v == '(') n = 10;
1712 else n = 11;
1713 ko = op;
1714 om += n;
1715 v = op += n;
1716 while (v >= ko + n)
1718 *v = *(v - n);
1719 v--;
1721 if (flags & PLUSPLUS) memcopy(aom, "(...))", 6);
1722 else if (n == 10) memcopy(aom, "(__VARARG__))", 13);
1723 else
1725 ko = strcopy(aom, " __PROTO__(");
1726 ko = memcopy(ko, aim, aie - aim);
1727 *ko = ')';
1728 if (++ko >= om)
1730 *ko++ = ')';
1731 om = ko;
1736 else if (flags & TYPEDEF)
1738 op = om;
1739 while (*--op == ' ' || *op == '\t' || *op == '\n');
1740 if (*op != ')')
1742 op = om += 14;
1743 *--op = ')';
1744 while ((x = *(op - 14)) >= 'A' && x <= 'Z' || x >= 'a' && x <= 'z' || x >= '0' && x <= '9' || x == '_')
1745 *--op = x;
1746 memcopy(op - 13, "(__OTORP__(*)", 13);
1749 if (flags & OTHER)
1751 else if (flags & PLUSPLUS)
1753 op = om;
1754 if (!(flags & TOKENS)) op = strcopy(op, "(...)");
1755 else op = memcopy(op, im, ie - im);
1756 PUTCHR(c);
1758 else
1760 if (flags & DECLARE) op = strcopy(om, "()");
1761 else if (!(flags & TOKENS)) op = strcopy(om, "(__VARARG__)");
1762 else
1764 op = strcopy(om, " __PROTO__(");
1765 op = memcopy(op, im, ie - im);
1766 PUTCHR(')');
1768 if (flags & EXTERNALIZE) memcpy(proto->ox, "extern", 6);
1769 PUTCHR(c);
1771 flags &= ~(MATCH|VARIADIC|VARIADIC2);
1772 if (c == ',' && !(flags & INDIRECT))
1774 call = 1;
1775 group = 0;
1776 break;
1779 else if (flags & (OTHER|SKIP)) call = 0;
1780 if (c == ';')
1782 flags &= ~(EXTERNALIZE|MANGLE|TOKENS|TYPEDEF);
1783 call = 0;
1784 if (flags & SLIDE)
1786 SYNC();
1787 return 0;
1790 else call = call > 1 && c == ',';
1791 group = 0;
1792 flags &= ~(IDID|INDIRECT|MATCH|OTHER|SKIP);
1794 else if (paren == 1 && group == 1 && !(flags & (IDID|MANGLE))) flags |= TOKENS|OTHER;
1795 break;
1796 case T_DO:
1797 case T_IF:
1798 flags |= TOKENS|SKIP;
1799 break;
1800 case T_EXTERN:
1801 #if PROTOMAIN
1802 if (flags & CLASSIC)
1804 if (proto->brace == 0)
1805 flags |= SKIP;
1807 else
1808 #endif
1809 if (paren == 0 && !(flags & TYPEDEF))
1811 flags |= MANGLE;
1812 if (!(flags & PLUSONLY) || proto->package)
1814 op = strcopy(op, " __MANGLE__");
1815 if (proto->package)
1817 op = strcopy(op - 1, proto->package);
1818 func = op + 1;
1819 op = strcopy(op, "_DATA__");
1822 else
1823 func = 0;
1825 break;
1826 case T_VARIADIC:
1827 if (paren == 0 && (flags & (DECLARE|VARIADIC)) == DECLARE)
1829 op -= 3;
1830 SYNC();
1831 return c;
1833 if (paren == 1 && !(flags & SKIP))
1834 flags |= VARIADIC;
1835 flags |= TOKENS;
1836 break;
1837 case T_VOID:
1838 goto fsm_id;
1839 case T_VA_START:
1840 if ((flags & (PLUSONLY|VARIADIC)) == VARIADIC)
1842 flags &= ~MATCH;
1843 line = proto->line;
1844 op = strcopy(op - 8, "__VA_START__");
1845 SYNC();
1846 for (;;)
1848 switch (lex(proto, (flags & GLOBAL) | RECURSIVE))
1850 case 0:
1851 case ';':
1852 break;
1853 case T_ID:
1854 if (!(flags & MATCH))
1856 flags |= MATCH;
1857 m = proto->tp;
1858 e = proto->ip;
1860 continue;
1861 default:
1862 continue;
1864 break;
1866 CACHE();
1867 if (flags & MATCH)
1869 v = m;
1870 n = e - m;
1872 else
1874 v = "ap";
1875 n = 2;
1877 op = strcopy(op, " __OTORP__(");
1878 proto->ip = proto->variadic;
1879 proto->op = op;
1880 flags &= ~MATCH;
1881 group = 0;
1882 bp = proto->ip + 1;
1883 if (*bp == 'r' && !strncmp(bp, "register", 8) && (*(bp + 8) == ' ' || *(bp + 8) == '\t')) bp += 9;
1884 for (;;)
1886 switch (lex(proto, (flags & GLOBAL) | RECURSIVE))
1888 case '(':
1889 if (paren++) group++;
1890 continue;
1891 case ')':
1892 if (--paren == 0)
1894 if (flags & MATCH)
1896 flags &= ~MATCH;
1897 if (!(flags & VARIADIC2))
1899 op = memcopy(op, m, e - m);
1900 op = strcopy(op, " = ");
1902 op = strcopy(op, "va_arg(");
1903 op = memcopy(op, v, n);
1904 PUTCHR(',');
1905 PUTCHR(' ');
1906 if (m > bp) op = memcopy(op, bp, m - bp);
1907 else op = strcopy(op, "int ");
1908 if (group > 1) op = strcopy(op, ")()");
1909 else op = memcopy(op, e, proto->ip - e - 1);
1910 PUTCHR(')');
1911 PUTCHR(';');
1913 group = 0;
1914 break;
1916 continue;
1917 case ',':
1918 if (paren == 1)
1920 if (flags & MATCH)
1922 flags &= ~MATCH;
1923 if (!(flags & VARIADIC2))
1925 op = memcopy(op, m, e - m);
1926 op = strcopy(op, " = ");
1928 op = strcopy(op, "va_arg(");
1929 op = memcopy(op, v, n);
1930 PUTCHR(',');
1931 PUTCHR(' ');
1932 if (m > bp) op = memcopy(op, bp, m - bp);
1933 else op = strcopy(op, "int ");
1934 if (group > 1) op = strcopy(op, ")()");
1935 else op = memcopy(op, e, proto->ip - e - 1);
1936 PUTCHR(')');
1937 PUTCHR(';');
1938 bp = proto->ip + 1;
1939 if (*bp == 'r' && !strncmp(bp, "register", 8) && (*(bp + 8) == ' ' || *(bp + 8) == '\t')) bp += 9;
1941 group = 0;
1942 proto->op = op;
1944 continue;
1945 case T_ID:
1946 if (group <= 1)
1948 flags |= MATCH;
1949 m = proto->tp;
1950 e = proto->ip;
1952 continue;
1953 default:
1954 continue;
1956 break;
1958 op = strcopy(op, ")");
1959 flags |= VARIADIC2;
1960 proto->line = line;
1961 call = 0;
1962 break;
1964 /*FALLTHROUGH*/
1965 case T_ID:
1966 fsm_id:
1967 #if PROTOMAIN
1968 if (flags & CLASSIC)
1970 if (!args && paren == 1) args++;
1971 break;
1973 #endif
1974 if (paren == 0)
1976 if (last == ')')
1978 if (proto->brace == 0 && !(flags & DECLARE)) flags |= SKIP;
1979 call = !call;
1981 else if ((flags & SKIP) || c == T_ID || c == T_VOID) call++;
1982 else flags |= SKIP;
1983 if (last == T_ID) flags |= IDID;
1985 c = T_ID;
1986 flags |= TOKENS;
1987 break;
1988 case T_INVALID:
1989 if (*proto->tp >= '0' && *proto->tp <= '9')
1991 n = 0;
1992 for (;; op--)
1994 switch (*(op - 1))
1996 case 'f':
1997 case 'F':
1998 t = op;
1999 while ((c = *--t) >= '0' && c <= '9' || c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z');
2000 if (*t == '.')
2001 op--;
2002 n = 0;
2003 break;
2004 case 'l':
2005 case 'L':
2006 if (!(n & 01))
2008 n |= 01;
2009 t = op;
2010 while ((c = *--t) >= '0' && c <= '9' || c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z');
2011 if (*t == '.')
2013 n = 0;
2014 op--;
2015 break;
2018 continue;
2019 case 'u':
2020 case 'U':
2021 n |= 02;
2022 continue;
2024 break;
2026 if (n & 01)
2027 *op++ = 'L';
2028 if (n & 02)
2030 m = op;
2031 t = op = m + 10;
2032 while ((c = *--m) >= '0' && c <= '9' || c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z')
2033 *--t = c;
2034 c = *t;
2035 strcopy(m + 1, "(unsigned)");
2036 *t = c;
2037 break;
2040 goto fsm_other;
2041 #if PROTOMAIN
2042 case '[':
2043 if ((flags & CLASSIC) && paren == 0 && group <= 2) flags |= SKIP;
2044 /*FALLTHROUGH*/
2045 #endif
2046 default:
2047 fsm_other:
2048 #if PROTOMAIN
2049 if (flags & CLASSIC) break;
2050 #endif
2051 flags |= TOKENS;
2052 if (paren == 0) flags |= OTHER;
2053 break;
2055 else if (c == '#' && *ip != '(') flags |= SHARP;
2056 last = c;
2057 #if PROTOMAIN
2058 if ((flags & (EXTERN|MATCH)) == (EXTERN|MATCH) && ((flags & (DIRECTIVE|SKIP)) || proto->brace || c != '(' && c != ')' && c != '*' && c != T_ID))
2059 CACHEOUT();
2060 else
2061 #endif
2062 SYNCOUT();
2063 goto fsm_start;
2065 else if (flags & (INIT_DEFINE|INIT_INCLUDE))
2067 #if PROTOMAIN
2068 if ((flags & YACC) && c == '%' && *ip == '{') t = 0;
2069 else
2070 #endif
2072 if (c == '#') for (t = ip; *t == ' ' || *t == '\t'; t++);
2073 else t = "";
2074 if (*t++ == 'i' && *t++ == 'f' && *t++ == 'n' && *t++ == 'd' && *t++ == 'e' && *t++ == 'f')
2076 #if !PROTOMAIN
2077 while (*t == ' ' || *t == '\t') t++;
2078 if (*t != '_')
2079 #endif
2080 t = 0;
2083 if (t)
2085 ip = bp;
2086 op = proto->op;
2088 else while (*ip != '\n') *op++ = *ip++;
2089 op = init(proto, op, flags);
2090 op = linesync(proto, op, proto->line);
2091 flags &= ~(INIT_DEFINE|INIT_INCLUDE);
2092 proto->flags &= ~(INIT_DEFINE|INIT_INCLUDE);
2093 goto fsm_start;
2095 SYNC();
2096 return c;
2100 * close a proto buffer stream
2103 void
2104 pppclose(char* iob)
2106 register struct proto* proto = (struct proto*)(iob - sizeof(struct proto));
2108 if (proto->flags & MORE) close(proto->fd);
2109 free((char*)proto); /* some ANSI cc's botch the free() prototype */
2113 * open a new proto buffer stream
2114 * read buffer pointer returned
2115 * 0 returned on error or if no magic
2117 * file !=0 file path to open, otherwise use fd
2118 * fd open file fd if file==0
2119 * notice !=0 copyright notice info commented at the top
2120 * options !=0 additional notice name=value pairs, space or ; separated
2121 * package !=0 generate header for this package
2124 char*
2125 pppopen(char* file, int fd, char* notice, char* options, char* package, char* comment, int flags)
2127 register struct proto* proto;
2128 register char* iob;
2129 register long n;
2130 register char* s;
2131 int pragma;
2132 char* b;
2133 #if PROTOMAIN
2134 int comlen;
2135 char com[80];
2136 #endif
2137 int m = 0;
2139 static int retain;
2142 * initialize proto
2145 #if PROTOMAIN
2146 if (flags & PROTO_CLASSIC) flags &= ~PROTO_INCLUDE;
2147 #endif
2148 if (flags & PROTO_RETAIN) flags &= ~retain;
2149 else retain &= PROTO_INITIALIZED;
2150 if (file && (fd = open(file, O_RDONLY)) < 0) return 0;
2151 #if !PROTOMAIN
2152 if ((n = lseek(fd, 0L, 2)) > 0)
2154 if (lseek(fd, 0L, 0)) return 0;
2155 if (n < CHUNK) n = CHUNK;
2156 else if (n > 2 * BLOCK) n = 0;
2157 m = 1;
2159 if (n > 0)
2162 * file read in one chunk
2165 if (!(proto = newof(0, struct proto, 1, 4 * n + 2)))
2166 return 0;
2167 proto->iz = n;
2168 proto->oz = 3 * n;
2169 n = 0;
2171 else
2172 #endif
2175 * file read in BLOCK chunks
2178 n = BLOCK;
2179 if (!(proto = newof(0, struct proto, 1, 5 * n + 2)))
2180 return 0;
2181 proto->iz = n;
2182 proto->oz = 3 * n;
2183 proto->flags |= MORE;
2185 proto->fd = fd;
2186 proto->package = package;
2187 iob = (char*)proto + sizeof(struct proto);
2188 proto->op = proto->ob = iob;
2189 proto->ip = proto->ib = iob + proto->oz + n;
2190 if (m) proto->options |= REGULAR;
2191 if (!comment)
2192 comment = "/*";
2193 if (!(proto->cc[0] = comment[0]))
2194 notice = options = 0;
2195 else if (comment[1])
2197 proto->cc[1] = comment[1];
2198 proto->cc[2] = comment[2] ? comment[2] : comment[0];
2200 else
2201 proto->cc[1] = proto->cc[2] = comment[0];
2204 * read the first chunk
2207 n = read(fd, proto->ip, proto->iz);
2208 if (!(proto->flags & MORE))
2209 close(fd);
2210 if (n < 0)
2212 pppclose(iob);
2213 return 0;
2215 *(proto->ip + n) = 0;
2218 * check for proto pragma in first block of lines
2219 * pragma blanked out if found
2221 * -1 no pragma
2222 * 0 #pragma noprototyped
2223 * 1 #pragma prototyped
2225 * NOTE: matches may occur inside comments and quotes
2228 #if PROTOMAIN
2229 if (!notice && !options || (comlen = astlicense(com, sizeof(com), NiL, "type=check", proto->cc[0], proto->cc[1], proto->cc[2])) <= 0)
2230 *com = 0;
2231 #endif
2232 pragma = -1;
2233 s = proto->ip;
2234 m = MAGICTOP;
2235 while (m-- > 0 && *s)
2237 while (*s == ' ' || *s == '\t') s++;
2238 if (*s == '#')
2240 b = s++;
2241 while (*s == ' ' || *s == '\t') s++;
2242 if (!strncmp(s, MAGICDIR, sizeof(MAGICDIR) - 1) && (*(s += sizeof(MAGICDIR) - 1) == ' ' || *s == '\t'))
2244 while (*s == ' ' || *s == '\t') s++;
2245 if (*s == 'n' && *(s + 1) == 'o')
2247 s += 2;
2248 pragma = -2;
2250 if (!strncmp(s, MAGICARG, sizeof(MAGICARG) - 1) && (*(s += sizeof(MAGICARG) - 1) == ' ' || *s == '\t' || *s == '\n' || *s == '\r'))
2251 while (*s)
2253 if ((*(s - 1) == ' ' || *(s - 1) == '\t') && *s == *MAGICOFF && !strncmp(s, MAGICOFF, sizeof(MAGICOFF) - 1))
2254 notice = options = 0;
2255 if (*s++ == '\n')
2257 pragma += 2;
2258 #if PROTOMAIN
2259 if (!(flags & PROTO_DISABLE) || (flags & PROTO_NOPRAGMA))
2260 #endif
2261 for (s--; b < s; *b++ = ' ');
2262 goto magic;
2265 pragma = -1;
2268 else if (*s == '/' && !strncmp(s, MAGICGEN, sizeof(MAGICGEN) - 1))
2270 pragma = 0;
2271 break;
2273 #if PROTOMAIN
2274 else if (*s == '%' && *(s + 1) == '{')
2275 proto->flags |= YACC;
2276 if (notice || options)
2278 if (*s == *com && !strncmp(s, com, comlen))
2279 notice = options = 0;
2280 else
2281 while (*s)
2283 if (*s == *NOTICED && !strncmp(s, NOTICED, sizeof(NOTICED) - 1))
2285 s += sizeof(NOTICED) - 1;
2286 while (*s == ' ' || *s == '\t')
2287 s++;
2288 if (*s == '(' && (*(s + 1) == 'c' || *(s + 1) == 'C') && *(s + 2) == ')' || *s >= '0' && *s <= '9' && *(s + 1) >= '0' && *(s + 1) <= '9')
2290 notice = options = 0;
2291 break;
2294 if (*s == *PUBLICDOMAIN && !strncmp(s, PUBLICDOMAIN, sizeof(PUBLICDOMAIN) - 1))
2296 notice = options = 0;
2297 break;
2299 else if (*s++ == '\n')
2301 s--;
2302 break;
2306 #endif
2307 while (*s && *s++ != '\n');
2309 magic:
2310 if (flags & PROTO_PLUSPLUS) proto->flags |= PLUSPLUS;
2311 if (flags & PROTO_TEST) proto->test = 1;
2312 if (flags & PROTO_EXTERNALIZE) proto->options |= EXTERNALIZE;
2313 #if PROTOMAIN
2314 if (flags & PROTO_CLASSIC) pragma = -pragma;
2315 if (flags & PROTO_DISABLE) pragma = 0;
2316 if (flags & PROTO_LINESYNC) proto->flags |= LINESYNC;
2317 if (!(proto->flags & YACC) && file && (m = strlen(file)) > 2 && file[--m] == 'y' && file[--m] == '.')
2318 proto->flags |= YACC;
2319 #endif
2320 if (pragma <= 0)
2322 if (flags & PROTO_PLUSPLUS)
2324 flags &= ~(PROTO_HEADER|PROTO_INCLUDE);
2325 proto->flags |= PLUSONLY;
2327 else if (!(flags & (PROTO_FORCE|PROTO_PASS)))
2329 pppclose(iob);
2330 return 0;
2332 else if ((flags & (PROTO_FORCE|PROTO_PASS)) == PROTO_PASS || !pragma)
2334 proto->flags |= PASS;
2335 if (proto->flags & MORE)
2336 proto->oz += proto->iz;
2337 proto->iz = n;
2338 if (notice || options)
2340 if (proto->cc[0] == '#' && proto->ip[0] == '#' && proto->ip[1] == '!')
2342 s = proto->ip;
2343 while (*s && *s++ != '\n');
2344 m = s - proto->ip;
2345 proto->op = memcopy(proto->op, proto->ip, m);
2346 proto->ip = s;
2347 proto->iz = n -= m;
2349 #if PROTOMAIN
2350 if (proto->cc[0])
2352 if ((comlen = astlicense(proto->op, proto->oz, notice, options, proto->cc[0], proto->cc[1], proto->cc[2])) < 0)
2353 proto_error((char*)proto + sizeof(struct proto), 1, proto->op, NiL);
2354 else
2355 proto->op += comlen;
2357 if (!(flags & PROTO_CLASSIC) && !(proto->flags & YACC))
2358 #endif
2359 proto->op = linesync(proto, proto->op, 1);
2360 proto->iz += proto->op - proto->ob;
2362 memcopy(proto->op, proto->ip, n);
2363 return iob;
2366 #if PROTOMAIN
2367 if (!(retain & PROTO_INITIALIZED))
2369 retain |= PROTO_INITIALIZED;
2370 ppfsm(FSM_INIT, NiL);
2372 #endif
2373 proto->line = 1;
2374 #if CHUNK >= 512
2375 if (notice || options || (flags & (PROTO_HEADER|PROTO_INCLUDE)))
2377 #if PROTOMAIN
2378 if (notice || options)
2380 if ((comlen = astlicense(proto->op, proto->oz, notice, options, proto->cc[0], proto->cc[1], proto->cc[2])) < 0)
2381 proto_error((char*)proto + sizeof(struct proto), 1, proto->op, NiL);
2382 else
2383 proto->op += comlen;
2385 #endif
2386 if (flags & PROTO_INCLUDE)
2388 proto->flags |= INIT_INCLUDE;
2389 if (flags & PROTO_RETAIN)
2390 retain |= PROTO_INCLUDE;
2392 else if (flags & PROTO_HEADER)
2394 if (flags & PROTO_RETAIN) retain |= PROTO_HEADER;
2395 #if PROTOMAIN
2396 if (flags & PROTO_CLASSIC)
2398 *proto->op++ = '#';
2399 proto->op = strcopy(proto->op, MAGICDIR);
2400 *proto->op++ = ' ';
2401 proto->op = strcopy(proto->op, MAGICARG);
2402 *proto->op++ = '\n';
2404 else
2405 #endif
2406 proto->flags |= INIT_DEFINE;
2408 #if PROTOMAIN
2409 if (!(flags & PROTO_CLASSIC))
2411 if (proto->flags & YACC)
2413 proto->op = strcopy(proto->op, "\n%{\n" + !notice);
2414 proto->op = strcopy(proto->op, MAGICGEN);
2415 proto->op = strcopy(proto->op, "%}\n");
2417 else
2419 if (n || notice || options)
2420 *proto->op++ = '\n';
2421 proto->op = strcopy(proto->op, MAGICGEN);
2422 if (n)
2423 proto->op = linesync(proto, proto->op, proto->line);
2424 else if (proto->flags & (INIT_DEFINE|INIT_INCLUDE))
2425 proto->op = init(proto, proto->op, proto->flags);
2428 #endif
2430 #endif
2431 #if PROTOMAIN
2432 proto->file = file;
2433 if (flags & PROTO_CLASSIC)
2435 proto->flags |= CLASSIC;
2436 if (!(flags & PROTO_HEADER)) proto->flags |= EXTERN;
2438 #endif
2439 return iob;
2443 * read next proto'd chunk into iob
2444 * the chunk is 0 terminated and its size is returned
2448 pppread(char* iob)
2450 register struct proto* proto = (struct proto*)(iob - sizeof(struct proto));
2451 register int n;
2453 if (proto->flags & PASS)
2455 if (proto->iz)
2457 n = proto->iz;
2458 proto->iz = 0;
2460 else if (!(proto->flags & MORE)) n = 0;
2461 else if ((n = read(proto->fd, proto->ob, proto->oz)) <= 0 || (proto->options & REGULAR) && n < proto->oz)
2463 proto->flags &= ~MORE;
2464 close(proto->fd);
2467 else
2469 if (proto->op == proto->ob)
2471 if (proto->flags & ERROR) return -1;
2472 #if PROTOMAIN
2473 if (proto->flags & YACC)
2475 register char* ip = proto->ip;
2476 register char* op = proto->ob;
2477 register char* ep = proto->ob + proto->oz - 2;
2479 if (!*ip)
2481 ip = proto->ip = proto->ib;
2482 if (!(proto->flags & MORE)) n = 0;
2483 else if ((n = read(proto->fd, ip, proto->iz)) <= 0 || (proto->options & REGULAR) && n < proto->iz)
2485 if (n < 0) n = 0;
2486 proto->flags &= ~MORE;
2487 close(proto->fd);
2489 ip[n] = 0;
2491 if (proto->flags & YACCSPLIT)
2493 proto->flags &= ~YACCSPLIT;
2494 if (*ip == '%')
2496 *op++ = *ip++;
2497 if (proto->flags & YACC2) proto->flags &= ~YACC;
2498 else proto->flags |= YACC2;
2501 if (proto->flags & YACC)
2502 while (op < ep && (n = *op++ = *ip))
2504 ip++;
2505 if (n == '%')
2507 if (*ip == '%' && (ip == proto->ip + 1 || *(ip - 2) == '\n'))
2509 *op++ = *ip++;
2510 if (proto->flags & YACC2) proto->flags &= ~YACC;
2511 else proto->flags |= YACC2;
2512 break;
2514 if (!*ip)
2516 *op++ = '%';
2517 proto->flags |= YACCSPLIT;
2518 break;
2521 else if (n == '\n') proto->line++;
2523 proto->op = memcopy(proto->ob, proto->ip, ip - proto->ip);
2524 proto->ip = ip;
2526 else
2527 #endif
2528 lex(proto, proto->flags);
2529 if ((proto->flags & (ERROR|MORE)) == ERROR)
2530 proto->op = strcopy(proto->op, "/* NOTE: some constructs may not have been converted */\n");
2532 n = proto->op - proto->ob;
2533 proto->op = proto->ob;
2535 return n;
2538 #if !PROTOMAIN
2541 * drop control of iob after first pppread()
2542 * return value is input fd
2543 * if fd<0 then all data in iob
2547 pppdrop(char* iob)
2549 register struct proto* proto = (struct proto*)(iob - sizeof(struct proto));
2551 if (proto->flags & MORE)
2553 proto->flags &= ~MORE;
2554 return proto->fd;
2556 return -1;
2559 #endif