xref: /aoo4110/main/rsc/source/rscpp/cpp5.c (revision b1cdbd2c)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #include	<stdio.h>
25 #include	<ctype.h>
26 #include	"cppdef.h"
27 #include	"cpp.h"
28 
29 /*
30  * Evaluate an #if expression.
31  */
32 
33 static char	*opname[] = {		/* For debug and error messages	*/
34 "end of expression", "val", "id",
35   "+",   "-",  "*",  "/",  "%",
36   "<<", ">>",  "&",  "|",  "^",
37   "==", "!=",  "<", "<=", ">=",  ">",
38   "&&", "||",  "?",  ":",  ",",
39   "unary +", "unary -", "~", "!",  "(",  ")", "(none)",
40 };
41 
42 /*
43  * opdope[] has the operator precedence:
44  *     Bits
45  *	  7	Unused (so the value is always positive)
46  *	6-2	Precedence (000x .. 017x)
47  *	1-0	Binary op. flags:
48  *	    01	The binop flag should be set/cleared when this op is seen.
49  *	    10	The new value of the binop flag.
50  * Note:  Expected, New binop
51  * constant	0	1	Binop, end, or ) should follow constants
52  * End of line	1	0	End may not be preceeded by an operator
53  * binary	1	0	Binary op follows a value, value follows.
54  * unary	0	0	Unary op doesn't follow a value, value follows
55  *   (		0	0	Doesn't follow value, value or unop follows
56  *   )		1	1	Follows value.  Op follows.
57  */
58 
59 static char	opdope[OP_MAX] = {
60   0001,					/* End of expression		*/
61   0002,					/* Digit			*/
62   0000,					/* Letter (identifier)		*/
63   0141, 0141, 0151, 0151, 0151,		/* ADD, SUB, MUL, DIV, MOD	*/
64   0131, 0131, 0101, 0071, 0071,		/* ASL, ASR, AND,  OR, XOR	*/
65   0111, 0111, 0121, 0121, 0121,	0121,	/*  EQ,  NE,  LT,  LE,  GE,  GT	*/
66   0061, 0051, 0041, 0041, 0031,		/* ANA, ORO, QUE, COL, CMA	*/
67 /*
68  * Unary op's follow
69  */
70   0160, 0160, 0160, 0160,		/* NEG, PLU, COM, NOT		*/
71   0170, 0013, 0023,			/* LPA, RPA, END		*/
72 };
73 /*
74  * OP_QUE and OP_RPA have alternate precedences:
75  */
76 #define	OP_RPA_PREC	0013
77 #define OP_QUE_PREC	0034
78 
79 /*
80  * S_ANDOR and S_QUEST signal "short-circuit" boolean evaluation, so that
81  *	#if FOO != 0 && 10 / FOO ...
82  * doesn't generate an error message.  They are stored in optab.skip.
83  */
84 #define S_ANDOR		2
85 #define S_QUEST		1
86 
87 typedef struct optab {
88     char	op;			/* Operator			*/
89     char	prec;			/* Its precedence		*/
90     char	skip;			/* Short-circuit: TRUE to skip	*/
91 } OPTAB;
92 static int	evalue;			/* Current value from evallex()	*/
93 
94 #ifdef	nomacargs
95 FILE_LOCAL int
isbinary(op)96 isbinary(op)
97 register int	op;
98 {
99 	return (op >= FIRST_BINOP && op <= LAST_BINOP);
100 }
101 
102 FILE_LOCAL int
isunary(op)103 isunary(op)
104 register int	op;
105 {
106 	return (op >= FIRST_UNOP && op <= LAST_UNOP);
107 }
108 #else
109 #define	isbinary(op)	(op >= FIRST_BINOP && op <= LAST_BINOP)
110 #define	isunary(op)	(op >= FIRST_UNOP  && op <= LAST_UNOP)
111 #endif
112 
113 /*
114  * The following definitions are used to specify basic variable sizes.
115  */
116 
117 #ifndef	S_CHAR
118 #define	S_CHAR		(sizeof (char))
119 #endif
120 #ifndef	S_SINT
121 #define	S_SINT		(sizeof (short int))
122 #endif
123 #ifndef	S_INT
124 #define	S_INT		(sizeof (int))
125 #endif
126 #ifndef	S_LINT
127 #define	S_LINT		(sizeof (long int))
128 #endif
129 #ifndef	S_FLOAT
130 #define	S_FLOAT		(sizeof (float))
131 #endif
132 #ifndef	S_DOUBLE
133 #define	S_DOUBLE	(sizeof (double))
134 #endif
135 #ifndef	S_PCHAR
136 #define	S_PCHAR		(sizeof (char *))
137 #endif
138 #ifndef	S_PSINT
139 #define	S_PSINT		(sizeof (short int *))
140 #endif
141 #ifndef	S_PINT
142 #define	S_PINT		(sizeof (int *))
143 #endif
144 #ifndef	S_PLINT
145 #define	S_PLINT		(sizeof (long int *))
146 #endif
147 #ifndef	S_PFLOAT
148 #define	S_PFLOAT	(sizeof (float *))
149 #endif
150 #ifndef	S_PDOUBLE
151 #define	S_PDOUBLE	(sizeof (double *))
152 #endif
153 #ifndef	S_PFPTR
154 #define S_PFPTR		(sizeof (int (*)()))
155 #endif
156 
157 typedef struct types {
158     short	type;			/* This is the bit if		*/
159     char	*name;			/* this is the token word	*/
160 } TYPES;
161 
162 static TYPES basic_types[] = {
163 	{ T_CHAR,	"char",		},
164 	{ T_INT,	"int",		},
165 	{ T_FLOAT,	"float",	},
166 	{ T_DOUBLE,	"double",	},
167 	{ T_SHORT,	"short",	},
168 	{ T_LONG,	"long",		},
169 	{ T_SIGNED,	"signed",	},
170 	{ T_UNSIGNED,	"unsigned",	},
171 	{ 0,		NULL,		},	/* Signal end		*/
172 };
173 
174 /*
175  * Test_table[] is used to test for illegal combinations.
176  */
177 static short test_table[] = {
178 	T_FLOAT | T_DOUBLE | T_LONG | T_SHORT,
179 	T_FLOAT | T_DOUBLE | T_CHAR | T_INT,
180 	T_FLOAT | T_DOUBLE | T_SIGNED | T_UNSIGNED,
181 	T_LONG  | T_SHORT  | T_CHAR,
182 	0						/* end marker	*/
183 };
184 
185 /*
186  * The order of this table is important -- it is also referenced by
187  * the command line processor to allow run-time overriding of the
188  * built-in size values.  The order must not be changed:
189  *	char, short, int, long, float, double (func pointer)
190  */
191 SIZES size_table[] = {
192     { T_CHAR,	S_CHAR,		S_PCHAR		},	/* char		*/
193     { T_SHORT,	S_SINT,		S_PSINT		},	/* short int	*/
194     { T_INT,	S_INT,		S_PINT		},	/* int		*/
195     { T_LONG,	S_LINT,		S_PLINT		},	/* long		*/
196     { T_FLOAT,	S_FLOAT,	S_PFLOAT	},	/* float	*/
197     { T_DOUBLE,	S_DOUBLE,	S_PDOUBLE	},	/* double	*/
198     { T_FPTR,	0,		S_PFPTR		},	/* int (*()) 	*/
199     { 0,	0,		0		},	/* End of table	*/
200 };
201 
202 
InitCpp5()203 void InitCpp5()
204 {
205 
206 }
207 
208 
209 
210 int
eval()211 eval()
212 /*
213  * Evaluate an expression.  Straight-forward operator precedence.
214  * This is called from control() on encountering an #if statement.
215  * It calls the following routines:
216  * evallex	Lexical analyser -- returns the type and value of
217  *		the next input token.
218  * evaleval	Evaluate the current operator, given the values on
219  *		the value stack.  Returns a pointer to the (new)
220  *		value stack.
221  * For compatiblity with older cpp's, this return returns 1 (TRUE)
222  * if a syntax error is detected.
223  */
224 {
225 	register int	op;		/* Current operator		*/
226 	register int	*valp;		/* -> value vector		*/
227 	register OPTAB	*opp;		/* Operator stack		*/
228 	int		prec;		/* Op precedence		*/
229 	int		binop;		/* Set if binary op. needed	*/
230 	int		op1;		/* Operand from stack		*/
231 	int		skip;		/* For short-circuit testing	*/
232 	int		value[NEXP];	/* Value stack			*/
233 	OPTAB		opstack[NEXP];	/* Operand stack		*/
234 #ifndef ZTC  /* BP */
235     extern int  *evaleval();    /* Does actual evaluation   */
236 #endif
237 	valp = value;
238 	opp = opstack;
239 	opp->op = OP_END;		/* Mark bottom of stack		*/
240 	opp->prec = opdope[OP_END];	/* And its precedence		*/
241 	opp->skip = 0;			/* Not skipping now		*/
242 	binop = 0;
243 again:	;
244 #ifdef	DEBUG_EVAL
245     fprintf( pCppOut, "In #if at again: skip = %d, binop = %d, line is: %s",
246 	    opp->skip, binop, infile->bptr);
247 #endif
248 	if ((op = evallex(opp->skip)) == OP_SUB && binop == 0)
249 	    op = OP_NEG;			/* Unary minus		*/
250 	else if (op == OP_ADD && binop == 0)
251 	    op = OP_PLU;			/* Unary plus		*/
252 	else if (op == OP_FAIL)
253 	    return (1);				/* Error in evallex	*/
254 #ifdef	DEBUG_EVAL
255     fprintf( pCppOut, "op = %s, opdope = %03o, binop = %d, skip = %d\n",
256 	    opname[op], opdope[op], binop, opp->skip);
257 #endif
258 	if (op == DIG) {			/* Value?		*/
259 	    if (binop != 0) {
260 		cerror("misplaced constant in #if", NULLST);
261 		return (1);
262 	    }
263 	    else if (valp >= &value[NEXP-1]) {
264 		cerror("#if value stack overflow", NULLST);
265 		return (1);
266 	    }
267 	    else {
268 #ifdef	DEBUG_EVAL
269         fprintf( pCppOut, "pushing %d onto value stack[%d]\n",
270 		    evalue, valp - value);
271 #endif
272 		*valp++ = evalue;
273 		binop = 1;
274 	    }
275 	    goto again;
276 	}
277 	else if (op > OP_END) {
278 	    cerror("Illegal #if line", NULLST);
279 	    return (1);
280 	}
281 	prec = opdope[op];
282 	if (binop != (prec & 1)) {
283 	    cerror("Operator %s in incorrect context", opname[op]);
284 	    return (1);
285 	}
286 	binop = (prec & 2) >> 1;
287 	for (;;) {
288 #ifdef	DEBUG_EVAL
289         fprintf( pCppOut, "op %s, prec %d., stacked op %s, prec %d, skip %d\n",
290 		opname[op], prec, opname[opp->op], opp->prec, opp->skip);
291 #endif
292 	    if (prec > opp->prec) {
293 		if (op == OP_LPA)
294 		    prec = OP_RPA_PREC;
295 		else if (op == OP_QUE)
296 		    prec = OP_QUE_PREC;
297 		op1 = opp->skip;		/* Save skip for test	*/
298 		/*
299 		 * Push operator onto op. stack.
300 		 */
301 		opp++;
302 		if (opp >= &opstack[NEXP]) {
303 		    cerror("expression stack overflow at op \"%s\"",
304 			opname[op]);
305 		    return (1);
306 		}
307 		opp->op = (char)op;
308 		opp->prec = (char)prec;
309 		skip = (valp[-1] != 0);		/* Short-circuit tester	*/
310 		/*
311 		 * Do the short-circuit stuff here.  Short-circuiting
312 		 * stops automagically when operators are evaluated.
313 		 */
314 		if ((op == OP_ANA && !skip)
315 		 || (op == OP_ORO && skip))
316 		    opp->skip = S_ANDOR;	/* And/or skip starts	*/
317 		else if (op == OP_QUE)		/* Start of ?: operator	*/
318 		    opp->skip = (char)((op1 & S_ANDOR) | ((!skip) ? S_QUEST : 0));
319 		else if (op == OP_COL) {	/* : inverts S_QUEST	*/
320 		    opp->skip = (char)((op1 & S_ANDOR)
321 			      | (((op1 & S_QUEST) != 0) ? 0 : S_QUEST));
322 		}
323 		else {				/* Other ops leave	*/
324 		    opp->skip = (char)op1;		/*  skipping unchanged.	*/
325 		}
326 #ifdef	DEBUG_EVAL
327         fprintf( pCppOut, "stacking %s, valp[-1] == %d at %s",
328 		    opname[op], valp[-1], infile->bptr);
329 		dumpstack(opstack, opp, value, valp);
330 #endif
331 		goto again;
332 	    }
333 	    /*
334 	     * Pop operator from op. stack and evaluate it.
335 	     * End of stack and '(' are specials.
336 	     */
337 	    skip = opp->skip;			/* Remember skip value	*/
338 	    switch ((op1 = opp->op)) {		/* Look at stacked op	*/
339 	    case OP_END:			/* Stack end marker	*/
340 		if (op == OP_EOE)
341 		    return (valp[-1]);		/* Finished ok.		*/
342 		goto again;			/* Read another op.	*/
343 
344 	    case OP_LPA:			/* ( on stack		*/
345 		if (op != OP_RPA) {		/* Matches ) on input	*/
346 		    cerror("unbalanced paren's, op is \"%s\"", opname[op]);
347 		    return (1);
348 		}
349 		opp--;				/* Unstack it		*/
350 		/* goto again;			-- Fall through		*/
351 
352 	    case OP_QUE:
353 		goto again;			/* Evaluate true expr.	*/
354 
355 	    case OP_COL:			/* : on stack.		*/
356 		opp--;				/* Unstack :		*/
357 		if (opp->op != OP_QUE) {	/* Matches ? on stack?	*/
358 		    cerror("Misplaced '?' or ':', previous operator is %s",
359                    opname[(int)opp->op]);
360 		    return (1);
361 		}
362 		/*
363 		 * Evaluate op1.
364 		 */
365 	    default:				/* Others:		*/
366 		opp--;				/* Unstack the operator	*/
367 #ifdef	DEBUG_EVAL
368         fprintf( pCppOut, "Stack before evaluation of %s\n", opname[op1]);
369 		dumpstack(opstack, opp, value, valp);
370 #endif
371 		valp = evaleval(valp, op1, skip);
372 #ifdef	DEBUG_EVAL
373         fprintf( pCppOut, "Stack after evaluation\n");
374 		dumpstack(opstack, opp, value, valp);
375 #endif
376 	    }					/* op1 switch end	*/
377 	}					/* Stack unwind loop	*/
378 }
379 
380 FILE_LOCAL int
evallex(int skip)381 evallex(int skip)
382 /*
383  * Return next eval operator or value.  Called from eval().  It
384  * calls a special-purpose routines for 'char' strings and
385  * numeric values:
386  * evalchar	called to evaluate 'x'
387  * evalnum	called to evaluate numbers.
388  */
389 {
390 	register int	c, c1, t;
391 
392 again:  do {					/* Collect the token	*/
393 	    c = skipws();
394 	    if ((c = macroid(c)) == EOF_CHAR || c == '\n') {
395 		unget();
396 		return (OP_EOE);		/* End of expression	*/
397 	    }
398 	} while ((t = type[c]) == LET && catenate());
399 	if (t == INV) {				/* Total nonsense	*/
400 	    if (!skip) {
401 		if (isascii(c) && isprint(c))
402 		    cierror("illegal character '%c' in #if", c);
403 		else
404 		    cierror("illegal character (%d decimal) in #if", c);
405 	    }
406 	    return (OP_FAIL);
407 	}
408 	else if (t == QUO) {			/* ' or "		*/
409 	    if (c == '\'') {			/* Character constant	*/
410 		evalue = evalchar(skip);	/* Somewhat messy	*/
411 #ifdef	DEBUG_EVAL
412         fprintf( pCppOut, "evalchar returns %d.\n", evalue);
413 #endif
414 		return (DIG);			/* Return a value	*/
415 	    }
416 	    cerror("Can't use a string in an #if", NULLST);
417 	    return (OP_FAIL);
418 	}
419 	else if (t == LET) {			/* ID must be a macro	*/
420 	    if (streq(token, "defined")) {	/* Or defined name	*/
421 		c1 = c = skipws();
422 		if (c == '(')			/* Allow defined(name)	*/
423 		    c = skipws();
424 		if (type[c] == LET) {
425 		    evalue = (lookid(c) != NULL);
426 		    if (c1 != '('		/* Need to balance	*/
427 		     || skipws() == ')')	/* Did we balance?	*/
428 			return (DIG);		/* Parsed ok		*/
429 		}
430 		cerror("Bad #if ... defined() syntax", NULLST);
431 		return (OP_FAIL);
432 	    }
433 	    else if (streq(token, "sizeof"))	/* New sizeof hackery	*/
434 		return (dosizeof());		/* Gets own routine	*/
435 	    /*
436 	     * The Draft ANSI C Standard says that an undefined symbol
437 	     * in an #if has the value zero.  We are a bit pickier,
438 	     * warning except where the programmer was careful to write
439 	     * 		#if defined(foo) ? foo : 0
440 	     */
441 #ifdef STRICT_UNDEF
442 	    if (!skip)
443 		cwarn("undefined symbol \"%s\" in #if, 0 used", token);
444 #endif
445 	    evalue = 0;
446 	    return (DIG);
447 	}
448 	else if (t == DIG) {			/* Numbers are harder	*/
449 	    evalue = evalnum(c);
450 #ifdef	DEBUG_EVAL
451         fprintf( pCppOut, "evalnum returns %d.\n", evalue);
452 #endif
453 	}
454 	else if (strchr("!=<>&|\\", c) != NULL) {
455 	    /*
456 	     * Process a possible multi-byte lexeme.
457 	     */
458 	    c1 = cget();			/* Peek at next char	*/
459 	    switch (c) {
460 	    case '!':
461 		if (c1 == '=')
462 		    return (OP_NE);
463 		break;
464 
465 	    case '=':
466 		if (c1 != '=') {		/* Can't say a=b in #if	*/
467 		    unget();
468 		    cerror("= not allowed in #if", NULLST);
469 		    return (OP_FAIL);
470 		}
471 		return (OP_EQ);
472 
473 	    case '>':
474 	    case '<':
475 		if (c1 == c)
476 		    return ((c == '<') ? OP_ASL : OP_ASR);
477 		else if (c1 == '=')
478 		    return ((c == '<') ? OP_LE  : OP_GE);
479 		break;
480 
481 	    case '|':
482 	    case '&':
483 		if (c1 == c)
484 		    return ((c == '|') ? OP_ORO : OP_ANA);
485 		break;
486 
487 	    case '\\':
488 		if (c1 == '\n')			/* Multi-line if	*/
489 		    goto again;
490 		cerror("Unexpected \\ in #if", NULLST);
491 		return (OP_FAIL);
492 	    }
493 	    unget();
494 	}
495 	return (t);
496 }
497 
498 FILE_LOCAL int
dosizeof()499 dosizeof()
500 /*
501  * Process the sizeof (basic type) operation in an #if string.
502  * Sets evalue to the size and returns
503  *	DIG		success
504  *	OP_FAIL		bad parse or something.
505  */
506 {
507 	register int	c;
508 	register TYPES	*tp;
509 	register SIZES	*sizp;
510 	register short	*testp;
511 	short		typecode;
512 
513 	if ((c = skipws()) != '(')
514 	    goto nogood;
515 	/*
516 	 * Scan off the tokens.
517 	 */
518 	typecode = 0;
519 	while (0 != (c = skipws())) {
520 	    if ((c = macroid(c)) == EOF_CHAR || c == '\n')
521 		goto nogood;			/* End of line is a bug	*/
522 	    else if (c == '(') {		/* thing (*)() func ptr	*/
523 		if (skipws() == '*'
524 		 && skipws() == ')') {		/* We found (*)		*/
525 		    if (skipws() != '(')	/* Let () be optional	*/
526 			unget();
527 		    else if (skipws() != ')')
528 			goto nogood;
529 		    typecode |= T_FPTR;		/* Function pointer	*/
530 		}
531 		else {				/* Junk is a bug	*/
532 		    goto nogood;
533 		}
534 	    }
535 	    else if (type[c] != LET)		/* Exit if not a type	*/
536 		break;
537 	    else if (!catenate()) {		/* Maybe combine tokens	*/
538 		/*
539 		 * Look for this unexpandable token in basic_types.
540 		 * The code accepts "int long" as well as "long int"
541 		 * which is a minor bug as bugs go (and one shared with
542 		 * a lot of C compilers).
543 		 */
544 		for (tp = basic_types; tp->name != NULLST; tp++) {
545 		    if (streq(token, tp->name))
546 			break;
547 		}
548 		if (tp->name == NULLST) {
549 		    cerror("#if sizeof, unknown type \"%s\"", token);
550 		    return (OP_FAIL);
551 		}
552 		typecode |= tp->type;		/* Or in the type bit	*/
553 	    }
554 	}
555 	/*
556 	 * We are at the end of the type scan.  Chew off '*' if necessary.
557 	 */
558 	if (c == '*') {
559 	    typecode |= T_PTR;
560 	    c = skipws();
561 	}
562 	if (c == ')') {				/* Last syntax check	*/
563 	    for (testp = test_table; *testp != 0; testp++) {
564 		if (!bittest(typecode & *testp)) {
565 		    cerror("#if ... sizeof: illegal type combination", NULLST);
566 		    return (OP_FAIL);
567 		}
568 	    }
569 	    /*
570 	     * We assume that all function pointers are the same size:
571 	     *		sizeof (int (*)()) == sizeof (float (*)())
572 	     * We assume that signed and unsigned don't change the size:
573 	     *		sizeof (signed int) == (sizeof unsigned int)
574 	     */
575 	    if ((typecode & T_FPTR) != 0)	/* Function pointer	*/
576 		typecode = T_FPTR | T_PTR;
577 	    else {				/* Var or var * datum	*/
578 		typecode &= ~(T_SIGNED | T_UNSIGNED);
579 		if ((typecode & (T_SHORT | T_LONG)) != 0)
580 		    typecode &= ~T_INT;
581 	    }
582 	    if ((typecode & ~T_PTR) == 0) {
583 		cerror("#if sizeof() error, no type specified", NULLST);
584 		return (OP_FAIL);
585 	    }
586 	    /*
587 	     * Exactly one bit (and possibly T_PTR) may be set.
588 	     */
589 	    for (sizp = size_table; sizp->bits != 0; sizp++) {
590 		if ((typecode & ~T_PTR) == sizp->bits) {
591 		    evalue = ((typecode & T_PTR) != 0)
592 			? sizp->psize : sizp->size;
593 		    return (DIG);
594 		}
595 	    }					/* We shouldn't fail	*/
596 	    cierror("#if ... sizeof: bug, unknown type code 0x%x", typecode);
597 	    return (OP_FAIL);
598 	}
599 
600 nogood:	unget();
601 	cerror("#if ... sizeof() syntax error", NULLST);
602 	return (OP_FAIL);
603 }
604 
605 FILE_LOCAL int
bittest(int value)606 bittest(int value)
607 /*
608  * TRUE if value is zero or exactly one bit is set in value.
609  */
610 {
611 #if (4096 & ~(-4096)) == 0
612 	return ((value & ~(-value)) == 0);
613 #else
614 	/*
615 	 * Do it the hard way (for non 2's complement machines)
616 	 */
617 	return (value == 0 || value ^ (value - 1) == (value * 2 - 1));
618 #endif
619 }
620 
621 FILE_LOCAL int
evalnum(int c)622 evalnum(int c)
623 /*
624  * Expand number for #if lexical analysis.  Note: evalnum recognizes
625  * the unsigned suffix, but only returns a signed int value.
626  */
627 {
628 	register int	value;
629 	register int	base;
630 	register int	c1;
631 
632 	if (c != '0')
633 	    base = 10;
634 	else if ((c = cget()) == 'x' || c == 'X') {
635 		base = 16;
636 		c = cget();
637 	}
638 	else base = 8;
639 	value = 0;
640 	for (;;) {
641 	    c1 = c;
642 	    if (isascii(c) && isupper(c1))
643 		c1 = tolower(c1);
644 #ifdef EBCDIC
645 	    if (c1 <= 'f')
646 #else
647 	    if (c1 >= 'a')
648 #endif
649 		c1 -= ('a' - 10);
650 	    else c1 -= '0';
651 	    if (c1 < 0 || c1 >= base)
652 		break;
653 	    value *= base;
654 	    value += c1;
655 	    c = cget();
656 	}
657 	if (c == 'u' || c == 'U')	/* Unsigned nonsense		*/
658 	    c = cget();
659 	unget();
660 	return (value);
661 }
662 
663 FILE_LOCAL int
evalchar(int skip)664 evalchar(int skip)
665 /*
666  * Get a character constant
667  */
668 {
669 	register int	c;
670 	register int	value;
671 	register int	count;
672 
673 	instring = TRUE;
674 	if ((c = cget()) == '\\') {
675 	    switch ((c = cget())) {
676 	    case 'a':				/* New in Standard	*/
677 #if ('a' == '\a' || '\a' == ALERT)
678 		value = ALERT;			/* Use predefined value	*/
679 #else
680 		value = '\a';			/* Use compiler's value	*/
681 #endif
682 		break;
683 
684 	    case 'b':
685 		value = '\b';
686 		break;
687 
688 	    case 'f':
689 		value = '\f';
690 		break;
691 
692 	    case 'n':
693 		value = '\n';
694 		break;
695 
696 	    case 'r':
697 		value = '\r';
698 		break;
699 
700 	    case 't':
701 		value = '\t';
702 		break;
703 
704 	    case 'v':				/* New in Standard	*/
705 #if ('v' == '\v' || '\v' == VT)
706 		value = VT;			/* Use predefined value	*/
707 #else
708 		value = '\v';			/* Use compiler's value	*/
709 #endif
710 		break;
711 
712 	    case 'x':				/* '\xFF'		*/
713 		count = 3;
714 		value = 0;
715 		while ((((c = get()) >= '0' && c <= '9')
716 		     || (c >= 'a' && c <= 'f')
717 		     || (c >= 'A' && c <= 'F'))
718 		    && (--count >= 0)) {
719 			value *= 16;
720 #ifdef EBCDIC
721 			value += (c <= '9') ? (c - '0') : ((c & 0xF) + 9);
722 #else
723 			value += (c >= '0') ? (c - '0') : ((c & 0xF) + 9);
724 #endif
725 		}
726 		unget();
727 		break;
728 
729 	    default:
730 		if (c >= '0' && c <= '7') {
731 		    count = 3;
732 		    value = 0;
733 		    while (c >= '0' && c <= '7' && --count >= 0) {
734 			value *= 8;
735 			value += (c - '0');
736 			c = get();
737 		    }
738 		    unget();
739 		}
740 		else value = c;
741 		break;
742 	    }
743 	}
744 	else if (c == '\'')
745 	    value = 0;
746 	else value = c;
747 	/*
748 	 * We warn on multi-byte constants and try to hack
749 	 * (big|little)endian machines.
750 	 */
751 #if BIG_ENDIAN
752 	count = 0;
753 #endif
754 	while ((c = get()) != '\'' && c != EOF_CHAR && c != '\n') {
755 	    if (!skip)
756 		ciwarn("multi-byte constant '%c' isn't portable", c);
757 #if BIG_ENDIAN
758 	    count += BITS_CHAR;
759 	    value += (c << count);
760 #else
761 	    value <<= BITS_CHAR;
762 	    value += c;
763 #endif
764 	}
765 	instring = FALSE;
766 	return (value);
767 }
768 
769 FILE_LOCAL int *
evaleval(int * valp,int op,int skip)770 evaleval(int* valp, int op, int skip)
771 /*
772  * Apply the argument operator to the data on the value stack.
773  * One or two values are popped from the value stack and the result
774  * is pushed onto the value stack.
775  *
776  * OP_COL is a special case.
777  *
778  * evaleval() returns the new pointer to the top of the value stack.
779  */
780 {
781 	register int	v1, v2 = 0;
782 
783 	if (isbinary(op))
784 	    v2 = *--valp;
785 	v1 = *--valp;
786 #ifdef	DEBUG_EVAL
787     fprintf( pCppOut, "%s op %s", (isbinary(op)) ? "binary" : "unary",
788 	    opname[op]);
789 	if (isbinary(op))
790         fprintf( pCppOut, ", v2 = %d.", v2);
791     fprintf( pCppOut, ", v1 = %d.\n", v1);
792 #endif
793 	switch (op) {
794 	case OP_EOE:
795 	     break;
796 
797 	case OP_ADD:
798 	    v1 += v2;
799 	    break;
800 
801 	case OP_SUB:
802 	    v1 -= v2;
803 	    break;
804 
805 	case OP_MUL:
806 	    v1 *= v2;
807 	    break;
808 
809 	case OP_DIV:
810 	case OP_MOD:
811 	    if (v2 == 0) {
812 		if (!skip) {
813 		    cwarn("%s by zero in #if, zero result assumed",
814 			(op == OP_DIV) ? "divide" : "mod");
815 		}
816 		v1 = 0;
817 	    }
818 	    else if (op == OP_DIV)
819 		v1 /= v2;
820 	    else
821 		v1 %= v2;
822 	    break;
823 
824 	case OP_ASL:
825 	    v1 <<= v2;
826 	    break;
827 
828 	case OP_ASR:
829 	    v1 >>= v2;
830 	    break;
831 
832 	case OP_AND:
833 	    v1 &= v2;
834 	    break;
835 
836 	case OP_OR:
837 	    v1 |= v2;
838 	    break;
839 
840 	case OP_XOR:
841 	    v1 ^= v2;
842 	    break;
843 
844 	case OP_EQ:
845 	    v1 = (v1 == v2);
846 	    break;
847 
848 	case OP_NE:
849 	    v1 = (v1 != v2);
850 	    break;
851 
852 	case OP_LT:
853 	    v1 = (v1 < v2);
854 	    break;
855 
856 	case OP_LE:
857 	    v1 = (v1 <= v2);
858 	    break;
859 
860 	case OP_GE:
861 	    v1 = (v1 >= v2);
862 	    break;
863 
864 	case OP_GT:
865 	    v1 = (v1 > v2);
866 	    break;
867 
868 	case OP_ANA:
869 	    v1 = (v1 && v2);
870 	    break;
871 
872 	case OP_ORO:
873 	    v1 = (v1 || v2);
874 	    break;
875 
876 	case OP_COL:
877 	    /*
878 	     * v1 has the "true" value, v2 the "false" value.
879 	     * The top of the value stack has the test.
880 	     */
881 	    v1 = (*--valp) ? v1 : v2;
882 	    break;
883 
884 	case OP_NEG:
885 	    v1 = (-v1);
886 	    break;
887 
888 	case OP_PLU:
889 	    break;
890 
891 	case OP_COM:
892 	    v1 = ~v1;
893 	    break;
894 
895 	case OP_NOT:
896 	    v1 = !v1;
897 	    break;
898 
899 	default:
900 	    cierror("#if bug, operand = %d.", op);
901 	    v1 = 0;
902 	}
903 	*valp++ = v1;
904 	return (valp);
905 }
906 
907 #ifdef	DEBUG_EVAL
dumpstack(opstack,opp,value,valp)908 dumpstack(opstack, opp, value, valp)
909 OPTAB		opstack[NEXP];	/* Operand stack		*/
910 register OPTAB	*opp;		/* Operator stack		*/
911 int		value[NEXP];	/* Value stack			*/
912 register int	*valp;		/* -> value vector		*/
913 {
914     fprintf( pCppOut, "index op prec skip name -- op stack at %s", infile->bptr);
915 	while (opp > opstack) {
916         fprintf( pCppOut, " [%2d] %2d  %03o    %d %s\n", opp - opstack,
917 		opp->op, opp->prec, opp->skip, opname[opp->op]);
918 	    opp--;
919 	}
920 	while (--valp >= value) {
921         fprintf( pCppOut, "value[%d] = %d\n", (valp - value), *valp);
922 	}
923 }
924 #endif
925 
926