xref: /freebsd/contrib/unifdef/unifdef.c (revision c63af363c2458aebc30c01cd0b93b4b902580019)
1fb3ef04dSDag-Erling Smørgrav /*
2fb3ef04dSDag-Erling Smørgrav  * Copyright (c) 2002 - 2020 Tony Finch <dot@dotat.at>
3fb3ef04dSDag-Erling Smørgrav  *
4fb3ef04dSDag-Erling Smørgrav  * Redistribution and use in source and binary forms, with or without
5fb3ef04dSDag-Erling Smørgrav  * modification, are permitted provided that the following conditions
6fb3ef04dSDag-Erling Smørgrav  * are met:
7fb3ef04dSDag-Erling Smørgrav  * 1. Redistributions of source code must retain the above copyright
8fb3ef04dSDag-Erling Smørgrav  *    notice, this list of conditions and the following disclaimer.
9fb3ef04dSDag-Erling Smørgrav  * 2. Redistributions in binary form must reproduce the above copyright
10fb3ef04dSDag-Erling Smørgrav  *    notice, this list of conditions and the following disclaimer in the
11fb3ef04dSDag-Erling Smørgrav  *    documentation and/or other materials provided with the distribution.
12fb3ef04dSDag-Erling Smørgrav  *
13fb3ef04dSDag-Erling Smørgrav  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14fb3ef04dSDag-Erling Smørgrav  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15fb3ef04dSDag-Erling Smørgrav  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16fb3ef04dSDag-Erling Smørgrav  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17fb3ef04dSDag-Erling Smørgrav  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18fb3ef04dSDag-Erling Smørgrav  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19fb3ef04dSDag-Erling Smørgrav  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20fb3ef04dSDag-Erling Smørgrav  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21fb3ef04dSDag-Erling Smørgrav  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22fb3ef04dSDag-Erling Smørgrav  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23fb3ef04dSDag-Erling Smørgrav  * SUCH DAMAGE.
24fb3ef04dSDag-Erling Smørgrav  */
25fb3ef04dSDag-Erling Smørgrav 
26fb3ef04dSDag-Erling Smørgrav /*
27fb3ef04dSDag-Erling Smørgrav  * unifdef - remove ifdef'ed lines
28fb3ef04dSDag-Erling Smørgrav  *
29fb3ef04dSDag-Erling Smørgrav  * This code was derived from software contributed to Berkeley by Dave Yost.
30fb3ef04dSDag-Erling Smørgrav  * It was rewritten to support ANSI C by Tony Finch. The original version
31fb3ef04dSDag-Erling Smørgrav  * of unifdef carried the 4-clause BSD copyright licence. None of its code
32fb3ef04dSDag-Erling Smørgrav  * remains in this version (though some of the names remain) so it now
33fb3ef04dSDag-Erling Smørgrav  * carries a more liberal licence.
34fb3ef04dSDag-Erling Smørgrav  *
35fb3ef04dSDag-Erling Smørgrav  *  Wishlist:
36fb3ef04dSDag-Erling Smørgrav  *      provide an option which will append the name of the
37fb3ef04dSDag-Erling Smørgrav  *        appropriate symbol after #else's and #endif's
38fb3ef04dSDag-Erling Smørgrav  *      provide an option which will check symbols after
39fb3ef04dSDag-Erling Smørgrav  *        #else's and #endif's to see that they match their
40fb3ef04dSDag-Erling Smørgrav  *        corresponding #ifdef or #ifndef
41fb3ef04dSDag-Erling Smørgrav  *
42fb3ef04dSDag-Erling Smørgrav  *   These require better buffer handling, which would also make
43fb3ef04dSDag-Erling Smørgrav  *   it possible to handle all "dodgy" directives correctly.
44fb3ef04dSDag-Erling Smørgrav  */
45fb3ef04dSDag-Erling Smørgrav 
46fb3ef04dSDag-Erling Smørgrav #include "unifdef.h"
47fb3ef04dSDag-Erling Smørgrav 
48fb3ef04dSDag-Erling Smørgrav static const char copyright[] =
49fb3ef04dSDag-Erling Smørgrav     #include "version.h"
50fb3ef04dSDag-Erling Smørgrav     "@(#) $Author: Tony Finch (dot@dotat.at) $\n"
51fb3ef04dSDag-Erling Smørgrav     "@(#) $URL: http://dotat.at/prog/unifdef $\n"
52fb3ef04dSDag-Erling Smørgrav ;
53fb3ef04dSDag-Erling Smørgrav 
54fb3ef04dSDag-Erling Smørgrav /* types of input lines: */
55fb3ef04dSDag-Erling Smørgrav typedef enum {
56fb3ef04dSDag-Erling Smørgrav 	LT_TRUEI,		/* a true #if with ignore flag */
57fb3ef04dSDag-Erling Smørgrav 	LT_FALSEI,		/* a false #if with ignore flag */
58fb3ef04dSDag-Erling Smørgrav 	LT_IF,			/* an unknown #if */
59fb3ef04dSDag-Erling Smørgrav 	LT_TRUE,		/* a true #if */
60fb3ef04dSDag-Erling Smørgrav 	LT_FALSE,		/* a false #if */
61fb3ef04dSDag-Erling Smørgrav 	LT_ELIF,		/* an unknown #elif */
62fb3ef04dSDag-Erling Smørgrav 	LT_ELTRUE,		/* a true #elif */
63fb3ef04dSDag-Erling Smørgrav 	LT_ELFALSE,		/* a false #elif */
64fb3ef04dSDag-Erling Smørgrav 	LT_ELSE,		/* #else */
65fb3ef04dSDag-Erling Smørgrav 	LT_ENDIF,		/* #endif */
66fb3ef04dSDag-Erling Smørgrav 	LT_DODGY,		/* flag: directive is not on one line */
67fb3ef04dSDag-Erling Smørgrav 	LT_DODGY_LAST = LT_DODGY + LT_ENDIF,
68fb3ef04dSDag-Erling Smørgrav 	LT_PLAIN,		/* ordinary line */
69fb3ef04dSDag-Erling Smørgrav 	LT_EOF,			/* end of file */
70fb3ef04dSDag-Erling Smørgrav 	LT_ERROR,		/* unevaluable #if */
71fb3ef04dSDag-Erling Smørgrav 	LT_COUNT
72fb3ef04dSDag-Erling Smørgrav } Linetype;
73fb3ef04dSDag-Erling Smørgrav 
74fb3ef04dSDag-Erling Smørgrav static char const * const linetype_name[] = {
75fb3ef04dSDag-Erling Smørgrav 	"TRUEI", "FALSEI", "IF", "TRUE", "FALSE",
76fb3ef04dSDag-Erling Smørgrav 	"ELIF", "ELTRUE", "ELFALSE", "ELSE", "ENDIF",
77fb3ef04dSDag-Erling Smørgrav 	"DODGY TRUEI", "DODGY FALSEI",
78fb3ef04dSDag-Erling Smørgrav 	"DODGY IF", "DODGY TRUE", "DODGY FALSE",
79fb3ef04dSDag-Erling Smørgrav 	"DODGY ELIF", "DODGY ELTRUE", "DODGY ELFALSE",
80fb3ef04dSDag-Erling Smørgrav 	"DODGY ELSE", "DODGY ENDIF",
81fb3ef04dSDag-Erling Smørgrav 	"PLAIN", "EOF", "ERROR"
82fb3ef04dSDag-Erling Smørgrav };
83fb3ef04dSDag-Erling Smørgrav 
84fb3ef04dSDag-Erling Smørgrav #define linetype_if2elif(lt) ((Linetype)(lt - LT_IF + LT_ELIF))
85fb3ef04dSDag-Erling Smørgrav #define linetype_2dodgy(lt) ((Linetype)(lt + LT_DODGY))
86fb3ef04dSDag-Erling Smørgrav 
87fb3ef04dSDag-Erling Smørgrav /* state of #if processing */
88fb3ef04dSDag-Erling Smørgrav typedef enum {
89fb3ef04dSDag-Erling Smørgrav 	IS_OUTSIDE,
90fb3ef04dSDag-Erling Smørgrav 	IS_FALSE_PREFIX,	/* false #if followed by false #elifs */
91fb3ef04dSDag-Erling Smørgrav 	IS_TRUE_PREFIX,		/* first non-false #(el)if is true */
92fb3ef04dSDag-Erling Smørgrav 	IS_PASS_MIDDLE,		/* first non-false #(el)if is unknown */
93fb3ef04dSDag-Erling Smørgrav 	IS_FALSE_MIDDLE,	/* a false #elif after a pass state */
94fb3ef04dSDag-Erling Smørgrav 	IS_TRUE_MIDDLE,		/* a true #elif after a pass state */
95fb3ef04dSDag-Erling Smørgrav 	IS_PASS_ELSE,		/* an else after a pass state */
96fb3ef04dSDag-Erling Smørgrav 	IS_FALSE_ELSE,		/* an else after a true state */
97fb3ef04dSDag-Erling Smørgrav 	IS_TRUE_ELSE,		/* an else after only false states */
98fb3ef04dSDag-Erling Smørgrav 	IS_FALSE_TRAILER,	/* #elifs after a true are false */
99fb3ef04dSDag-Erling Smørgrav 	IS_COUNT
100fb3ef04dSDag-Erling Smørgrav } Ifstate;
101fb3ef04dSDag-Erling Smørgrav 
102fb3ef04dSDag-Erling Smørgrav static char const * const ifstate_name[] = {
103fb3ef04dSDag-Erling Smørgrav 	"OUTSIDE", "FALSE_PREFIX", "TRUE_PREFIX",
104fb3ef04dSDag-Erling Smørgrav 	"PASS_MIDDLE", "FALSE_MIDDLE", "TRUE_MIDDLE",
105fb3ef04dSDag-Erling Smørgrav 	"PASS_ELSE", "FALSE_ELSE", "TRUE_ELSE",
106fb3ef04dSDag-Erling Smørgrav 	"FALSE_TRAILER"
107fb3ef04dSDag-Erling Smørgrav };
108fb3ef04dSDag-Erling Smørgrav 
109fb3ef04dSDag-Erling Smørgrav /* state of comment parser */
110fb3ef04dSDag-Erling Smørgrav typedef enum {
111fb3ef04dSDag-Erling Smørgrav 	NO_COMMENT = false,	/* outside a comment */
112fb3ef04dSDag-Erling Smørgrav 	C_COMMENT,		/* in a comment like this one */
113fb3ef04dSDag-Erling Smørgrav 	CXX_COMMENT,		/* between // and end of line */
114fb3ef04dSDag-Erling Smørgrav 	STARTING_COMMENT,	/* just after slash-backslash-newline */
115fb3ef04dSDag-Erling Smørgrav 	FINISHING_COMMENT,	/* star-backslash-newline in a C comment */
116fb3ef04dSDag-Erling Smørgrav 	CHAR_LITERAL,		/* inside '' */
117fb3ef04dSDag-Erling Smørgrav 	STRING_LITERAL,		/* inside "" */
118fb3ef04dSDag-Erling Smørgrav 	RAW_STRING_LITERAL	/* inside R"()" */
119fb3ef04dSDag-Erling Smørgrav } Comment_state;
120fb3ef04dSDag-Erling Smørgrav 
121fb3ef04dSDag-Erling Smørgrav static char const * const comment_name[] = {
122fb3ef04dSDag-Erling Smørgrav 	"NO", "C", "CXX", "STARTING", "FINISHING", "CHAR", "STRING"
123fb3ef04dSDag-Erling Smørgrav };
124fb3ef04dSDag-Erling Smørgrav 
125fb3ef04dSDag-Erling Smørgrav /* state of preprocessor line parser */
126fb3ef04dSDag-Erling Smørgrav typedef enum {
127fb3ef04dSDag-Erling Smørgrav 	LS_START,		/* only space and comments on this line */
128fb3ef04dSDag-Erling Smørgrav 	LS_HASH,		/* only space, comments, and a hash */
129fb3ef04dSDag-Erling Smørgrav 	LS_DIRTY		/* this line can't be a preprocessor line */
130fb3ef04dSDag-Erling Smørgrav } Line_state;
131fb3ef04dSDag-Erling Smørgrav 
132fb3ef04dSDag-Erling Smørgrav static char const * const linestate_name[] = {
133fb3ef04dSDag-Erling Smørgrav 	"START", "HASH", "DIRTY"
134fb3ef04dSDag-Erling Smørgrav };
135fb3ef04dSDag-Erling Smørgrav 
136fb3ef04dSDag-Erling Smørgrav /*
137fb3ef04dSDag-Erling Smørgrav  * Minimum translation limits from ISO/IEC 9899:1999 5.2.4.1
138fb3ef04dSDag-Erling Smørgrav  */
139fb3ef04dSDag-Erling Smørgrav #define	MAXDEPTH        64			/* maximum #if nesting */
140fb3ef04dSDag-Erling Smørgrav #define	MAXLINE         4096			/* maximum length of line */
141fb3ef04dSDag-Erling Smørgrav 
142fb3ef04dSDag-Erling Smørgrav /*
143fb3ef04dSDag-Erling Smørgrav  * Sometimes when editing a keyword the replacement text is longer, so
144fb3ef04dSDag-Erling Smørgrav  * we leave some space at the end of the tline buffer to accommodate this.
145fb3ef04dSDag-Erling Smørgrav  */
146fb3ef04dSDag-Erling Smørgrav #define	EDITSLOP        10
147fb3ef04dSDag-Erling Smørgrav 
148fb3ef04dSDag-Erling Smørgrav /*
149343b776fSDag-Erling Smørgrav  * C17/18 allow 63 characters per macro name, but up to 127 arbitrarily large
150343b776fSDag-Erling Smørgrav  * parameters.
151343b776fSDag-Erling Smørgrav  */
152343b776fSDag-Erling Smørgrav struct macro {
153343b776fSDag-Erling Smørgrav 	RB_ENTRY(macro)	entry;
154343b776fSDag-Erling Smørgrav 	const char	*name;
155343b776fSDag-Erling Smørgrav 	const char	*value;
156343b776fSDag-Erling Smørgrav 	bool		ignore;		/* -iDsym or -iUsym */
157343b776fSDag-Erling Smørgrav };
158343b776fSDag-Erling Smørgrav 
159343b776fSDag-Erling Smørgrav static int
macro_cmp(struct macro * a,struct macro * b)160343b776fSDag-Erling Smørgrav macro_cmp(struct macro *a, struct macro *b)
161343b776fSDag-Erling Smørgrav {
162343b776fSDag-Erling Smørgrav 	return (strcmp(a->name, b->name));
163343b776fSDag-Erling Smørgrav }
164343b776fSDag-Erling Smørgrav 
165343b776fSDag-Erling Smørgrav static RB_HEAD(MACROMAP, macro) macro_tree = RB_INITIALIZER(&macro_tree);
166343b776fSDag-Erling Smørgrav RB_GENERATE_STATIC(MACROMAP, macro, entry, macro_cmp);
167343b776fSDag-Erling Smørgrav 
168343b776fSDag-Erling Smørgrav /*
169fb3ef04dSDag-Erling Smørgrav  * Globals.
170fb3ef04dSDag-Erling Smørgrav  */
171fb3ef04dSDag-Erling Smørgrav 
172fb3ef04dSDag-Erling Smørgrav static bool             compblank;		/* -B: compress blank lines */
173fb3ef04dSDag-Erling Smørgrav static bool             lnblank;		/* -b: blank deleted lines */
174fb3ef04dSDag-Erling Smørgrav static bool             complement;		/* -c: do the complement */
175fb3ef04dSDag-Erling Smørgrav static bool             debugging;		/* -d: debugging reports */
176fb3ef04dSDag-Erling Smørgrav static bool             inplace;		/* -m: modify in place */
177fb3ef04dSDag-Erling Smørgrav static bool             iocccok;		/* -e: fewer IOCCC errors */
178fb3ef04dSDag-Erling Smørgrav static bool             strictlogic;		/* -K: keep ambiguous #ifs */
179fb3ef04dSDag-Erling Smørgrav static bool             killconsts;		/* -k: eval constant #ifs */
180fb3ef04dSDag-Erling Smørgrav static bool             lnnum;			/* -n: add #line directives */
181fb3ef04dSDag-Erling Smørgrav static bool             symlist;		/* -s: output symbol list */
182fb3ef04dSDag-Erling Smørgrav static bool             symdepth;		/* -S: output symbol depth */
183fb3ef04dSDag-Erling Smørgrav static bool             text;			/* -t: this is a text file */
184fb3ef04dSDag-Erling Smørgrav 
185fb3ef04dSDag-Erling Smørgrav static FILE            *input;			/* input file pointer */
186fb3ef04dSDag-Erling Smørgrav static const char      *filename;		/* input file name */
187fb3ef04dSDag-Erling Smørgrav static int              linenum;		/* current line number */
188fb3ef04dSDag-Erling Smørgrav static const char      *linefile;		/* file name for #line */
189fb3ef04dSDag-Erling Smørgrav static FILE            *output;			/* output file pointer */
190fb3ef04dSDag-Erling Smørgrav static const char      *ofilename;		/* output file name */
191fb3ef04dSDag-Erling Smørgrav static const char      *backext;		/* backup extension */
192fb3ef04dSDag-Erling Smørgrav static char            *tempname;		/* avoid splatting input */
193fb3ef04dSDag-Erling Smørgrav 
194fb3ef04dSDag-Erling Smørgrav static char             tline[MAXLINE+EDITSLOP];/* input buffer plus space */
195fb3ef04dSDag-Erling Smørgrav static char            *keyword;		/* used for editing #elif's */
196fb3ef04dSDag-Erling Smørgrav 
197fb3ef04dSDag-Erling Smørgrav /*
198fb3ef04dSDag-Erling Smørgrav  * When processing a file, the output's newline style will match the
199fb3ef04dSDag-Erling Smørgrav  * input's, and unifdef correctly handles CRLF or LF endings whatever
200fb3ef04dSDag-Erling Smørgrav  * the platform's native style. The stdio streams are opened in binary
201fb3ef04dSDag-Erling Smørgrav  * mode to accommodate platforms whose native newline style is CRLF.
202fb3ef04dSDag-Erling Smørgrav  * When the output isn't a processed input file (when it is error /
203fb3ef04dSDag-Erling Smørgrav  * debug / diagnostic messages) then unifdef uses native line endings.
204fb3ef04dSDag-Erling Smørgrav  */
205fb3ef04dSDag-Erling Smørgrav 
206fb3ef04dSDag-Erling Smørgrav static const char      *newline;		/* input file format */
207fb3ef04dSDag-Erling Smørgrav static const char       newline_unix[] = "\n";
208fb3ef04dSDag-Erling Smørgrav static const char       newline_crlf[] = "\r\n";
209fb3ef04dSDag-Erling Smørgrav 
210fb3ef04dSDag-Erling Smørgrav static Comment_state    incomment;		/* comment parser state */
211fb3ef04dSDag-Erling Smørgrav static Line_state       linestate;		/* #if line parser state */
212fb3ef04dSDag-Erling Smørgrav static Ifstate          ifstate[MAXDEPTH];	/* #if processor state */
213fb3ef04dSDag-Erling Smørgrav static bool             ignoring[MAXDEPTH];	/* ignore comments state */
214fb3ef04dSDag-Erling Smørgrav static int              stifline[MAXDEPTH];	/* start of current #if */
215fb3ef04dSDag-Erling Smørgrav static int              depth;			/* current #if nesting */
216fb3ef04dSDag-Erling Smørgrav static int              delcount;		/* count of deleted lines */
217fb3ef04dSDag-Erling Smørgrav static unsigned         blankcount;		/* count of blank lines */
218fb3ef04dSDag-Erling Smørgrav static unsigned         blankmax;		/* maximum recent blankcount */
219fb3ef04dSDag-Erling Smørgrav static bool             constexpr;		/* constant #if expression */
220fb3ef04dSDag-Erling Smørgrav static bool             zerosyms;		/* to format symdepth output */
221fb3ef04dSDag-Erling Smørgrav static bool             firstsym;		/* ditto */
222fb3ef04dSDag-Erling Smørgrav 
223fb3ef04dSDag-Erling Smørgrav static int              exitmode;		/* exit status mode */
224fb3ef04dSDag-Erling Smørgrav static int              exitstat;		/* program exit status */
225fb3ef04dSDag-Erling Smørgrav static bool             altered;		/* was this file modified? */
226fb3ef04dSDag-Erling Smørgrav 
227fb3ef04dSDag-Erling Smørgrav static void             addsym1(bool, bool, char *);
228fb3ef04dSDag-Erling Smørgrav static void             addsym2(bool, const char *, const char *);
229fb3ef04dSDag-Erling Smørgrav static char            *astrcat(const char *, const char *);
230fb3ef04dSDag-Erling Smørgrav static void             cleantemp(void);
231fb3ef04dSDag-Erling Smørgrav static void             closeio(void);
232fb3ef04dSDag-Erling Smørgrav static void             debug(const char *, ...);
233343b776fSDag-Erling Smørgrav static void             debugsym(const char *, const struct macro *);
234fb3ef04dSDag-Erling Smørgrav static bool             defundef(void);
235fb3ef04dSDag-Erling Smørgrav static void             defundefile(const char *);
236fb3ef04dSDag-Erling Smørgrav static void             done(void);
237fb3ef04dSDag-Erling Smørgrav static void             error(const char *);
238343b776fSDag-Erling Smørgrav static struct macro    *findsym(const char **);
239fb3ef04dSDag-Erling Smørgrav static void             flushline(bool);
240fb3ef04dSDag-Erling Smørgrav static void             hashline(void);
241fb3ef04dSDag-Erling Smørgrav static void             help(void);
242fb3ef04dSDag-Erling Smørgrav static Linetype         ifeval(const char **);
243fb3ef04dSDag-Erling Smørgrav static void             ignoreoff(void);
244fb3ef04dSDag-Erling Smørgrav static void             ignoreon(void);
245fb3ef04dSDag-Erling Smørgrav static void             indirectsym(void);
246fb3ef04dSDag-Erling Smørgrav static void             keywordedit(const char *);
247fb3ef04dSDag-Erling Smørgrav static const char      *matchsym(const char *, const char *);
248fb3ef04dSDag-Erling Smørgrav static void             nest(void);
249fb3ef04dSDag-Erling Smørgrav static Linetype         parseline(void);
250fb3ef04dSDag-Erling Smørgrav static void             process(void);
251fb3ef04dSDag-Erling Smørgrav static void             processinout(const char *, const char *);
252fb3ef04dSDag-Erling Smørgrav static const char      *skipargs(const char *);
253fb3ef04dSDag-Erling Smørgrav static const char      *skipcomment(const char *);
254fb3ef04dSDag-Erling Smørgrav static const char      *skiphash(void);
255fb3ef04dSDag-Erling Smørgrav static const char      *skipline(const char *);
256fb3ef04dSDag-Erling Smørgrav static const char      *skipsym(const char *);
257fb3ef04dSDag-Erling Smørgrav static void             state(Ifstate);
258fb3ef04dSDag-Erling Smørgrav static void             unnest(void);
259fb3ef04dSDag-Erling Smørgrav static void             usage(void);
260fb3ef04dSDag-Erling Smørgrav static void             version(void);
261fb3ef04dSDag-Erling Smørgrav static const char      *xstrdup(const char *, const char *);
262fb3ef04dSDag-Erling Smørgrav 
263fb3ef04dSDag-Erling Smørgrav #define endsym(c) (!isalnum((unsigned char)c) && c != '_')
264fb3ef04dSDag-Erling Smørgrav 
265fb3ef04dSDag-Erling Smørgrav /*
266fb3ef04dSDag-Erling Smørgrav  * The main program.
267fb3ef04dSDag-Erling Smørgrav  */
268fb3ef04dSDag-Erling Smørgrav int
main(int argc,char * argv[])269fb3ef04dSDag-Erling Smørgrav main(int argc, char *argv[])
270fb3ef04dSDag-Erling Smørgrav {
271fb3ef04dSDag-Erling Smørgrav 	int opt;
272fb3ef04dSDag-Erling Smørgrav 
273fb3ef04dSDag-Erling Smørgrav 	while ((opt = getopt(argc, argv, "i:D:U:f:I:M:o:x:bBcdehKklmnsStV")) != -1)
274fb3ef04dSDag-Erling Smørgrav 		switch (opt) {
275fb3ef04dSDag-Erling Smørgrav 		case 'i': /* treat stuff controlled by these symbols as text */
276fb3ef04dSDag-Erling Smørgrav 			/*
277fb3ef04dSDag-Erling Smørgrav 			 * For strict backwards-compatibility the U or D
278fb3ef04dSDag-Erling Smørgrav 			 * should be immediately after the -i but it doesn't
279fb3ef04dSDag-Erling Smørgrav 			 * matter much if we relax that requirement.
280fb3ef04dSDag-Erling Smørgrav 			 */
281fb3ef04dSDag-Erling Smørgrav 			opt = *optarg++;
282fb3ef04dSDag-Erling Smørgrav 			if (opt == 'D')
283fb3ef04dSDag-Erling Smørgrav 				addsym1(true, true, optarg);
284fb3ef04dSDag-Erling Smørgrav 			else if (opt == 'U')
285fb3ef04dSDag-Erling Smørgrav 				addsym1(true, false, optarg);
286fb3ef04dSDag-Erling Smørgrav 			else
287fb3ef04dSDag-Erling Smørgrav 				usage();
288fb3ef04dSDag-Erling Smørgrav 			break;
289fb3ef04dSDag-Erling Smørgrav 		case 'D': /* define a symbol */
290fb3ef04dSDag-Erling Smørgrav 			addsym1(false, true, optarg);
291fb3ef04dSDag-Erling Smørgrav 			break;
292fb3ef04dSDag-Erling Smørgrav 		case 'U': /* undef a symbol */
293fb3ef04dSDag-Erling Smørgrav 			addsym1(false, false, optarg);
294fb3ef04dSDag-Erling Smørgrav 			break;
295fb3ef04dSDag-Erling Smørgrav 		case 'I': /* no-op for compatibility with cpp */
296fb3ef04dSDag-Erling Smørgrav 			break;
297fb3ef04dSDag-Erling Smørgrav 		case 'b': /* blank deleted lines instead of omitting them */
298fb3ef04dSDag-Erling Smørgrav 		case 'l': /* backwards compatibility */
299fb3ef04dSDag-Erling Smørgrav 			lnblank = true;
300fb3ef04dSDag-Erling Smørgrav 			break;
301fb3ef04dSDag-Erling Smørgrav 		case 'B': /* compress blank lines around removed section */
302fb3ef04dSDag-Erling Smørgrav 			compblank = true;
303fb3ef04dSDag-Erling Smørgrav 			break;
304fb3ef04dSDag-Erling Smørgrav 		case 'c': /* treat -D as -U and vice versa */
305fb3ef04dSDag-Erling Smørgrav 			complement = true;
306fb3ef04dSDag-Erling Smørgrav 			break;
307fb3ef04dSDag-Erling Smørgrav 		case 'd':
308fb3ef04dSDag-Erling Smørgrav 			debugging = true;
309fb3ef04dSDag-Erling Smørgrav 			break;
310fb3ef04dSDag-Erling Smørgrav 		case 'e': /* fewer errors from dodgy lines */
311fb3ef04dSDag-Erling Smørgrav 			iocccok = true;
312fb3ef04dSDag-Erling Smørgrav 			break;
313fb3ef04dSDag-Erling Smørgrav 		case 'f': /* definitions file */
314fb3ef04dSDag-Erling Smørgrav 			defundefile(optarg);
315fb3ef04dSDag-Erling Smørgrav 			break;
316fb3ef04dSDag-Erling Smørgrav 		case 'h':
317fb3ef04dSDag-Erling Smørgrav 			help();
318fb3ef04dSDag-Erling Smørgrav 			break;
319fb3ef04dSDag-Erling Smørgrav 		case 'K': /* keep ambiguous #ifs */
320fb3ef04dSDag-Erling Smørgrav 			strictlogic = true;
321fb3ef04dSDag-Erling Smørgrav 			break;
322fb3ef04dSDag-Erling Smørgrav 		case 'k': /* process constant #ifs */
323fb3ef04dSDag-Erling Smørgrav 			killconsts = true;
324fb3ef04dSDag-Erling Smørgrav 			break;
325fb3ef04dSDag-Erling Smørgrav 		case 'm': /* modify in place */
326fb3ef04dSDag-Erling Smørgrav 			inplace = true;
327fb3ef04dSDag-Erling Smørgrav 			break;
328fb3ef04dSDag-Erling Smørgrav 		case 'M': /* modify in place and keep backup */
329fb3ef04dSDag-Erling Smørgrav 			inplace = true;
330fb3ef04dSDag-Erling Smørgrav 			if (strlen(optarg) > 0)
331fb3ef04dSDag-Erling Smørgrav 				backext = optarg;
332fb3ef04dSDag-Erling Smørgrav 			break;
333fb3ef04dSDag-Erling Smørgrav 		case 'n': /* add #line directive after deleted lines */
334fb3ef04dSDag-Erling Smørgrav 			lnnum = true;
335fb3ef04dSDag-Erling Smørgrav 			break;
336fb3ef04dSDag-Erling Smørgrav 		case 'o': /* output to a file */
337fb3ef04dSDag-Erling Smørgrav 			ofilename = optarg;
338fb3ef04dSDag-Erling Smørgrav 			break;
339fb3ef04dSDag-Erling Smørgrav 		case 's': /* only output list of symbols that control #ifs */
340fb3ef04dSDag-Erling Smørgrav 			symlist = true;
341fb3ef04dSDag-Erling Smørgrav 			break;
342fb3ef04dSDag-Erling Smørgrav 		case 'S': /* list symbols with their nesting depth */
343fb3ef04dSDag-Erling Smørgrav 			symlist = symdepth = true;
344fb3ef04dSDag-Erling Smørgrav 			break;
345fb3ef04dSDag-Erling Smørgrav 		case 't': /* don't parse C comments */
346fb3ef04dSDag-Erling Smørgrav 			text = true;
347fb3ef04dSDag-Erling Smørgrav 			break;
348fb3ef04dSDag-Erling Smørgrav 		case 'V':
349fb3ef04dSDag-Erling Smørgrav 			version();
350fb3ef04dSDag-Erling Smørgrav 			break;
351fb3ef04dSDag-Erling Smørgrav 		case 'x':
352fb3ef04dSDag-Erling Smørgrav 			exitmode = atoi(optarg);
353fb3ef04dSDag-Erling Smørgrav 			if(exitmode < 0 || exitmode > 2)
354fb3ef04dSDag-Erling Smørgrav 				usage();
355fb3ef04dSDag-Erling Smørgrav 			break;
356fb3ef04dSDag-Erling Smørgrav 		default:
357fb3ef04dSDag-Erling Smørgrav 			usage();
358fb3ef04dSDag-Erling Smørgrav 		}
359fb3ef04dSDag-Erling Smørgrav 	argc -= optind;
360fb3ef04dSDag-Erling Smørgrav 	argv += optind;
361fb3ef04dSDag-Erling Smørgrav 	if (compblank && lnblank)
362fb3ef04dSDag-Erling Smørgrav 		errx(2, "-B and -b are mutually exclusive");
363fb3ef04dSDag-Erling Smørgrav 	if (symlist && (ofilename != NULL || inplace || argc > 1))
364fb3ef04dSDag-Erling Smørgrav 		errx(2, "-s only works with one input file");
365fb3ef04dSDag-Erling Smørgrav 	if (argc > 1 && ofilename != NULL)
366fb3ef04dSDag-Erling Smørgrav 		errx(2, "-o cannot be used with multiple input files");
367fb3ef04dSDag-Erling Smørgrav 	if (argc > 1 && !inplace)
368fb3ef04dSDag-Erling Smørgrav 		errx(2, "multiple input files require -m or -M");
369fb3ef04dSDag-Erling Smørgrav 	if (argc == 0 && inplace)
370fb3ef04dSDag-Erling Smørgrav 		errx(2, "-m requires an input file");
371fb3ef04dSDag-Erling Smørgrav 	if (argc == 0)
372fb3ef04dSDag-Erling Smørgrav 		argc = 1;
373fb3ef04dSDag-Erling Smørgrav 	if (argc == 1 && !inplace && ofilename == NULL)
374fb3ef04dSDag-Erling Smørgrav 		ofilename = "-";
375fb3ef04dSDag-Erling Smørgrav 	indirectsym();
376fb3ef04dSDag-Erling Smørgrav 
377fb3ef04dSDag-Erling Smørgrav 	atexit(cleantemp);
378fb3ef04dSDag-Erling Smørgrav 	if (ofilename != NULL)
379fb3ef04dSDag-Erling Smørgrav 		processinout(*argv, ofilename);
380fb3ef04dSDag-Erling Smørgrav 	else while (argc-- > 0) {
381fb3ef04dSDag-Erling Smørgrav 		processinout(*argv, *argv);
382fb3ef04dSDag-Erling Smørgrav 		argv++;
383fb3ef04dSDag-Erling Smørgrav 	}
384fb3ef04dSDag-Erling Smørgrav 	switch(exitmode) {
385fb3ef04dSDag-Erling Smørgrav 	case(0): exit(exitstat);
386fb3ef04dSDag-Erling Smørgrav 	case(1): exit(!exitstat);
387fb3ef04dSDag-Erling Smørgrav 	case(2): exit(0);
388fb3ef04dSDag-Erling Smørgrav 	default: abort(); /* bug */
389fb3ef04dSDag-Erling Smørgrav 	}
390fb3ef04dSDag-Erling Smørgrav }
391fb3ef04dSDag-Erling Smørgrav 
392fb3ef04dSDag-Erling Smørgrav /*
393fb3ef04dSDag-Erling Smørgrav  * File logistics.
394fb3ef04dSDag-Erling Smørgrav  */
395fb3ef04dSDag-Erling Smørgrav static void
processinout(const char * ifn,const char * ofn)396fb3ef04dSDag-Erling Smørgrav processinout(const char *ifn, const char *ofn)
397fb3ef04dSDag-Erling Smørgrav {
398fb3ef04dSDag-Erling Smørgrav 	struct stat st;
399fb3ef04dSDag-Erling Smørgrav 
400fb3ef04dSDag-Erling Smørgrav 	if (ifn == NULL || strcmp(ifn, "-") == 0) {
401fb3ef04dSDag-Erling Smørgrav 		filename = "[stdin]";
402fb3ef04dSDag-Erling Smørgrav 		linefile = NULL;
403fb3ef04dSDag-Erling Smørgrav 		input = fbinmode(stdin);
404fb3ef04dSDag-Erling Smørgrav 	} else {
405fb3ef04dSDag-Erling Smørgrav 		filename = ifn;
406fb3ef04dSDag-Erling Smørgrav 		linefile = ifn;
407fb3ef04dSDag-Erling Smørgrav 		input = fopen(ifn, "rb");
408fb3ef04dSDag-Erling Smørgrav 		if (input == NULL)
409fb3ef04dSDag-Erling Smørgrav 			err(2, "can't open %s", ifn);
410fb3ef04dSDag-Erling Smørgrav 	}
411fb3ef04dSDag-Erling Smørgrav 	if (strcmp(ofn, "-") == 0) {
412fb3ef04dSDag-Erling Smørgrav 		output = fbinmode(stdout);
413fb3ef04dSDag-Erling Smørgrav 		process();
414fb3ef04dSDag-Erling Smørgrav 		return;
415fb3ef04dSDag-Erling Smørgrav 	}
416fb3ef04dSDag-Erling Smørgrav 	if (stat(ofn, &st) < 0) {
417fb3ef04dSDag-Erling Smørgrav 		output = fopen(ofn, "wb");
418fb3ef04dSDag-Erling Smørgrav 		if (output == NULL)
419fb3ef04dSDag-Erling Smørgrav 			err(2, "can't create %s", ofn);
420fb3ef04dSDag-Erling Smørgrav 		process();
421fb3ef04dSDag-Erling Smørgrav 		return;
422fb3ef04dSDag-Erling Smørgrav 	}
423fb3ef04dSDag-Erling Smørgrav 
424fb3ef04dSDag-Erling Smørgrav 	tempname = astrcat(ofn, ".XXXXXX");
425fb3ef04dSDag-Erling Smørgrav 	output = mktempmode(tempname, st.st_mode);
426fb3ef04dSDag-Erling Smørgrav 	if (output == NULL)
427fb3ef04dSDag-Erling Smørgrav 		err(2, "can't create %s", tempname);
428fb3ef04dSDag-Erling Smørgrav 
429fb3ef04dSDag-Erling Smørgrav 	process();
430fb3ef04dSDag-Erling Smørgrav 
431fb3ef04dSDag-Erling Smørgrav 	if (backext != NULL) {
432fb3ef04dSDag-Erling Smørgrav 		char *backname = astrcat(ofn, backext);
433fb3ef04dSDag-Erling Smørgrav 		if (rename(ofn, backname) < 0)
434fb3ef04dSDag-Erling Smørgrav 			err(2, "can't rename \"%s\" to \"%s\"", ofn, backname);
435fb3ef04dSDag-Erling Smørgrav 		free(backname);
436fb3ef04dSDag-Erling Smørgrav 	}
437fb3ef04dSDag-Erling Smørgrav 	/* leave file unmodified if unifdef made no changes */
438fb3ef04dSDag-Erling Smørgrav 	if (!altered && backext == NULL) {
439fb3ef04dSDag-Erling Smørgrav 		if (remove(tempname) < 0)
440fb3ef04dSDag-Erling Smørgrav 			warn("can't remove \"%s\"", tempname);
441fb3ef04dSDag-Erling Smørgrav 	} else if (replace(tempname, ofn) < 0)
442fb3ef04dSDag-Erling Smørgrav 		err(2, "can't rename \"%s\" to \"%s\"", tempname, ofn);
443fb3ef04dSDag-Erling Smørgrav 	free(tempname);
444fb3ef04dSDag-Erling Smørgrav 	tempname = NULL;
445fb3ef04dSDag-Erling Smørgrav }
446fb3ef04dSDag-Erling Smørgrav 
447fb3ef04dSDag-Erling Smørgrav /*
448fb3ef04dSDag-Erling Smørgrav  * For cleaning up if there is an error.
449fb3ef04dSDag-Erling Smørgrav  */
450fb3ef04dSDag-Erling Smørgrav static void
cleantemp(void)451fb3ef04dSDag-Erling Smørgrav cleantemp(void)
452fb3ef04dSDag-Erling Smørgrav {
453fb3ef04dSDag-Erling Smørgrav 	if (tempname != NULL)
454fb3ef04dSDag-Erling Smørgrav 		remove(tempname);
455fb3ef04dSDag-Erling Smørgrav }
456fb3ef04dSDag-Erling Smørgrav 
457fb3ef04dSDag-Erling Smørgrav /*
458fb3ef04dSDag-Erling Smørgrav  * Self-identification functions.
459fb3ef04dSDag-Erling Smørgrav  */
460fb3ef04dSDag-Erling Smørgrav 
461fb3ef04dSDag-Erling Smørgrav static void
version(void)462fb3ef04dSDag-Erling Smørgrav version(void)
463fb3ef04dSDag-Erling Smørgrav {
464fb3ef04dSDag-Erling Smørgrav 	const char *c = copyright;
465fb3ef04dSDag-Erling Smørgrav 	for (;;) {
466fb3ef04dSDag-Erling Smørgrav 		while (*++c != '$')
467fb3ef04dSDag-Erling Smørgrav 			if (*c == '\0')
468fb3ef04dSDag-Erling Smørgrav 				exit(0);
469fb3ef04dSDag-Erling Smørgrav 		while (*++c != '$')
470fb3ef04dSDag-Erling Smørgrav 			putc(*c, stderr);
471fb3ef04dSDag-Erling Smørgrav 		putc('\n', stderr);
472fb3ef04dSDag-Erling Smørgrav 	}
473fb3ef04dSDag-Erling Smørgrav }
474fb3ef04dSDag-Erling Smørgrav 
475fb3ef04dSDag-Erling Smørgrav static void
synopsis(FILE * fp)476fb3ef04dSDag-Erling Smørgrav synopsis(FILE *fp)
477fb3ef04dSDag-Erling Smørgrav {
478fb3ef04dSDag-Erling Smørgrav 	fprintf(fp,
479fb3ef04dSDag-Erling Smørgrav 	    "usage:	unifdef [-bBcdehKkmnsStV] [-x{012}] [-Mext] [-opath] \\\n"
480fb3ef04dSDag-Erling Smørgrav 	    "		[-[i]Dsym[=val]] [-[i]Usym] [-fpath] ... [file] ...\n");
481fb3ef04dSDag-Erling Smørgrav }
482fb3ef04dSDag-Erling Smørgrav 
483fb3ef04dSDag-Erling Smørgrav static void
usage(void)484fb3ef04dSDag-Erling Smørgrav usage(void)
485fb3ef04dSDag-Erling Smørgrav {
486fb3ef04dSDag-Erling Smørgrav 	synopsis(stderr);
487fb3ef04dSDag-Erling Smørgrav 	exit(2);
488fb3ef04dSDag-Erling Smørgrav }
489fb3ef04dSDag-Erling Smørgrav 
490fb3ef04dSDag-Erling Smørgrav static void
help(void)491fb3ef04dSDag-Erling Smørgrav help(void)
492fb3ef04dSDag-Erling Smørgrav {
493fb3ef04dSDag-Erling Smørgrav 	synopsis(stdout);
494fb3ef04dSDag-Erling Smørgrav 	printf(
495fb3ef04dSDag-Erling Smørgrav 	    "	-Dsym=val  define preprocessor symbol with given value\n"
496fb3ef04dSDag-Erling Smørgrav 	    "	-Dsym      define preprocessor symbol with value 1\n"
497fb3ef04dSDag-Erling Smørgrav 	    "	-Usym	   preprocessor symbol is undefined\n"
498fb3ef04dSDag-Erling Smørgrav 	    "	-iDsym=val \\  ignore C strings and comments\n"
499fb3ef04dSDag-Erling Smørgrav 	    "	-iDsym      ) in sections controlled by these\n"
500fb3ef04dSDag-Erling Smørgrav 	    "	-iUsym	   /  preprocessor symbols\n"
501fb3ef04dSDag-Erling Smørgrav 	    "	-fpath	file containing #define and #undef directives\n"
502fb3ef04dSDag-Erling Smørgrav 	    "	-b	blank lines instead of deleting them\n"
503fb3ef04dSDag-Erling Smørgrav 	    "	-B	compress blank lines around deleted section\n"
504fb3ef04dSDag-Erling Smørgrav 	    "	-c	complement (invert) keep vs. delete\n"
505fb3ef04dSDag-Erling Smørgrav 	    "	-d	debugging mode\n"
506fb3ef04dSDag-Erling Smørgrav 	    "	-e	ignore multiline preprocessor directives\n"
507fb3ef04dSDag-Erling Smørgrav 	    "	-h	print help\n"
508fb3ef04dSDag-Erling Smørgrav 	    "	-Ipath	extra include file path (ignored)\n"
509fb3ef04dSDag-Erling Smørgrav 	    "	-K	disable && and || short-circuiting\n"
510fb3ef04dSDag-Erling Smørgrav 	    "	-k	process constant #if expressions\n"
511fb3ef04dSDag-Erling Smørgrav 	    "	-Mext	modify in place and keep backups\n"
512fb3ef04dSDag-Erling Smørgrav 	    "	-m	modify input files in place\n"
513fb3ef04dSDag-Erling Smørgrav 	    "	-n	add #line directives to output\n"
514fb3ef04dSDag-Erling Smørgrav 	    "	-opath	output file name\n"
515fb3ef04dSDag-Erling Smørgrav 	    "	-S	list #if control symbols with nesting\n"
516fb3ef04dSDag-Erling Smørgrav 	    "	-s	list #if control symbols\n"
517fb3ef04dSDag-Erling Smørgrav 	    "	-t	ignore C strings and comments\n"
518fb3ef04dSDag-Erling Smørgrav 	    "	-V	print version\n"
519fb3ef04dSDag-Erling Smørgrav 	    "	-x{012}	exit status mode\n"
520fb3ef04dSDag-Erling Smørgrav 	);
521fb3ef04dSDag-Erling Smørgrav 	exit(0);
522fb3ef04dSDag-Erling Smørgrav }
523fb3ef04dSDag-Erling Smørgrav 
524fb3ef04dSDag-Erling Smørgrav /*
525fb3ef04dSDag-Erling Smørgrav  * A state transition function alters the global #if processing state
526fb3ef04dSDag-Erling Smørgrav  * in a particular way. The table below is indexed by the current
527fb3ef04dSDag-Erling Smørgrav  * processing state and the type of the current line.
528fb3ef04dSDag-Erling Smørgrav  *
529fb3ef04dSDag-Erling Smørgrav  * Nesting is handled by keeping a stack of states; some transition
530fb3ef04dSDag-Erling Smørgrav  * functions increase or decrease the depth. They also maintain the
531fb3ef04dSDag-Erling Smørgrav  * ignore state on a stack. In some complicated cases they have to
532fb3ef04dSDag-Erling Smørgrav  * alter the preprocessor directive, as follows.
533fb3ef04dSDag-Erling Smørgrav  *
534fb3ef04dSDag-Erling Smørgrav  * When we have processed a group that starts off with a known-false
535fb3ef04dSDag-Erling Smørgrav  * #if/#elif sequence (which has therefore been deleted) followed by a
536fb3ef04dSDag-Erling Smørgrav  * #elif that we don't understand and therefore must keep, we edit the
537fb3ef04dSDag-Erling Smørgrav  * latter into a #if to keep the nesting correct. We use memcpy() to
538fb3ef04dSDag-Erling Smørgrav  * overwrite the 4 byte token "elif" with "if  " without a '\0' byte.
539fb3ef04dSDag-Erling Smørgrav  *
540fb3ef04dSDag-Erling Smørgrav  * When we find a true #elif in a group, the following block will
541fb3ef04dSDag-Erling Smørgrav  * always be kept and the rest of the sequence after the next #elif or
542fb3ef04dSDag-Erling Smørgrav  * #else will be discarded. We edit the #elif into a #else and the
543fb3ef04dSDag-Erling Smørgrav  * following directive to #endif since this has the desired behaviour.
544fb3ef04dSDag-Erling Smørgrav  *
545fb3ef04dSDag-Erling Smørgrav  * "Dodgy" directives are split across multiple lines, the most common
546fb3ef04dSDag-Erling Smørgrav  * example being a multi-line comment hanging off the right of the
547fb3ef04dSDag-Erling Smørgrav  * directive. We can handle them correctly only if there is no change
548fb3ef04dSDag-Erling Smørgrav  * from printing to dropping (or vice versa) caused by that directive.
549fb3ef04dSDag-Erling Smørgrav  * If the directive is the first of a group we have a choice between
550fb3ef04dSDag-Erling Smørgrav  * failing with an error, or passing it through unchanged instead of
551fb3ef04dSDag-Erling Smørgrav  * evaluating it. The latter is not the default to avoid questions from
552fb3ef04dSDag-Erling Smørgrav  * users about unifdef unexpectedly leaving behind preprocessor directives.
553fb3ef04dSDag-Erling Smørgrav  */
554fb3ef04dSDag-Erling Smørgrav typedef void state_fn(void);
555fb3ef04dSDag-Erling Smørgrav 
556fb3ef04dSDag-Erling Smørgrav /* report an error */
Eelif(void)557fb3ef04dSDag-Erling Smørgrav static void Eelif (void) { error("Inappropriate #elif"); }
Eelse(void)558fb3ef04dSDag-Erling Smørgrav static void Eelse (void) { error("Inappropriate #else"); }
Eendif(void)559fb3ef04dSDag-Erling Smørgrav static void Eendif(void) { error("Inappropriate #endif"); }
Eeof(void)560fb3ef04dSDag-Erling Smørgrav static void Eeof  (void) { error("Premature EOF"); }
Eioccc(void)561fb3ef04dSDag-Erling Smørgrav static void Eioccc(void) { error("Obfuscated preprocessor control line"); }
562fb3ef04dSDag-Erling Smørgrav /* plain line handling */
print(void)563fb3ef04dSDag-Erling Smørgrav static void print (void) { flushline(true); }
drop(void)564fb3ef04dSDag-Erling Smørgrav static void drop  (void) { flushline(false); }
565fb3ef04dSDag-Erling Smørgrav /* output lacks group's start line */
Strue(void)566fb3ef04dSDag-Erling Smørgrav static void Strue (void) { drop();  ignoreoff(); state(IS_TRUE_PREFIX); }
Sfalse(void)567fb3ef04dSDag-Erling Smørgrav static void Sfalse(void) { drop();  ignoreoff(); state(IS_FALSE_PREFIX); }
Selse(void)568fb3ef04dSDag-Erling Smørgrav static void Selse (void) { drop();               state(IS_TRUE_ELSE); }
569fb3ef04dSDag-Erling Smørgrav /* print/pass this block */
Pelif(void)570fb3ef04dSDag-Erling Smørgrav static void Pelif (void) { print(); ignoreoff(); state(IS_PASS_MIDDLE); }
Pelse(void)571fb3ef04dSDag-Erling Smørgrav static void Pelse (void) { print();              state(IS_PASS_ELSE); }
Pendif(void)572fb3ef04dSDag-Erling Smørgrav static void Pendif(void) { print(); unnest(); }
573fb3ef04dSDag-Erling Smørgrav /* discard this block */
Dfalse(void)574fb3ef04dSDag-Erling Smørgrav static void Dfalse(void) { drop();  ignoreoff(); state(IS_FALSE_TRAILER); }
Delif(void)575fb3ef04dSDag-Erling Smørgrav static void Delif (void) { drop();  ignoreoff(); state(IS_FALSE_MIDDLE); }
Delse(void)576fb3ef04dSDag-Erling Smørgrav static void Delse (void) { drop();               state(IS_FALSE_ELSE); }
Dendif(void)577fb3ef04dSDag-Erling Smørgrav static void Dendif(void) { drop();  unnest(); }
578fb3ef04dSDag-Erling Smørgrav /* first line of group */
Fdrop(void)579fb3ef04dSDag-Erling Smørgrav static void Fdrop (void) { nest();  Dfalse(); }
Fpass(void)580fb3ef04dSDag-Erling Smørgrav static void Fpass (void) { nest();  Pelif(); }
Ftrue(void)581fb3ef04dSDag-Erling Smørgrav static void Ftrue (void) { nest();  Strue(); }
Ffalse(void)582fb3ef04dSDag-Erling Smørgrav static void Ffalse(void) { nest();  Sfalse(); }
583fb3ef04dSDag-Erling Smørgrav /* variable pedantry for obfuscated lines */
Oiffy(void)584fb3ef04dSDag-Erling Smørgrav static void Oiffy (void) { if (!iocccok) Eioccc(); Fpass(); ignoreon(); }
Oif(void)585fb3ef04dSDag-Erling Smørgrav static void Oif   (void) { if (!iocccok) Eioccc(); Fpass(); }
Oelif(void)586fb3ef04dSDag-Erling Smørgrav static void Oelif (void) { if (!iocccok) Eioccc(); Pelif(); }
587fb3ef04dSDag-Erling Smørgrav /* ignore comments in this block */
Idrop(void)588fb3ef04dSDag-Erling Smørgrav static void Idrop (void) { Fdrop();  ignoreon(); }
Itrue(void)589fb3ef04dSDag-Erling Smørgrav static void Itrue (void) { Ftrue();  ignoreon(); }
Ifalse(void)590fb3ef04dSDag-Erling Smørgrav static void Ifalse(void) { Ffalse(); ignoreon(); }
591fb3ef04dSDag-Erling Smørgrav /* modify this line */
Mpass(void)592fb3ef04dSDag-Erling Smørgrav static void Mpass (void) { memcpy(keyword, "if  ", 4); Pelif(); }
Mtrue(void)593fb3ef04dSDag-Erling Smørgrav static void Mtrue (void) { keywordedit("else");  state(IS_TRUE_MIDDLE); }
Melif(void)594fb3ef04dSDag-Erling Smørgrav static void Melif (void) { keywordedit("endif"); state(IS_FALSE_TRAILER); }
Melse(void)595fb3ef04dSDag-Erling Smørgrav static void Melse (void) { keywordedit("endif"); state(IS_FALSE_ELSE); }
596fb3ef04dSDag-Erling Smørgrav 
597fb3ef04dSDag-Erling Smørgrav static state_fn * const trans_table[IS_COUNT][LT_COUNT] = {
598fb3ef04dSDag-Erling Smørgrav /* IS_OUTSIDE */
599fb3ef04dSDag-Erling Smørgrav { Itrue, Ifalse,Fpass, Ftrue, Ffalse,Eelif, Eelif, Eelif, Eelse, Eendif,
600fb3ef04dSDag-Erling Smørgrav   Oiffy, Oiffy, Fpass, Oif,   Oif,   Eelif, Eelif, Eelif, Eelse, Eendif,
601fb3ef04dSDag-Erling Smørgrav   print, done,  abort },
602fb3ef04dSDag-Erling Smørgrav /* IS_FALSE_PREFIX */
603fb3ef04dSDag-Erling Smørgrav { Idrop, Idrop, Fdrop, Fdrop, Fdrop, Mpass, Strue, Sfalse,Selse, Dendif,
604fb3ef04dSDag-Erling Smørgrav   Idrop, Idrop, Fdrop, Fdrop, Fdrop, Mpass, Eioccc,Eioccc,Eioccc,Eioccc,
605fb3ef04dSDag-Erling Smørgrav   drop,  Eeof,  abort },
606fb3ef04dSDag-Erling Smørgrav /* IS_TRUE_PREFIX */
607fb3ef04dSDag-Erling Smørgrav { Itrue, Ifalse,Fpass, Ftrue, Ffalse,Dfalse,Dfalse,Dfalse,Delse, Dendif,
608fb3ef04dSDag-Erling Smørgrav   Oiffy, Oiffy, Fpass, Oif,   Oif,   Eioccc,Eioccc,Eioccc,Eioccc,Eioccc,
609fb3ef04dSDag-Erling Smørgrav   print, Eeof,  abort },
610fb3ef04dSDag-Erling Smørgrav /* IS_PASS_MIDDLE */
611fb3ef04dSDag-Erling Smørgrav { Itrue, Ifalse,Fpass, Ftrue, Ffalse,Pelif, Mtrue, Delif, Pelse, Pendif,
612fb3ef04dSDag-Erling Smørgrav   Oiffy, Oiffy, Fpass, Oif,   Oif,   Pelif, Oelif, Oelif, Pelse, Pendif,
613fb3ef04dSDag-Erling Smørgrav   print, Eeof,  abort },
614fb3ef04dSDag-Erling Smørgrav /* IS_FALSE_MIDDLE */
615fb3ef04dSDag-Erling Smørgrav { Idrop, Idrop, Fdrop, Fdrop, Fdrop, Pelif, Mtrue, Delif, Pelse, Pendif,
616fb3ef04dSDag-Erling Smørgrav   Idrop, Idrop, Fdrop, Fdrop, Fdrop, Eioccc,Eioccc,Eioccc,Eioccc,Eioccc,
617fb3ef04dSDag-Erling Smørgrav   drop,  Eeof,  abort },
618fb3ef04dSDag-Erling Smørgrav /* IS_TRUE_MIDDLE */
619fb3ef04dSDag-Erling Smørgrav { Itrue, Ifalse,Fpass, Ftrue, Ffalse,Melif, Melif, Melif, Melse, Pendif,
620fb3ef04dSDag-Erling Smørgrav   Oiffy, Oiffy, Fpass, Oif,   Oif,   Eioccc,Eioccc,Eioccc,Eioccc,Pendif,
621fb3ef04dSDag-Erling Smørgrav   print, Eeof,  abort },
622fb3ef04dSDag-Erling Smørgrav /* IS_PASS_ELSE */
623fb3ef04dSDag-Erling Smørgrav { Itrue, Ifalse,Fpass, Ftrue, Ffalse,Eelif, Eelif, Eelif, Eelse, Pendif,
624fb3ef04dSDag-Erling Smørgrav   Oiffy, Oiffy, Fpass, Oif,   Oif,   Eelif, Eelif, Eelif, Eelse, Pendif,
625fb3ef04dSDag-Erling Smørgrav   print, Eeof,  abort },
626fb3ef04dSDag-Erling Smørgrav /* IS_FALSE_ELSE */
627fb3ef04dSDag-Erling Smørgrav { Idrop, Idrop, Fdrop, Fdrop, Fdrop, Eelif, Eelif, Eelif, Eelse, Dendif,
628fb3ef04dSDag-Erling Smørgrav   Idrop, Idrop, Fdrop, Fdrop, Fdrop, Eelif, Eelif, Eelif, Eelse, Eioccc,
629fb3ef04dSDag-Erling Smørgrav   drop,  Eeof,  abort },
630fb3ef04dSDag-Erling Smørgrav /* IS_TRUE_ELSE */
631fb3ef04dSDag-Erling Smørgrav { Itrue, Ifalse,Fpass, Ftrue, Ffalse,Eelif, Eelif, Eelif, Eelse, Dendif,
632fb3ef04dSDag-Erling Smørgrav   Oiffy, Oiffy, Fpass, Oif,   Oif,   Eelif, Eelif, Eelif, Eelse, Eioccc,
633fb3ef04dSDag-Erling Smørgrav   print, Eeof,  abort },
634fb3ef04dSDag-Erling Smørgrav /* IS_FALSE_TRAILER */
635fb3ef04dSDag-Erling Smørgrav { Idrop, Idrop, Fdrop, Fdrop, Fdrop, Dfalse,Dfalse,Dfalse,Delse, Dendif,
636fb3ef04dSDag-Erling Smørgrav   Idrop, Idrop, Fdrop, Fdrop, Fdrop, Dfalse,Dfalse,Dfalse,Delse, Eioccc,
637fb3ef04dSDag-Erling Smørgrav   drop,  Eeof,  abort }
638fb3ef04dSDag-Erling Smørgrav /*TRUEI  FALSEI IF     TRUE   FALSE  ELIF   ELTRUE ELFALSE ELSE  ENDIF
639fb3ef04dSDag-Erling Smørgrav   TRUEI  FALSEI IF     TRUE   FALSE  ELIF   ELTRUE ELFALSE ELSE  ENDIF (DODGY)
640fb3ef04dSDag-Erling Smørgrav   PLAIN  EOF    ERROR */
641fb3ef04dSDag-Erling Smørgrav };
642fb3ef04dSDag-Erling Smørgrav 
643fb3ef04dSDag-Erling Smørgrav /*
644fb3ef04dSDag-Erling Smørgrav  * State machine utility functions
645fb3ef04dSDag-Erling Smørgrav  */
646fb3ef04dSDag-Erling Smørgrav static void
ignoreoff(void)647fb3ef04dSDag-Erling Smørgrav ignoreoff(void)
648fb3ef04dSDag-Erling Smørgrav {
649fb3ef04dSDag-Erling Smørgrav 	if (depth == 0)
650fb3ef04dSDag-Erling Smørgrav 		abort(); /* bug */
651fb3ef04dSDag-Erling Smørgrav 	ignoring[depth] = ignoring[depth-1];
652fb3ef04dSDag-Erling Smørgrav }
653fb3ef04dSDag-Erling Smørgrav static void
ignoreon(void)654fb3ef04dSDag-Erling Smørgrav ignoreon(void)
655fb3ef04dSDag-Erling Smørgrav {
656fb3ef04dSDag-Erling Smørgrav 	ignoring[depth] = true;
657fb3ef04dSDag-Erling Smørgrav }
658fb3ef04dSDag-Erling Smørgrav static void
keywordedit(const char * replacement)659fb3ef04dSDag-Erling Smørgrav keywordedit(const char *replacement)
660fb3ef04dSDag-Erling Smørgrav {
661fb3ef04dSDag-Erling Smørgrav 	snprintf(keyword, tline + sizeof(tline) - keyword,
662fb3ef04dSDag-Erling Smørgrav 	    "%s%s", replacement, newline);
663fb3ef04dSDag-Erling Smørgrav 	altered = true;
664fb3ef04dSDag-Erling Smørgrav 	print();
665fb3ef04dSDag-Erling Smørgrav }
666fb3ef04dSDag-Erling Smørgrav static void
nest(void)667fb3ef04dSDag-Erling Smørgrav nest(void)
668fb3ef04dSDag-Erling Smørgrav {
669fb3ef04dSDag-Erling Smørgrav 	if (depth > MAXDEPTH-1)
670fb3ef04dSDag-Erling Smørgrav 		abort(); /* bug */
671fb3ef04dSDag-Erling Smørgrav 	if (depth == MAXDEPTH-1)
672fb3ef04dSDag-Erling Smørgrav 		error("Too many levels of nesting");
673fb3ef04dSDag-Erling Smørgrav 	depth += 1;
674fb3ef04dSDag-Erling Smørgrav 	stifline[depth] = linenum;
675fb3ef04dSDag-Erling Smørgrav }
676fb3ef04dSDag-Erling Smørgrav static void
unnest(void)677fb3ef04dSDag-Erling Smørgrav unnest(void)
678fb3ef04dSDag-Erling Smørgrav {
679fb3ef04dSDag-Erling Smørgrav 	if (depth == 0)
680fb3ef04dSDag-Erling Smørgrav 		abort(); /* bug */
681fb3ef04dSDag-Erling Smørgrav 	depth -= 1;
682fb3ef04dSDag-Erling Smørgrav }
683fb3ef04dSDag-Erling Smørgrav static void
state(Ifstate is)684fb3ef04dSDag-Erling Smørgrav state(Ifstate is)
685fb3ef04dSDag-Erling Smørgrav {
686fb3ef04dSDag-Erling Smørgrav 	ifstate[depth] = is;
687fb3ef04dSDag-Erling Smørgrav }
688fb3ef04dSDag-Erling Smørgrav 
689fb3ef04dSDag-Erling Smørgrav /*
690fb3ef04dSDag-Erling Smørgrav  * The last state transition function. When this is called,
691fb3ef04dSDag-Erling Smørgrav  * lineval == LT_EOF, so the process() loop will terminate.
692fb3ef04dSDag-Erling Smørgrav  */
693fb3ef04dSDag-Erling Smørgrav static void
done(void)694fb3ef04dSDag-Erling Smørgrav done(void)
695fb3ef04dSDag-Erling Smørgrav {
696fb3ef04dSDag-Erling Smørgrav 	if (incomment)
697fb3ef04dSDag-Erling Smørgrav 		error("EOF in comment");
698fb3ef04dSDag-Erling Smørgrav 	closeio();
699fb3ef04dSDag-Erling Smørgrav }
700fb3ef04dSDag-Erling Smørgrav 
701fb3ef04dSDag-Erling Smørgrav /*
702fb3ef04dSDag-Erling Smørgrav  * Write a line to the output or not, according to command line options.
703fb3ef04dSDag-Erling Smørgrav  * If writing fails, closeio() will print the error and exit.
704fb3ef04dSDag-Erling Smørgrav  */
705fb3ef04dSDag-Erling Smørgrav static void
flushline(bool keep)706fb3ef04dSDag-Erling Smørgrav flushline(bool keep)
707fb3ef04dSDag-Erling Smørgrav {
708fb3ef04dSDag-Erling Smørgrav 	if (symlist)
709fb3ef04dSDag-Erling Smørgrav 		return;
710fb3ef04dSDag-Erling Smørgrav 	if (keep ^ complement) {
711fb3ef04dSDag-Erling Smørgrav 		bool blankline = tline[strspn(tline, " \t\r\n")] == '\0';
712fb3ef04dSDag-Erling Smørgrav 		if (blankline && compblank && blankcount != blankmax) {
713fb3ef04dSDag-Erling Smørgrav 			delcount += 1;
714fb3ef04dSDag-Erling Smørgrav 			blankcount += 1;
715fb3ef04dSDag-Erling Smørgrav 		} else {
716fb3ef04dSDag-Erling Smørgrav 			if (lnnum && delcount > 0)
717fb3ef04dSDag-Erling Smørgrav 				hashline();
718fb3ef04dSDag-Erling Smørgrav 			if (fputs(tline, output) == EOF)
719fb3ef04dSDag-Erling Smørgrav 				closeio();
720fb3ef04dSDag-Erling Smørgrav 			delcount = 0;
721fb3ef04dSDag-Erling Smørgrav 			blankmax = blankcount = blankline ? blankcount + 1 : 0;
722fb3ef04dSDag-Erling Smørgrav 		}
723fb3ef04dSDag-Erling Smørgrav 	} else {
724fb3ef04dSDag-Erling Smørgrav 		if (lnblank && fputs(newline, output) == EOF)
725fb3ef04dSDag-Erling Smørgrav 			closeio();
726fb3ef04dSDag-Erling Smørgrav 		altered = true;
727fb3ef04dSDag-Erling Smørgrav 		delcount += 1;
728fb3ef04dSDag-Erling Smørgrav 		blankcount = 0;
729fb3ef04dSDag-Erling Smørgrav 	}
730fb3ef04dSDag-Erling Smørgrav 	if (debugging && fflush(output) == EOF)
731fb3ef04dSDag-Erling Smørgrav 		closeio();
732fb3ef04dSDag-Erling Smørgrav }
733fb3ef04dSDag-Erling Smørgrav 
734fb3ef04dSDag-Erling Smørgrav /*
735fb3ef04dSDag-Erling Smørgrav  * Format of #line directives depends on whether we know the input filename.
736fb3ef04dSDag-Erling Smørgrav  */
737fb3ef04dSDag-Erling Smørgrav static void
hashline(void)738fb3ef04dSDag-Erling Smørgrav hashline(void)
739fb3ef04dSDag-Erling Smørgrav {
740fb3ef04dSDag-Erling Smørgrav 	int e;
741fb3ef04dSDag-Erling Smørgrav 
742fb3ef04dSDag-Erling Smørgrav 	if (linefile == NULL)
743fb3ef04dSDag-Erling Smørgrav 		e = fprintf(output, "#line %d%s", linenum, newline);
744fb3ef04dSDag-Erling Smørgrav 	else
745fb3ef04dSDag-Erling Smørgrav 		e = fprintf(output, "#line %d \"%s\"%s",
746fb3ef04dSDag-Erling Smørgrav 		    linenum, linefile, newline);
747fb3ef04dSDag-Erling Smørgrav 	if (e < 0)
748fb3ef04dSDag-Erling Smørgrav 		closeio();
749fb3ef04dSDag-Erling Smørgrav }
750fb3ef04dSDag-Erling Smørgrav 
751fb3ef04dSDag-Erling Smørgrav /*
752fb3ef04dSDag-Erling Smørgrav  * Flush the output and handle errors.
753fb3ef04dSDag-Erling Smørgrav  */
754fb3ef04dSDag-Erling Smørgrav static void
closeio(void)755fb3ef04dSDag-Erling Smørgrav closeio(void)
756fb3ef04dSDag-Erling Smørgrav {
757fb3ef04dSDag-Erling Smørgrav 	/* Tidy up after findsym(). */
758fb3ef04dSDag-Erling Smørgrav 	if (symdepth && !zerosyms)
759fb3ef04dSDag-Erling Smørgrav 		printf("\n");
760fb3ef04dSDag-Erling Smørgrav 	if (output != NULL && (ferror(output) || fclose(output) == EOF))
761fb3ef04dSDag-Erling Smørgrav 			err(2, "%s: can't write to output", filename);
762fb3ef04dSDag-Erling Smørgrav 	fclose(input);
763fb3ef04dSDag-Erling Smørgrav }
764fb3ef04dSDag-Erling Smørgrav 
765fb3ef04dSDag-Erling Smørgrav /*
766fb3ef04dSDag-Erling Smørgrav  * The driver for the state machine.
767fb3ef04dSDag-Erling Smørgrav  */
768fb3ef04dSDag-Erling Smørgrav static void
process(void)769fb3ef04dSDag-Erling Smørgrav process(void)
770fb3ef04dSDag-Erling Smørgrav {
771fb3ef04dSDag-Erling Smørgrav 	Linetype lineval = LT_PLAIN;
772fb3ef04dSDag-Erling Smørgrav 	/* When compressing blank lines, act as if the file
773fb3ef04dSDag-Erling Smørgrav 	   is preceded by a large number of blank lines. */
774fb3ef04dSDag-Erling Smørgrav 	blankmax = blankcount = 1000;
775fb3ef04dSDag-Erling Smørgrav 	zerosyms = true;
776fb3ef04dSDag-Erling Smørgrav 	newline = NULL;
777fb3ef04dSDag-Erling Smørgrav 	linenum = 0;
778fb3ef04dSDag-Erling Smørgrav 	altered = false;
779fb3ef04dSDag-Erling Smørgrav 	while (lineval != LT_EOF) {
780fb3ef04dSDag-Erling Smørgrav 		lineval = parseline();
781fb3ef04dSDag-Erling Smørgrav 		trans_table[ifstate[depth]][lineval]();
782fb3ef04dSDag-Erling Smørgrav 		debug("process line %d %s -> %s depth %d",
783fb3ef04dSDag-Erling Smørgrav 		    linenum, linetype_name[lineval],
784fb3ef04dSDag-Erling Smørgrav 		    ifstate_name[ifstate[depth]], depth);
785fb3ef04dSDag-Erling Smørgrav 	}
786fb3ef04dSDag-Erling Smørgrav 	exitstat |= altered;
787fb3ef04dSDag-Erling Smørgrav }
788fb3ef04dSDag-Erling Smørgrav 
789fb3ef04dSDag-Erling Smørgrav /*
790fb3ef04dSDag-Erling Smørgrav  * Parse a line and determine its type. We keep the preprocessor line
791fb3ef04dSDag-Erling Smørgrav  * parser state between calls in the global variable linestate, with
792fb3ef04dSDag-Erling Smørgrav  * help from skipcomment().
793fb3ef04dSDag-Erling Smørgrav  */
794fb3ef04dSDag-Erling Smørgrav static Linetype
parseline(void)795fb3ef04dSDag-Erling Smørgrav parseline(void)
796fb3ef04dSDag-Erling Smørgrav {
797fb3ef04dSDag-Erling Smørgrav 	const char *cp;
798343b776fSDag-Erling Smørgrav 	struct macro *cursym;
799fb3ef04dSDag-Erling Smørgrav 	Linetype retval;
800fb3ef04dSDag-Erling Smørgrav 	Comment_state wascomment;
801fb3ef04dSDag-Erling Smørgrav 
802fb3ef04dSDag-Erling Smørgrav 	wascomment = incomment;
803fb3ef04dSDag-Erling Smørgrav 	cp = skiphash();
804fb3ef04dSDag-Erling Smørgrav 	if (cp == NULL)
805fb3ef04dSDag-Erling Smørgrav 		return (LT_EOF);
806fb3ef04dSDag-Erling Smørgrav 	if (newline == NULL) {
807fb3ef04dSDag-Erling Smørgrav 		if (strrchr(tline, '\n') == strrchr(tline, '\r') + 1)
808fb3ef04dSDag-Erling Smørgrav 			newline = newline_crlf;
809fb3ef04dSDag-Erling Smørgrav 		else
810fb3ef04dSDag-Erling Smørgrav 			newline = newline_unix;
811fb3ef04dSDag-Erling Smørgrav 	}
812fb3ef04dSDag-Erling Smørgrav 	if (*cp == '\0') {
813fb3ef04dSDag-Erling Smørgrav 		retval = LT_PLAIN;
814fb3ef04dSDag-Erling Smørgrav 		goto done;
815fb3ef04dSDag-Erling Smørgrav 	}
816fb3ef04dSDag-Erling Smørgrav 	keyword = tline + (cp - tline);
817fb3ef04dSDag-Erling Smørgrav 	if ((cp = matchsym("ifdef", keyword)) != NULL ||
818fb3ef04dSDag-Erling Smørgrav 	    (cp = matchsym("ifndef", keyword)) != NULL) {
819fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp);
820343b776fSDag-Erling Smørgrav 		if ((cursym = findsym(&cp)) == NULL)
821fb3ef04dSDag-Erling Smørgrav 			retval = LT_IF;
822fb3ef04dSDag-Erling Smørgrav 		else {
823fb3ef04dSDag-Erling Smørgrav 			retval = (keyword[2] == 'n')
824fb3ef04dSDag-Erling Smørgrav 			    ? LT_FALSE : LT_TRUE;
825343b776fSDag-Erling Smørgrav 			if (cursym->value == NULL)
826fb3ef04dSDag-Erling Smørgrav 				retval = (retval == LT_TRUE)
827fb3ef04dSDag-Erling Smørgrav 				    ? LT_FALSE : LT_TRUE;
828343b776fSDag-Erling Smørgrav 			if (cursym->ignore)
829fb3ef04dSDag-Erling Smørgrav 				retval = (retval == LT_TRUE)
830fb3ef04dSDag-Erling Smørgrav 				    ? LT_TRUEI : LT_FALSEI;
831fb3ef04dSDag-Erling Smørgrav 		}
832fb3ef04dSDag-Erling Smørgrav 	} else if ((cp = matchsym("if", keyword)) != NULL)
833fb3ef04dSDag-Erling Smørgrav 		retval = ifeval(&cp);
834fb3ef04dSDag-Erling Smørgrav 	else if ((cp = matchsym("elif", keyword)) != NULL)
835fb3ef04dSDag-Erling Smørgrav 		retval = linetype_if2elif(ifeval(&cp));
836fb3ef04dSDag-Erling Smørgrav 	else if ((cp = matchsym("else", keyword)) != NULL)
837fb3ef04dSDag-Erling Smørgrav 		retval = LT_ELSE;
838fb3ef04dSDag-Erling Smørgrav 	else if ((cp = matchsym("endif", keyword)) != NULL)
839fb3ef04dSDag-Erling Smørgrav 		retval = LT_ENDIF;
840fb3ef04dSDag-Erling Smørgrav 	else {
841fb3ef04dSDag-Erling Smørgrav 		cp = skipsym(keyword);
842fb3ef04dSDag-Erling Smørgrav 		/* no way can we deal with a continuation inside a keyword */
843fb3ef04dSDag-Erling Smørgrav 		if (strncmp(cp, "\\\r\n", 3) == 0 ||
844fb3ef04dSDag-Erling Smørgrav 		    strncmp(cp, "\\\n", 2) == 0)
845fb3ef04dSDag-Erling Smørgrav 			Eioccc();
846fb3ef04dSDag-Erling Smørgrav 		cp = skipline(cp);
847fb3ef04dSDag-Erling Smørgrav 		retval = LT_PLAIN;
848fb3ef04dSDag-Erling Smørgrav 		goto done;
849fb3ef04dSDag-Erling Smørgrav 	}
850fb3ef04dSDag-Erling Smørgrav 	cp = skipcomment(cp);
851fb3ef04dSDag-Erling Smørgrav 	if (*cp != '\0') {
852fb3ef04dSDag-Erling Smørgrav 		cp = skipline(cp);
853fb3ef04dSDag-Erling Smørgrav 		if (retval == LT_TRUE || retval == LT_FALSE ||
854fb3ef04dSDag-Erling Smørgrav 		    retval == LT_TRUEI || retval == LT_FALSEI)
855fb3ef04dSDag-Erling Smørgrav 			retval = LT_IF;
856fb3ef04dSDag-Erling Smørgrav 		if (retval == LT_ELTRUE || retval == LT_ELFALSE)
857fb3ef04dSDag-Erling Smørgrav 			retval = LT_ELIF;
858fb3ef04dSDag-Erling Smørgrav 	}
859fb3ef04dSDag-Erling Smørgrav 	/* the following can happen if the last line of the file lacks a
860fb3ef04dSDag-Erling Smørgrav 	   newline or if there is too much whitespace in a directive */
861fb3ef04dSDag-Erling Smørgrav 	if (linestate == LS_HASH) {
862fb3ef04dSDag-Erling Smørgrav 		long len = cp - tline;
863fb3ef04dSDag-Erling Smørgrav 		if (fgets(tline + len, MAXLINE - len, input) == NULL) {
864fb3ef04dSDag-Erling Smørgrav 			if (ferror(input))
865fb3ef04dSDag-Erling Smørgrav 				err(2, "can't read %s", filename);
866fb3ef04dSDag-Erling Smørgrav 			debug("parser insert newline at EOF", linenum);
867fb3ef04dSDag-Erling Smørgrav 			strcpy(tline + len, newline);
868fb3ef04dSDag-Erling Smørgrav 			cp += strlen(newline);
869fb3ef04dSDag-Erling Smørgrav 			linestate = LS_START;
870fb3ef04dSDag-Erling Smørgrav 		} else {
871fb3ef04dSDag-Erling Smørgrav 			debug("parser concatenate dangling whitespace");
872fb3ef04dSDag-Erling Smørgrav 			++linenum;
873fb3ef04dSDag-Erling Smørgrav 			cp = skipcomment(cp);
874fb3ef04dSDag-Erling Smørgrav 		}
875fb3ef04dSDag-Erling Smørgrav 	}
876fb3ef04dSDag-Erling Smørgrav 	if (retval != LT_PLAIN && (wascomment || linestate != LS_START)) {
877fb3ef04dSDag-Erling Smørgrav 		retval = linetype_2dodgy(retval);
878fb3ef04dSDag-Erling Smørgrav 		linestate = LS_DIRTY;
879fb3ef04dSDag-Erling Smørgrav 	}
880fb3ef04dSDag-Erling Smørgrav done:
881fb3ef04dSDag-Erling Smørgrav 	debug("parser line %d state %s comment %s line", linenum,
882fb3ef04dSDag-Erling Smørgrav 	    comment_name[incomment], linestate_name[linestate]);
883fb3ef04dSDag-Erling Smørgrav 	return (retval);
884fb3ef04dSDag-Erling Smørgrav }
885fb3ef04dSDag-Erling Smørgrav 
886fb3ef04dSDag-Erling Smørgrav /*
887fb3ef04dSDag-Erling Smørgrav  * These are the binary operators that are supported by the expression
888fb3ef04dSDag-Erling Smørgrav  * evaluator.
889fb3ef04dSDag-Erling Smørgrav  */
op_strict(long * p,long v,Linetype at,Linetype bt)890fb3ef04dSDag-Erling Smørgrav static Linetype op_strict(long *p, long v, Linetype at, Linetype bt) {
891fb3ef04dSDag-Erling Smørgrav 	if(at == LT_IF || bt == LT_IF) return (LT_IF);
892fb3ef04dSDag-Erling Smørgrav 	return (*p = v, v ? LT_TRUE : LT_FALSE);
893fb3ef04dSDag-Erling Smørgrav }
op_lt(long * p,Linetype at,long a,Linetype bt,long b)894fb3ef04dSDag-Erling Smørgrav static Linetype op_lt(long *p, Linetype at, long a, Linetype bt, long b) {
895fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a < b, at, bt);
896fb3ef04dSDag-Erling Smørgrav }
op_gt(long * p,Linetype at,long a,Linetype bt,long b)897fb3ef04dSDag-Erling Smørgrav static Linetype op_gt(long *p, Linetype at, long a, Linetype bt, long b) {
898fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a > b, at, bt);
899fb3ef04dSDag-Erling Smørgrav }
op_le(long * p,Linetype at,long a,Linetype bt,long b)900fb3ef04dSDag-Erling Smørgrav static Linetype op_le(long *p, Linetype at, long a, Linetype bt, long b) {
901fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a <= b, at, bt);
902fb3ef04dSDag-Erling Smørgrav }
op_ge(long * p,Linetype at,long a,Linetype bt,long b)903fb3ef04dSDag-Erling Smørgrav static Linetype op_ge(long *p, Linetype at, long a, Linetype bt, long b) {
904fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a >= b, at, bt);
905fb3ef04dSDag-Erling Smørgrav }
op_eq(long * p,Linetype at,long a,Linetype bt,long b)906fb3ef04dSDag-Erling Smørgrav static Linetype op_eq(long *p, Linetype at, long a, Linetype bt, long b) {
907fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a == b, at, bt);
908fb3ef04dSDag-Erling Smørgrav }
op_ne(long * p,Linetype at,long a,Linetype bt,long b)909fb3ef04dSDag-Erling Smørgrav static Linetype op_ne(long *p, Linetype at, long a, Linetype bt, long b) {
910fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a != b, at, bt);
911fb3ef04dSDag-Erling Smørgrav }
op_or(long * p,Linetype at,long a,Linetype bt,long b)912fb3ef04dSDag-Erling Smørgrav static Linetype op_or(long *p, Linetype at, long a, Linetype bt, long b) {
913fb3ef04dSDag-Erling Smørgrav 	if (!strictlogic && (at == LT_TRUE || bt == LT_TRUE))
914fb3ef04dSDag-Erling Smørgrav 		return (*p = 1, LT_TRUE);
915fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a || b, at, bt);
916fb3ef04dSDag-Erling Smørgrav }
op_and(long * p,Linetype at,long a,Linetype bt,long b)917fb3ef04dSDag-Erling Smørgrav static Linetype op_and(long *p, Linetype at, long a, Linetype bt, long b) {
918fb3ef04dSDag-Erling Smørgrav 	if (!strictlogic && (at == LT_FALSE || bt == LT_FALSE))
919fb3ef04dSDag-Erling Smørgrav 		return (*p = 0, LT_FALSE);
920fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a && b, at, bt);
921fb3ef04dSDag-Erling Smørgrav }
op_blsh(long * p,Linetype at,long a,Linetype bt,long b)922fb3ef04dSDag-Erling Smørgrav static Linetype op_blsh(long *p, Linetype at, long a, Linetype bt, long b) {
923fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a << b, at, bt);
924fb3ef04dSDag-Erling Smørgrav }
op_brsh(long * p,Linetype at,long a,Linetype bt,long b)925fb3ef04dSDag-Erling Smørgrav static Linetype op_brsh(long *p, Linetype at, long a, Linetype bt, long b) {
926fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a >> b, at, bt);
927fb3ef04dSDag-Erling Smørgrav }
op_add(long * p,Linetype at,long a,Linetype bt,long b)928fb3ef04dSDag-Erling Smørgrav static Linetype op_add(long *p, Linetype at, long a, Linetype bt, long b) {
929fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a + b, at, bt);
930fb3ef04dSDag-Erling Smørgrav }
op_sub(long * p,Linetype at,long a,Linetype bt,long b)931fb3ef04dSDag-Erling Smørgrav static Linetype op_sub(long *p, Linetype at, long a, Linetype bt, long b) {
932fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a - b, at, bt);
933fb3ef04dSDag-Erling Smørgrav }
op_mul(long * p,Linetype at,long a,Linetype bt,long b)934fb3ef04dSDag-Erling Smørgrav static Linetype op_mul(long *p, Linetype at, long a, Linetype bt, long b) {
935fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a * b, at, bt);
936fb3ef04dSDag-Erling Smørgrav }
op_div(long * p,Linetype at,long a,Linetype bt,long b)937fb3ef04dSDag-Erling Smørgrav static Linetype op_div(long *p, Linetype at, long a, Linetype bt, long b) {
938fb3ef04dSDag-Erling Smørgrav 	if (bt != LT_TRUE) {
939fb3ef04dSDag-Erling Smørgrav 		debug("eval division by zero");
940fb3ef04dSDag-Erling Smørgrav 		return (LT_ERROR);
941fb3ef04dSDag-Erling Smørgrav 	}
942fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a / b, at, bt);
943fb3ef04dSDag-Erling Smørgrav }
op_mod(long * p,Linetype at,long a,Linetype bt,long b)944fb3ef04dSDag-Erling Smørgrav static Linetype op_mod(long *p, Linetype at, long a, Linetype bt, long b) {
945fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a % b, at, bt);
946fb3ef04dSDag-Erling Smørgrav }
op_bor(long * p,Linetype at,long a,Linetype bt,long b)947fb3ef04dSDag-Erling Smørgrav static Linetype op_bor(long *p, Linetype at, long a, Linetype bt, long b) {
948fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a | b, at, bt);
949fb3ef04dSDag-Erling Smørgrav }
op_bxor(long * p,Linetype at,long a,Linetype bt,long b)950fb3ef04dSDag-Erling Smørgrav static Linetype op_bxor(long *p, Linetype at, long a, Linetype bt, long b) {
951fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a ^ b, at, bt);
952fb3ef04dSDag-Erling Smørgrav }
op_band(long * p,Linetype at,long a,Linetype bt,long b)953fb3ef04dSDag-Erling Smørgrav static Linetype op_band(long *p, Linetype at, long a, Linetype bt, long b) {
954fb3ef04dSDag-Erling Smørgrav 	return op_strict(p, a & b, at, bt);
955fb3ef04dSDag-Erling Smørgrav }
956fb3ef04dSDag-Erling Smørgrav 
957fb3ef04dSDag-Erling Smørgrav /*
958fb3ef04dSDag-Erling Smørgrav  * An evaluation function takes three arguments, as follows: (1) a pointer to
959fb3ef04dSDag-Erling Smørgrav  * an element of the precedence table which lists the operators at the current
960fb3ef04dSDag-Erling Smørgrav  * level of precedence; (2) a pointer to an integer which will receive the
961fb3ef04dSDag-Erling Smørgrav  * value of the expression; and (3) a pointer to a char* that points to the
962fb3ef04dSDag-Erling Smørgrav  * expression to be evaluated and that is updated to the end of the expression
963fb3ef04dSDag-Erling Smørgrav  * when evaluation is complete. The function returns LT_FALSE if the value of
964fb3ef04dSDag-Erling Smørgrav  * the expression is zero, LT_TRUE if it is non-zero, LT_IF if the expression
965fb3ef04dSDag-Erling Smørgrav  * depends on an unknown symbol, or LT_ERROR if there is a parse failure.
966fb3ef04dSDag-Erling Smørgrav  */
967fb3ef04dSDag-Erling Smørgrav struct ops;
968fb3ef04dSDag-Erling Smørgrav 
969fb3ef04dSDag-Erling Smørgrav typedef Linetype eval_fn(const struct ops *, long *, const char **);
970fb3ef04dSDag-Erling Smørgrav 
971fb3ef04dSDag-Erling Smørgrav static eval_fn eval_table, eval_unary;
972fb3ef04dSDag-Erling Smørgrav 
973fb3ef04dSDag-Erling Smørgrav /*
974fb3ef04dSDag-Erling Smørgrav  * The precedence table. Expressions involving binary operators are evaluated
975fb3ef04dSDag-Erling Smørgrav  * in a table-driven way by eval_table. When it evaluates a subexpression it
976fb3ef04dSDag-Erling Smørgrav  * calls the inner function with its first argument pointing to the next
977fb3ef04dSDag-Erling Smørgrav  * element of the table. Innermost expressions have special non-table-driven
978fb3ef04dSDag-Erling Smørgrav  * handling.
979fb3ef04dSDag-Erling Smørgrav  *
980fb3ef04dSDag-Erling Smørgrav  * The stop characters help with lexical analysis: an operator is not
981fb3ef04dSDag-Erling Smørgrav  * recognized if it is followed by one of the stop characters because
982fb3ef04dSDag-Erling Smørgrav  * that would make it a different operator.
983fb3ef04dSDag-Erling Smørgrav  */
984fb3ef04dSDag-Erling Smørgrav struct op {
985fb3ef04dSDag-Erling Smørgrav 	const char *str;
986fb3ef04dSDag-Erling Smørgrav 	Linetype (*fn)(long *, Linetype, long, Linetype, long);
987fb3ef04dSDag-Erling Smørgrav 	const char *stop;
988fb3ef04dSDag-Erling Smørgrav };
989fb3ef04dSDag-Erling Smørgrav struct ops {
990fb3ef04dSDag-Erling Smørgrav 	eval_fn *inner;
991fb3ef04dSDag-Erling Smørgrav 	struct op op[5];
992fb3ef04dSDag-Erling Smørgrav };
993fb3ef04dSDag-Erling Smørgrav static const struct ops eval_ops[] = {
994fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "||", op_or,   NULL } } },
995fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "&&", op_and,  NULL } } },
996fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "|",  op_bor,  "|" } } },
997fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "^",  op_bxor, NULL } } },
998fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "&",  op_band, "&" } } },
999fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "==", op_eq,   NULL },
1000fb3ef04dSDag-Erling Smørgrav 			{ "!=", op_ne,   NULL } } },
1001fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "<=", op_le,   NULL },
1002fb3ef04dSDag-Erling Smørgrav 			{ ">=", op_ge,   NULL },
1003fb3ef04dSDag-Erling Smørgrav 			{ "<",  op_lt,   "<=" },
1004fb3ef04dSDag-Erling Smørgrav 			{ ">",  op_gt,   ">=" } } },
1005fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "<<", op_blsh, NULL },
1006fb3ef04dSDag-Erling Smørgrav 			{ ">>", op_brsh, NULL } } },
1007fb3ef04dSDag-Erling Smørgrav 	{ eval_table, { { "+",  op_add,  NULL },
1008fb3ef04dSDag-Erling Smørgrav 			{ "-",  op_sub,  NULL } } },
1009fb3ef04dSDag-Erling Smørgrav 	{ eval_unary, { { "*",  op_mul,  NULL },
1010fb3ef04dSDag-Erling Smørgrav 			{ "/",  op_div,  NULL },
1011fb3ef04dSDag-Erling Smørgrav 			{ "%",  op_mod,  NULL } } },
1012fb3ef04dSDag-Erling Smørgrav };
1013fb3ef04dSDag-Erling Smørgrav 
1014fb3ef04dSDag-Erling Smørgrav /* Current operator precedence level */
prec(const struct ops * ops)1015fb3ef04dSDag-Erling Smørgrav static long prec(const struct ops *ops)
1016fb3ef04dSDag-Erling Smørgrav {
1017fb3ef04dSDag-Erling Smørgrav 	return (ops - eval_ops);
1018fb3ef04dSDag-Erling Smørgrav }
1019fb3ef04dSDag-Erling Smørgrav 
1020fb3ef04dSDag-Erling Smørgrav /*
1021fb3ef04dSDag-Erling Smørgrav  * Function for evaluating the innermost parts of expressions,
1022fb3ef04dSDag-Erling Smørgrav  * viz. !expr (expr) number defined(symbol) symbol
1023fb3ef04dSDag-Erling Smørgrav  * We reset the constexpr flag in the last two cases.
1024fb3ef04dSDag-Erling Smørgrav  */
1025fb3ef04dSDag-Erling Smørgrav static Linetype
eval_unary(const struct ops * ops,long * valp,const char ** cpp)1026fb3ef04dSDag-Erling Smørgrav eval_unary(const struct ops *ops, long *valp, const char **cpp)
1027fb3ef04dSDag-Erling Smørgrav {
1028fb3ef04dSDag-Erling Smørgrav 	const char *cp;
1029fb3ef04dSDag-Erling Smørgrav 	char *ep;
1030343b776fSDag-Erling Smørgrav 	struct macro *sym;
1031fb3ef04dSDag-Erling Smørgrav 	bool defparen;
1032fb3ef04dSDag-Erling Smørgrav 	Linetype lt;
1033fb3ef04dSDag-Erling Smørgrav 
1034fb3ef04dSDag-Erling Smørgrav 	cp = skipcomment(*cpp);
1035fb3ef04dSDag-Erling Smørgrav 	if (*cp == '!') {
1036fb3ef04dSDag-Erling Smørgrav 		debug("eval%d !", prec(ops));
1037fb3ef04dSDag-Erling Smørgrav 		cp++;
1038fb3ef04dSDag-Erling Smørgrav 		lt = eval_unary(ops, valp, &cp);
1039fb3ef04dSDag-Erling Smørgrav 		if (lt == LT_ERROR)
1040fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1041fb3ef04dSDag-Erling Smørgrav 		if (lt != LT_IF) {
1042fb3ef04dSDag-Erling Smørgrav 			*valp = !*valp;
1043fb3ef04dSDag-Erling Smørgrav 			lt = *valp ? LT_TRUE : LT_FALSE;
1044fb3ef04dSDag-Erling Smørgrav 		}
1045fb3ef04dSDag-Erling Smørgrav 	} else if (*cp == '~') {
1046fb3ef04dSDag-Erling Smørgrav 		debug("eval%d ~", prec(ops));
1047fb3ef04dSDag-Erling Smørgrav 		cp++;
1048fb3ef04dSDag-Erling Smørgrav 		lt = eval_unary(ops, valp, &cp);
1049fb3ef04dSDag-Erling Smørgrav 		if (lt == LT_ERROR)
1050fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1051fb3ef04dSDag-Erling Smørgrav 		if (lt != LT_IF) {
1052fb3ef04dSDag-Erling Smørgrav 			*valp = ~(*valp);
1053fb3ef04dSDag-Erling Smørgrav 			lt = *valp ? LT_TRUE : LT_FALSE;
1054fb3ef04dSDag-Erling Smørgrav 		}
1055fb3ef04dSDag-Erling Smørgrav 	} else if (*cp == '-') {
1056fb3ef04dSDag-Erling Smørgrav 		debug("eval%d -", prec(ops));
1057fb3ef04dSDag-Erling Smørgrav 		cp++;
1058fb3ef04dSDag-Erling Smørgrav 		lt = eval_unary(ops, valp, &cp);
1059fb3ef04dSDag-Erling Smørgrav 		if (lt == LT_ERROR)
1060fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1061fb3ef04dSDag-Erling Smørgrav 		if (lt != LT_IF) {
1062fb3ef04dSDag-Erling Smørgrav 			*valp = -(*valp);
1063fb3ef04dSDag-Erling Smørgrav 			lt = *valp ? LT_TRUE : LT_FALSE;
1064fb3ef04dSDag-Erling Smørgrav 		}
1065fb3ef04dSDag-Erling Smørgrav 	} else if (*cp == '(') {
1066fb3ef04dSDag-Erling Smørgrav 		cp++;
1067fb3ef04dSDag-Erling Smørgrav 		debug("eval%d (", prec(ops));
1068fb3ef04dSDag-Erling Smørgrav 		lt = eval_table(eval_ops, valp, &cp);
1069fb3ef04dSDag-Erling Smørgrav 		if (lt == LT_ERROR)
1070fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1071fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp);
1072fb3ef04dSDag-Erling Smørgrav 		if (*cp++ != ')')
1073fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1074fb3ef04dSDag-Erling Smørgrav 	} else if (isdigit((unsigned char)*cp)) {
1075fb3ef04dSDag-Erling Smørgrav 		debug("eval%d number", prec(ops));
1076fb3ef04dSDag-Erling Smørgrav 		*valp = strtol(cp, &ep, 0);
1077fb3ef04dSDag-Erling Smørgrav 		if (ep == cp)
1078fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1079fb3ef04dSDag-Erling Smørgrav 		lt = *valp ? LT_TRUE : LT_FALSE;
1080fb3ef04dSDag-Erling Smørgrav 		cp = ep;
1081fb3ef04dSDag-Erling Smørgrav 	} else if (matchsym("defined", cp) != NULL) {
1082fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp+7);
1083fb3ef04dSDag-Erling Smørgrav 		if (*cp == '(') {
1084fb3ef04dSDag-Erling Smørgrav 			cp = skipcomment(cp+1);
1085fb3ef04dSDag-Erling Smørgrav 			defparen = true;
1086fb3ef04dSDag-Erling Smørgrav 		} else {
1087fb3ef04dSDag-Erling Smørgrav 			defparen = false;
1088fb3ef04dSDag-Erling Smørgrav 		}
1089fb3ef04dSDag-Erling Smørgrav 		sym = findsym(&cp);
1090fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp);
1091fb3ef04dSDag-Erling Smørgrav 		if (defparen && *cp++ != ')') {
1092fb3ef04dSDag-Erling Smørgrav 			debug("eval%d defined missing ')'", prec(ops));
1093fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1094fb3ef04dSDag-Erling Smørgrav 		}
1095343b776fSDag-Erling Smørgrav 		if (sym == NULL) {
1096fb3ef04dSDag-Erling Smørgrav 			debug("eval%d defined unknown", prec(ops));
1097fb3ef04dSDag-Erling Smørgrav 			lt = LT_IF;
1098fb3ef04dSDag-Erling Smørgrav 		} else {
1099343b776fSDag-Erling Smørgrav 			debug("eval%d defined %s", prec(ops), sym->name);
1100343b776fSDag-Erling Smørgrav 			*valp = (sym->value != NULL);
1101fb3ef04dSDag-Erling Smørgrav 			lt = *valp ? LT_TRUE : LT_FALSE;
1102fb3ef04dSDag-Erling Smørgrav 		}
1103fb3ef04dSDag-Erling Smørgrav 		constexpr = false;
1104fb3ef04dSDag-Erling Smørgrav 	} else if (!endsym(*cp)) {
1105fb3ef04dSDag-Erling Smørgrav 		debug("eval%d symbol", prec(ops));
1106fb3ef04dSDag-Erling Smørgrav 		sym = findsym(&cp);
1107343b776fSDag-Erling Smørgrav 		if (sym == NULL) {
1108fb3ef04dSDag-Erling Smørgrav 			lt = LT_IF;
1109fb3ef04dSDag-Erling Smørgrav 			cp = skipargs(cp);
1110343b776fSDag-Erling Smørgrav 		} else if (sym->value == NULL) {
1111fb3ef04dSDag-Erling Smørgrav 			*valp = 0;
1112fb3ef04dSDag-Erling Smørgrav 			lt = LT_FALSE;
1113fb3ef04dSDag-Erling Smørgrav 		} else {
1114343b776fSDag-Erling Smørgrav 			*valp = strtol(sym->value, &ep, 0);
1115343b776fSDag-Erling Smørgrav 			if (*ep != '\0' || ep == sym->value)
1116fb3ef04dSDag-Erling Smørgrav 				return (LT_ERROR);
1117fb3ef04dSDag-Erling Smørgrav 			lt = *valp ? LT_TRUE : LT_FALSE;
1118fb3ef04dSDag-Erling Smørgrav 			cp = skipargs(cp);
1119fb3ef04dSDag-Erling Smørgrav 		}
1120fb3ef04dSDag-Erling Smørgrav 		constexpr = false;
1121fb3ef04dSDag-Erling Smørgrav 	} else {
1122fb3ef04dSDag-Erling Smørgrav 		debug("eval%d bad expr", prec(ops));
1123fb3ef04dSDag-Erling Smørgrav 		return (LT_ERROR);
1124fb3ef04dSDag-Erling Smørgrav 	}
1125fb3ef04dSDag-Erling Smørgrav 
1126fb3ef04dSDag-Erling Smørgrav 	*cpp = cp;
1127fb3ef04dSDag-Erling Smørgrav 	debug("eval%d = %d", prec(ops), *valp);
1128fb3ef04dSDag-Erling Smørgrav 	return (lt);
1129fb3ef04dSDag-Erling Smørgrav }
1130fb3ef04dSDag-Erling Smørgrav 
1131fb3ef04dSDag-Erling Smørgrav /*
1132fb3ef04dSDag-Erling Smørgrav  * Table-driven evaluation of binary operators.
1133fb3ef04dSDag-Erling Smørgrav  */
1134fb3ef04dSDag-Erling Smørgrav static Linetype
eval_table(const struct ops * ops,long * valp,const char ** cpp)1135fb3ef04dSDag-Erling Smørgrav eval_table(const struct ops *ops, long *valp, const char **cpp)
1136fb3ef04dSDag-Erling Smørgrav {
1137fb3ef04dSDag-Erling Smørgrav 	const struct op *op;
1138fb3ef04dSDag-Erling Smørgrav 	const char *cp;
1139fb3ef04dSDag-Erling Smørgrav 	long val = 0;
1140fb3ef04dSDag-Erling Smørgrav 	Linetype lt, rt;
1141fb3ef04dSDag-Erling Smørgrav 
1142fb3ef04dSDag-Erling Smørgrav 	debug("eval%d", prec(ops));
1143fb3ef04dSDag-Erling Smørgrav 	cp = *cpp;
1144fb3ef04dSDag-Erling Smørgrav 	lt = ops->inner(ops+1, valp, &cp);
1145fb3ef04dSDag-Erling Smørgrav 	if (lt == LT_ERROR)
1146fb3ef04dSDag-Erling Smørgrav 		return (LT_ERROR);
1147fb3ef04dSDag-Erling Smørgrav 	for (;;) {
1148fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp);
1149fb3ef04dSDag-Erling Smørgrav 		for (op = ops->op; op->str != NULL; op++) {
1150fb3ef04dSDag-Erling Smørgrav 			if (strncmp(cp, op->str, strlen(op->str)) == 0) {
1151fb3ef04dSDag-Erling Smørgrav 				/* assume only one-char operators have stop chars */
1152fb3ef04dSDag-Erling Smørgrav 				if (op->stop != NULL && cp[1] != '\0' &&
1153fb3ef04dSDag-Erling Smørgrav 				    strchr(op->stop, cp[1]) != NULL)
1154fb3ef04dSDag-Erling Smørgrav 					continue;
1155fb3ef04dSDag-Erling Smørgrav 				else
1156fb3ef04dSDag-Erling Smørgrav 					break;
1157fb3ef04dSDag-Erling Smørgrav 			}
1158fb3ef04dSDag-Erling Smørgrav 		}
1159fb3ef04dSDag-Erling Smørgrav 		if (op->str == NULL)
1160fb3ef04dSDag-Erling Smørgrav 			break;
1161fb3ef04dSDag-Erling Smørgrav 		cp += strlen(op->str);
1162fb3ef04dSDag-Erling Smørgrav 		debug("eval%d %s", prec(ops), op->str);
1163fb3ef04dSDag-Erling Smørgrav 		rt = ops->inner(ops+1, &val, &cp);
1164fb3ef04dSDag-Erling Smørgrav 		if (rt == LT_ERROR)
1165fb3ef04dSDag-Erling Smørgrav 			return (LT_ERROR);
1166fb3ef04dSDag-Erling Smørgrav 		lt = op->fn(valp, lt, *valp, rt, val);
1167fb3ef04dSDag-Erling Smørgrav 	}
1168fb3ef04dSDag-Erling Smørgrav 
1169fb3ef04dSDag-Erling Smørgrav 	*cpp = cp;
1170fb3ef04dSDag-Erling Smørgrav 	debug("eval%d = %d", prec(ops), *valp);
1171fb3ef04dSDag-Erling Smørgrav 	debug("eval%d lt = %s", prec(ops), linetype_name[lt]);
1172fb3ef04dSDag-Erling Smørgrav 	return (lt);
1173fb3ef04dSDag-Erling Smørgrav }
1174fb3ef04dSDag-Erling Smørgrav 
1175fb3ef04dSDag-Erling Smørgrav /*
1176fb3ef04dSDag-Erling Smørgrav  * Evaluate the expression on a #if or #elif line. If we can work out
1177fb3ef04dSDag-Erling Smørgrav  * the result we return LT_TRUE or LT_FALSE accordingly, otherwise we
1178fb3ef04dSDag-Erling Smørgrav  * return just a generic LT_IF.
1179fb3ef04dSDag-Erling Smørgrav  */
1180fb3ef04dSDag-Erling Smørgrav static Linetype
ifeval(const char ** cpp)1181fb3ef04dSDag-Erling Smørgrav ifeval(const char **cpp)
1182fb3ef04dSDag-Erling Smørgrav {
1183fb3ef04dSDag-Erling Smørgrav 	Linetype ret;
1184fb3ef04dSDag-Erling Smørgrav 	long val = 0;
1185fb3ef04dSDag-Erling Smørgrav 
1186fb3ef04dSDag-Erling Smørgrav 	debug("eval %s", *cpp);
1187fb3ef04dSDag-Erling Smørgrav 	constexpr = killconsts ? false : true;
1188fb3ef04dSDag-Erling Smørgrav 	ret = eval_table(eval_ops, &val, cpp);
1189fb3ef04dSDag-Erling Smørgrav 	debug("eval = %d", val);
1190fb3ef04dSDag-Erling Smørgrav 	return (constexpr ? LT_IF : ret == LT_ERROR ? LT_IF : ret);
1191fb3ef04dSDag-Erling Smørgrav }
1192fb3ef04dSDag-Erling Smørgrav 
1193fb3ef04dSDag-Erling Smørgrav /*
1194fb3ef04dSDag-Erling Smørgrav  * Read a line and examine its initial part to determine if it is a
1195fb3ef04dSDag-Erling Smørgrav  * preprocessor directive. Returns NULL on EOF, or a pointer to a
1196fb3ef04dSDag-Erling Smørgrav  * preprocessor directive name, or a pointer to the zero byte at the
1197fb3ef04dSDag-Erling Smørgrav  * end of the line.
1198fb3ef04dSDag-Erling Smørgrav  */
1199fb3ef04dSDag-Erling Smørgrav static const char *
skiphash(void)1200fb3ef04dSDag-Erling Smørgrav skiphash(void)
1201fb3ef04dSDag-Erling Smørgrav {
1202fb3ef04dSDag-Erling Smørgrav 	const char *cp;
1203fb3ef04dSDag-Erling Smørgrav 
1204fb3ef04dSDag-Erling Smørgrav 	linenum++;
1205fb3ef04dSDag-Erling Smørgrav 	if (fgets(tline, MAXLINE, input) == NULL) {
1206fb3ef04dSDag-Erling Smørgrav 		if (ferror(input))
1207fb3ef04dSDag-Erling Smørgrav 			err(2, "can't read %s", filename);
1208fb3ef04dSDag-Erling Smørgrav 		else
1209fb3ef04dSDag-Erling Smørgrav 			return (NULL);
1210fb3ef04dSDag-Erling Smørgrav 	}
1211fb3ef04dSDag-Erling Smørgrav 	cp = skipcomment(tline);
1212fb3ef04dSDag-Erling Smørgrav 	if (linestate == LS_START && *cp == '#') {
1213fb3ef04dSDag-Erling Smørgrav 		linestate = LS_HASH;
1214fb3ef04dSDag-Erling Smørgrav 		return (skipcomment(cp + 1));
1215fb3ef04dSDag-Erling Smørgrav 	} else if (*cp == '\0') {
1216fb3ef04dSDag-Erling Smørgrav 		return (cp);
1217fb3ef04dSDag-Erling Smørgrav 	} else {
1218fb3ef04dSDag-Erling Smørgrav 		return (skipline(cp));
1219fb3ef04dSDag-Erling Smørgrav 	}
1220fb3ef04dSDag-Erling Smørgrav }
1221fb3ef04dSDag-Erling Smørgrav 
1222fb3ef04dSDag-Erling Smørgrav /*
1223fb3ef04dSDag-Erling Smørgrav  * Mark a line dirty and consume the rest of it, keeping track of the
1224fb3ef04dSDag-Erling Smørgrav  * lexical state.
1225fb3ef04dSDag-Erling Smørgrav  */
1226fb3ef04dSDag-Erling Smørgrav static const char *
skipline(const char * cp)1227fb3ef04dSDag-Erling Smørgrav skipline(const char *cp)
1228fb3ef04dSDag-Erling Smørgrav {
1229fb3ef04dSDag-Erling Smørgrav 	const char *pcp;
1230fb3ef04dSDag-Erling Smørgrav 	if (*cp != '\0')
1231fb3ef04dSDag-Erling Smørgrav 		linestate = LS_DIRTY;
1232fb3ef04dSDag-Erling Smørgrav 	while (*cp != '\0') {
1233fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(pcp = cp);
1234fb3ef04dSDag-Erling Smørgrav 		if (pcp == cp)
1235fb3ef04dSDag-Erling Smørgrav 			cp++;
1236fb3ef04dSDag-Erling Smørgrav 	}
1237fb3ef04dSDag-Erling Smørgrav 	return (cp);
1238fb3ef04dSDag-Erling Smørgrav }
1239fb3ef04dSDag-Erling Smørgrav 
1240fb3ef04dSDag-Erling Smørgrav /*
1241fb3ef04dSDag-Erling Smørgrav  * Skip over comments, strings, and character literals and stop at the
1242fb3ef04dSDag-Erling Smørgrav  * next character position that is not whitespace. Between calls we keep
1243fb3ef04dSDag-Erling Smørgrav  * the comment state in the global variable incomment, and we also adjust
1244fb3ef04dSDag-Erling Smørgrav  * the global variable linestate when we see a newline.
1245fb3ef04dSDag-Erling Smørgrav  * XXX: doesn't cope with the buffer splitting inside a state transition.
1246fb3ef04dSDag-Erling Smørgrav  */
1247fb3ef04dSDag-Erling Smørgrav static const char *
skipcomment(const char * cp)1248fb3ef04dSDag-Erling Smørgrav skipcomment(const char *cp)
1249fb3ef04dSDag-Erling Smørgrav {
1250fb3ef04dSDag-Erling Smørgrav 	if (text || ignoring[depth]) {
1251fb3ef04dSDag-Erling Smørgrav 		for (; isspace((unsigned char)*cp); cp++)
1252fb3ef04dSDag-Erling Smørgrav 			if (*cp == '\n')
1253fb3ef04dSDag-Erling Smørgrav 				linestate = LS_START;
1254fb3ef04dSDag-Erling Smørgrav 		return (cp);
1255fb3ef04dSDag-Erling Smørgrav 	}
1256fb3ef04dSDag-Erling Smørgrav 	while (*cp != '\0')
1257fb3ef04dSDag-Erling Smørgrav 		/* don't reset to LS_START after a line continuation */
1258fb3ef04dSDag-Erling Smørgrav 		if (strncmp(cp, "\\\r\n", 3) == 0)
1259fb3ef04dSDag-Erling Smørgrav 			cp += 3;
1260fb3ef04dSDag-Erling Smørgrav 		else if (strncmp(cp, "\\\n", 2) == 0)
1261fb3ef04dSDag-Erling Smørgrav 			cp += 2;
1262fb3ef04dSDag-Erling Smørgrav 		else switch (incomment) {
1263fb3ef04dSDag-Erling Smørgrav 		case NO_COMMENT:
1264fb3ef04dSDag-Erling Smørgrav 			if (strncmp(cp, "/\\\r\n", 4) == 0) {
1265fb3ef04dSDag-Erling Smørgrav 				incomment = STARTING_COMMENT;
1266fb3ef04dSDag-Erling Smørgrav 				cp += 4;
1267fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "/\\\n", 3) == 0) {
1268fb3ef04dSDag-Erling Smørgrav 				incomment = STARTING_COMMENT;
1269fb3ef04dSDag-Erling Smørgrav 				cp += 3;
1270fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "/*", 2) == 0) {
1271fb3ef04dSDag-Erling Smørgrav 				incomment = C_COMMENT;
1272fb3ef04dSDag-Erling Smørgrav 				cp += 2;
1273fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "//", 2) == 0) {
1274fb3ef04dSDag-Erling Smørgrav 				incomment = CXX_COMMENT;
1275fb3ef04dSDag-Erling Smørgrav 				cp += 2;
1276fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "\'", 1) == 0) {
1277fb3ef04dSDag-Erling Smørgrav 				incomment = CHAR_LITERAL;
1278fb3ef04dSDag-Erling Smørgrav 				linestate = LS_DIRTY;
1279fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1280fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "\"", 1) == 0) {
1281fb3ef04dSDag-Erling Smørgrav 				incomment = STRING_LITERAL;
1282fb3ef04dSDag-Erling Smørgrav 				linestate = LS_DIRTY;
1283fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1284fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "R\"(", 3) == 0) {
1285fb3ef04dSDag-Erling Smørgrav 				incomment = RAW_STRING_LITERAL;
1286fb3ef04dSDag-Erling Smørgrav 				linestate = LS_DIRTY;
1287fb3ef04dSDag-Erling Smørgrav 				cp += 3;
1288fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "\n", 1) == 0) {
1289fb3ef04dSDag-Erling Smørgrav 				linestate = LS_START;
1290fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1291fb3ef04dSDag-Erling Smørgrav 			} else if (strchr(" \r\t", *cp) != NULL) {
1292fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1293fb3ef04dSDag-Erling Smørgrav 			} else
1294fb3ef04dSDag-Erling Smørgrav 				return (cp);
1295fb3ef04dSDag-Erling Smørgrav 			continue;
1296fb3ef04dSDag-Erling Smørgrav 		case CXX_COMMENT:
1297fb3ef04dSDag-Erling Smørgrav 			if (strncmp(cp, "\n", 1) == 0) {
1298fb3ef04dSDag-Erling Smørgrav 				incomment = NO_COMMENT;
1299fb3ef04dSDag-Erling Smørgrav 				linestate = LS_START;
1300fb3ef04dSDag-Erling Smørgrav 			}
1301fb3ef04dSDag-Erling Smørgrav 			cp += 1;
1302fb3ef04dSDag-Erling Smørgrav 			continue;
1303fb3ef04dSDag-Erling Smørgrav 		case CHAR_LITERAL:
1304fb3ef04dSDag-Erling Smørgrav 		case STRING_LITERAL:
1305fb3ef04dSDag-Erling Smørgrav 			if ((incomment == CHAR_LITERAL && cp[0] == '\'') ||
1306fb3ef04dSDag-Erling Smørgrav 			    (incomment == STRING_LITERAL && cp[0] == '\"')) {
1307fb3ef04dSDag-Erling Smørgrav 				incomment = NO_COMMENT;
1308fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1309fb3ef04dSDag-Erling Smørgrav 			} else if (cp[0] == '\\') {
1310fb3ef04dSDag-Erling Smørgrav 				if (cp[1] == '\0')
1311fb3ef04dSDag-Erling Smørgrav 					cp += 1;
1312fb3ef04dSDag-Erling Smørgrav 				else
1313fb3ef04dSDag-Erling Smørgrav 					cp += 2;
1314fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "\n", 1) == 0) {
1315fb3ef04dSDag-Erling Smørgrav 				if (incomment == CHAR_LITERAL)
1316fb3ef04dSDag-Erling Smørgrav 					error("Unterminated char literal");
1317fb3ef04dSDag-Erling Smørgrav 				else
1318fb3ef04dSDag-Erling Smørgrav 					error("Unterminated string literal");
1319fb3ef04dSDag-Erling Smørgrav 			} else
1320fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1321fb3ef04dSDag-Erling Smørgrav 			continue;
1322fb3ef04dSDag-Erling Smørgrav 		case RAW_STRING_LITERAL:
1323fb3ef04dSDag-Erling Smørgrav 			if (strncmp(cp, ")\"", 2) == 0) {
1324fb3ef04dSDag-Erling Smørgrav 				incomment = NO_COMMENT;
1325fb3ef04dSDag-Erling Smørgrav 				cp += 2;
1326fb3ef04dSDag-Erling Smørgrav 			} else
1327fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1328fb3ef04dSDag-Erling Smørgrav 			continue;
1329fb3ef04dSDag-Erling Smørgrav 		case C_COMMENT:
1330fb3ef04dSDag-Erling Smørgrav 			if (strncmp(cp, "*\\\r\n", 4) == 0) {
1331fb3ef04dSDag-Erling Smørgrav 				incomment = FINISHING_COMMENT;
1332fb3ef04dSDag-Erling Smørgrav 				cp += 4;
1333fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "*\\\n", 3) == 0) {
1334fb3ef04dSDag-Erling Smørgrav 				incomment = FINISHING_COMMENT;
1335fb3ef04dSDag-Erling Smørgrav 				cp += 3;
1336fb3ef04dSDag-Erling Smørgrav 			} else if (strncmp(cp, "*/", 2) == 0) {
1337fb3ef04dSDag-Erling Smørgrav 				incomment = NO_COMMENT;
1338fb3ef04dSDag-Erling Smørgrav 				cp += 2;
1339fb3ef04dSDag-Erling Smørgrav 			} else
1340fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1341fb3ef04dSDag-Erling Smørgrav 			continue;
1342fb3ef04dSDag-Erling Smørgrav 		case STARTING_COMMENT:
1343fb3ef04dSDag-Erling Smørgrav 			if (*cp == '*') {
1344fb3ef04dSDag-Erling Smørgrav 				incomment = C_COMMENT;
1345fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1346fb3ef04dSDag-Erling Smørgrav 			} else if (*cp == '/') {
1347fb3ef04dSDag-Erling Smørgrav 				incomment = CXX_COMMENT;
1348fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1349fb3ef04dSDag-Erling Smørgrav 			} else {
1350fb3ef04dSDag-Erling Smørgrav 				incomment = NO_COMMENT;
1351fb3ef04dSDag-Erling Smørgrav 				linestate = LS_DIRTY;
1352fb3ef04dSDag-Erling Smørgrav 			}
1353fb3ef04dSDag-Erling Smørgrav 			continue;
1354fb3ef04dSDag-Erling Smørgrav 		case FINISHING_COMMENT:
1355fb3ef04dSDag-Erling Smørgrav 			if (*cp == '/') {
1356fb3ef04dSDag-Erling Smørgrav 				incomment = NO_COMMENT;
1357fb3ef04dSDag-Erling Smørgrav 				cp += 1;
1358fb3ef04dSDag-Erling Smørgrav 			} else
1359fb3ef04dSDag-Erling Smørgrav 				incomment = C_COMMENT;
1360fb3ef04dSDag-Erling Smørgrav 			continue;
1361fb3ef04dSDag-Erling Smørgrav 		default:
1362fb3ef04dSDag-Erling Smørgrav 			abort(); /* bug */
1363fb3ef04dSDag-Erling Smørgrav 		}
1364fb3ef04dSDag-Erling Smørgrav 	return (cp);
1365fb3ef04dSDag-Erling Smørgrav }
1366fb3ef04dSDag-Erling Smørgrav 
1367fb3ef04dSDag-Erling Smørgrav /*
1368fb3ef04dSDag-Erling Smørgrav  * Skip macro arguments.
1369fb3ef04dSDag-Erling Smørgrav  */
1370fb3ef04dSDag-Erling Smørgrav static const char *
skipargs(const char * cp)1371fb3ef04dSDag-Erling Smørgrav skipargs(const char *cp)
1372fb3ef04dSDag-Erling Smørgrav {
1373fb3ef04dSDag-Erling Smørgrav 	const char *ocp = cp;
1374fb3ef04dSDag-Erling Smørgrav 	int level = 0;
1375fb3ef04dSDag-Erling Smørgrav 	cp = skipcomment(cp);
1376fb3ef04dSDag-Erling Smørgrav 	if (*cp != '(')
1377fb3ef04dSDag-Erling Smørgrav 		return (cp);
1378fb3ef04dSDag-Erling Smørgrav 	do {
1379fb3ef04dSDag-Erling Smørgrav 		if (*cp == '(')
1380fb3ef04dSDag-Erling Smørgrav 			level++;
1381fb3ef04dSDag-Erling Smørgrav 		if (*cp == ')')
1382fb3ef04dSDag-Erling Smørgrav 			level--;
1383fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp+1);
1384fb3ef04dSDag-Erling Smørgrav 	} while (level != 0 && *cp != '\0');
1385fb3ef04dSDag-Erling Smørgrav 	if (level == 0)
1386fb3ef04dSDag-Erling Smørgrav 		return (cp);
1387fb3ef04dSDag-Erling Smørgrav 	else
1388fb3ef04dSDag-Erling Smørgrav 	/* Rewind and re-detect the syntax error later. */
1389fb3ef04dSDag-Erling Smørgrav 		return (ocp);
1390fb3ef04dSDag-Erling Smørgrav }
1391fb3ef04dSDag-Erling Smørgrav 
1392fb3ef04dSDag-Erling Smørgrav /*
1393fb3ef04dSDag-Erling Smørgrav  * Skip over an identifier.
1394fb3ef04dSDag-Erling Smørgrav  */
1395fb3ef04dSDag-Erling Smørgrav static const char *
skipsym(const char * cp)1396fb3ef04dSDag-Erling Smørgrav skipsym(const char *cp)
1397fb3ef04dSDag-Erling Smørgrav {
1398fb3ef04dSDag-Erling Smørgrav 	while (!endsym(*cp))
1399fb3ef04dSDag-Erling Smørgrav 		++cp;
1400fb3ef04dSDag-Erling Smørgrav 	return (cp);
1401fb3ef04dSDag-Erling Smørgrav }
1402fb3ef04dSDag-Erling Smørgrav 
1403fb3ef04dSDag-Erling Smørgrav /*
1404fb3ef04dSDag-Erling Smørgrav  * Skip whitespace and take a copy of any following identifier.
1405fb3ef04dSDag-Erling Smørgrav  */
1406fb3ef04dSDag-Erling Smørgrav static const char *
getsym(const char ** cpp)1407fb3ef04dSDag-Erling Smørgrav getsym(const char **cpp)
1408fb3ef04dSDag-Erling Smørgrav {
1409fb3ef04dSDag-Erling Smørgrav 	const char *cp = *cpp, *sym;
1410fb3ef04dSDag-Erling Smørgrav 
1411fb3ef04dSDag-Erling Smørgrav 	cp = skipcomment(cp);
1412fb3ef04dSDag-Erling Smørgrav 	cp = skipsym(sym = cp);
1413fb3ef04dSDag-Erling Smørgrav 	if (cp == sym)
1414fb3ef04dSDag-Erling Smørgrav 		return NULL;
1415fb3ef04dSDag-Erling Smørgrav 	*cpp = cp;
1416fb3ef04dSDag-Erling Smørgrav 	return (xstrdup(sym, cp));
1417fb3ef04dSDag-Erling Smørgrav }
1418fb3ef04dSDag-Erling Smørgrav 
1419fb3ef04dSDag-Erling Smørgrav /*
1420fb3ef04dSDag-Erling Smørgrav  * Check that s (a symbol) matches the start of t, and that the
1421fb3ef04dSDag-Erling Smørgrav  * following character in t is not a symbol character. Returns a
1422fb3ef04dSDag-Erling Smørgrav  * pointer to the following character in t if there is a match,
1423fb3ef04dSDag-Erling Smørgrav  * otherwise NULL.
1424fb3ef04dSDag-Erling Smørgrav  */
1425fb3ef04dSDag-Erling Smørgrav static const char *
matchsym(const char * s,const char * t)1426fb3ef04dSDag-Erling Smørgrav matchsym(const char *s, const char *t)
1427fb3ef04dSDag-Erling Smørgrav {
1428fb3ef04dSDag-Erling Smørgrav 	while (*s != '\0' && *t != '\0')
1429fb3ef04dSDag-Erling Smørgrav 		if (*s != *t)
1430fb3ef04dSDag-Erling Smørgrav 			return (NULL);
1431fb3ef04dSDag-Erling Smørgrav 		else
1432fb3ef04dSDag-Erling Smørgrav 			++s, ++t;
1433fb3ef04dSDag-Erling Smørgrav 	if (*s == '\0' && endsym(*t))
1434fb3ef04dSDag-Erling Smørgrav 		return(t);
1435fb3ef04dSDag-Erling Smørgrav 	else
1436fb3ef04dSDag-Erling Smørgrav 		return(NULL);
1437fb3ef04dSDag-Erling Smørgrav }
1438fb3ef04dSDag-Erling Smørgrav 
1439fb3ef04dSDag-Erling Smørgrav /*
1440fb3ef04dSDag-Erling Smørgrav  * Look for the symbol in the symbol table. If it is found, we return
1441fb3ef04dSDag-Erling Smørgrav  * the symbol table index, else we return -1.
1442fb3ef04dSDag-Erling Smørgrav  */
1443343b776fSDag-Erling Smørgrav static struct macro *
findsym(const char ** strp)1444fb3ef04dSDag-Erling Smørgrav findsym(const char **strp)
1445fb3ef04dSDag-Erling Smørgrav {
1446fb3ef04dSDag-Erling Smørgrav 	const char *str;
1447343b776fSDag-Erling Smørgrav 	char *strkey;
1448343b776fSDag-Erling Smørgrav 	struct macro key, *res;
1449fb3ef04dSDag-Erling Smørgrav 
1450fb3ef04dSDag-Erling Smørgrav 	str = *strp;
1451fb3ef04dSDag-Erling Smørgrav 	*strp = skipsym(str);
1452fb3ef04dSDag-Erling Smørgrav 	if (symlist) {
1453fb3ef04dSDag-Erling Smørgrav 		if (*strp == str)
1454343b776fSDag-Erling Smørgrav 			return (NULL);
1455fb3ef04dSDag-Erling Smørgrav 		if (symdepth && firstsym)
1456fb3ef04dSDag-Erling Smørgrav 			printf("%s%3d", zerosyms ? "" : "\n", depth);
1457fb3ef04dSDag-Erling Smørgrav 		firstsym = zerosyms = false;
1458fb3ef04dSDag-Erling Smørgrav 		printf("%s%.*s%s",
1459fb3ef04dSDag-Erling Smørgrav 		       symdepth ? " " : "",
1460fb3ef04dSDag-Erling Smørgrav 		       (int)(*strp-str), str,
1461fb3ef04dSDag-Erling Smørgrav 		       symdepth ? "" : "\n");
1462fb3ef04dSDag-Erling Smørgrav 		/* we don't care about the value of the symbol */
1463343b776fSDag-Erling Smørgrav 		return (NULL);
1464fb3ef04dSDag-Erling Smørgrav 	}
1465343b776fSDag-Erling Smørgrav 
1466343b776fSDag-Erling Smørgrav 	/*
1467343b776fSDag-Erling Smørgrav 	 * 'str' just points into the current mid-parse input and is not
1468343b776fSDag-Erling Smørgrav 	 * nul-terminated.  We know the length of the symbol, *strp - str, but
1469343b776fSDag-Erling Smørgrav 	 * need to provide a nul-terminated lookup key for RB_FIND's comparison
1470343b776fSDag-Erling Smørgrav 	 * function.  Create one here.
1471343b776fSDag-Erling Smørgrav 	 */
1472343b776fSDag-Erling Smørgrav 	strkey = malloc(*strp - str + 1);
1473343b776fSDag-Erling Smørgrav 	memcpy(strkey, str, *strp - str);
1474343b776fSDag-Erling Smørgrav 	strkey[*strp - str] = 0;
1475343b776fSDag-Erling Smørgrav 
1476343b776fSDag-Erling Smørgrav 	key.name = strkey;
1477343b776fSDag-Erling Smørgrav 	res = RB_FIND(MACROMAP, &macro_tree, &key);
1478343b776fSDag-Erling Smørgrav 	if (res != NULL)
1479343b776fSDag-Erling Smørgrav 		debugsym("findsym", res);
1480343b776fSDag-Erling Smørgrav 
1481343b776fSDag-Erling Smørgrav 	free(strkey);
1482343b776fSDag-Erling Smørgrav 	return (res);
1483fb3ef04dSDag-Erling Smørgrav }
1484fb3ef04dSDag-Erling Smørgrav 
1485fb3ef04dSDag-Erling Smørgrav /*
1486fb3ef04dSDag-Erling Smørgrav  * Resolve indirect symbol values to their final definitions.
1487fb3ef04dSDag-Erling Smørgrav  */
1488fb3ef04dSDag-Erling Smørgrav static void
indirectsym(void)1489fb3ef04dSDag-Erling Smørgrav indirectsym(void)
1490fb3ef04dSDag-Erling Smørgrav {
1491*c63af363SDag-Erling Smørgrav 	struct macro key = { 0 };
1492343b776fSDag-Erling Smørgrav 	int changed;
1493343b776fSDag-Erling Smørgrav 	struct macro *sym, *ind;
1494fb3ef04dSDag-Erling Smørgrav 
1495fb3ef04dSDag-Erling Smørgrav 	do {
1496fb3ef04dSDag-Erling Smørgrav 		changed = 0;
1497343b776fSDag-Erling Smørgrav 		RB_FOREACH(sym, MACROMAP, &macro_tree) {
1498343b776fSDag-Erling Smørgrav 			if (sym->value == NULL)
1499fb3ef04dSDag-Erling Smørgrav 				continue;
1500*c63af363SDag-Erling Smørgrav 			key.name = sym->value;
1501*c63af363SDag-Erling Smørgrav 			ind = RB_FIND(MACROMAP, &macro_tree, &key);
1502343b776fSDag-Erling Smørgrav 			if (ind == NULL || ind == sym ||
1503343b776fSDag-Erling Smørgrav 			    ind->value == NULL ||
1504343b776fSDag-Erling Smørgrav 			    ind->value == sym->value)
1505fb3ef04dSDag-Erling Smørgrav 				continue;
1506fb3ef04dSDag-Erling Smørgrav 			debugsym("indir...", sym);
1507343b776fSDag-Erling Smørgrav 			sym->value = ind->value;
1508fb3ef04dSDag-Erling Smørgrav 			debugsym("...ectsym", sym);
1509fb3ef04dSDag-Erling Smørgrav 			changed++;
1510fb3ef04dSDag-Erling Smørgrav 		}
1511fb3ef04dSDag-Erling Smørgrav 	} while (changed);
1512fb3ef04dSDag-Erling Smørgrav }
1513fb3ef04dSDag-Erling Smørgrav 
1514fb3ef04dSDag-Erling Smørgrav /*
1515fb3ef04dSDag-Erling Smørgrav  * Add a symbol to the symbol table, specified with the format sym=val
1516fb3ef04dSDag-Erling Smørgrav  */
1517fb3ef04dSDag-Erling Smørgrav static void
addsym1(bool ignorethis,bool definethis,char * symval)1518fb3ef04dSDag-Erling Smørgrav addsym1(bool ignorethis, bool definethis, char *symval)
1519fb3ef04dSDag-Erling Smørgrav {
1520fb3ef04dSDag-Erling Smørgrav 	const char *sym, *val;
1521fb3ef04dSDag-Erling Smørgrav 
1522fb3ef04dSDag-Erling Smørgrav 	sym = symval;
1523fb3ef04dSDag-Erling Smørgrav 	val = skipsym(sym);
1524fb3ef04dSDag-Erling Smørgrav 	if (definethis && *val == '=') {
1525fb3ef04dSDag-Erling Smørgrav 		symval[val - sym] = '\0';
1526fb3ef04dSDag-Erling Smørgrav 		val = val + 1;
1527fb3ef04dSDag-Erling Smørgrav 	} else if (*val == '\0') {
1528fb3ef04dSDag-Erling Smørgrav 		val = definethis ? "1" : NULL;
1529fb3ef04dSDag-Erling Smørgrav 	} else {
1530fb3ef04dSDag-Erling Smørgrav 		usage();
1531fb3ef04dSDag-Erling Smørgrav 	}
1532fb3ef04dSDag-Erling Smørgrav 	addsym2(ignorethis, sym, val);
1533fb3ef04dSDag-Erling Smørgrav }
1534fb3ef04dSDag-Erling Smørgrav 
1535fb3ef04dSDag-Erling Smørgrav /*
1536fb3ef04dSDag-Erling Smørgrav  * Add a symbol to the symbol table.
1537fb3ef04dSDag-Erling Smørgrav  */
1538fb3ef04dSDag-Erling Smørgrav static void
addsym2(bool ignorethis,const char * symname,const char * val)1539343b776fSDag-Erling Smørgrav addsym2(bool ignorethis, const char *symname, const char *val)
1540fb3ef04dSDag-Erling Smørgrav {
1541*c63af363SDag-Erling Smørgrav 	struct macro key = { .name = symname };
1542343b776fSDag-Erling Smørgrav 	struct macro *sym, *r;
1543fb3ef04dSDag-Erling Smørgrav 
1544*c63af363SDag-Erling Smørgrav 	sym = RB_FIND(MACROMAP, &macro_tree, &key);
1545343b776fSDag-Erling Smørgrav 	if (sym == NULL) {
1546343b776fSDag-Erling Smørgrav 		sym = calloc(1, sizeof(*sym));
1547343b776fSDag-Erling Smørgrav 		sym->ignore = ignorethis;
1548343b776fSDag-Erling Smørgrav 		sym->name = symname;
1549343b776fSDag-Erling Smørgrav 		sym->value = val;
1550343b776fSDag-Erling Smørgrav 		r = RB_INSERT(MACROMAP, &macro_tree, sym);
1551343b776fSDag-Erling Smørgrav 		assert(r == NULL);
1552343b776fSDag-Erling Smørgrav 		debugsym("addsym", sym);
1553aacbe738SDag-Erling Smørgrav 	} else {
1554aacbe738SDag-Erling Smørgrav 		sym->ignore = ignorethis;
1555aacbe738SDag-Erling Smørgrav 		sym->value = val;
1556aacbe738SDag-Erling Smørgrav 		debugsym("updsym", sym);
1557aacbe738SDag-Erling Smørgrav 	}
1558fb3ef04dSDag-Erling Smørgrav }
1559fb3ef04dSDag-Erling Smørgrav 
1560fb3ef04dSDag-Erling Smørgrav static void
debugsym(const char * why,const struct macro * sym)1561343b776fSDag-Erling Smørgrav debugsym(const char *why, const struct macro *sym)
1562fb3ef04dSDag-Erling Smørgrav {
1563343b776fSDag-Erling Smørgrav 	debug("%s %s%c%s", why, sym->name,
1564343b776fSDag-Erling Smørgrav 	    sym->value ? '=' : ' ',
1565343b776fSDag-Erling Smørgrav 	    sym->value ? sym->value : "undef");
1566fb3ef04dSDag-Erling Smørgrav }
1567fb3ef04dSDag-Erling Smørgrav 
1568fb3ef04dSDag-Erling Smørgrav /*
1569fb3ef04dSDag-Erling Smørgrav  * Add symbols to the symbol table from a file containing
1570fb3ef04dSDag-Erling Smørgrav  * #define and #undef preprocessor directives.
1571fb3ef04dSDag-Erling Smørgrav  */
1572fb3ef04dSDag-Erling Smørgrav static void
defundefile(const char * fn)1573fb3ef04dSDag-Erling Smørgrav defundefile(const char *fn)
1574fb3ef04dSDag-Erling Smørgrav {
1575fb3ef04dSDag-Erling Smørgrav 	filename = fn;
1576fb3ef04dSDag-Erling Smørgrav 	input = fopen(fn, "rb");
1577fb3ef04dSDag-Erling Smørgrav 	if (input == NULL)
1578fb3ef04dSDag-Erling Smørgrav 		err(2, "can't open %s", fn);
1579fb3ef04dSDag-Erling Smørgrav 	linenum = 0;
1580fb3ef04dSDag-Erling Smørgrav 	while (defundef())
1581fb3ef04dSDag-Erling Smørgrav 		;
1582fb3ef04dSDag-Erling Smørgrav 	if (ferror(input))
1583fb3ef04dSDag-Erling Smørgrav 		err(2, "can't read %s", filename);
1584fb3ef04dSDag-Erling Smørgrav 	else
1585fb3ef04dSDag-Erling Smørgrav 		fclose(input);
1586fb3ef04dSDag-Erling Smørgrav 	if (incomment)
1587fb3ef04dSDag-Erling Smørgrav 		error("EOF in comment");
1588fb3ef04dSDag-Erling Smørgrav }
1589fb3ef04dSDag-Erling Smørgrav 
1590fb3ef04dSDag-Erling Smørgrav /*
1591fb3ef04dSDag-Erling Smørgrav  * Read and process one #define or #undef directive
1592fb3ef04dSDag-Erling Smørgrav  */
1593fb3ef04dSDag-Erling Smørgrav static bool
defundef(void)1594fb3ef04dSDag-Erling Smørgrav defundef(void)
1595fb3ef04dSDag-Erling Smørgrav {
1596fb3ef04dSDag-Erling Smørgrav 	const char *cp, *kw, *sym, *val, *end;
1597fb3ef04dSDag-Erling Smørgrav 
1598fb3ef04dSDag-Erling Smørgrav 	cp = skiphash();
1599fb3ef04dSDag-Erling Smørgrav 	if (cp == NULL)
1600fb3ef04dSDag-Erling Smørgrav 		return (false);
1601fb3ef04dSDag-Erling Smørgrav 	if (*cp == '\0')
1602fb3ef04dSDag-Erling Smørgrav 		goto done;
1603fb3ef04dSDag-Erling Smørgrav 	/* strip trailing whitespace, and do a fairly rough check to
1604fb3ef04dSDag-Erling Smørgrav 	   avoid unsupported multi-line preprocessor directives */
1605fb3ef04dSDag-Erling Smørgrav 	end = cp + strlen(cp);
1606fb3ef04dSDag-Erling Smørgrav 	while (end > tline && strchr(" \t\n\r", end[-1]) != NULL)
1607fb3ef04dSDag-Erling Smørgrav 		--end;
1608fb3ef04dSDag-Erling Smørgrav 	if (end > tline && end[-1] == '\\')
1609fb3ef04dSDag-Erling Smørgrav 		Eioccc();
1610fb3ef04dSDag-Erling Smørgrav 
1611fb3ef04dSDag-Erling Smørgrav 	kw = cp;
1612fb3ef04dSDag-Erling Smørgrav 	if ((cp = matchsym("define", kw)) != NULL) {
1613fb3ef04dSDag-Erling Smørgrav 		sym = getsym(&cp);
1614fb3ef04dSDag-Erling Smørgrav 		if (sym == NULL)
1615fb3ef04dSDag-Erling Smørgrav 			error("Missing macro name in #define");
1616fb3ef04dSDag-Erling Smørgrav 		if (*cp == '(') {
1617fb3ef04dSDag-Erling Smørgrav 			val = "1";
1618fb3ef04dSDag-Erling Smørgrav 		} else {
1619fb3ef04dSDag-Erling Smørgrav 			cp = skipcomment(cp);
1620fb3ef04dSDag-Erling Smørgrav 			val = (cp < end) ? xstrdup(cp, end) : "";
1621fb3ef04dSDag-Erling Smørgrav 		}
1622fb3ef04dSDag-Erling Smørgrav 		debug("#define");
1623fb3ef04dSDag-Erling Smørgrav 		addsym2(false, sym, val);
1624fb3ef04dSDag-Erling Smørgrav 	} else if ((cp = matchsym("undef", kw)) != NULL) {
1625fb3ef04dSDag-Erling Smørgrav 		sym = getsym(&cp);
1626fb3ef04dSDag-Erling Smørgrav 		if (sym == NULL)
1627fb3ef04dSDag-Erling Smørgrav 			error("Missing macro name in #undef");
1628fb3ef04dSDag-Erling Smørgrav 		cp = skipcomment(cp);
1629fb3ef04dSDag-Erling Smørgrav 		debug("#undef");
1630fb3ef04dSDag-Erling Smørgrav 		addsym2(false, sym, NULL);
1631fb3ef04dSDag-Erling Smørgrav 	} else {
1632fb3ef04dSDag-Erling Smørgrav 		error("Unrecognized preprocessor directive");
1633fb3ef04dSDag-Erling Smørgrav 	}
1634fb3ef04dSDag-Erling Smørgrav 	skipline(cp);
1635fb3ef04dSDag-Erling Smørgrav done:
1636fb3ef04dSDag-Erling Smørgrav 	debug("parser line %d state %s comment %s line", linenum,
1637fb3ef04dSDag-Erling Smørgrav 	    comment_name[incomment], linestate_name[linestate]);
1638fb3ef04dSDag-Erling Smørgrav 	return (true);
1639fb3ef04dSDag-Erling Smørgrav }
1640fb3ef04dSDag-Erling Smørgrav 
1641fb3ef04dSDag-Erling Smørgrav /*
1642fb3ef04dSDag-Erling Smørgrav  * Concatenate two strings into new memory, checking for failure.
1643fb3ef04dSDag-Erling Smørgrav  */
1644fb3ef04dSDag-Erling Smørgrav static char *
astrcat(const char * s1,const char * s2)1645fb3ef04dSDag-Erling Smørgrav astrcat(const char *s1, const char *s2)
1646fb3ef04dSDag-Erling Smørgrav {
1647fb3ef04dSDag-Erling Smørgrav 	char *s;
1648fb3ef04dSDag-Erling Smørgrav 	int len;
1649fb3ef04dSDag-Erling Smørgrav 	size_t size;
1650fb3ef04dSDag-Erling Smørgrav 
1651fb3ef04dSDag-Erling Smørgrav 	len = snprintf(NULL, 0, "%s%s", s1, s2);
1652fb3ef04dSDag-Erling Smørgrav 	if (len < 0)
1653fb3ef04dSDag-Erling Smørgrav 		err(2, "snprintf");
1654fb3ef04dSDag-Erling Smørgrav 	size = (size_t)len + 1;
1655fb3ef04dSDag-Erling Smørgrav 	s = (char *)malloc(size);
1656fb3ef04dSDag-Erling Smørgrav 	if (s == NULL)
1657fb3ef04dSDag-Erling Smørgrav 		err(2, "malloc");
1658fb3ef04dSDag-Erling Smørgrav 	snprintf(s, size, "%s%s", s1, s2);
1659fb3ef04dSDag-Erling Smørgrav 	return (s);
1660fb3ef04dSDag-Erling Smørgrav }
1661fb3ef04dSDag-Erling Smørgrav 
1662fb3ef04dSDag-Erling Smørgrav /*
1663fb3ef04dSDag-Erling Smørgrav  * Duplicate a segment of a string, checking for failure.
1664fb3ef04dSDag-Erling Smørgrav  */
1665fb3ef04dSDag-Erling Smørgrav static const char *
xstrdup(const char * start,const char * end)1666fb3ef04dSDag-Erling Smørgrav xstrdup(const char *start, const char *end)
1667fb3ef04dSDag-Erling Smørgrav {
1668fb3ef04dSDag-Erling Smørgrav 	size_t n;
1669fb3ef04dSDag-Erling Smørgrav 	char *s;
1670fb3ef04dSDag-Erling Smørgrav 
1671fb3ef04dSDag-Erling Smørgrav 	if (end < start) abort(); /* bug */
1672fb3ef04dSDag-Erling Smørgrav 	n = (size_t)(end - start) + 1;
1673fb3ef04dSDag-Erling Smørgrav 	s = (char *)malloc(n);
1674fb3ef04dSDag-Erling Smørgrav 	if (s == NULL)
1675fb3ef04dSDag-Erling Smørgrav 		err(2, "malloc");
1676fb3ef04dSDag-Erling Smørgrav 	snprintf(s, n, "%s", start);
1677fb3ef04dSDag-Erling Smørgrav 	return (s);
1678fb3ef04dSDag-Erling Smørgrav }
1679fb3ef04dSDag-Erling Smørgrav 
1680fb3ef04dSDag-Erling Smørgrav /*
1681fb3ef04dSDag-Erling Smørgrav  * Diagnostics.
1682fb3ef04dSDag-Erling Smørgrav  */
1683fb3ef04dSDag-Erling Smørgrav static void
debug(const char * msg,...)1684fb3ef04dSDag-Erling Smørgrav debug(const char *msg, ...)
1685fb3ef04dSDag-Erling Smørgrav {
1686fb3ef04dSDag-Erling Smørgrav 	va_list ap;
1687fb3ef04dSDag-Erling Smørgrav 
1688fb3ef04dSDag-Erling Smørgrav 	if (debugging) {
1689fb3ef04dSDag-Erling Smørgrav 		va_start(ap, msg);
1690fb3ef04dSDag-Erling Smørgrav 		vwarnx(msg, ap);
1691fb3ef04dSDag-Erling Smørgrav 		va_end(ap);
1692fb3ef04dSDag-Erling Smørgrav 	}
1693fb3ef04dSDag-Erling Smørgrav }
1694fb3ef04dSDag-Erling Smørgrav 
1695fb3ef04dSDag-Erling Smørgrav static void
error(const char * msg)1696fb3ef04dSDag-Erling Smørgrav error(const char *msg)
1697fb3ef04dSDag-Erling Smørgrav {
1698fb3ef04dSDag-Erling Smørgrav 	if (depth == 0)
1699fb3ef04dSDag-Erling Smørgrav 		warnx("%s: %d: %s", filename, linenum, msg);
1700fb3ef04dSDag-Erling Smørgrav 	else
1701fb3ef04dSDag-Erling Smørgrav 		warnx("%s: %d: %s (#if line %d depth %d)",
1702fb3ef04dSDag-Erling Smørgrav 		    filename, linenum, msg, stifline[depth], depth);
1703fb3ef04dSDag-Erling Smørgrav 	closeio();
1704fb3ef04dSDag-Erling Smørgrav 	errx(2, "Output may be truncated");
1705fb3ef04dSDag-Erling Smørgrav }
1706