xref: /freebsd/contrib/bmake/cond.c (revision 6a7405f5a6b639682cacf01e35d561411ff556aa)
1*6a7405f5SSimon J. Gerraty /*	$NetBSD: cond.c,v 1.371 2025/01/11 21:21:33 rillig Exp $	*/
23955d011SMarcel Moolenaar 
33955d011SMarcel Moolenaar /*
43955d011SMarcel Moolenaar  * Copyright (c) 1988, 1989, 1990 The Regents of the University of California.
53955d011SMarcel Moolenaar  * All rights reserved.
63955d011SMarcel Moolenaar  *
73955d011SMarcel Moolenaar  * This code is derived from software contributed to Berkeley by
83955d011SMarcel Moolenaar  * Adam de Boor.
93955d011SMarcel Moolenaar  *
103955d011SMarcel Moolenaar  * Redistribution and use in source and binary forms, with or without
113955d011SMarcel Moolenaar  * modification, are permitted provided that the following conditions
123955d011SMarcel Moolenaar  * are met:
133955d011SMarcel Moolenaar  * 1. Redistributions of source code must retain the above copyright
143955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer.
153955d011SMarcel Moolenaar  * 2. Redistributions in binary form must reproduce the above copyright
163955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer in the
173955d011SMarcel Moolenaar  *    documentation and/or other materials provided with the distribution.
183955d011SMarcel Moolenaar  * 3. Neither the name of the University nor the names of its contributors
193955d011SMarcel Moolenaar  *    may be used to endorse or promote products derived from this software
203955d011SMarcel Moolenaar  *    without specific prior written permission.
213955d011SMarcel Moolenaar  *
223955d011SMarcel Moolenaar  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
233955d011SMarcel Moolenaar  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
243955d011SMarcel Moolenaar  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
253955d011SMarcel Moolenaar  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
263955d011SMarcel Moolenaar  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
273955d011SMarcel Moolenaar  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
283955d011SMarcel Moolenaar  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
293955d011SMarcel Moolenaar  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
303955d011SMarcel Moolenaar  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
313955d011SMarcel Moolenaar  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
323955d011SMarcel Moolenaar  * SUCH DAMAGE.
333955d011SMarcel Moolenaar  */
343955d011SMarcel Moolenaar 
353955d011SMarcel Moolenaar /*
363955d011SMarcel Moolenaar  * Copyright (c) 1988, 1989 by Adam de Boor
373955d011SMarcel Moolenaar  * Copyright (c) 1989 by Berkeley Softworks
383955d011SMarcel Moolenaar  * All rights reserved.
393955d011SMarcel Moolenaar  *
403955d011SMarcel Moolenaar  * This code is derived from software contributed to Berkeley by
413955d011SMarcel Moolenaar  * Adam de Boor.
423955d011SMarcel Moolenaar  *
433955d011SMarcel Moolenaar  * Redistribution and use in source and binary forms, with or without
443955d011SMarcel Moolenaar  * modification, are permitted provided that the following conditions
453955d011SMarcel Moolenaar  * are met:
463955d011SMarcel Moolenaar  * 1. Redistributions of source code must retain the above copyright
473955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer.
483955d011SMarcel Moolenaar  * 2. Redistributions in binary form must reproduce the above copyright
493955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer in the
503955d011SMarcel Moolenaar  *    documentation and/or other materials provided with the distribution.
513955d011SMarcel Moolenaar  * 3. All advertising materials mentioning features or use of this software
523955d011SMarcel Moolenaar  *    must display the following acknowledgement:
533955d011SMarcel Moolenaar  *	This product includes software developed by the University of
543955d011SMarcel Moolenaar  *	California, Berkeley and its contributors.
553955d011SMarcel Moolenaar  * 4. Neither the name of the University nor the names of its contributors
563955d011SMarcel Moolenaar  *    may be used to endorse or promote products derived from this software
573955d011SMarcel Moolenaar  *    without specific prior written permission.
583955d011SMarcel Moolenaar  *
593955d011SMarcel Moolenaar  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
603955d011SMarcel Moolenaar  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
613955d011SMarcel Moolenaar  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
623955d011SMarcel Moolenaar  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
633955d011SMarcel Moolenaar  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
643955d011SMarcel Moolenaar  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
653955d011SMarcel Moolenaar  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
663955d011SMarcel Moolenaar  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
673955d011SMarcel Moolenaar  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
683955d011SMarcel Moolenaar  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
693955d011SMarcel Moolenaar  * SUCH DAMAGE.
703955d011SMarcel Moolenaar  */
713955d011SMarcel Moolenaar 
7206b9b3e0SSimon J. Gerraty /*
7306b9b3e0SSimon J. Gerraty  * Handling of conditionals in a makefile.
743955d011SMarcel Moolenaar  *
753955d011SMarcel Moolenaar  * Interface:
76e2eeea75SSimon J. Gerraty  *	Cond_EvalLine   Evaluate the conditional directive, such as
77e2eeea75SSimon J. Gerraty  *			'.if <cond>', '.elifnmake <cond>', '.else', '.endif'.
783955d011SMarcel Moolenaar  *
79956e45f6SSimon J. Gerraty  *	Cond_EvalCondition
80956e45f6SSimon J. Gerraty  *			Evaluate the conditional, which is either the argument
81956e45f6SSimon J. Gerraty  *			of one of the .if directives or the condition in a
82956e45f6SSimon J. Gerraty  *			':?then:else' variable modifier.
83956e45f6SSimon J. Gerraty  *
84d5e0a182SSimon J. Gerraty  *	Cond_EndFile	At the end of reading a makefile, ensure that the
854fde40d9SSimon J. Gerraty  *			conditional directives are well-balanced.
863955d011SMarcel Moolenaar  */
873955d011SMarcel Moolenaar 
882c3632d1SSimon J. Gerraty #include <errno.h>
893955d011SMarcel Moolenaar 
903955d011SMarcel Moolenaar #include "make.h"
913955d011SMarcel Moolenaar #include "dir.h"
923955d011SMarcel Moolenaar 
93956e45f6SSimon J. Gerraty /*	"@(#)cond.c	8.2 (Berkeley) 1/2/94"	*/
94*6a7405f5SSimon J. Gerraty MAKE_RCSID("$NetBSD: cond.c,v 1.371 2025/01/11 21:21:33 rillig Exp $");
95956e45f6SSimon J. Gerraty 
963955d011SMarcel Moolenaar /*
979f45a3c8SSimon J. Gerraty  * Conditional expressions conform to this grammar:
9812904384SSimon J. Gerraty  *	Or -> And ('||' And)*
9912904384SSimon J. Gerraty  *	And -> Term ('&&' Term)*
100dba7b0efSSimon J. Gerraty  *	Term -> Function '(' Argument ')'
101dba7b0efSSimon J. Gerraty  *	Term -> Leaf Operator Leaf
102dba7b0efSSimon J. Gerraty  *	Term -> Leaf
103dba7b0efSSimon J. Gerraty  *	Term -> '(' Or ')'
104dba7b0efSSimon J. Gerraty  *	Term -> '!' Term
105dba7b0efSSimon J. Gerraty  *	Leaf -> "string"
106dba7b0efSSimon J. Gerraty  *	Leaf -> Number
107dba7b0efSSimon J. Gerraty  *	Leaf -> VariableExpression
1089f45a3c8SSimon J. Gerraty  *	Leaf -> BareWord
109dba7b0efSSimon J. Gerraty  *	Operator -> '==' | '!=' | '>' | '<' | '>=' | '<='
1103955d011SMarcel Moolenaar  *
1119f45a3c8SSimon J. Gerraty  * BareWord is an unquoted string literal, its evaluation depends on the kind
1129f45a3c8SSimon J. Gerraty  * of '.if' directive.
1133955d011SMarcel Moolenaar  *
1149f45a3c8SSimon J. Gerraty  * The tokens are scanned by CondParser_Token, which returns:
115dba7b0efSSimon J. Gerraty  *	TOK_AND		for '&&'
116dba7b0efSSimon J. Gerraty  *	TOK_OR		for '||'
117956e45f6SSimon J. Gerraty  *	TOK_NOT		for '!'
118956e45f6SSimon J. Gerraty  *	TOK_LPAREN	for '('
119956e45f6SSimon J. Gerraty  *	TOK_RPAREN	for ')'
120dba7b0efSSimon J. Gerraty  *
121956e45f6SSimon J. Gerraty  * Other terminal symbols are evaluated using either the default function or
1229f45a3c8SSimon J. Gerraty  * the function given in the terminal, they return either TOK_TRUE, TOK_FALSE
1239f45a3c8SSimon J. Gerraty  * or TOK_ERROR.
1243955d011SMarcel Moolenaar  */
125956e45f6SSimon J. Gerraty typedef enum Token {
126dba7b0efSSimon J. Gerraty 	TOK_FALSE, TOK_TRUE, TOK_AND, TOK_OR, TOK_NOT,
1273955d011SMarcel Moolenaar 	TOK_LPAREN, TOK_RPAREN, TOK_EOF, TOK_NONE, TOK_ERROR
1283955d011SMarcel Moolenaar } Token;
1293955d011SMarcel Moolenaar 
130dba7b0efSSimon J. Gerraty typedef enum ComparisonOp {
131dba7b0efSSimon J. Gerraty 	LT, LE, GT, GE, EQ, NE
132dba7b0efSSimon J. Gerraty } ComparisonOp;
133dba7b0efSSimon J. Gerraty 
134956e45f6SSimon J. Gerraty typedef struct CondParser {
135dba7b0efSSimon J. Gerraty 
136dba7b0efSSimon J. Gerraty 	/*
137dba7b0efSSimon J. Gerraty 	 * The plain '.if ${VAR}' evaluates to true if the value of the
138148ee845SSimon J. Gerraty 	 * expression has length > 0 and is not numerically zero.  The other
139148ee845SSimon J. Gerraty 	 * '.if' variants delegate to evalBare instead, for example '.ifdef
140148ee845SSimon J. Gerraty 	 * ${VAR}' is equivalent to '.if defined(${VAR})', checking whether
141148ee845SSimon J. Gerraty 	 * the variable named by the expression '${VAR}' is defined.
142dba7b0efSSimon J. Gerraty 	 */
143b0c40a00SSimon J. Gerraty 	bool plain;
144dba7b0efSSimon J. Gerraty 
145dba7b0efSSimon J. Gerraty 	/* The function to apply on unquoted bare words. */
14612904384SSimon J. Gerraty 	bool (*evalBare)(const char *);
147b0c40a00SSimon J. Gerraty 	bool negateEvalBare;
148dba7b0efSSimon J. Gerraty 
14912904384SSimon J. Gerraty 	/*
15012904384SSimon J. Gerraty 	 * Whether the left-hand side of a comparison may be an unquoted
15112904384SSimon J. Gerraty 	 * string.  This is allowed for expressions of the form
15212904384SSimon J. Gerraty 	 * ${condition:?:}, see ApplyModifier_IfElse.  Such a condition is
15312904384SSimon J. Gerraty 	 * expanded before it is evaluated, due to ease of implementation.
15412904384SSimon J. Gerraty 	 * This means that at the point where the condition is evaluated,
15512904384SSimon J. Gerraty 	 * make cannot know anymore whether the left-hand side had originally
156d5e0a182SSimon J. Gerraty 	 * been an expression or a plain word.
15712904384SSimon J. Gerraty 	 *
1581d3f2ddcSSimon J. Gerraty 	 * In conditional directives like '.if', the left-hand side must
159d5e0a182SSimon J. Gerraty 	 * either be an expression, a quoted string or a number.
16012904384SSimon J. Gerraty 	 */
16112904384SSimon J. Gerraty 	bool leftUnquotedOK;
16212904384SSimon J. Gerraty 
163956e45f6SSimon J. Gerraty 	const char *p;		/* The remaining condition to parse */
164956e45f6SSimon J. Gerraty 	Token curr;		/* Single push-back token used in parsing */
1653955d011SMarcel Moolenaar 
16606b9b3e0SSimon J. Gerraty 	/*
16706b9b3e0SSimon J. Gerraty 	 * Whether an error message has already been printed for this
168d5e0a182SSimon J. Gerraty 	 * condition.
16906b9b3e0SSimon J. Gerraty 	 */
170b0c40a00SSimon J. Gerraty 	bool printedError;
171956e45f6SSimon J. Gerraty } CondParser;
172956e45f6SSimon J. Gerraty 
173148ee845SSimon J. Gerraty static CondResult CondParser_Or(CondParser *, bool);
1743955d011SMarcel Moolenaar 
1754fde40d9SSimon J. Gerraty unsigned int cond_depth = 0;	/* current .if nesting level */
1763955d011SMarcel Moolenaar 
17712904384SSimon J. Gerraty /* Names for ComparisonOp. */
17812904384SSimon J. Gerraty static const char opname[][3] = { "<", "<=", ">", ">=", "==", "!=" };
17928a6bc81SSimon J. Gerraty 
1809f45a3c8SSimon J. Gerraty MAKE_INLINE bool
skip_string(const char ** pp,const char * str)1819f45a3c8SSimon J. Gerraty skip_string(const char **pp, const char *str)
1823955d011SMarcel Moolenaar {
1839f45a3c8SSimon J. Gerraty 	size_t len = strlen(str);
1849f45a3c8SSimon J. Gerraty 	bool ok = strncmp(*pp, str, len) == 0;
1859f45a3c8SSimon J. Gerraty 	if (ok)
1869f45a3c8SSimon J. Gerraty 		*pp += len;
1879f45a3c8SSimon J. Gerraty 	return ok;
1883955d011SMarcel Moolenaar }
1893955d011SMarcel Moolenaar 
190e2eeea75SSimon J. Gerraty static Token
ToToken(bool cond)191b0c40a00SSimon J. Gerraty ToToken(bool cond)
192e2eeea75SSimon J. Gerraty {
193e2eeea75SSimon J. Gerraty 	return cond ? TOK_TRUE : TOK_FALSE;
194e2eeea75SSimon J. Gerraty }
195e2eeea75SSimon J. Gerraty 
196956e45f6SSimon J. Gerraty static void
CondParser_SkipWhitespace(CondParser * par)197956e45f6SSimon J. Gerraty CondParser_SkipWhitespace(CondParser *par)
1983955d011SMarcel Moolenaar {
199956e45f6SSimon J. Gerraty 	cpp_skip_whitespace(&par->p);
200956e45f6SSimon J. Gerraty }
201956e45f6SSimon J. Gerraty 
20206b9b3e0SSimon J. Gerraty /*
20312904384SSimon J. Gerraty  * Parse a single word, taking into account balanced parentheses as well as
20412904384SSimon J. Gerraty  * embedded expressions.  Used for the argument of a built-in function as
20512904384SSimon J. Gerraty  * well as for bare words, which are then passed to the default function.
20606b9b3e0SSimon J. Gerraty  */
2079f45a3c8SSimon J. Gerraty static char *
ParseWord(const char ** pp,bool doEval)2089f45a3c8SSimon J. Gerraty ParseWord(const char **pp, bool doEval)
20906b9b3e0SSimon J. Gerraty {
210956e45f6SSimon J. Gerraty 	const char *p = *pp;
2111d3f2ddcSSimon J. Gerraty 	Buffer word;
212d5e0a182SSimon J. Gerraty 	int depth;
2133955d011SMarcel Moolenaar 
214d5e0a182SSimon J. Gerraty 	Buf_Init(&word);
2153955d011SMarcel Moolenaar 
216d5e0a182SSimon J. Gerraty 	depth = 0;
2173955d011SMarcel Moolenaar 	for (;;) {
218956e45f6SSimon J. Gerraty 		char ch = *p;
219e2eeea75SSimon J. Gerraty 		if (ch == '\0' || ch == ' ' || ch == '\t')
2203955d011SMarcel Moolenaar 			break;
221d5e0a182SSimon J. Gerraty 		if ((ch == '&' || ch == '|') && depth == 0)
2223955d011SMarcel Moolenaar 			break;
2239f45a3c8SSimon J. Gerraty 		if (ch == '$') {
224*6a7405f5SSimon J. Gerraty 			VarEvalMode emode = doEval ? VARE_EVAL : VARE_PARSE;
2258c973ee2SSimon J. Gerraty 			FStr nestedVal = Var_Parse(&p, SCOPE_CMDLINE, emode);
226956e45f6SSimon J. Gerraty 			/* TODO: handle errors */
2271d3f2ddcSSimon J. Gerraty 			Buf_AddStr(&word, nestedVal.str);
22806b9b3e0SSimon J. Gerraty 			FStr_Done(&nestedVal);
2293955d011SMarcel Moolenaar 			continue;
2303955d011SMarcel Moolenaar 		}
2313955d011SMarcel Moolenaar 		if (ch == '(')
232d5e0a182SSimon J. Gerraty 			depth++;
233d5e0a182SSimon J. Gerraty 		else if (ch == ')' && --depth < 0)
2343955d011SMarcel Moolenaar 			break;
2351d3f2ddcSSimon J. Gerraty 		Buf_AddByte(&word, ch);
236956e45f6SSimon J. Gerraty 		p++;
2373955d011SMarcel Moolenaar 	}
2383955d011SMarcel Moolenaar 
2399f45a3c8SSimon J. Gerraty 	*pp = p;
2403955d011SMarcel Moolenaar 
2411d3f2ddcSSimon J. Gerraty 	return Buf_DoneData(&word);
2429f45a3c8SSimon J. Gerraty }
2439f45a3c8SSimon J. Gerraty 
2449f45a3c8SSimon J. Gerraty /* Parse the function argument, including the surrounding parentheses. */
2459f45a3c8SSimon J. Gerraty static char *
ParseFuncArg(CondParser * par,const char ** pp,bool doEval,const char * func)2469f45a3c8SSimon J. Gerraty ParseFuncArg(CondParser *par, const char **pp, bool doEval, const char *func)
2479f45a3c8SSimon J. Gerraty {
248*6a7405f5SSimon J. Gerraty 	const char *p = *pp, *argStart, *argEnd;
2499f45a3c8SSimon J. Gerraty 	char *res;
2509f45a3c8SSimon J. Gerraty 
251d5e0a182SSimon J. Gerraty 	p++;			/* skip the '(' */
2529f45a3c8SSimon J. Gerraty 	cpp_skip_hspace(&p);
253*6a7405f5SSimon J. Gerraty 	argStart = p;
2549f45a3c8SSimon J. Gerraty 	res = ParseWord(&p, doEval);
255*6a7405f5SSimon J. Gerraty 	argEnd = p;
256e2eeea75SSimon J. Gerraty 	cpp_skip_hspace(&p);
2573955d011SMarcel Moolenaar 
2589f45a3c8SSimon J. Gerraty 	if (*p++ != ')') {
2599f45a3c8SSimon J. Gerraty 		int len = 0;
2609f45a3c8SSimon J. Gerraty 		while (ch_isalpha(func[len]))
2619f45a3c8SSimon J. Gerraty 			len++;
2629f45a3c8SSimon J. Gerraty 
263dba7b0efSSimon J. Gerraty 		Parse_Error(PARSE_FATAL,
264*6a7405f5SSimon J. Gerraty 		    "Missing ')' after argument '%.*s' for '%.*s'",
265*6a7405f5SSimon J. Gerraty 		    (int)(argEnd - argStart), argStart, len, func);
266b0c40a00SSimon J. Gerraty 		par->printedError = true;
2679f45a3c8SSimon J. Gerraty 		free(res);
2689f45a3c8SSimon J. Gerraty 		return NULL;
2693955d011SMarcel Moolenaar 	}
2703955d011SMarcel Moolenaar 
271956e45f6SSimon J. Gerraty 	*pp = p;
2729f45a3c8SSimon J. Gerraty 	return res;
2733955d011SMarcel Moolenaar }
2742c3632d1SSimon J. Gerraty 
2751d3f2ddcSSimon J. Gerraty /* See if the given variable is defined. */
276b0c40a00SSimon J. Gerraty static bool
FuncDefined(const char * var)2779f45a3c8SSimon J. Gerraty FuncDefined(const char *var)
2783955d011SMarcel Moolenaar {
2799f45a3c8SSimon J. Gerraty 	return Var_Exists(SCOPE_CMDLINE, var);
2803955d011SMarcel Moolenaar }
2812c3632d1SSimon J. Gerraty 
2821d3f2ddcSSimon J. Gerraty /* See if a target matching targetPattern is requested to be made. */
283b0c40a00SSimon J. Gerraty static bool
FuncMake(const char * targetPattern)2841d3f2ddcSSimon J. Gerraty FuncMake(const char *targetPattern)
2853955d011SMarcel Moolenaar {
286956e45f6SSimon J. Gerraty 	StringListNode *ln;
287148ee845SSimon J. Gerraty 	bool warned = false;
288956e45f6SSimon J. Gerraty 
289148ee845SSimon J. Gerraty 	for (ln = opts.create.first; ln != NULL; ln = ln->next) {
290148ee845SSimon J. Gerraty 		StrMatchResult res = Str_Match(ln->datum, targetPattern);
291148ee845SSimon J. Gerraty 		if (res.error != NULL && !warned) {
292148ee845SSimon J. Gerraty 			warned = true;
293148ee845SSimon J. Gerraty 			Parse_Error(PARSE_WARNING,
294148ee845SSimon J. Gerraty 			    "%s in pattern argument '%s' to function 'make'",
295148ee845SSimon J. Gerraty 			    res.error, targetPattern);
296148ee845SSimon J. Gerraty 		}
297148ee845SSimon J. Gerraty 		if (res.matched)
298b0c40a00SSimon J. Gerraty 			return true;
299148ee845SSimon J. Gerraty 	}
300b0c40a00SSimon J. Gerraty 	return false;
3013955d011SMarcel Moolenaar }
3022c3632d1SSimon J. Gerraty 
3032c3632d1SSimon J. Gerraty /* See if the given file exists. */
304b0c40a00SSimon J. Gerraty static bool
FuncExists(const char * file)3059f45a3c8SSimon J. Gerraty FuncExists(const char *file)
3063955d011SMarcel Moolenaar {
307b0c40a00SSimon J. Gerraty 	bool result;
3083955d011SMarcel Moolenaar 	char *path;
3093955d011SMarcel Moolenaar 
3109f45a3c8SSimon J. Gerraty 	path = Dir_FindFile(file, &dirSearchPath);
311e2eeea75SSimon J. Gerraty 	DEBUG2(COND, "exists(%s) result is \"%s\"\n",
3129f45a3c8SSimon J. Gerraty 	    file, path != NULL ? path : "");
313e2eeea75SSimon J. Gerraty 	result = path != NULL;
3143955d011SMarcel Moolenaar 	free(path);
3153841c287SSimon J. Gerraty 	return result;
3163955d011SMarcel Moolenaar }
3172c3632d1SSimon J. Gerraty 
3182c3632d1SSimon J. Gerraty /* See if the given node exists and is an actual target. */
319b0c40a00SSimon J. Gerraty static bool
FuncTarget(const char * node)3209f45a3c8SSimon J. Gerraty FuncTarget(const char *node)
3213955d011SMarcel Moolenaar {
3229f45a3c8SSimon J. Gerraty 	GNode *gn = Targ_FindNode(node);
323956e45f6SSimon J. Gerraty 	return gn != NULL && GNode_IsTarget(gn);
3243955d011SMarcel Moolenaar }
3253955d011SMarcel Moolenaar 
32606b9b3e0SSimon J. Gerraty /*
32706b9b3e0SSimon J. Gerraty  * See if the given node exists and is an actual target with commands
32806b9b3e0SSimon J. Gerraty  * associated with it.
32906b9b3e0SSimon J. Gerraty  */
330b0c40a00SSimon J. Gerraty static bool
FuncCommands(const char * node)3319f45a3c8SSimon J. Gerraty FuncCommands(const char *node)
3323955d011SMarcel Moolenaar {
3339f45a3c8SSimon J. Gerraty 	GNode *gn = Targ_FindNode(node);
3349f45a3c8SSimon J. Gerraty 	return gn != NULL && GNode_IsTarget(gn) &&
3359f45a3c8SSimon J. Gerraty 	       !Lst_IsEmpty(&gn->commands);
3363955d011SMarcel Moolenaar }
3372c3632d1SSimon J. Gerraty 
338e2eeea75SSimon J. Gerraty /*
339148ee845SSimon J. Gerraty  * Convert the string to a floating point number.  Accepted formats are
3409f45a3c8SSimon J. Gerraty  * base-10 integer, base-16 integer and finite floating point numbers.
3413955d011SMarcel Moolenaar  */
342b0c40a00SSimon J. Gerraty static bool
TryParseNumber(const char * str,double * out_value)343e2eeea75SSimon J. Gerraty TryParseNumber(const char *str, double *out_value)
3443955d011SMarcel Moolenaar {
345e2eeea75SSimon J. Gerraty 	char *end;
346e2eeea75SSimon J. Gerraty 	unsigned long ul_val;
347e2eeea75SSimon J. Gerraty 	double dbl_val;
3483955d011SMarcel Moolenaar 
349e2eeea75SSimon J. Gerraty 	if (str[0] == '\0') {	/* XXX: why is an empty string a number? */
350e2eeea75SSimon J. Gerraty 		*out_value = 0.0;
351b0c40a00SSimon J. Gerraty 		return true;
352ac3446e9SSimon J. Gerraty 	}
353e2eeea75SSimon J. Gerraty 
35412904384SSimon J. Gerraty 	errno = 0;
355e2eeea75SSimon J. Gerraty 	ul_val = strtoul(str, &end, str[1] == 'x' ? 16 : 10);
356e2eeea75SSimon J. Gerraty 	if (*end == '\0' && errno != ERANGE) {
357e2eeea75SSimon J. Gerraty 		*out_value = str[0] == '-' ? -(double)-ul_val : (double)ul_val;
358b0c40a00SSimon J. Gerraty 		return true;
3593955d011SMarcel Moolenaar 	}
3603955d011SMarcel Moolenaar 
361e2eeea75SSimon J. Gerraty 	if (*end != '\0' && *end != '.' && *end != 'e' && *end != 'E')
362b0c40a00SSimon J. Gerraty 		return false;	/* skip the expensive strtod call */
363e2eeea75SSimon J. Gerraty 	dbl_val = strtod(str, &end);
364e2eeea75SSimon J. Gerraty 	if (*end != '\0')
365b0c40a00SSimon J. Gerraty 		return false;
366e2eeea75SSimon J. Gerraty 
367e2eeea75SSimon J. Gerraty 	*out_value = dbl_val;
368b0c40a00SSimon J. Gerraty 	return true;
3693955d011SMarcel Moolenaar }
3703955d011SMarcel Moolenaar 
371b0c40a00SSimon J. Gerraty static bool
is_separator(char ch)372956e45f6SSimon J. Gerraty is_separator(char ch)
373956e45f6SSimon J. Gerraty {
374b0c40a00SSimon J. Gerraty 	return ch == '\0' || ch_isspace(ch) || ch == '!' || ch == '=' ||
375b0c40a00SSimon J. Gerraty 	       ch == '>' || ch == '<' || ch == ')' /* but not '(' */;
376956e45f6SSimon J. Gerraty }
377956e45f6SSimon J. Gerraty 
378dba7b0efSSimon J. Gerraty /*
379d5e0a182SSimon J. Gerraty  * In a quoted or unquoted string literal or a number, parse an
3808c973ee2SSimon J. Gerraty  * expression and add its value to the buffer.
3818c973ee2SSimon J. Gerraty  *
3828c973ee2SSimon J. Gerraty  * Return whether to continue parsing the leaf.
383dba7b0efSSimon J. Gerraty  *
384dba7b0efSSimon J. Gerraty  * Example: .if x${CENTER}y == "${PREFIX}${SUFFIX}" || 0x${HEX}
385dba7b0efSSimon J. Gerraty  */
386b0c40a00SSimon J. Gerraty static bool
CondParser_StringExpr(CondParser * par,const char * start,bool doEval,bool quoted,Buffer * buf,FStr * inout_str)387dba7b0efSSimon J. Gerraty CondParser_StringExpr(CondParser *par, const char *start,
38812904384SSimon J. Gerraty 		      bool doEval, bool quoted,
38912904384SSimon J. Gerraty 		      Buffer *buf, FStr *inout_str)
390dba7b0efSSimon J. Gerraty {
391b0c40a00SSimon J. Gerraty 	VarEvalMode emode;
3929f45a3c8SSimon J. Gerraty 	const char *p;
393c59c3bf3SSimon J. Gerraty 	bool atStart;		/* true means an expression outside quotes */
394dba7b0efSSimon J. Gerraty 
3958d5c8e21SSimon J. Gerraty 	emode = doEval && quoted ? VARE_EVAL
396*6a7405f5SSimon J. Gerraty 	    : doEval ? VARE_EVAL_DEFINED_LOUD
3978d5c8e21SSimon J. Gerraty 	    : VARE_PARSE;
398dba7b0efSSimon J. Gerraty 
3999f45a3c8SSimon J. Gerraty 	p = par->p;
4009f45a3c8SSimon J. Gerraty 	atStart = p == start;
4018c973ee2SSimon J. Gerraty 	*inout_str = Var_Parse(&p, SCOPE_CMDLINE, emode);
402dba7b0efSSimon J. Gerraty 	/* TODO: handle errors */
403dba7b0efSSimon J. Gerraty 	if (inout_str->str == var_Error) {
404dba7b0efSSimon J. Gerraty 		FStr_Done(inout_str);
405dba7b0efSSimon J. Gerraty 		*inout_str = FStr_InitRefer(NULL);
406b0c40a00SSimon J. Gerraty 		return false;
407dba7b0efSSimon J. Gerraty 	}
4089f45a3c8SSimon J. Gerraty 	par->p = p;
409dba7b0efSSimon J. Gerraty 
410dba7b0efSSimon J. Gerraty 	if (atStart && is_separator(par->p[0]))
411b0c40a00SSimon J. Gerraty 		return false;
412dba7b0efSSimon J. Gerraty 
413dba7b0efSSimon J. Gerraty 	Buf_AddStr(buf, inout_str->str);
414dba7b0efSSimon J. Gerraty 	FStr_Done(inout_str);
415dba7b0efSSimon J. Gerraty 	*inout_str = FStr_InitRefer(NULL);	/* not finished yet */
416b0c40a00SSimon J. Gerraty 	return true;
417dba7b0efSSimon J. Gerraty }
418dba7b0efSSimon J. Gerraty 
419dba7b0efSSimon J. Gerraty /*
420d5e0a182SSimon J. Gerraty  * Parse a string from an expression or an optionally quoted string,
4219f45a3c8SSimon J. Gerraty  * on the left-hand and right-hand sides of comparisons.
4223955d011SMarcel Moolenaar  *
423548bfc56SSimon J. Gerraty  * Return the string without any enclosing quotes, or NULL on error.
424b0c40a00SSimon J. Gerraty  * Sets out_quoted if the leaf was a quoted string literal.
4253955d011SMarcel Moolenaar  */
426548bfc56SSimon J. Gerraty static FStr
CondParser_Leaf(CondParser * par,bool doEval,bool unquotedOK,bool * out_quoted)42712904384SSimon J. Gerraty CondParser_Leaf(CondParser *par, bool doEval, bool unquotedOK,
428548bfc56SSimon J. Gerraty 		bool *out_quoted)
4293955d011SMarcel Moolenaar {
4303955d011SMarcel Moolenaar 	Buffer buf;
43106b9b3e0SSimon J. Gerraty 	FStr str;
432b0c40a00SSimon J. Gerraty 	bool quoted;
4332c3632d1SSimon J. Gerraty 	const char *start;
4343955d011SMarcel Moolenaar 
435e2eeea75SSimon J. Gerraty 	Buf_Init(&buf);
43606b9b3e0SSimon J. Gerraty 	str = FStr_InitRefer(NULL);
437e2eeea75SSimon J. Gerraty 	*out_quoted = quoted = par->p[0] == '"';
438956e45f6SSimon J. Gerraty 	start = par->p;
439e2eeea75SSimon J. Gerraty 	if (quoted)
440956e45f6SSimon J. Gerraty 		par->p++;
44106b9b3e0SSimon J. Gerraty 
44206b9b3e0SSimon J. Gerraty 	while (par->p[0] != '\0' && str.str == NULL) {
443956e45f6SSimon J. Gerraty 		switch (par->p[0]) {
4443955d011SMarcel Moolenaar 		case '\\':
445956e45f6SSimon J. Gerraty 			par->p++;
446956e45f6SSimon J. Gerraty 			if (par->p[0] != '\0') {
447956e45f6SSimon J. Gerraty 				Buf_AddByte(&buf, par->p[0]);
448956e45f6SSimon J. Gerraty 				par->p++;
4493955d011SMarcel Moolenaar 			}
450956e45f6SSimon J. Gerraty 			continue;
4513955d011SMarcel Moolenaar 		case '"':
452956e45f6SSimon J. Gerraty 			par->p++;
453dba7b0efSSimon J. Gerraty 			if (quoted)
4549f45a3c8SSimon J. Gerraty 				goto return_buf;	/* skip the closing quote */
455dba7b0efSSimon J. Gerraty 			Buf_AddByte(&buf, '"');
456956e45f6SSimon J. Gerraty 			continue;
457e2eeea75SSimon J. Gerraty 		case ')':	/* see is_separator */
4583955d011SMarcel Moolenaar 		case '!':
4593955d011SMarcel Moolenaar 		case '=':
4603955d011SMarcel Moolenaar 		case '>':
4613955d011SMarcel Moolenaar 		case '<':
4623955d011SMarcel Moolenaar 		case ' ':
4633955d011SMarcel Moolenaar 		case '\t':
464e2eeea75SSimon J. Gerraty 			if (!quoted)
4659f45a3c8SSimon J. Gerraty 				goto return_buf;
466956e45f6SSimon J. Gerraty 			Buf_AddByte(&buf, par->p[0]);
467956e45f6SSimon J. Gerraty 			par->p++;
468956e45f6SSimon J. Gerraty 			continue;
4693955d011SMarcel Moolenaar 		case '$':
470dba7b0efSSimon J. Gerraty 			if (!CondParser_StringExpr(par,
471dba7b0efSSimon J. Gerraty 			    start, doEval, quoted, &buf, &str))
4729f45a3c8SSimon J. Gerraty 				goto return_str;
473956e45f6SSimon J. Gerraty 			continue;
4743955d011SMarcel Moolenaar 		default:
47512904384SSimon J. Gerraty 			if (!unquotedOK && !quoted && *start != '$' &&
47606b9b3e0SSimon J. Gerraty 			    !ch_isdigit(*start)) {
47706b9b3e0SSimon J. Gerraty 				str = FStr_InitRefer(NULL);
4789f45a3c8SSimon J. Gerraty 				goto return_str;
47928a6bc81SSimon J. Gerraty 			}
480956e45f6SSimon J. Gerraty 			Buf_AddByte(&buf, par->p[0]);
481956e45f6SSimon J. Gerraty 			par->p++;
482956e45f6SSimon J. Gerraty 			continue;
4833955d011SMarcel Moolenaar 		}
4843955d011SMarcel Moolenaar 	}
4859f45a3c8SSimon J. Gerraty return_buf:
486dba7b0efSSimon J. Gerraty 	str = FStr_InitOwn(buf.data);
48712904384SSimon J. Gerraty 	buf.data = NULL;
4889f45a3c8SSimon J. Gerraty return_str:
48912904384SSimon J. Gerraty 	Buf_Done(&buf);
490548bfc56SSimon J. Gerraty 	return str;
4913955d011SMarcel Moolenaar }
4922c3632d1SSimon J. Gerraty 
49306b9b3e0SSimon J. Gerraty /*
49406b9b3e0SSimon J. Gerraty  * Evaluate a "comparison without operator", such as in ".if ${VAR}" or
49506b9b3e0SSimon J. Gerraty  * ".if 0".
49606b9b3e0SSimon J. Gerraty  */
497b0c40a00SSimon J. Gerraty static bool
EvalTruthy(CondParser * par,const char * value,bool quoted)498148ee845SSimon J. Gerraty EvalTruthy(CondParser *par, const char *value, bool quoted)
4993955d011SMarcel Moolenaar {
500e2eeea75SSimon J. Gerraty 	double num;
501956e45f6SSimon J. Gerraty 
502e2eeea75SSimon J. Gerraty 	if (quoted)
503e2eeea75SSimon J. Gerraty 		return value[0] != '\0';
504e2eeea75SSimon J. Gerraty 	if (TryParseNumber(value, &num))
505e2eeea75SSimon J. Gerraty 		return num != 0.0;
506dba7b0efSSimon J. Gerraty 	if (par->plain)
507e2eeea75SSimon J. Gerraty 		return value[0] != '\0';
5089f45a3c8SSimon J. Gerraty 	return par->evalBare(value) != par->negateEvalBare;
509956e45f6SSimon J. Gerraty }
510956e45f6SSimon J. Gerraty 
511956e45f6SSimon J. Gerraty /* Evaluate a numerical comparison, such as in ".if ${VAR} >= 9". */
512b0c40a00SSimon J. Gerraty static bool
EvalCompareNum(double lhs,ComparisonOp op,double rhs)513dba7b0efSSimon J. Gerraty EvalCompareNum(double lhs, ComparisonOp op, double rhs)
514956e45f6SSimon J. Gerraty {
5151d3f2ddcSSimon J. Gerraty 	DEBUG3(COND, "Comparing %f %s %f\n", lhs, opname[op], rhs);
516956e45f6SSimon J. Gerraty 
517dba7b0efSSimon J. Gerraty 	switch (op) {
518dba7b0efSSimon J. Gerraty 	case LT:
519dba7b0efSSimon J. Gerraty 		return lhs < rhs;
520dba7b0efSSimon J. Gerraty 	case LE:
521dba7b0efSSimon J. Gerraty 		return lhs <= rhs;
522dba7b0efSSimon J. Gerraty 	case GT:
523dba7b0efSSimon J. Gerraty 		return lhs > rhs;
524dba7b0efSSimon J. Gerraty 	case GE:
525dba7b0efSSimon J. Gerraty 		return lhs >= rhs;
5264fde40d9SSimon J. Gerraty 	case EQ:
527dba7b0efSSimon J. Gerraty 		return lhs == rhs;
5284fde40d9SSimon J. Gerraty 	default:
5294fde40d9SSimon J. Gerraty 		return lhs != rhs;
530956e45f6SSimon J. Gerraty 	}
531956e45f6SSimon J. Gerraty }
532956e45f6SSimon J. Gerraty 
533956e45f6SSimon J. Gerraty static Token
EvalCompareStr(CondParser * par,const char * lhs,ComparisonOp op,const char * rhs)534dba7b0efSSimon J. Gerraty EvalCompareStr(CondParser *par, const char *lhs,
535dba7b0efSSimon J. Gerraty 	       ComparisonOp op, const char *rhs)
536956e45f6SSimon J. Gerraty {
537dba7b0efSSimon J. Gerraty 	if (op != EQ && op != NE) {
538dba7b0efSSimon J. Gerraty 		Parse_Error(PARSE_FATAL,
5394fde40d9SSimon J. Gerraty 		    "Comparison with '%s' requires both operands "
5404fde40d9SSimon J. Gerraty 		    "'%s' and '%s' to be numeric",
5414fde40d9SSimon J. Gerraty 		    opname[op], lhs, rhs);
542b0c40a00SSimon J. Gerraty 		par->printedError = true;
543956e45f6SSimon J. Gerraty 		return TOK_ERROR;
544956e45f6SSimon J. Gerraty 	}
545956e45f6SSimon J. Gerraty 
5461d3f2ddcSSimon J. Gerraty 	DEBUG3(COND, "Comparing \"%s\" %s \"%s\"\n", lhs, opname[op], rhs);
547dba7b0efSSimon J. Gerraty 	return ToToken((op == EQ) == (strcmp(lhs, rhs) == 0));
548956e45f6SSimon J. Gerraty }
549956e45f6SSimon J. Gerraty 
550956e45f6SSimon J. Gerraty /* Evaluate a comparison, such as "${VAR} == 12345". */
551956e45f6SSimon J. Gerraty static Token
EvalCompare(CondParser * par,const char * lhs,bool lhsQuoted,ComparisonOp op,const char * rhs,bool rhsQuoted)552b0c40a00SSimon J. Gerraty EvalCompare(CondParser *par, const char *lhs, bool lhsQuoted,
553b0c40a00SSimon J. Gerraty 	    ComparisonOp op, const char *rhs, bool rhsQuoted)
554956e45f6SSimon J. Gerraty {
5553955d011SMarcel Moolenaar 	double left, right;
5563955d011SMarcel Moolenaar 
557956e45f6SSimon J. Gerraty 	if (!rhsQuoted && !lhsQuoted)
558956e45f6SSimon J. Gerraty 		if (TryParseNumber(lhs, &left) && TryParseNumber(rhs, &right))
559dba7b0efSSimon J. Gerraty 			return ToToken(EvalCompareNum(left, op, right));
560956e45f6SSimon J. Gerraty 
561dba7b0efSSimon J. Gerraty 	return EvalCompareStr(par, lhs, op, rhs);
562dba7b0efSSimon J. Gerraty }
563dba7b0efSSimon J. Gerraty 
564b0c40a00SSimon J. Gerraty static bool
CondParser_ComparisonOp(CondParser * par,ComparisonOp * out_op)565dba7b0efSSimon J. Gerraty CondParser_ComparisonOp(CondParser *par, ComparisonOp *out_op)
566dba7b0efSSimon J. Gerraty {
567dba7b0efSSimon J. Gerraty 	const char *p = par->p;
568dba7b0efSSimon J. Gerraty 
5699f45a3c8SSimon J. Gerraty 	if (p[0] == '<' && p[1] == '=')
5709f45a3c8SSimon J. Gerraty 		return par->p += 2, *out_op = LE, true;
5719f45a3c8SSimon J. Gerraty 	if (p[0] == '<')
5729f45a3c8SSimon J. Gerraty 		return par->p += 1, *out_op = LT, true;
5739f45a3c8SSimon J. Gerraty 	if (p[0] == '>' && p[1] == '=')
5749f45a3c8SSimon J. Gerraty 		return par->p += 2, *out_op = GE, true;
5759f45a3c8SSimon J. Gerraty 	if (p[0] == '>')
5769f45a3c8SSimon J. Gerraty 		return par->p += 1, *out_op = GT, true;
5779f45a3c8SSimon J. Gerraty 	if (p[0] == '=' && p[1] == '=')
5789f45a3c8SSimon J. Gerraty 		return par->p += 2, *out_op = EQ, true;
5799f45a3c8SSimon J. Gerraty 	if (p[0] == '!' && p[1] == '=')
5809f45a3c8SSimon J. Gerraty 		return par->p += 2, *out_op = NE, true;
581b0c40a00SSimon J. Gerraty 	return false;
582956e45f6SSimon J. Gerraty }
583956e45f6SSimon J. Gerraty 
58406b9b3e0SSimon J. Gerraty /*
58506b9b3e0SSimon J. Gerraty  * Parse a comparison condition such as:
586956e45f6SSimon J. Gerraty  *
587956e45f6SSimon J. Gerraty  *	0
588956e45f6SSimon J. Gerraty  *	${VAR:Mpattern}
589956e45f6SSimon J. Gerraty  *	${VAR} == value
590956e45f6SSimon J. Gerraty  *	${VAR:U0} < 12345
591956e45f6SSimon J. Gerraty  */
592956e45f6SSimon J. Gerraty static Token
CondParser_Comparison(CondParser * par,bool doEval)593b0c40a00SSimon J. Gerraty CondParser_Comparison(CondParser *par, bool doEval)
594956e45f6SSimon J. Gerraty {
595956e45f6SSimon J. Gerraty 	Token t = TOK_ERROR;
59606b9b3e0SSimon J. Gerraty 	FStr lhs, rhs;
597dba7b0efSSimon J. Gerraty 	ComparisonOp op;
598b0c40a00SSimon J. Gerraty 	bool lhsQuoted, rhsQuoted;
599956e45f6SSimon J. Gerraty 
600548bfc56SSimon J. Gerraty 	lhs = CondParser_Leaf(par, doEval, par->leftUnquotedOK, &lhsQuoted);
60106b9b3e0SSimon J. Gerraty 	if (lhs.str == NULL)
602e2eeea75SSimon J. Gerraty 		goto done_lhs;
6033955d011SMarcel Moolenaar 
604956e45f6SSimon J. Gerraty 	CondParser_SkipWhitespace(par);
6053955d011SMarcel Moolenaar 
606dba7b0efSSimon J. Gerraty 	if (!CondParser_ComparisonOp(par, &op)) {
607148ee845SSimon J. Gerraty 		t = ToToken(doEval && EvalTruthy(par, lhs.str, lhsQuoted));
608e2eeea75SSimon J. Gerraty 		goto done_lhs;
6093955d011SMarcel Moolenaar 	}
6103955d011SMarcel Moolenaar 
611956e45f6SSimon J. Gerraty 	CondParser_SkipWhitespace(par);
6123955d011SMarcel Moolenaar 
613956e45f6SSimon J. Gerraty 	if (par->p[0] == '\0') {
614dba7b0efSSimon J. Gerraty 		Parse_Error(PARSE_FATAL,
61512904384SSimon J. Gerraty 		    "Missing right-hand side of operator '%s'", opname[op]);
616b0c40a00SSimon J. Gerraty 		par->printedError = true;
617e2eeea75SSimon J. Gerraty 		goto done_lhs;
6183955d011SMarcel Moolenaar 	}
6193955d011SMarcel Moolenaar 
620548bfc56SSimon J. Gerraty 	rhs = CondParser_Leaf(par, doEval, true, &rhsQuoted);
6211d3f2ddcSSimon J. Gerraty 	t = rhs.str == NULL ? TOK_ERROR
6221d3f2ddcSSimon J. Gerraty 	    : !doEval ? TOK_FALSE
6231d3f2ddcSSimon J. Gerraty 	    : EvalCompare(par, lhs.str, lhsQuoted, op, rhs.str, rhsQuoted);
62406b9b3e0SSimon J. Gerraty 	FStr_Done(&rhs);
6251d3f2ddcSSimon J. Gerraty 
626e2eeea75SSimon J. Gerraty done_lhs:
62706b9b3e0SSimon J. Gerraty 	FStr_Done(&lhs);
6283955d011SMarcel Moolenaar 	return t;
6293955d011SMarcel Moolenaar }
6303955d011SMarcel Moolenaar 
63106b9b3e0SSimon J. Gerraty /*
63206b9b3e0SSimon J. Gerraty  * The argument to empty() is a variable name, optionally followed by
63306b9b3e0SSimon J. Gerraty  * variable modifiers.
63406b9b3e0SSimon J. Gerraty  */
63512904384SSimon J. Gerraty static bool
CondParser_FuncCallEmpty(CondParser * par,bool doEval,Token * out_token)63612904384SSimon J. Gerraty CondParser_FuncCallEmpty(CondParser *par, bool doEval, Token *out_token)
6373955d011SMarcel Moolenaar {
638d5e0a182SSimon J. Gerraty 	const char *p = par->p;
63912904384SSimon J. Gerraty 	Token tok;
64006b9b3e0SSimon J. Gerraty 	FStr val;
6413955d011SMarcel Moolenaar 
642d5e0a182SSimon J. Gerraty 	if (!skip_string(&p, "empty"))
64312904384SSimon J. Gerraty 		return false;
6443955d011SMarcel Moolenaar 
645d5e0a182SSimon J. Gerraty 	cpp_skip_whitespace(&p);
646d5e0a182SSimon J. Gerraty 	if (*p != '(')
64712904384SSimon J. Gerraty 		return false;
64812904384SSimon J. Gerraty 
649d5e0a182SSimon J. Gerraty 	p--;			/* Make p[1] point to the '('. */
6508d5c8e21SSimon J. Gerraty 	val = Var_Parse(&p, SCOPE_CMDLINE, doEval ? VARE_EVAL : VARE_PARSE);
651956e45f6SSimon J. Gerraty 	/* TODO: handle errors */
6523955d011SMarcel Moolenaar 
65312904384SSimon J. Gerraty 	if (val.str == var_Error)
65412904384SSimon J. Gerraty 		tok = TOK_ERROR;
65512904384SSimon J. Gerraty 	else {
65606b9b3e0SSimon J. Gerraty 		cpp_skip_whitespace(&val.str);
6579f45a3c8SSimon J. Gerraty 		tok = ToToken(doEval && val.str[0] == '\0');
6583955d011SMarcel Moolenaar 	}
6593955d011SMarcel Moolenaar 
66012904384SSimon J. Gerraty 	FStr_Done(&val);
66112904384SSimon J. Gerraty 	*out_token = tok;
662d5e0a182SSimon J. Gerraty 	par->p = p;
66312904384SSimon J. Gerraty 	return true;
6643955d011SMarcel Moolenaar }
6653955d011SMarcel Moolenaar 
6662f2a5ecdSSimon J. Gerraty /* Parse a function call expression, such as 'exists(${file})'. */
667b0c40a00SSimon J. Gerraty static bool
CondParser_FuncCall(CondParser * par,bool doEval,Token * out_token)668b0c40a00SSimon J. Gerraty CondParser_FuncCall(CondParser *par, bool doEval, Token *out_token)
6693955d011SMarcel Moolenaar {
6709f45a3c8SSimon J. Gerraty 	char *arg;
6719f45a3c8SSimon J. Gerraty 	const char *p = par->p;
6729f45a3c8SSimon J. Gerraty 	bool (*fn)(const char *);
6739f45a3c8SSimon J. Gerraty 	const char *fn_name = p;
674e2eeea75SSimon J. Gerraty 
6759f45a3c8SSimon J. Gerraty 	if (skip_string(&p, "defined"))
6769f45a3c8SSimon J. Gerraty 		fn = FuncDefined;
6779f45a3c8SSimon J. Gerraty 	else if (skip_string(&p, "make"))
6789f45a3c8SSimon J. Gerraty 		fn = FuncMake;
6799f45a3c8SSimon J. Gerraty 	else if (skip_string(&p, "exists"))
6809f45a3c8SSimon J. Gerraty 		fn = FuncExists;
6819f45a3c8SSimon J. Gerraty 	else if (skip_string(&p, "target"))
6829f45a3c8SSimon J. Gerraty 		fn = FuncTarget;
6839f45a3c8SSimon J. Gerraty 	else if (skip_string(&p, "commands"))
6849f45a3c8SSimon J. Gerraty 		fn = FuncCommands;
6859f45a3c8SSimon J. Gerraty 	else
68612904384SSimon J. Gerraty 		return false;
687e2eeea75SSimon J. Gerraty 
6889f45a3c8SSimon J. Gerraty 	cpp_skip_whitespace(&p);
6899f45a3c8SSimon J. Gerraty 	if (*p != '(')
69012904384SSimon J. Gerraty 		return false;
691e2eeea75SSimon J. Gerraty 
6929f45a3c8SSimon J. Gerraty 	arg = ParseFuncArg(par, &p, doEval, fn_name);
6939f45a3c8SSimon J. Gerraty 	*out_token = ToToken(doEval &&
6949f45a3c8SSimon J. Gerraty 	    arg != NULL && arg[0] != '\0' && fn(arg));
695e2eeea75SSimon J. Gerraty 	free(arg);
6969f45a3c8SSimon J. Gerraty 
6979f45a3c8SSimon J. Gerraty 	par->p = p;
698b0c40a00SSimon J. Gerraty 	return true;
699e2eeea75SSimon J. Gerraty }
700e2eeea75SSimon J. Gerraty 
70106b9b3e0SSimon J. Gerraty /*
70212904384SSimon J. Gerraty  * Parse a comparison that neither starts with '"' nor '$', such as the
70312904384SSimon J. Gerraty  * unusual 'bare == right' or '3 == ${VAR}', or a simple leaf without
704d5e0a182SSimon J. Gerraty  * operator, which is a number, an expression or a string literal.
70512904384SSimon J. Gerraty  *
70612904384SSimon J. Gerraty  * TODO: Can this be merged into CondParser_Comparison?
70706b9b3e0SSimon J. Gerraty  */
708e2eeea75SSimon J. Gerraty static Token
CondParser_ComparisonOrLeaf(CondParser * par,bool doEval)709b0c40a00SSimon J. Gerraty CondParser_ComparisonOrLeaf(CondParser *par, bool doEval)
710e2eeea75SSimon J. Gerraty {
7113955d011SMarcel Moolenaar 	Token t;
7129f45a3c8SSimon J. Gerraty 	char *arg;
713d5e0a182SSimon J. Gerraty 	const char *p;
7143955d011SMarcel Moolenaar 
715d5e0a182SSimon J. Gerraty 	p = par->p;
716d5e0a182SSimon J. Gerraty 	if (ch_isdigit(p[0]) || p[0] == '-' || p[0] == '+')
717956e45f6SSimon J. Gerraty 		return CondParser_Comparison(par, doEval);
7183955d011SMarcel Moolenaar 
7193955d011SMarcel Moolenaar 	/*
720d5e0a182SSimon J. Gerraty 	 * Most likely we have a bare word to apply the default function to.
721d5e0a182SSimon J. Gerraty 	 * However, ".if a == b" gets here when the "a" is unquoted and
722d5e0a182SSimon J. Gerraty 	 * doesn't start with a '$'. This surprises people.
72306b9b3e0SSimon J. Gerraty 	 * If what follows the function argument is a '=' or '!' then the
72406b9b3e0SSimon J. Gerraty 	 * syntax would be invalid if we did "defined(a)" - so instead treat
72506b9b3e0SSimon J. Gerraty 	 * as an expression.
7263955d011SMarcel Moolenaar 	 */
727b0c40a00SSimon J. Gerraty 	/*
728d5e0a182SSimon J. Gerraty 	 * XXX: In edge cases, an expression may be evaluated twice,
7299f45a3c8SSimon J. Gerraty 	 *  see cond-token-plain.mk, keyword 'twice'.
730b0c40a00SSimon J. Gerraty 	 */
731d5e0a182SSimon J. Gerraty 	arg = ParseWord(&p, doEval);
7329f45a3c8SSimon J. Gerraty 	assert(arg[0] != '\0');
733*6a7405f5SSimon J. Gerraty 	cpp_skip_hspace(&p);
7349f45a3c8SSimon J. Gerraty 
7358d5c8e21SSimon J. Gerraty 	if (*p == '=' || *p == '!' || *p == '<' || *p == '>') {
7368d5c8e21SSimon J. Gerraty 		free(arg);
737956e45f6SSimon J. Gerraty 		return CondParser_Comparison(par, doEval);
7388d5c8e21SSimon J. Gerraty 	}
739d5e0a182SSimon J. Gerraty 	par->p = p;
7403955d011SMarcel Moolenaar 
7413955d011SMarcel Moolenaar 	/*
7423955d011SMarcel Moolenaar 	 * Evaluate the argument using the default function.
743956e45f6SSimon J. Gerraty 	 * This path always treats .if as .ifdef. To get here, the character
7443955d011SMarcel Moolenaar 	 * after .if must have been taken literally, so the argument cannot
745d5e0a182SSimon J. Gerraty 	 * be empty - even if it contained an expression.
7463955d011SMarcel Moolenaar 	 */
7479f45a3c8SSimon J. Gerraty 	t = ToToken(doEval && par->evalBare(arg) != par->negateEvalBare);
7483955d011SMarcel Moolenaar 	free(arg);
7493955d011SMarcel Moolenaar 	return t;
7503955d011SMarcel Moolenaar }
7513955d011SMarcel Moolenaar 
752956e45f6SSimon J. Gerraty /* Return the next token or comparison result from the parser. */
7533955d011SMarcel Moolenaar static Token
CondParser_Token(CondParser * par,bool doEval)754b0c40a00SSimon J. Gerraty CondParser_Token(CondParser *par, bool doEval)
7553955d011SMarcel Moolenaar {
7563955d011SMarcel Moolenaar 	Token t;
7573955d011SMarcel Moolenaar 
758956e45f6SSimon J. Gerraty 	t = par->curr;
7593955d011SMarcel Moolenaar 	if (t != TOK_NONE) {
760956e45f6SSimon J. Gerraty 		par->curr = TOK_NONE;
7613955d011SMarcel Moolenaar 		return t;
7623955d011SMarcel Moolenaar 	}
7633955d011SMarcel Moolenaar 
764e2eeea75SSimon J. Gerraty 	cpp_skip_hspace(&par->p);
7653955d011SMarcel Moolenaar 
766956e45f6SSimon J. Gerraty 	switch (par->p[0]) {
7673955d011SMarcel Moolenaar 
7683955d011SMarcel Moolenaar 	case '(':
769956e45f6SSimon J. Gerraty 		par->p++;
7703955d011SMarcel Moolenaar 		return TOK_LPAREN;
7713955d011SMarcel Moolenaar 
7723955d011SMarcel Moolenaar 	case ')':
773956e45f6SSimon J. Gerraty 		par->p++;
7743955d011SMarcel Moolenaar 		return TOK_RPAREN;
7753955d011SMarcel Moolenaar 
7763955d011SMarcel Moolenaar 	case '|':
777956e45f6SSimon J. Gerraty 		par->p++;
778e2eeea75SSimon J. Gerraty 		if (par->p[0] == '|')
779956e45f6SSimon J. Gerraty 			par->p++;
78022619282SSimon J. Gerraty 		else {
781e2eeea75SSimon J. Gerraty 			Parse_Error(PARSE_FATAL, "Unknown operator '|'");
782b0c40a00SSimon J. Gerraty 			par->printedError = true;
783e2eeea75SSimon J. Gerraty 			return TOK_ERROR;
7843955d011SMarcel Moolenaar 		}
7853955d011SMarcel Moolenaar 		return TOK_OR;
7863955d011SMarcel Moolenaar 
7873955d011SMarcel Moolenaar 	case '&':
788956e45f6SSimon J. Gerraty 		par->p++;
789e2eeea75SSimon J. Gerraty 		if (par->p[0] == '&')
790956e45f6SSimon J. Gerraty 			par->p++;
79122619282SSimon J. Gerraty 		else {
792e2eeea75SSimon J. Gerraty 			Parse_Error(PARSE_FATAL, "Unknown operator '&'");
793b0c40a00SSimon J. Gerraty 			par->printedError = true;
794e2eeea75SSimon J. Gerraty 			return TOK_ERROR;
7953955d011SMarcel Moolenaar 		}
7963955d011SMarcel Moolenaar 		return TOK_AND;
7973955d011SMarcel Moolenaar 
7983955d011SMarcel Moolenaar 	case '!':
799956e45f6SSimon J. Gerraty 		par->p++;
8003955d011SMarcel Moolenaar 		return TOK_NOT;
8013955d011SMarcel Moolenaar 
802e2eeea75SSimon J. Gerraty 	case '#':		/* XXX: see unit-tests/cond-token-plain.mk */
803e2eeea75SSimon J. Gerraty 	case '\n':		/* XXX: why should this end the condition? */
804e2eeea75SSimon J. Gerraty 		/* Probably obsolete now, from 1993-03-21. */
8053955d011SMarcel Moolenaar 	case '\0':
8063955d011SMarcel Moolenaar 		return TOK_EOF;
8073955d011SMarcel Moolenaar 
8083955d011SMarcel Moolenaar 	case '"':
8093955d011SMarcel Moolenaar 	case '$':
810956e45f6SSimon J. Gerraty 		return CondParser_Comparison(par, doEval);
8113955d011SMarcel Moolenaar 
8123955d011SMarcel Moolenaar 	default:
81312904384SSimon J. Gerraty 		if (CondParser_FuncCallEmpty(par, doEval, &t))
81412904384SSimon J. Gerraty 			return t;
815b0c40a00SSimon J. Gerraty 		if (CondParser_FuncCall(par, doEval, &t))
816b0c40a00SSimon J. Gerraty 			return t;
817b0c40a00SSimon J. Gerraty 		return CondParser_ComparisonOrLeaf(par, doEval);
8183955d011SMarcel Moolenaar 	}
8193955d011SMarcel Moolenaar }
8203955d011SMarcel Moolenaar 
82112904384SSimon J. Gerraty /* Skip the next token if it equals t. */
82212904384SSimon J. Gerraty static bool
CondParser_Skip(CondParser * par,Token t)82312904384SSimon J. Gerraty CondParser_Skip(CondParser *par, Token t)
82412904384SSimon J. Gerraty {
82512904384SSimon J. Gerraty 	Token actual;
82612904384SSimon J. Gerraty 
82712904384SSimon J. Gerraty 	actual = CondParser_Token(par, false);
82812904384SSimon J. Gerraty 	if (actual == t)
82912904384SSimon J. Gerraty 		return true;
83012904384SSimon J. Gerraty 
83112904384SSimon J. Gerraty 	assert(par->curr == TOK_NONE);
83212904384SSimon J. Gerraty 	assert(actual != TOK_NONE);
83312904384SSimon J. Gerraty 	par->curr = actual;
83412904384SSimon J. Gerraty 	return false;
83512904384SSimon J. Gerraty }
83612904384SSimon J. Gerraty 
83706b9b3e0SSimon J. Gerraty /*
838dba7b0efSSimon J. Gerraty  * Term -> '(' Or ')'
839dba7b0efSSimon J. Gerraty  * Term -> '!' Term
840dba7b0efSSimon J. Gerraty  * Term -> Leaf Operator Leaf
841dba7b0efSSimon J. Gerraty  * Term -> Leaf
8423955d011SMarcel Moolenaar  */
843dba7b0efSSimon J. Gerraty static CondResult
CondParser_Term(CondParser * par,bool doEval)844b0c40a00SSimon J. Gerraty CondParser_Term(CondParser *par, bool doEval)
8453955d011SMarcel Moolenaar {
846dba7b0efSSimon J. Gerraty 	CondResult res;
8473955d011SMarcel Moolenaar 	Token t;
848c59c3bf3SSimon J. Gerraty 	bool neg = false;
8493955d011SMarcel Moolenaar 
850c59c3bf3SSimon J. Gerraty 	while ((t = CondParser_Token(par, doEval)) == TOK_NOT)
851c59c3bf3SSimon J. Gerraty 		neg = !neg;
852c59c3bf3SSimon J. Gerraty 
853c59c3bf3SSimon J. Gerraty 	if (t == TOK_TRUE || t == TOK_FALSE)
854c59c3bf3SSimon J. Gerraty 		return neg == (t == TOK_FALSE) ? CR_TRUE : CR_FALSE;
8553955d011SMarcel Moolenaar 
856dba7b0efSSimon J. Gerraty 	if (t == TOK_LPAREN) {
857dba7b0efSSimon J. Gerraty 		res = CondParser_Or(par, doEval);
858dba7b0efSSimon J. Gerraty 		if (res == CR_ERROR)
859dba7b0efSSimon J. Gerraty 			return CR_ERROR;
860dba7b0efSSimon J. Gerraty 		if (CondParser_Token(par, doEval) != TOK_RPAREN)
861dba7b0efSSimon J. Gerraty 			return CR_ERROR;
862c59c3bf3SSimon J. Gerraty 		return neg == (res == CR_FALSE) ? CR_TRUE : CR_FALSE;
8633955d011SMarcel Moolenaar 	}
864dba7b0efSSimon J. Gerraty 
865dba7b0efSSimon J. Gerraty 	return CR_ERROR;
8663955d011SMarcel Moolenaar }
8672c3632d1SSimon J. Gerraty 
86806b9b3e0SSimon J. Gerraty /*
86912904384SSimon J. Gerraty  * And -> Term ('&&' Term)*
8703955d011SMarcel Moolenaar  */
871dba7b0efSSimon J. Gerraty static CondResult
CondParser_And(CondParser * par,bool doEval)872b0c40a00SSimon J. Gerraty CondParser_And(CondParser *par, bool doEval)
8733955d011SMarcel Moolenaar {
87412904384SSimon J. Gerraty 	CondResult res, rhs;
8753955d011SMarcel Moolenaar 
87612904384SSimon J. Gerraty 	res = CR_TRUE;
87712904384SSimon J. Gerraty 	do {
87812904384SSimon J. Gerraty 		if ((rhs = CondParser_Term(par, doEval)) == CR_ERROR)
879dba7b0efSSimon J. Gerraty 			return CR_ERROR;
88012904384SSimon J. Gerraty 		if (rhs == CR_FALSE) {
88112904384SSimon J. Gerraty 			res = CR_FALSE;
88212904384SSimon J. Gerraty 			doEval = false;
8833955d011SMarcel Moolenaar 		}
88412904384SSimon J. Gerraty 	} while (CondParser_Skip(par, TOK_AND));
885dba7b0efSSimon J. Gerraty 
886dba7b0efSSimon J. Gerraty 	return res;
8873955d011SMarcel Moolenaar }
8882c3632d1SSimon J. Gerraty 
88906b9b3e0SSimon J. Gerraty /*
89012904384SSimon J. Gerraty  * Or -> And ('||' And)*
8913955d011SMarcel Moolenaar  */
892dba7b0efSSimon J. Gerraty static CondResult
CondParser_Or(CondParser * par,bool doEval)893b0c40a00SSimon J. Gerraty CondParser_Or(CondParser *par, bool doEval)
8943955d011SMarcel Moolenaar {
89512904384SSimon J. Gerraty 	CondResult res, rhs;
8963955d011SMarcel Moolenaar 
89712904384SSimon J. Gerraty 	res = CR_FALSE;
89812904384SSimon J. Gerraty 	do {
89912904384SSimon J. Gerraty 		if ((rhs = CondParser_And(par, doEval)) == CR_ERROR)
900dba7b0efSSimon J. Gerraty 			return CR_ERROR;
90112904384SSimon J. Gerraty 		if (rhs == CR_TRUE) {
90212904384SSimon J. Gerraty 			res = CR_TRUE;
90312904384SSimon J. Gerraty 			doEval = false;
9043955d011SMarcel Moolenaar 		}
90512904384SSimon J. Gerraty 	} while (CondParser_Skip(par, TOK_OR));
906dba7b0efSSimon J. Gerraty 
907dba7b0efSSimon J. Gerraty 	return res;
9083955d011SMarcel Moolenaar }
9093955d011SMarcel Moolenaar 
91006b9b3e0SSimon J. Gerraty /*
911d5e0a182SSimon J. Gerraty  * Evaluate the condition, including any side effects from the
912956e45f6SSimon J. Gerraty  * expressions in the condition. The condition consists of &&, ||, !,
913956e45f6SSimon J. Gerraty  * function(arg), comparisons and parenthetical groupings thereof.
9143955d011SMarcel Moolenaar  */
9159f45a3c8SSimon J. Gerraty static CondResult
CondEvalExpression(const char * cond,bool plain,bool (* evalBare)(const char *),bool negate,bool eprint,bool leftUnquotedOK)9169f45a3c8SSimon J. Gerraty CondEvalExpression(const char *cond, bool plain,
91712904384SSimon J. Gerraty 		   bool (*evalBare)(const char *), bool negate,
91812904384SSimon J. Gerraty 		   bool eprint, bool leftUnquotedOK)
9193955d011SMarcel Moolenaar {
920956e45f6SSimon J. Gerraty 	CondParser par;
9219f45a3c8SSimon J. Gerraty 	CondResult rval;
922*6a7405f5SSimon J. Gerraty 	int parseErrorsBefore = parseErrors;
9233955d011SMarcel Moolenaar 
924e2eeea75SSimon J. Gerraty 	cpp_skip_hspace(&cond);
9253955d011SMarcel Moolenaar 
926dba7b0efSSimon J. Gerraty 	par.plain = plain;
927dba7b0efSSimon J. Gerraty 	par.evalBare = evalBare;
928dba7b0efSSimon J. Gerraty 	par.negateEvalBare = negate;
92912904384SSimon J. Gerraty 	par.leftUnquotedOK = leftUnquotedOK;
930956e45f6SSimon J. Gerraty 	par.p = cond;
931956e45f6SSimon J. Gerraty 	par.curr = TOK_NONE;
932b0c40a00SSimon J. Gerraty 	par.printedError = false;
9333955d011SMarcel Moolenaar 
934c59c3bf3SSimon J. Gerraty 	DEBUG1(COND, "CondParser_Eval: %s\n", par.p);
935c59c3bf3SSimon J. Gerraty 	rval = CondParser_Or(&par, true);
936c59c3bf3SSimon J. Gerraty 	if (par.curr != TOK_EOF)
937c59c3bf3SSimon J. Gerraty 		rval = CR_ERROR;
9383955d011SMarcel Moolenaar 
939*6a7405f5SSimon J. Gerraty 	if (rval == CR_ERROR && eprint && !par.printedError
940*6a7405f5SSimon J. Gerraty 	    && parseErrors == parseErrorsBefore)
941*6a7405f5SSimon J. Gerraty 		Parse_Error(PARSE_FATAL, "Malformed conditional '%s'", cond);
9423955d011SMarcel Moolenaar 
9433955d011SMarcel Moolenaar 	return rval;
9443955d011SMarcel Moolenaar }
9453955d011SMarcel Moolenaar 
94606b9b3e0SSimon J. Gerraty /*
94706b9b3e0SSimon J. Gerraty  * Evaluate a condition in a :? modifier, such as
94806b9b3e0SSimon J. Gerraty  * ${"${VAR}" == value:?yes:no}.
94906b9b3e0SSimon J. Gerraty  */
9509f45a3c8SSimon J. Gerraty CondResult
Cond_EvalCondition(const char * cond)9519f45a3c8SSimon J. Gerraty Cond_EvalCondition(const char *cond)
952956e45f6SSimon J. Gerraty {
9539f45a3c8SSimon J. Gerraty 	return CondEvalExpression(cond, true,
95412904384SSimon J. Gerraty 	    FuncDefined, false, false, true);
955956e45f6SSimon J. Gerraty }
9563955d011SMarcel Moolenaar 
957b0c40a00SSimon J. Gerraty static bool
IsEndif(const char * p)95806b9b3e0SSimon J. Gerraty IsEndif(const char *p)
95906b9b3e0SSimon J. Gerraty {
96006b9b3e0SSimon J. Gerraty 	return p[0] == 'e' && p[1] == 'n' && p[2] == 'd' &&
96106b9b3e0SSimon J. Gerraty 	       p[3] == 'i' && p[4] == 'f' && !ch_isalpha(p[5]);
96206b9b3e0SSimon J. Gerraty }
96306b9b3e0SSimon J. Gerraty 
964b0c40a00SSimon J. Gerraty static bool
DetermineKindOfConditional(const char ** pp,bool * out_plain,bool (** out_evalBare)(const char *),bool * out_negate)965b0c40a00SSimon J. Gerraty DetermineKindOfConditional(const char **pp, bool *out_plain,
96612904384SSimon J. Gerraty 			   bool (**out_evalBare)(const char *),
967b0c40a00SSimon J. Gerraty 			   bool *out_negate)
968dba7b0efSSimon J. Gerraty {
9699f45a3c8SSimon J. Gerraty 	const char *p = *pp + 2;
970dba7b0efSSimon J. Gerraty 
971b0c40a00SSimon J. Gerraty 	*out_plain = false;
972dba7b0efSSimon J. Gerraty 	*out_evalBare = FuncDefined;
9739f45a3c8SSimon J. Gerraty 	*out_negate = skip_string(&p, "n");
9749f45a3c8SSimon J. Gerraty 
9759f45a3c8SSimon J. Gerraty 	if (skip_string(&p, "def")) {		/* .ifdef and .ifndef */
9769f45a3c8SSimon J. Gerraty 	} else if (skip_string(&p, "make"))	/* .ifmake and .ifnmake */
977dba7b0efSSimon J. Gerraty 		*out_evalBare = FuncMake;
9789f45a3c8SSimon J. Gerraty 	else if (!*out_negate)			/* plain .if */
979b0c40a00SSimon J. Gerraty 		*out_plain = true;
9809f45a3c8SSimon J. Gerraty 	else
9819f45a3c8SSimon J. Gerraty 		goto unknown_directive;
9829f45a3c8SSimon J. Gerraty 	if (ch_isalpha(*p))
9839f45a3c8SSimon J. Gerraty 		goto unknown_directive;
9849f45a3c8SSimon J. Gerraty 
9859f45a3c8SSimon J. Gerraty 	*pp = p;
9869f45a3c8SSimon J. Gerraty 	return true;
9879f45a3c8SSimon J. Gerraty 
9889f45a3c8SSimon J. Gerraty unknown_directive:
989b0c40a00SSimon J. Gerraty 	return false;
990dba7b0efSSimon J. Gerraty }
991dba7b0efSSimon J. Gerraty 
99206b9b3e0SSimon J. Gerraty /*
99306b9b3e0SSimon J. Gerraty  * Evaluate the conditional directive in the line, which is one of:
9943955d011SMarcel Moolenaar  *
995e2eeea75SSimon J. Gerraty  *	.if <cond>
996e2eeea75SSimon J. Gerraty  *	.ifmake <cond>
997e2eeea75SSimon J. Gerraty  *	.ifnmake <cond>
998e2eeea75SSimon J. Gerraty  *	.ifdef <cond>
999e2eeea75SSimon J. Gerraty  *	.ifndef <cond>
1000e2eeea75SSimon J. Gerraty  *	.elif <cond>
1001e2eeea75SSimon J. Gerraty  *	.elifmake <cond>
1002e2eeea75SSimon J. Gerraty  *	.elifnmake <cond>
1003e2eeea75SSimon J. Gerraty  *	.elifdef <cond>
1004e2eeea75SSimon J. Gerraty  *	.elifndef <cond>
1005e2eeea75SSimon J. Gerraty  *	.else
1006e2eeea75SSimon J. Gerraty  *	.endif
1007e2eeea75SSimon J. Gerraty  *
1008e2eeea75SSimon J. Gerraty  * In these directives, <cond> consists of &&, ||, !, function(arg),
1009e2eeea75SSimon J. Gerraty  * comparisons, expressions, bare words, numbers and strings, and
1010e2eeea75SSimon J. Gerraty  * parenthetical groupings thereof.
1011956e45f6SSimon J. Gerraty  *
1012956e45f6SSimon J. Gerraty  * Results:
10139f45a3c8SSimon J. Gerraty  *	CR_TRUE		to continue parsing the lines that follow the
1014b0c40a00SSimon J. Gerraty  *			conditional (when <cond> evaluates to true)
10159f45a3c8SSimon J. Gerraty  *	CR_FALSE	to skip the lines after the conditional
1016b0c40a00SSimon J. Gerraty  *			(when <cond> evaluates to false, or when a previous
1017d5e0a182SSimon J. Gerraty  *			branch was already taken)
10189f45a3c8SSimon J. Gerraty  *	CR_ERROR	if the conditional was not valid, either because of
1019956e45f6SSimon J. Gerraty  *			a syntax error or because some variable was undefined
1020956e45f6SSimon J. Gerraty  *			or because the condition could not be evaluated
10213955d011SMarcel Moolenaar  */
10229f45a3c8SSimon J. Gerraty CondResult
Cond_EvalLine(const char * line)102306b9b3e0SSimon J. Gerraty Cond_EvalLine(const char *line)
10243955d011SMarcel Moolenaar {
1025e2eeea75SSimon J. Gerraty 	typedef enum IfState {
1026e2eeea75SSimon J. Gerraty 
1027b0c40a00SSimon J. Gerraty 		/* None of the previous <cond> evaluated to true. */
1028e2eeea75SSimon J. Gerraty 		IFS_INITIAL	= 0,
1029e2eeea75SSimon J. Gerraty 
10309f45a3c8SSimon J. Gerraty 		/*
10319f45a3c8SSimon J. Gerraty 		 * The previous <cond> evaluated to true. The lines following
10329f45a3c8SSimon J. Gerraty 		 * this condition are interpreted.
10339f45a3c8SSimon J. Gerraty 		 */
1034e2eeea75SSimon J. Gerraty 		IFS_ACTIVE	= 1 << 0,
1035e2eeea75SSimon J. Gerraty 
1036e2eeea75SSimon J. Gerraty 		/* The previous directive was an '.else'. */
1037e2eeea75SSimon J. Gerraty 		IFS_SEEN_ELSE	= 1 << 1,
1038e2eeea75SSimon J. Gerraty 
1039b0c40a00SSimon J. Gerraty 		/* One of the previous <cond> evaluated to true. */
1040e2eeea75SSimon J. Gerraty 		IFS_WAS_ACTIVE	= 1 << 2
1041e2eeea75SSimon J. Gerraty 
1042e2eeea75SSimon J. Gerraty 	} IfState;
1043e2eeea75SSimon J. Gerraty 
1044e2eeea75SSimon J. Gerraty 	static enum IfState *cond_states = NULL;
1045e2eeea75SSimon J. Gerraty 	static unsigned int cond_states_cap = 128;
10463955d011SMarcel Moolenaar 
1047b0c40a00SSimon J. Gerraty 	bool plain;
104812904384SSimon J. Gerraty 	bool (*evalBare)(const char *);
1049b0c40a00SSimon J. Gerraty 	bool negate;
1050b0c40a00SSimon J. Gerraty 	bool isElif;
10519f45a3c8SSimon J. Gerraty 	CondResult res;
1052e2eeea75SSimon J. Gerraty 	IfState state;
1053e2eeea75SSimon J. Gerraty 	const char *p = line;
10543955d011SMarcel Moolenaar 
1055e2eeea75SSimon J. Gerraty 	if (cond_states == NULL) {
105606b9b3e0SSimon J. Gerraty 		cond_states = bmake_malloc(
105706b9b3e0SSimon J. Gerraty 		    cond_states_cap * sizeof *cond_states);
1058e2eeea75SSimon J. Gerraty 		cond_states[0] = IFS_ACTIVE;
105959a02420SSimon J. Gerraty 	}
10603955d011SMarcel Moolenaar 
1061e2eeea75SSimon J. Gerraty 	p++;			/* skip the leading '.' */
1062e2eeea75SSimon J. Gerraty 	cpp_skip_hspace(&p);
1063e2eeea75SSimon J. Gerraty 
1064d5e0a182SSimon J. Gerraty 	if (IsEndif(p)) {
106506b9b3e0SSimon J. Gerraty 		if (p[5] != '\0') {
106606b9b3e0SSimon J. Gerraty 			Parse_Error(PARSE_FATAL,
106712904384SSimon J. Gerraty 			    "The .endif directive does not take arguments");
1068e2eeea75SSimon J. Gerraty 		}
1069e2eeea75SSimon J. Gerraty 
10704fde40d9SSimon J. Gerraty 		if (cond_depth == CurFile_CondMinDepth()) {
1071956e45f6SSimon J. Gerraty 			Parse_Error(PARSE_FATAL, "if-less endif");
10729f45a3c8SSimon J. Gerraty 			return CR_TRUE;
10733955d011SMarcel Moolenaar 		}
1074e2eeea75SSimon J. Gerraty 
10753955d011SMarcel Moolenaar 		/* Return state for previous conditional */
10763955d011SMarcel Moolenaar 		cond_depth--;
1077148ee845SSimon J. Gerraty 		Parse_GuardEndif();
1078e2eeea75SSimon J. Gerraty 		return cond_states[cond_depth] & IFS_ACTIVE
10799f45a3c8SSimon J. Gerraty 		    ? CR_TRUE : CR_FALSE;
10803955d011SMarcel Moolenaar 	}
10813955d011SMarcel Moolenaar 
108206b9b3e0SSimon J. Gerraty 	/* Parse the name of the directive, such as 'if', 'elif', 'endif'. */
108306b9b3e0SSimon J. Gerraty 	if (p[0] == 'e') {
1084d5e0a182SSimon J. Gerraty 		if (p[1] != 'l')
10859f45a3c8SSimon J. Gerraty 			return CR_ERROR;
108606b9b3e0SSimon J. Gerraty 
10873955d011SMarcel Moolenaar 		/* Quite likely this is 'else' or 'elif' */
1088e2eeea75SSimon J. Gerraty 		p += 2;
10899f45a3c8SSimon J. Gerraty 		if (strncmp(p, "se", 2) == 0 && !ch_isalpha(p[2])) {
109006b9b3e0SSimon J. Gerraty 			if (p[2] != '\0')
1091e2eeea75SSimon J. Gerraty 				Parse_Error(PARSE_FATAL,
109206b9b3e0SSimon J. Gerraty 				    "The .else directive "
109312904384SSimon J. Gerraty 				    "does not take arguments");
1094e2eeea75SSimon J. Gerraty 
10954fde40d9SSimon J. Gerraty 			if (cond_depth == CurFile_CondMinDepth()) {
1096956e45f6SSimon J. Gerraty 				Parse_Error(PARSE_FATAL, "if-less else");
10979f45a3c8SSimon J. Gerraty 				return CR_TRUE;
10983955d011SMarcel Moolenaar 			}
1099148ee845SSimon J. Gerraty 			Parse_GuardElse();
11003955d011SMarcel Moolenaar 
1101e2eeea75SSimon J. Gerraty 			state = cond_states[cond_depth];
1102e2eeea75SSimon J. Gerraty 			if (state == IFS_INITIAL) {
1103e2eeea75SSimon J. Gerraty 				state = IFS_ACTIVE | IFS_SEEN_ELSE;
1104e2eeea75SSimon J. Gerraty 			} else {
1105e2eeea75SSimon J. Gerraty 				if (state & IFS_SEEN_ELSE)
110606b9b3e0SSimon J. Gerraty 					Parse_Error(PARSE_WARNING,
110706b9b3e0SSimon J. Gerraty 					    "extra else");
1108e2eeea75SSimon J. Gerraty 				state = IFS_WAS_ACTIVE | IFS_SEEN_ELSE;
11093955d011SMarcel Moolenaar 			}
1110e2eeea75SSimon J. Gerraty 			cond_states[cond_depth] = state;
1111e2eeea75SSimon J. Gerraty 
11129f45a3c8SSimon J. Gerraty 			return state & IFS_ACTIVE ? CR_TRUE : CR_FALSE;
11133955d011SMarcel Moolenaar 		}
11143955d011SMarcel Moolenaar 		/* Assume for now it is an elif */
1115b0c40a00SSimon J. Gerraty 		isElif = true;
11163955d011SMarcel Moolenaar 	} else
1117b0c40a00SSimon J. Gerraty 		isElif = false;
11183955d011SMarcel Moolenaar 
1119d5e0a182SSimon J. Gerraty 	if (p[0] != 'i' || p[1] != 'f')
1120d5e0a182SSimon J. Gerraty 		return CR_ERROR;
11213955d011SMarcel Moolenaar 
1122dba7b0efSSimon J. Gerraty 	if (!DetermineKindOfConditional(&p, &plain, &evalBare, &negate))
11239f45a3c8SSimon J. Gerraty 		return CR_ERROR;
11243955d011SMarcel Moolenaar 
11253955d011SMarcel Moolenaar 	if (isElif) {
11264fde40d9SSimon J. Gerraty 		if (cond_depth == CurFile_CondMinDepth()) {
1127956e45f6SSimon J. Gerraty 			Parse_Error(PARSE_FATAL, "if-less elif");
11289f45a3c8SSimon J. Gerraty 			return CR_TRUE;
11293955d011SMarcel Moolenaar 		}
1130148ee845SSimon J. Gerraty 		Parse_GuardElse();
1131e2eeea75SSimon J. Gerraty 		state = cond_states[cond_depth];
1132e2eeea75SSimon J. Gerraty 		if (state & IFS_SEEN_ELSE) {
11333955d011SMarcel Moolenaar 			Parse_Error(PARSE_WARNING, "extra elif");
113406b9b3e0SSimon J. Gerraty 			cond_states[cond_depth] =
113506b9b3e0SSimon J. Gerraty 			    IFS_WAS_ACTIVE | IFS_SEEN_ELSE;
11369f45a3c8SSimon J. Gerraty 			return CR_FALSE;
11373955d011SMarcel Moolenaar 		}
1138e2eeea75SSimon J. Gerraty 		if (state != IFS_INITIAL) {
1139e2eeea75SSimon J. Gerraty 			cond_states[cond_depth] = IFS_WAS_ACTIVE;
11409f45a3c8SSimon J. Gerraty 			return CR_FALSE;
11413955d011SMarcel Moolenaar 		}
11423955d011SMarcel Moolenaar 	} else {
11433955d011SMarcel Moolenaar 		/* Normal .if */
1144e2eeea75SSimon J. Gerraty 		if (cond_depth + 1 >= cond_states_cap) {
114559a02420SSimon J. Gerraty 			/*
114659a02420SSimon J. Gerraty 			 * This is rare, but not impossible.
114759a02420SSimon J. Gerraty 			 * In meta mode, dirdeps.mk (only runs at level 0)
114859a02420SSimon J. Gerraty 			 * can need more than the default.
114959a02420SSimon J. Gerraty 			 */
1150e2eeea75SSimon J. Gerraty 			cond_states_cap += 32;
1151e2eeea75SSimon J. Gerraty 			cond_states = bmake_realloc(cond_states,
11529f45a3c8SSimon J. Gerraty 			    cond_states_cap * sizeof *cond_states);
11533955d011SMarcel Moolenaar 		}
1154e2eeea75SSimon J. Gerraty 		state = cond_states[cond_depth];
11553955d011SMarcel Moolenaar 		cond_depth++;
1156e2eeea75SSimon J. Gerraty 		if (!(state & IFS_ACTIVE)) {
1157e2eeea75SSimon J. Gerraty 			cond_states[cond_depth] = IFS_WAS_ACTIVE;
11589f45a3c8SSimon J. Gerraty 			return CR_FALSE;
11593955d011SMarcel Moolenaar 		}
11603955d011SMarcel Moolenaar 	}
11613955d011SMarcel Moolenaar 
11629f45a3c8SSimon J. Gerraty 	res = CondEvalExpression(p, plain, evalBare, negate, true, false);
11639f45a3c8SSimon J. Gerraty 	if (res == CR_ERROR) {
11649f45a3c8SSimon J. Gerraty 		/* Syntax error, error message already output. */
11659f45a3c8SSimon J. Gerraty 		/* Skip everything to the matching '.endif'. */
11669f45a3c8SSimon J. Gerraty 		/* An extra '.else' is not detected in this case. */
1167e2eeea75SSimon J. Gerraty 		cond_states[cond_depth] = IFS_WAS_ACTIVE;
11689f45a3c8SSimon J. Gerraty 		return CR_FALSE;
11693955d011SMarcel Moolenaar 	}
11703955d011SMarcel Moolenaar 
11719f45a3c8SSimon J. Gerraty 	cond_states[cond_depth] = res == CR_TRUE ? IFS_ACTIVE : IFS_INITIAL;
11729f45a3c8SSimon J. Gerraty 	return res;
11733955d011SMarcel Moolenaar }
11743955d011SMarcel Moolenaar 
1175148ee845SSimon J. Gerraty static bool
ParseVarnameGuard(const char ** pp,const char ** varname)1176148ee845SSimon J. Gerraty ParseVarnameGuard(const char **pp, const char **varname)
1177148ee845SSimon J. Gerraty {
1178148ee845SSimon J. Gerraty 	const char *p = *pp;
1179148ee845SSimon J. Gerraty 
1180148ee845SSimon J. Gerraty 	if (ch_isalpha(*p) || *p == '_') {
1181148ee845SSimon J. Gerraty 		while (ch_isalnum(*p) || *p == '_')
1182148ee845SSimon J. Gerraty 			p++;
1183148ee845SSimon J. Gerraty 		*varname = *pp;
1184148ee845SSimon J. Gerraty 		*pp = p;
1185148ee845SSimon J. Gerraty 		return true;
1186148ee845SSimon J. Gerraty 	}
1187148ee845SSimon J. Gerraty 	return false;
1188148ee845SSimon J. Gerraty }
1189148ee845SSimon J. Gerraty 
1190148ee845SSimon J. Gerraty /* Extracts the multiple-inclusion guard from a conditional, if any. */
1191148ee845SSimon J. Gerraty Guard *
Cond_ExtractGuard(const char * line)1192148ee845SSimon J. Gerraty Cond_ExtractGuard(const char *line)
1193148ee845SSimon J. Gerraty {
1194148ee845SSimon J. Gerraty 	const char *p, *varname;
1195148ee845SSimon J. Gerraty 	Substring dir;
1196148ee845SSimon J. Gerraty 	Guard *guard;
1197148ee845SSimon J. Gerraty 
1198148ee845SSimon J. Gerraty 	p = line + 1;		/* skip the '.' */
1199148ee845SSimon J. Gerraty 	cpp_skip_hspace(&p);
1200148ee845SSimon J. Gerraty 
1201148ee845SSimon J. Gerraty 	dir.start = p;
1202148ee845SSimon J. Gerraty 	while (ch_isalpha(*p))
1203148ee845SSimon J. Gerraty 		p++;
1204148ee845SSimon J. Gerraty 	dir.end = p;
1205148ee845SSimon J. Gerraty 	cpp_skip_hspace(&p);
1206148ee845SSimon J. Gerraty 
1207148ee845SSimon J. Gerraty 	if (Substring_Equals(dir, "if")) {
1208148ee845SSimon J. Gerraty 		if (skip_string(&p, "!defined(")) {
1209148ee845SSimon J. Gerraty 			if (ParseVarnameGuard(&p, &varname)
1210148ee845SSimon J. Gerraty 			    && strcmp(p, ")") == 0)
1211148ee845SSimon J. Gerraty 				goto found_variable;
1212148ee845SSimon J. Gerraty 		} else if (skip_string(&p, "!target(")) {
1213148ee845SSimon J. Gerraty 			const char *arg_p = p;
1214148ee845SSimon J. Gerraty 			free(ParseWord(&p, false));
1215148ee845SSimon J. Gerraty 			if (strcmp(p, ")") == 0) {
1216148ee845SSimon J. Gerraty 				guard = bmake_malloc(sizeof(*guard));
1217148ee845SSimon J. Gerraty 				guard->kind = GK_TARGET;
121898875883SSimon J. Gerraty 				guard->name = ParseWord(&arg_p, true);
1219148ee845SSimon J. Gerraty 				return guard;
1220148ee845SSimon J. Gerraty 			}
1221148ee845SSimon J. Gerraty 		}
1222148ee845SSimon J. Gerraty 	} else if (Substring_Equals(dir, "ifndef")) {
1223148ee845SSimon J. Gerraty 		if (ParseVarnameGuard(&p, &varname) && *p == '\0')
1224148ee845SSimon J. Gerraty 			goto found_variable;
1225148ee845SSimon J. Gerraty 	}
1226148ee845SSimon J. Gerraty 	return NULL;
1227148ee845SSimon J. Gerraty 
1228148ee845SSimon J. Gerraty found_variable:
1229148ee845SSimon J. Gerraty 	guard = bmake_malloc(sizeof(*guard));
123098875883SSimon J. Gerraty 	guard->kind = GK_VARIABLE;
1231148ee845SSimon J. Gerraty 	guard->name = bmake_strsedup(varname, p);
1232148ee845SSimon J. Gerraty 	return guard;
1233148ee845SSimon J. Gerraty }
1234148ee845SSimon J. Gerraty 
12353955d011SMarcel Moolenaar void
Cond_EndFile(void)12364fde40d9SSimon J. Gerraty Cond_EndFile(void)
12373955d011SMarcel Moolenaar {
12384fde40d9SSimon J. Gerraty 	unsigned int open_conds = cond_depth - CurFile_CondMinDepth();
12393955d011SMarcel Moolenaar 
12404fde40d9SSimon J. Gerraty 	if (open_conds != 0) {
124106b9b3e0SSimon J. Gerraty 		Parse_Error(PARSE_FATAL, "%u open conditional%s",
124206b9b3e0SSimon J. Gerraty 		    open_conds, open_conds == 1 ? "" : "s");
12434fde40d9SSimon J. Gerraty 		cond_depth = CurFile_CondMinDepth();
12443955d011SMarcel Moolenaar 	}
12453955d011SMarcel Moolenaar }
1246