xref: /freebsd/contrib/bmake/make.h (revision d9a65c5de1c9f30ae71ce0be8fb88be9d20d216d)
1*d9a65c5dSSimon J. Gerraty /*	$NetBSD: make.h,v 1.350 2025/03/07 06:50:34 rillig Exp $	*/
23955d011SMarcel Moolenaar 
33955d011SMarcel Moolenaar /*
43955d011SMarcel Moolenaar  * Copyright (c) 1988, 1989, 1990, 1993
53955d011SMarcel Moolenaar  *	The Regents of the University of California.  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  *	from: @(#)make.h	8.3 (Berkeley) 6/13/95
353955d011SMarcel Moolenaar  */
363955d011SMarcel Moolenaar 
373955d011SMarcel Moolenaar /*
383955d011SMarcel Moolenaar  * Copyright (c) 1989 by Berkeley Softworks
393955d011SMarcel Moolenaar  * All rights reserved.
403955d011SMarcel Moolenaar  *
413955d011SMarcel Moolenaar  * This code is derived from software contributed to Berkeley by
423955d011SMarcel Moolenaar  * Adam de Boor.
433955d011SMarcel Moolenaar  *
443955d011SMarcel Moolenaar  * Redistribution and use in source and binary forms, with or without
453955d011SMarcel Moolenaar  * modification, are permitted provided that the following conditions
463955d011SMarcel Moolenaar  * are met:
473955d011SMarcel Moolenaar  * 1. Redistributions of source code must retain the above copyright
483955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer.
493955d011SMarcel Moolenaar  * 2. Redistributions in binary form must reproduce the above copyright
503955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer in the
513955d011SMarcel Moolenaar  *    documentation and/or other materials provided with the distribution.
523955d011SMarcel Moolenaar  * 3. All advertising materials mentioning features or use of this software
533955d011SMarcel Moolenaar  *    must display the following acknowledgement:
543955d011SMarcel Moolenaar  *	This product includes software developed by the University of
553955d011SMarcel Moolenaar  *	California, Berkeley and its contributors.
563955d011SMarcel Moolenaar  * 4. Neither the name of the University nor the names of its contributors
573955d011SMarcel Moolenaar  *    may be used to endorse or promote products derived from this software
583955d011SMarcel Moolenaar  *    without specific prior written permission.
593955d011SMarcel Moolenaar  *
603955d011SMarcel Moolenaar  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
613955d011SMarcel Moolenaar  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
623955d011SMarcel Moolenaar  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
633955d011SMarcel Moolenaar  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
643955d011SMarcel Moolenaar  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
653955d011SMarcel Moolenaar  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
663955d011SMarcel Moolenaar  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
673955d011SMarcel Moolenaar  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
683955d011SMarcel Moolenaar  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
693955d011SMarcel Moolenaar  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
703955d011SMarcel Moolenaar  * SUCH DAMAGE.
713955d011SMarcel Moolenaar  *
723955d011SMarcel Moolenaar  *	from: @(#)make.h	8.3 (Berkeley) 6/13/95
733955d011SMarcel Moolenaar  */
743955d011SMarcel Moolenaar 
75dba7b0efSSimon J. Gerraty /*
763955d011SMarcel Moolenaar  * make.h --
779f45a3c8SSimon J. Gerraty  *	The global definitions for make
783955d011SMarcel Moolenaar  */
793955d011SMarcel Moolenaar 
802c3632d1SSimon J. Gerraty #ifndef MAKE_MAKE_H
812c3632d1SSimon J. Gerraty #define MAKE_MAKE_H
823955d011SMarcel Moolenaar 
833955d011SMarcel Moolenaar #ifdef HAVE_CONFIG_H
843955d011SMarcel Moolenaar # include "config.h"
853955d011SMarcel Moolenaar #endif
863955d011SMarcel Moolenaar 
873955d011SMarcel Moolenaar #include <sys/types.h>
883955d011SMarcel Moolenaar #include <sys/param.h>
892c3632d1SSimon J. Gerraty #include <sys/stat.h>
903955d011SMarcel Moolenaar 
912c3632d1SSimon J. Gerraty #include <assert.h>
923955d011SMarcel Moolenaar #include <ctype.h>
93be19d90bSSimon J. Gerraty #include <fcntl.h>
94956e45f6SSimon J. Gerraty #include <stdarg.h>
953955d011SMarcel Moolenaar #include <stdio.h>
963955d011SMarcel Moolenaar #include <stdlib.h>
973955d011SMarcel Moolenaar #ifdef HAVE_STRING_H
983955d011SMarcel Moolenaar #include <string.h>
993955d011SMarcel Moolenaar #else
1003955d011SMarcel Moolenaar #include <strings.h>
1013955d011SMarcel Moolenaar #endif
1023955d011SMarcel Moolenaar #include <unistd.h>
1033955d011SMarcel Moolenaar #include <sys/cdefs.h>
1043955d011SMarcel Moolenaar 
105be19d90bSSimon J. Gerraty #ifndef FD_CLOEXEC
106be19d90bSSimon J. Gerraty #define FD_CLOEXEC 1
107be19d90bSSimon J. Gerraty #endif
108be19d90bSSimon J. Gerraty 
1093955d011SMarcel Moolenaar #if defined(__GNUC__)
1103955d011SMarcel Moolenaar #define MAKE_GNUC_PREREQ(x, y)						\
1113955d011SMarcel Moolenaar 	((__GNUC__ == (x) && __GNUC_MINOR__ >= (y)) ||			\
1123955d011SMarcel Moolenaar 	 (__GNUC__ > (x)))
1139f45a3c8SSimon J. Gerraty #else
1141748de26SSimon J. Gerraty #define MAKE_GNUC_PREREQ(x, y)	0
1159f45a3c8SSimon J. Gerraty #endif
1163955d011SMarcel Moolenaar 
11722619282SSimon J. Gerraty #if MAKE_GNUC_PREREQ(2, 7) || lint
1183955d011SMarcel Moolenaar #define MAKE_ATTR_UNUSED	__attribute__((__unused__))
1193955d011SMarcel Moolenaar #else
1203955d011SMarcel Moolenaar #define MAKE_ATTR_UNUSED	/* delete */
1213955d011SMarcel Moolenaar #endif
1223955d011SMarcel Moolenaar 
1233955d011SMarcel Moolenaar #if MAKE_GNUC_PREREQ(2, 5)
1243955d011SMarcel Moolenaar #define MAKE_ATTR_DEAD		__attribute__((__noreturn__))
1253955d011SMarcel Moolenaar #elif defined(__GNUC__)
1263955d011SMarcel Moolenaar #define MAKE_ATTR_DEAD		__volatile
1273955d011SMarcel Moolenaar #else
1283955d011SMarcel Moolenaar #define MAKE_ATTR_DEAD		/* delete */
1293955d011SMarcel Moolenaar #endif
1303955d011SMarcel Moolenaar 
1313955d011SMarcel Moolenaar #if MAKE_GNUC_PREREQ(2, 7)
1323955d011SMarcel Moolenaar #define MAKE_ATTR_PRINTFLIKE(fmtarg, firstvararg)	\
1333955d011SMarcel Moolenaar 	    __attribute__((__format__ (__printf__, fmtarg, firstvararg)))
1343955d011SMarcel Moolenaar #else
1353955d011SMarcel Moolenaar #define MAKE_ATTR_PRINTFLIKE(fmtarg, firstvararg)	/* delete */
1363955d011SMarcel Moolenaar #endif
1373955d011SMarcel Moolenaar 
1389f45a3c8SSimon J. Gerraty #if MAKE_GNUC_PREREQ(4, 0)
1399f45a3c8SSimon J. Gerraty #define MAKE_ATTR_USE		__attribute__((__warn_unused_result__))
1409f45a3c8SSimon J. Gerraty #else
1419f45a3c8SSimon J. Gerraty #define MAKE_ATTR_USE		/* delete */
1429f45a3c8SSimon J. Gerraty #endif
1439f45a3c8SSimon J. Gerraty 
144548bfc56SSimon J. Gerraty #if MAKE_GNUC_PREREQ(8, 0)
145548bfc56SSimon J. Gerraty #define MAKE_ATTR_NOINLINE		__attribute__((__noinline__))
146548bfc56SSimon J. Gerraty #else
147548bfc56SSimon J. Gerraty #define MAKE_ATTR_NOINLINE		/* delete */
148548bfc56SSimon J. Gerraty #endif
149548bfc56SSimon J. Gerraty 
1501d3f2ddcSSimon J. Gerraty #if __STDC_VERSION__ >= 199901L || defined(lint)
151e2eeea75SSimon J. Gerraty #define MAKE_INLINE static inline MAKE_ATTR_UNUSED
1529f45a3c8SSimon J. Gerraty #else
1539f45a3c8SSimon J. Gerraty #define MAKE_INLINE static MAKE_ATTR_UNUSED
1549f45a3c8SSimon J. Gerraty #endif
15512904384SSimon J. Gerraty 
15612904384SSimon J. Gerraty /* MAKE_STATIC marks a function that may or may not be inlined. */
15712904384SSimon J. Gerraty #if defined(lint)
15812904384SSimon J. Gerraty /* As of 2021-07-31, NetBSD lint ignores __attribute__((unused)). */
15912904384SSimon J. Gerraty #define MAKE_STATIC MAKE_INLINE
16012904384SSimon J. Gerraty #else
161b0c40a00SSimon J. Gerraty #define MAKE_STATIC static MAKE_ATTR_UNUSED
16212904384SSimon J. Gerraty #endif
163e2eeea75SSimon J. Gerraty 
164b0c40a00SSimon J. Gerraty #if __STDC_VERSION__ >= 199901L || defined(lint) || defined(USE_C99_BOOLEAN)
16506b9b3e0SSimon J. Gerraty #include <stdbool.h>
16612904384SSimon J. Gerraty #elif defined(__bool_true_false_are_defined)
16712904384SSimon J. Gerraty /*
16812904384SSimon J. Gerraty  * All files of make must be compiled with the same definition of bool.
16912904384SSimon J. Gerraty  * Since one of the files includes <stdbool.h>, that means the header is
17012904384SSimon J. Gerraty  * available on this platform.  Recompile everything with -DUSE_C99_BOOLEAN.
17112904384SSimon J. Gerraty  */
17212904384SSimon J. Gerraty #error "<stdbool.h> is included in pre-C99 mode"
17312904384SSimon J. Gerraty #elif defined(bool) || defined(true) || defined(false)
17412904384SSimon J. Gerraty /*
17512904384SSimon J. Gerraty  * In pre-C99 mode, make does not expect that bool is already defined.
17612904384SSimon J. Gerraty  * You need to ensure that all translation units use the same definition for
17712904384SSimon J. Gerraty  * bool.
17812904384SSimon J. Gerraty  */
17912904384SSimon J. Gerraty #error "bool/true/false is defined in pre-C99 mode"
1802c3632d1SSimon J. Gerraty #else
18112904384SSimon J. Gerraty typedef unsigned char bool;
182b0c40a00SSimon J. Gerraty #define true	1
183b0c40a00SSimon J. Gerraty #define false	0
184956e45f6SSimon J. Gerraty #endif
1852c3632d1SSimon J. Gerraty 
1866a7405f5SSimon J. Gerraty /*
1876a7405f5SSimon J. Gerraty  * In code coverage mode with gcc>=12, calling vfork/exec does not mark any
1886a7405f5SSimon J. Gerraty  * further code from the parent process as covered. gcc-10.5.0 is fine, as
1896a7405f5SSimon J. Gerraty  * are fork/exec calls, as well as posix_spawn.
1906a7405f5SSimon J. Gerraty  */
1916a7405f5SSimon J. Gerraty #ifndef FORK_FUNCTION
1926a7405f5SSimon J. Gerraty #define FORK_FUNCTION vfork
1936a7405f5SSimon J. Gerraty #endif
1946a7405f5SSimon J. Gerraty 
1953955d011SMarcel Moolenaar #include "lst.h"
196b0c40a00SSimon J. Gerraty #include "make_malloc.h"
197b0c40a00SSimon J. Gerraty #include "str.h"
1983955d011SMarcel Moolenaar #include "hash.h"
1993955d011SMarcel Moolenaar #include "make-conf.h"
2003955d011SMarcel Moolenaar #include "buf.h"
2013955d011SMarcel Moolenaar 
2023955d011SMarcel Moolenaar /*
2033955d011SMarcel Moolenaar  * some vendors don't have this --sjg
2043955d011SMarcel Moolenaar  */
2053955d011SMarcel Moolenaar #if defined(S_IFDIR) && !defined(S_ISDIR)
2063955d011SMarcel Moolenaar # define S_ISDIR(m) (((m) & S_IFMT) == S_IFDIR)
2073955d011SMarcel Moolenaar #endif
2083955d011SMarcel Moolenaar 
2093955d011SMarcel Moolenaar #if defined(sun) && (defined(__svr4__) || defined(__SVR4))
2103955d011SMarcel Moolenaar # define POSIX_SIGNALS
2113955d011SMarcel Moolenaar #endif
2123955d011SMarcel Moolenaar 
21306b9b3e0SSimon J. Gerraty /*
21498875883SSimon J. Gerraty  * IRIX defines OP_NONE in sys/fcntl.h
21598875883SSimon J. Gerraty  */
21698875883SSimon J. Gerraty #if defined(OP_NONE)
21798875883SSimon J. Gerraty # undef OP_NONE
21898875883SSimon J. Gerraty #endif
21998875883SSimon J. Gerraty 
22098875883SSimon J. Gerraty /*
22106b9b3e0SSimon J. Gerraty  * The typical flow of states is:
22206b9b3e0SSimon J. Gerraty  *
22306b9b3e0SSimon J. Gerraty  * The direct successful path:
22406b9b3e0SSimon J. Gerraty  * UNMADE -> BEINGMADE -> MADE.
22506b9b3e0SSimon J. Gerraty  *
22606b9b3e0SSimon J. Gerraty  * The direct error path:
22706b9b3e0SSimon J. Gerraty  * UNMADE -> BEINGMADE -> ERROR.
22806b9b3e0SSimon J. Gerraty  *
22906b9b3e0SSimon J. Gerraty  * The successful path when dependencies need to be made first:
23006b9b3e0SSimon J. Gerraty  * UNMADE -> DEFERRED -> REQUESTED -> BEINGMADE -> MADE.
23106b9b3e0SSimon J. Gerraty  *
23206b9b3e0SSimon J. Gerraty  * A node that has dependencies, and one of the dependencies cannot be made:
23306b9b3e0SSimon J. Gerraty  * UNMADE -> DEFERRED -> ABORTED.
23406b9b3e0SSimon J. Gerraty  *
23506b9b3e0SSimon J. Gerraty  * A node that turns out to be up-to-date:
23606b9b3e0SSimon J. Gerraty  * UNMADE -> BEINGMADE -> UPTODATE.
23706b9b3e0SSimon J. Gerraty  */
238e2eeea75SSimon J. Gerraty typedef enum GNodeMade {
23906b9b3e0SSimon J. Gerraty 	/* Not examined yet. */
24006b9b3e0SSimon J. Gerraty 	UNMADE,
2419f45a3c8SSimon J. Gerraty 	/*
2429f45a3c8SSimon J. Gerraty 	 * The node has been examined but is not yet ready since its
2439f45a3c8SSimon J. Gerraty 	 * dependencies have to be made first.
2449f45a3c8SSimon J. Gerraty 	 */
24506b9b3e0SSimon J. Gerraty 	DEFERRED,
24606b9b3e0SSimon J. Gerraty 
24706b9b3e0SSimon J. Gerraty 	/* The node is on the toBeMade list. */
24806b9b3e0SSimon J. Gerraty 	REQUESTED,
24906b9b3e0SSimon J. Gerraty 
2509f45a3c8SSimon J. Gerraty 	/*
2519f45a3c8SSimon J. Gerraty 	 * The node is already being made. Trying to build a node in this
2529f45a3c8SSimon J. Gerraty 	 * state indicates a cycle in the graph.
2539f45a3c8SSimon J. Gerraty 	 */
25406b9b3e0SSimon J. Gerraty 	BEINGMADE,
25506b9b3e0SSimon J. Gerraty 
25606b9b3e0SSimon J. Gerraty 	/* Was out-of-date and has been made. */
25706b9b3e0SSimon J. Gerraty 	MADE,
25806b9b3e0SSimon J. Gerraty 	/* Was already up-to-date, does not need to be made. */
25906b9b3e0SSimon J. Gerraty 	UPTODATE,
2609f45a3c8SSimon J. Gerraty 	/*
2619f45a3c8SSimon J. Gerraty 	 * An error occurred while it was being made. Used only in compat
2629f45a3c8SSimon J. Gerraty 	 * mode.
2639f45a3c8SSimon J. Gerraty 	 */
26406b9b3e0SSimon J. Gerraty 	ERROR,
2659f45a3c8SSimon J. Gerraty 	/*
2669f45a3c8SSimon J. Gerraty 	 * The target was aborted due to an error making a dependency. Used
2679f45a3c8SSimon J. Gerraty 	 * only in compat mode.
2689f45a3c8SSimon J. Gerraty 	 */
26906b9b3e0SSimon J. Gerraty 	ABORTED
2702c3632d1SSimon J. Gerraty } GNodeMade;
2713955d011SMarcel Moolenaar 
27206b9b3e0SSimon J. Gerraty /*
27306b9b3e0SSimon J. Gerraty  * The OP_ constants are used when parsing a dependency line as a way of
2742c3632d1SSimon J. Gerraty  * communicating to other parts of the program the way in which a target
2752c3632d1SSimon J. Gerraty  * should be made.
2762c3632d1SSimon J. Gerraty  *
27706b9b3e0SSimon J. Gerraty  * Some of the OP_ constants can be combined, others cannot.
278b0c40a00SSimon J. Gerraty  *
279b0c40a00SSimon J. Gerraty  * See the tests depsrc-*.mk and deptgt-*.mk.
28006b9b3e0SSimon J. Gerraty  */
281956e45f6SSimon J. Gerraty typedef enum GNodeType {
282e2eeea75SSimon J. Gerraty 	OP_NONE		= 0,
283e2eeea75SSimon J. Gerraty 
2849f45a3c8SSimon J. Gerraty 	/*
2859f45a3c8SSimon J. Gerraty 	 * The dependency operator ':' is the most common one.  The commands
2869f45a3c8SSimon J. Gerraty 	 * of this node are executed if any child is out-of-date.
2879f45a3c8SSimon J. Gerraty 	 */
2882c3632d1SSimon J. Gerraty 	OP_DEPENDS	= 1 << 0,
2899f45a3c8SSimon J. Gerraty 	/*
2909f45a3c8SSimon J. Gerraty 	 * The dependency operator '!' always executes its commands, even if
2919f45a3c8SSimon J. Gerraty 	 * its children are up-to-date.
2929f45a3c8SSimon J. Gerraty 	 */
2932c3632d1SSimon J. Gerraty 	OP_FORCE	= 1 << 1,
2949f45a3c8SSimon J. Gerraty 	/*
2959f45a3c8SSimon J. Gerraty 	 * The dependency operator '::' behaves like ':', except that it
29606b9b3e0SSimon J. Gerraty 	 * allows multiple dependency groups to be defined.  Each of these
2979f45a3c8SSimon J. Gerraty 	 * groups is executed on its own, independently from the others. Each
2989f45a3c8SSimon J. Gerraty 	 * individual dependency group is called a cohort.
2999f45a3c8SSimon J. Gerraty 	 */
3002c3632d1SSimon J. Gerraty 	OP_DOUBLEDEP	= 1 << 2,
3012c3632d1SSimon J. Gerraty 
302956e45f6SSimon J. Gerraty 	/* Matches the dependency operators ':', '!' and '::'. */
3032c3632d1SSimon J. Gerraty 	OP_OPMASK	= OP_DEPENDS | OP_FORCE | OP_DOUBLEDEP,
3042c3632d1SSimon J. Gerraty 
30506b9b3e0SSimon J. Gerraty 	/* Don't care if the target doesn't exist and can't be created. */
3062c3632d1SSimon J. Gerraty 	OP_OPTIONAL	= 1 << 3,
30706b9b3e0SSimon J. Gerraty 	/* Use associated commands for parents. */
3082c3632d1SSimon J. Gerraty 	OP_USE		= 1 << 4,
3099f45a3c8SSimon J. Gerraty 	/*
3109f45a3c8SSimon J. Gerraty 	 * Target is never out of date, but always execute commands anyway.
3119f45a3c8SSimon J. Gerraty 	 * Its time doesn't matter, so it has none...sort of.
3129f45a3c8SSimon J. Gerraty 	 */
3132c3632d1SSimon J. Gerraty 	OP_EXEC		= 1 << 5,
3149f45a3c8SSimon J. Gerraty 	/*
3159f45a3c8SSimon J. Gerraty 	 * Ignore non-zero exit status from shell commands when creating the
3169f45a3c8SSimon J. Gerraty 	 * node.
3179f45a3c8SSimon J. Gerraty 	 */
3182c3632d1SSimon J. Gerraty 	OP_IGNORE	= 1 << 6,
31906b9b3e0SSimon J. Gerraty 	/* Don't remove the target when interrupted. */
3202c3632d1SSimon J. Gerraty 	OP_PRECIOUS	= 1 << 7,
32106b9b3e0SSimon J. Gerraty 	/* Don't echo commands when executed. */
3222c3632d1SSimon J. Gerraty 	OP_SILENT	= 1 << 8,
3239f45a3c8SSimon J. Gerraty 	/*
3249f45a3c8SSimon J. Gerraty 	 * Target is a recursive make so its commands should always be
3259f45a3c8SSimon J. Gerraty 	 * executed when it is out of date, regardless of the state of the -n
3269f45a3c8SSimon J. Gerraty 	 * or -t flags.
3279f45a3c8SSimon J. Gerraty 	 */
3282c3632d1SSimon J. Gerraty 	OP_MAKE		= 1 << 9,
3299f45a3c8SSimon J. Gerraty 	/*
3309f45a3c8SSimon J. Gerraty 	 * Target is out-of-date only if any of its children was out-of-date.
3319f45a3c8SSimon J. Gerraty 	 */
3322c3632d1SSimon J. Gerraty 	OP_JOIN		= 1 << 10,
33306b9b3e0SSimon J. Gerraty 	/* Assume the children of the node have been already made. */
3342c3632d1SSimon J. Gerraty 	OP_MADE		= 1 << 11,
33506b9b3e0SSimon J. Gerraty 	/* Special .BEGIN, .END or .INTERRUPT. */
3362c3632d1SSimon J. Gerraty 	OP_SPECIAL	= 1 << 12,
33706b9b3e0SSimon J. Gerraty 	/* Like .USE, only prepend commands. */
3382c3632d1SSimon J. Gerraty 	OP_USEBEFORE	= 1 << 13,
3399f45a3c8SSimon J. Gerraty 	/*
3409f45a3c8SSimon J. Gerraty 	 * The node is invisible to its parents. I.e. it doesn't show up in
3419f45a3c8SSimon J. Gerraty 	 * the parents' local variables (.IMPSRC, .ALLSRC).
3429f45a3c8SSimon J. Gerraty 	 */
3432c3632d1SSimon J. Gerraty 	OP_INVISIBLE	= 1 << 14,
3449f45a3c8SSimon J. Gerraty 	/*
3459f45a3c8SSimon J. Gerraty 	 * The node does not become the main target, even if it is the first
3469f45a3c8SSimon J. Gerraty 	 * target in the first makefile.
3479f45a3c8SSimon J. Gerraty 	 */
3482c3632d1SSimon J. Gerraty 	OP_NOTMAIN	= 1 << 15,
34906b9b3e0SSimon J. Gerraty 	/* Not a file target; run always. */
3502c3632d1SSimon J. Gerraty 	OP_PHONY	= 1 << 16,
35106b9b3e0SSimon J. Gerraty 	/* Don't search for the file in the path. */
3522c3632d1SSimon J. Gerraty 	OP_NOPATH	= 1 << 17,
3539f45a3c8SSimon J. Gerraty 	/*
3549f45a3c8SSimon J. Gerraty 	 * In a dependency line "target: source1 .WAIT source2", source1 is
35506b9b3e0SSimon J. Gerraty 	 * made first, including its children.  Once that is finished,
35606b9b3e0SSimon J. Gerraty 	 * source2 is made, including its children.  The .WAIT keyword may
3579f45a3c8SSimon J. Gerraty 	 * appear more than once in a single dependency declaration.
3589f45a3c8SSimon J. Gerraty 	 */
3592c3632d1SSimon J. Gerraty 	OP_WAIT		= 1 << 18,
3602c3632d1SSimon J. Gerraty 	/* .NOMETA do not create a .meta file */
3612c3632d1SSimon J. Gerraty 	OP_NOMETA	= 1 << 19,
3622c3632d1SSimon J. Gerraty 	/* .META we _do_ want a .meta file */
3632c3632d1SSimon J. Gerraty 	OP_META		= 1 << 20,
3642c3632d1SSimon J. Gerraty 	/* Do not compare commands in .meta file */
3652c3632d1SSimon J. Gerraty 	OP_NOMETA_CMP	= 1 << 21,
3662c3632d1SSimon J. Gerraty 	/* Possibly a submake node */
3672c3632d1SSimon J. Gerraty 	OP_SUBMAKE	= 1 << 22,
3682c3632d1SSimon J. Gerraty 
3692c3632d1SSimon J. Gerraty 	/* Attributes applied by PMake */
3702c3632d1SSimon J. Gerraty 
371e2eeea75SSimon J. Gerraty 	/* The node is a transformation rule, such as ".c.o". */
37206b9b3e0SSimon J. Gerraty 	OP_TRANSFORM	= 1 << 30,
3732c3632d1SSimon J. Gerraty 	/* Target is a member of an archive */
374956e45f6SSimon J. Gerraty 	/* XXX: How does this differ from OP_ARCHV? */
37506b9b3e0SSimon J. Gerraty 	OP_MEMBER	= 1 << 29,
3769f45a3c8SSimon J. Gerraty 	/*
3779f45a3c8SSimon J. Gerraty 	 * The node is a library, its name has the form "-l<libname>".
3789f45a3c8SSimon J. Gerraty 	 */
37906b9b3e0SSimon J. Gerraty 	OP_LIB		= 1 << 28,
3809f45a3c8SSimon J. Gerraty 	/*
3819f45a3c8SSimon J. Gerraty 	 * The node is an archive member, its name has the form
3829f45a3c8SSimon J. Gerraty 	 * "archive(member)".
3839f45a3c8SSimon J. Gerraty 	 */
384956e45f6SSimon J. Gerraty 	/* XXX: How does this differ from OP_MEMBER? */
38506b9b3e0SSimon J. Gerraty 	OP_ARCHV	= 1 << 27,
3869f45a3c8SSimon J. Gerraty 	/*
3879f45a3c8SSimon J. Gerraty 	 * Target has all the commands it should. Used when parsing to catch
38806b9b3e0SSimon J. Gerraty 	 * multiple command groups for a target.  Only applies to the
3899f45a3c8SSimon J. Gerraty 	 * dependency operators ':' and '!', but not to '::'.
3909f45a3c8SSimon J. Gerraty 	 */
39106b9b3e0SSimon J. Gerraty 	OP_HAS_COMMANDS	= 1 << 26,
3929f45a3c8SSimon J. Gerraty 	/*
3939f45a3c8SSimon J. Gerraty 	 * The special command "..." has been seen. All further commands from
3949f45a3c8SSimon J. Gerraty 	 * this node will be saved on the .END node instead, to be executed
3959f45a3c8SSimon J. Gerraty 	 * at the very end.
3969f45a3c8SSimon J. Gerraty 	 */
39706b9b3e0SSimon J. Gerraty 	OP_SAVE_CMDS	= 1 << 25,
3989f45a3c8SSimon J. Gerraty 	/*
3999f45a3c8SSimon J. Gerraty 	 * Already processed by Suff_FindDeps, to find dependencies from
4009f45a3c8SSimon J. Gerraty 	 * suffix transformation rules.
4019f45a3c8SSimon J. Gerraty 	 */
40206b9b3e0SSimon J. Gerraty 	OP_DEPS_FOUND	= 1 << 24,
4032c3632d1SSimon J. Gerraty 	/* Node found while expanding .ALLSRC */
4049f45a3c8SSimon J. Gerraty 	OP_MARK		= 1 << 23
4052c3632d1SSimon J. Gerraty } GNodeType;
4062c3632d1SSimon J. Gerraty 
40712904384SSimon J. Gerraty typedef struct GNodeFlags {
40806b9b3e0SSimon J. Gerraty 	/* this target needs to be (re)made */
40912904384SSimon J. Gerraty 	bool remake:1;
41006b9b3e0SSimon J. Gerraty 	/* children of this target were made */
41112904384SSimon J. Gerraty 	bool childMade:1;
41206b9b3e0SSimon J. Gerraty 	/* children don't exist, and we pretend made */
41312904384SSimon J. Gerraty 	bool force:1;
41406b9b3e0SSimon J. Gerraty 	/* Set by Make_ProcessWait() */
41512904384SSimon J. Gerraty 	bool doneWait:1;
41606b9b3e0SSimon J. Gerraty 	/* Build requested by .ORDER processing */
41712904384SSimon J. Gerraty 	bool doneOrder:1;
41806b9b3e0SSimon J. Gerraty 	/* Node created from .depend */
41912904384SSimon J. Gerraty 	bool fromDepend:1;
42006b9b3e0SSimon J. Gerraty 	/* We do it once only */
42112904384SSimon J. Gerraty 	bool doneAllsrc:1;
42206b9b3e0SSimon J. Gerraty 	/* Used by MakePrintStatus */
42312904384SSimon J. Gerraty 	bool cycle:1;
42406b9b3e0SSimon J. Gerraty 	/* Used by MakePrintStatus */
42512904384SSimon J. Gerraty 	bool doneCycle:1;
4262c3632d1SSimon J. Gerraty } GNodeFlags;
4272c3632d1SSimon J. Gerraty 
428956e45f6SSimon J. Gerraty typedef struct List StringList;
429956e45f6SSimon J. Gerraty typedef struct ListNode StringListNode;
430956e45f6SSimon J. Gerraty 
431956e45f6SSimon J. Gerraty typedef struct List GNodeList;
432956e45f6SSimon J. Gerraty typedef struct ListNode GNodeListNode;
433956e45f6SSimon J. Gerraty 
434dba7b0efSSimon J. Gerraty typedef struct SearchPath {
435dba7b0efSSimon J. Gerraty 	List /* of CachedDir */ dirs;
436dba7b0efSSimon J. Gerraty } SearchPath;
437956e45f6SSimon J. Gerraty 
43806b9b3e0SSimon J. Gerraty /*
43906b9b3e0SSimon J. Gerraty  * A graph node represents a target that can possibly be made, including its
440548bfc56SSimon J. Gerraty  * relation to other targets.
44106b9b3e0SSimon J. Gerraty  */
4422c3632d1SSimon J. Gerraty typedef struct GNode {
4432c3632d1SSimon J. Gerraty 	/* The target's name, such as "clean" or "make.c" */
4442c3632d1SSimon J. Gerraty 	char *name;
4452c3632d1SSimon J. Gerraty 	/* The unexpanded name of a .USE node */
4462c3632d1SSimon J. Gerraty 	char *uname;
4479f45a3c8SSimon J. Gerraty 	/*
4489f45a3c8SSimon J. Gerraty 	 * The full pathname of the file belonging to the target.
4499f45a3c8SSimon J. Gerraty 	 *
45006b9b3e0SSimon J. Gerraty 	 * XXX: What about .PHONY targets? These don't have an associated
4519f45a3c8SSimon J. Gerraty 	 * path.
4529f45a3c8SSimon J. Gerraty 	 */
4532c3632d1SSimon J. Gerraty 	char *path;
4542c3632d1SSimon J. Gerraty 
4559f45a3c8SSimon J. Gerraty 	/*
4569f45a3c8SSimon J. Gerraty 	 * The type of operator used to define the sources (see the OP flags
45706b9b3e0SSimon J. Gerraty 	 * below).
4589f45a3c8SSimon J. Gerraty 	 *
4599f45a3c8SSimon J. Gerraty 	 * XXX: This looks like a wild mixture of type and flags.
4609f45a3c8SSimon J. Gerraty 	 */
4612c3632d1SSimon J. Gerraty 	GNodeType type;
4622c3632d1SSimon J. Gerraty 	GNodeFlags flags;
4632c3632d1SSimon J. Gerraty 
4642c3632d1SSimon J. Gerraty 	/* The state of processing on this node */
4652c3632d1SSimon J. Gerraty 	GNodeMade made;
46606b9b3e0SSimon J. Gerraty 	/* The number of unmade children */
46706b9b3e0SSimon J. Gerraty 	int unmade;
4683955d011SMarcel Moolenaar 
4699f45a3c8SSimon J. Gerraty 	/*
4709f45a3c8SSimon J. Gerraty 	 * The modification time; 0 means the node does not have a
4719f45a3c8SSimon J. Gerraty 	 * corresponding file; see GNode_IsOODate.
4729f45a3c8SSimon J. Gerraty 	 */
473956e45f6SSimon J. Gerraty 	time_t mtime;
474956e45f6SSimon J. Gerraty 	struct GNode *youngestChild;
4753955d011SMarcel Moolenaar 
4769f45a3c8SSimon J. Gerraty 	/*
4779f45a3c8SSimon J. Gerraty 	 * The GNodes for which this node is an implied source. May be empty.
4789f45a3c8SSimon J. Gerraty 	 * For example, when there is an inference rule for .c.o, the node
4799f45a3c8SSimon J. Gerraty 	 * for file.c has the node for file.o in this list.
4809f45a3c8SSimon J. Gerraty 	 */
48106b9b3e0SSimon J. Gerraty 	GNodeList implicitParents;
4823955d011SMarcel Moolenaar 
4839f45a3c8SSimon J. Gerraty 	/*
4849f45a3c8SSimon J. Gerraty 	 * The nodes that depend on this one, or in other words, the nodes
4859f45a3c8SSimon J. Gerraty 	 * for which this is a source.
4869f45a3c8SSimon J. Gerraty 	 */
48706b9b3e0SSimon J. Gerraty 	GNodeList parents;
4882c3632d1SSimon J. Gerraty 	/* The nodes on which this one depends. */
48906b9b3e0SSimon J. Gerraty 	GNodeList children;
4903955d011SMarcel Moolenaar 
4919f45a3c8SSimon J. Gerraty 	/*
4929f45a3c8SSimon J. Gerraty 	 * .ORDER nodes we need made. The nodes that must be made (if they're
4932c3632d1SSimon J. Gerraty 	 * made) before this node can be made, but that do not enter into the
4949f45a3c8SSimon J. Gerraty 	 * datedness of this node.
4959f45a3c8SSimon J. Gerraty 	 */
49606b9b3e0SSimon J. Gerraty 	GNodeList order_pred;
4979f45a3c8SSimon J. Gerraty 	/*
4989f45a3c8SSimon J. Gerraty 	 * .ORDER nodes who need us. The nodes that must be made (if they're
49906b9b3e0SSimon J. Gerraty 	 * made at all) after this node is made, but that do not depend on
5009f45a3c8SSimon J. Gerraty 	 * this node, in the normal sense.
5019f45a3c8SSimon J. Gerraty 	 */
50206b9b3e0SSimon J. Gerraty 	GNodeList order_succ;
5032c3632d1SSimon J. Gerraty 
504dba7b0efSSimon J. Gerraty 	/*
505dba7b0efSSimon J. Gerraty 	 * Other nodes of the same name, for targets that were defined using
506dba7b0efSSimon J. Gerraty 	 * the '::' dependency operator (OP_DOUBLEDEP).
507dba7b0efSSimon J. Gerraty 	 */
50806b9b3e0SSimon J. Gerraty 	GNodeList cohorts;
509956e45f6SSimon J. Gerraty 	/* The "#n" suffix for this cohort, or "" for other nodes */
5102c3632d1SSimon J. Gerraty 	char cohort_num[8];
5112c3632d1SSimon J. Gerraty 	/* The number of unmade instances on the cohorts list */
5122c3632d1SSimon J. Gerraty 	int unmade_cohorts;
5139f45a3c8SSimon J. Gerraty 	/*
5149f45a3c8SSimon J. Gerraty 	 * Pointer to the first instance of a '::' node; only set when on a
5159f45a3c8SSimon J. Gerraty 	 * cohorts list
5169f45a3c8SSimon J. Gerraty 	 */
5172c3632d1SSimon J. Gerraty 	struct GNode *centurion;
5182c3632d1SSimon J. Gerraty 
5192c3632d1SSimon J. Gerraty 	/* Last time (sequence number) we tried to make this node */
520956e45f6SSimon J. Gerraty 	unsigned int checked_seqno;
5212c3632d1SSimon J. Gerraty 
522dba7b0efSSimon J. Gerraty 	/*
523dba7b0efSSimon J. Gerraty 	 * The "local" variables that are specific to this target and this
52406b9b3e0SSimon J. Gerraty 	 * target only, such as $@, $<, $?.
525956e45f6SSimon J. Gerraty 	 *
526dba7b0efSSimon J. Gerraty 	 * Also used for the global variable scopes SCOPE_GLOBAL,
527dba7b0efSSimon J. Gerraty 	 * SCOPE_CMDLINE, SCOPE_INTERNAL, which contain variables with
528dba7b0efSSimon J. Gerraty 	 * arbitrary names.
529dba7b0efSSimon J. Gerraty 	 */
53006b9b3e0SSimon J. Gerraty 	HashTable /* of Var pointer */ vars;
5312c3632d1SSimon J. Gerraty 
5322c3632d1SSimon J. Gerraty 	/* The commands to be given to a shell to create this target. */
53306b9b3e0SSimon J. Gerraty 	StringList commands;
5342c3632d1SSimon J. Gerraty 
5359f45a3c8SSimon J. Gerraty 	/*
5369f45a3c8SSimon J. Gerraty 	 * Suffix for the node (determined by Suff_FindDeps and opaque to
5379f45a3c8SSimon J. Gerraty 	 * everyone but the Suff module)
5389f45a3c8SSimon J. Gerraty 	 */
53906b9b3e0SSimon J. Gerraty 	struct Suffix *suffix;
5402c3632d1SSimon J. Gerraty 
5419f45a3c8SSimon J. Gerraty 	/* Filename where the GNode got defined, unlimited lifetime */
5422c3632d1SSimon J. Gerraty 	const char *fname;
5439f45a3c8SSimon J. Gerraty 	/* Line number where the GNode got defined, 1-based */
5449f45a3c8SSimon J. Gerraty 	unsigned lineno;
545c59c3bf3SSimon J. Gerraty 	int exit_status;
5463955d011SMarcel Moolenaar } GNode;
5473955d011SMarcel Moolenaar 
5482f2a5ecdSSimon J. Gerraty /*
5492f2a5ecdSSimon J. Gerraty  * Keep track of whether to include <posix.mk> when parsing the line
5502f2a5ecdSSimon J. Gerraty  * '.POSIX:'.
5512f2a5ecdSSimon J. Gerraty  */
5522f2a5ecdSSimon J. Gerraty extern enum PosixState {
5532f2a5ecdSSimon J. Gerraty 	PS_NOT_YET,
5542f2a5ecdSSimon J. Gerraty 	PS_MAYBE_NEXT_LINE,
5552f2a5ecdSSimon J. Gerraty 	PS_NOW_OR_NEVER,
5562f2a5ecdSSimon J. Gerraty 	PS_TOO_LATE
5572f2a5ecdSSimon J. Gerraty } posix_state;
5582f2a5ecdSSimon J. Gerraty 
559e2eeea75SSimon J. Gerraty /* Error levels for diagnostics during parsing. */
560956e45f6SSimon J. Gerraty typedef enum ParseErrorLevel {
5619f45a3c8SSimon J. Gerraty 	/*
5629f45a3c8SSimon J. Gerraty 	 * Exit when the current top-level makefile has been parsed
5639f45a3c8SSimon J. Gerraty 	 * completely.
5649f45a3c8SSimon J. Gerraty 	 */
565956e45f6SSimon J. Gerraty 	PARSE_FATAL = 1,
566e2eeea75SSimon J. Gerraty 	/* Print "warning"; may be upgraded to fatal by the -w option. */
567956e45f6SSimon J. Gerraty 	PARSE_WARNING,
568e2eeea75SSimon J. Gerraty 	/* Informational, mainly used during development of makefiles. */
569956e45f6SSimon J. Gerraty 	PARSE_INFO
570956e45f6SSimon J. Gerraty } ParseErrorLevel;
5713955d011SMarcel Moolenaar 
5723955d011SMarcel Moolenaar /*
573956e45f6SSimon J. Gerraty  * Values returned by Cond_EvalLine and Cond_EvalCondition.
5743955d011SMarcel Moolenaar  */
5759f45a3c8SSimon J. Gerraty typedef enum CondResult {
5769f45a3c8SSimon J. Gerraty 	CR_TRUE,		/* Parse the next lines */
5779f45a3c8SSimon J. Gerraty 	CR_FALSE,		/* Skip the next lines */
5789f45a3c8SSimon J. Gerraty 	CR_ERROR		/* Unknown directive or parse error */
5799f45a3c8SSimon J. Gerraty } CondResult;
5803955d011SMarcel Moolenaar 
581148ee845SSimon J. Gerraty typedef struct {
582148ee845SSimon J. Gerraty 	enum GuardKind {
583148ee845SSimon J. Gerraty 		GK_VARIABLE,
584148ee845SSimon J. Gerraty 		GK_TARGET
585148ee845SSimon J. Gerraty 	} kind;
586148ee845SSimon J. Gerraty 	char *name;
587148ee845SSimon J. Gerraty } Guard;
588148ee845SSimon J. Gerraty 
589e2eeea75SSimon J. Gerraty /* Names of the variables that are "local" to a specific target. */
5903955d011SMarcel Moolenaar #define TARGET	"@"		/* Target of dependency */
5913955d011SMarcel Moolenaar #define OODATE	"?"		/* All out-of-date sources */
5923955d011SMarcel Moolenaar #define ALLSRC	">"		/* All sources */
5933955d011SMarcel Moolenaar #define IMPSRC	"<"		/* Source implied by transformation */
5943955d011SMarcel Moolenaar #define PREFIX	"*"		/* Common prefix */
5953955d011SMarcel Moolenaar #define ARCHIVE	"!"		/* Archive in "archive(member)" syntax */
5963955d011SMarcel Moolenaar #define MEMBER	"%"		/* Member in "archive(member)" syntax */
5973955d011SMarcel Moolenaar 
5983955d011SMarcel Moolenaar /*
5993955d011SMarcel Moolenaar  * Global Variables
6003955d011SMarcel Moolenaar  */
6013955d011SMarcel Moolenaar 
602e2eeea75SSimon J. Gerraty /* True if every target is precious */
603b0c40a00SSimon J. Gerraty extern bool allPrecious;
604e2eeea75SSimon J. Gerraty /* True if failed targets should be deleted */
605b0c40a00SSimon J. Gerraty extern bool deleteOnError;
606b0c40a00SSimon J. Gerraty /* true while processing .depend */
607b0c40a00SSimon J. Gerraty extern bool doing_depend;
608e2eeea75SSimon J. Gerraty /* .DEFAULT rule */
609e2eeea75SSimon J. Gerraty extern GNode *defaultNode;
6103955d011SMarcel Moolenaar 
61106b9b3e0SSimon J. Gerraty /*
61206b9b3e0SSimon J. Gerraty  * Variables defined internally by make which should not override those set
61306b9b3e0SSimon J. Gerraty  * by makefiles.
61406b9b3e0SSimon J. Gerraty  */
615dba7b0efSSimon J. Gerraty extern GNode *SCOPE_INTERNAL;
616dba7b0efSSimon J. Gerraty /* Variables defined in a global scope, e.g in the makefile itself. */
617dba7b0efSSimon J. Gerraty extern GNode *SCOPE_GLOBAL;
618e2eeea75SSimon J. Gerraty /* Variables defined on the command line. */
619dba7b0efSSimon J. Gerraty extern GNode *SCOPE_CMDLINE;
620e2eeea75SSimon J. Gerraty 
62106b9b3e0SSimon J. Gerraty /*
622548bfc56SSimon J. Gerraty  * Value returned by Var_Parse when an error is encountered. It points to an
623548bfc56SSimon J. Gerraty  * empty string, so naive callers needn't worry about it.
62406b9b3e0SSimon J. Gerraty  */
625e2eeea75SSimon J. Gerraty extern char var_Error[];
626e2eeea75SSimon J. Gerraty 
627e2eeea75SSimon J. Gerraty /* The time at the start of this whole process */
628e2eeea75SSimon J. Gerraty extern time_t now;
629e2eeea75SSimon J. Gerraty 
630e2eeea75SSimon J. Gerraty /*
63106b9b3e0SSimon J. Gerraty  * The list of directories to search when looking for targets (set by the
63206b9b3e0SSimon J. Gerraty  * special target .PATH).
6331bbe5942SSimon J. Gerraty  */
63406b9b3e0SSimon J. Gerraty extern SearchPath dirSearchPath;
635e2eeea75SSimon J. Gerraty /* Used for .include "...". */
636e2eeea75SSimon J. Gerraty extern SearchPath *parseIncPath;
63706b9b3e0SSimon J. Gerraty /*
638dba7b0efSSimon J. Gerraty  * Used for .include <...>, for the built-in sys.mk and for makefiles from
639dba7b0efSSimon J. Gerraty  * the command line arguments.
64006b9b3e0SSimon J. Gerraty  */
641e2eeea75SSimon J. Gerraty extern SearchPath *sysIncPath;
642e2eeea75SSimon J. Gerraty /* The default for sysIncPath. */
643e2eeea75SSimon J. Gerraty extern SearchPath *defSysIncPath;
6443955d011SMarcel Moolenaar 
645e2eeea75SSimon J. Gerraty /* Startup directory */
646e2eeea75SSimon J. Gerraty extern char curdir[];
647e2eeea75SSimon J. Gerraty /* The basename of the program name, suffixed with [n] for sub-makes.  */
64806b9b3e0SSimon J. Gerraty extern const char *progname;
649dba7b0efSSimon J. Gerraty extern int makelevel;
650e2eeea75SSimon J. Gerraty /* Name of the .depend makefile */
651e2eeea75SSimon J. Gerraty extern char *makeDependfile;
652e2eeea75SSimon J. Gerraty /* If we replaced environ, this will be non-NULL. */
653e2eeea75SSimon J. Gerraty extern char **savedEnv;
6549f45a3c8SSimon J. Gerraty extern GNode *mainNode;
6553955d011SMarcel Moolenaar 
6563955d011SMarcel Moolenaar extern pid_t myPid;
6573955d011SMarcel Moolenaar 
6583955d011SMarcel Moolenaar #define MAKEFLAGS	".MAKEFLAGS"
65951ee2c1cSSimon J. Gerraty #ifndef MAKE_LEVEL_ENV
66051ee2c1cSSimon J. Gerraty # define MAKE_LEVEL_ENV	"MAKELEVEL"
6613955d011SMarcel Moolenaar #endif
6623955d011SMarcel Moolenaar 
6639f45a3c8SSimon J. Gerraty typedef struct DebugFlags {
6649f45a3c8SSimon J. Gerraty 	bool DEBUG_ARCH:1;
6659f45a3c8SSimon J. Gerraty 	bool DEBUG_COND:1;
6669f45a3c8SSimon J. Gerraty 	bool DEBUG_CWD:1;
6679f45a3c8SSimon J. Gerraty 	bool DEBUG_DIR:1;
6689f45a3c8SSimon J. Gerraty 	bool DEBUG_ERROR:1;
6699f45a3c8SSimon J. Gerraty 	bool DEBUG_FOR:1;
6709f45a3c8SSimon J. Gerraty 	bool DEBUG_GRAPH1:1;
6719f45a3c8SSimon J. Gerraty 	bool DEBUG_GRAPH2:1;
6729f45a3c8SSimon J. Gerraty 	bool DEBUG_GRAPH3:1;
6739f45a3c8SSimon J. Gerraty 	bool DEBUG_HASH:1;
6749f45a3c8SSimon J. Gerraty 	bool DEBUG_JOB:1;
6759f45a3c8SSimon J. Gerraty 	bool DEBUG_LOUD:1;
6769f45a3c8SSimon J. Gerraty 	bool DEBUG_MAKE:1;
6779f45a3c8SSimon J. Gerraty 	bool DEBUG_META:1;
6789f45a3c8SSimon J. Gerraty 	bool DEBUG_PARSE:1;
6799f45a3c8SSimon J. Gerraty 	bool DEBUG_SCRIPT:1;
6809f45a3c8SSimon J. Gerraty 	bool DEBUG_SHELL:1;
6819f45a3c8SSimon J. Gerraty 	bool DEBUG_SUFF:1;
6829f45a3c8SSimon J. Gerraty 	bool DEBUG_TARG:1;
6839f45a3c8SSimon J. Gerraty 	bool DEBUG_VAR:1;
684956e45f6SSimon J. Gerraty } DebugFlags;
6852c3632d1SSimon J. Gerraty 
6863955d011SMarcel Moolenaar #define CONCAT(a, b) a##b
6873955d011SMarcel Moolenaar 
6889f45a3c8SSimon J. Gerraty #define DEBUG(module) (opts.debug.CONCAT(DEBUG_, module))
689956e45f6SSimon J. Gerraty 
690956e45f6SSimon J. Gerraty void debug_printf(const char *, ...) MAKE_ATTR_PRINTFLIKE(1, 2);
691956e45f6SSimon J. Gerraty 
69206b9b3e0SSimon J. Gerraty #define DEBUG_IMPL(module, args) \
69306b9b3e0SSimon J. Gerraty 	do { \
69406b9b3e0SSimon J. Gerraty 		if (DEBUG(module)) \
69506b9b3e0SSimon J. Gerraty 			debug_printf args; \
69612904384SSimon J. Gerraty 	} while (false)
69706b9b3e0SSimon J. Gerraty 
6989f45a3c8SSimon J. Gerraty #define DEBUG0(module, fmt) \
6999f45a3c8SSimon J. Gerraty 	DEBUG_IMPL(module, (fmt))
700956e45f6SSimon J. Gerraty #define DEBUG1(module, fmt, arg1) \
70106b9b3e0SSimon J. Gerraty 	DEBUG_IMPL(module, (fmt, arg1))
702956e45f6SSimon J. Gerraty #define DEBUG2(module, fmt, arg1, arg2) \
70306b9b3e0SSimon J. Gerraty 	DEBUG_IMPL(module, (fmt, arg1, arg2))
704956e45f6SSimon J. Gerraty #define DEBUG3(module, fmt, arg1, arg2, arg3) \
70506b9b3e0SSimon J. Gerraty 	DEBUG_IMPL(module, (fmt, arg1, arg2, arg3))
706956e45f6SSimon J. Gerraty #define DEBUG4(module, fmt, arg1, arg2, arg3, arg4) \
70706b9b3e0SSimon J. Gerraty 	DEBUG_IMPL(module, (fmt, arg1, arg2, arg3, arg4))
708956e45f6SSimon J. Gerraty #define DEBUG5(module, fmt, arg1, arg2, arg3, arg4, arg5) \
70906b9b3e0SSimon J. Gerraty 	DEBUG_IMPL(module, (fmt, arg1, arg2, arg3, arg4, arg5))
710956e45f6SSimon J. Gerraty 
711956e45f6SSimon J. Gerraty typedef enum PrintVarsMode {
712e2eeea75SSimon J. Gerraty 	PVM_NONE,
713e2eeea75SSimon J. Gerraty 	PVM_UNEXPANDED,
714e2eeea75SSimon J. Gerraty 	PVM_EXPANDED
715956e45f6SSimon J. Gerraty } PrintVarsMode;
716956e45f6SSimon J. Gerraty 
717956e45f6SSimon J. Gerraty /* Command line options */
718956e45f6SSimon J. Gerraty typedef struct CmdOpts {
719548bfc56SSimon J. Gerraty 	/* -B: whether to be compatible to traditional make */
720b0c40a00SSimon J. Gerraty 	bool compatMake;
721956e45f6SSimon J. Gerraty 
7229f45a3c8SSimon J. Gerraty 	/*
723548bfc56SSimon J. Gerraty 	 * -d: debug control: There is one flag per module.  It is up to the
7249f45a3c8SSimon J. Gerraty 	 * module what debug information to print.
7259f45a3c8SSimon J. Gerraty 	 */
726956e45f6SSimon J. Gerraty 	DebugFlags debug;
727956e45f6SSimon J. Gerraty 
7286a7405f5SSimon J. Gerraty 	/* -dF: debug output is written here - default stderr */
729956e45f6SSimon J. Gerraty 	FILE *debug_file;
730956e45f6SSimon J. Gerraty 
7319f45a3c8SSimon J. Gerraty 	/*
7329f45a3c8SSimon J. Gerraty 	 * -dL: lint mode
733e2eeea75SSimon J. Gerraty 	 *
734e2eeea75SSimon J. Gerraty 	 * Runs make in strict mode, with additional checks and better error
7359f45a3c8SSimon J. Gerraty 	 * handling.
7369f45a3c8SSimon J. Gerraty 	 */
737b0c40a00SSimon J. Gerraty 	bool strict;
738e2eeea75SSimon J. Gerraty 
739956e45f6SSimon J. Gerraty 	/* -dV: for the -V option, print unexpanded variable values */
740b0c40a00SSimon J. Gerraty 	bool debugVflag;
741956e45f6SSimon J. Gerraty 
742956e45f6SSimon J. Gerraty 	/* -e: check environment variables before global variables */
743b0c40a00SSimon J. Gerraty 	bool checkEnvFirst;
744956e45f6SSimon J. Gerraty 
745956e45f6SSimon J. Gerraty 	/* -f: the makefiles to read */
74606b9b3e0SSimon J. Gerraty 	StringList makefiles;
747956e45f6SSimon J. Gerraty 
748956e45f6SSimon J. Gerraty 	/* -i: if true, ignore all errors from shell commands */
749b0c40a00SSimon J. Gerraty 	bool ignoreErrors;
750956e45f6SSimon J. Gerraty 
7519f45a3c8SSimon J. Gerraty 	/*
7529f45a3c8SSimon J. Gerraty 	 * -j: the maximum number of jobs that can run in parallel; this is
7539f45a3c8SSimon J. Gerraty 	 * coordinated with the submakes
7549f45a3c8SSimon J. Gerraty 	 */
755956e45f6SSimon J. Gerraty 	int maxJobs;
756956e45f6SSimon J. Gerraty 
7579f45a3c8SSimon J. Gerraty 	/*
7589f45a3c8SSimon J. Gerraty 	 * -k: if true and an error occurs while making a node, continue
7599f45a3c8SSimon J. Gerraty 	 * making nodes that do not depend on the erroneous node
7609f45a3c8SSimon J. Gerraty 	 */
761b0c40a00SSimon J. Gerraty 	bool keepgoing;
762956e45f6SSimon J. Gerraty 
763956e45f6SSimon J. Gerraty 	/* -N: execute no commands from the targets */
764b0c40a00SSimon J. Gerraty 	bool noRecursiveExecute;
765956e45f6SSimon J. Gerraty 
766956e45f6SSimon J. Gerraty 	/* -n: execute almost no commands from the targets */
767b0c40a00SSimon J. Gerraty 	bool noExecute;
768956e45f6SSimon J. Gerraty 
769dba7b0efSSimon J. Gerraty 	/*
770dba7b0efSSimon J. Gerraty 	 * -q: if true, do not really make anything, just see if the targets
771dba7b0efSSimon J. Gerraty 	 * are out-of-date
772dba7b0efSSimon J. Gerraty 	 */
7739f45a3c8SSimon J. Gerraty 	bool query;
774956e45f6SSimon J. Gerraty 
775dba7b0efSSimon J. Gerraty 	/* -r: raw mode, do not load the builtin rules. */
776b0c40a00SSimon J. Gerraty 	bool noBuiltins;
777956e45f6SSimon J. Gerraty 
778956e45f6SSimon J. Gerraty 	/* -s: don't echo the shell commands before executing them */
7799f45a3c8SSimon J. Gerraty 	bool silent;
780956e45f6SSimon J. Gerraty 
7819f45a3c8SSimon J. Gerraty 	/*
7829f45a3c8SSimon J. Gerraty 	 * -t: touch the targets if they are out-of-date, but don't actually
7839f45a3c8SSimon J. Gerraty 	 * make them
7849f45a3c8SSimon J. Gerraty 	 */
7859f45a3c8SSimon J. Gerraty 	bool touch;
786956e45f6SSimon J. Gerraty 
787956e45f6SSimon J. Gerraty 	/* -[Vv]: print expanded or unexpanded selected variables */
788956e45f6SSimon J. Gerraty 	PrintVarsMode printVars;
789956e45f6SSimon J. Gerraty 	/* -[Vv]: the variables to print */
79006b9b3e0SSimon J. Gerraty 	StringList variables;
791956e45f6SSimon J. Gerraty 
792956e45f6SSimon J. Gerraty 	/* -W: if true, makefile parsing warnings are treated as errors */
793b0c40a00SSimon J. Gerraty 	bool parseWarnFatal;
794956e45f6SSimon J. Gerraty 
795dba7b0efSSimon J. Gerraty 	/* -w: print 'Entering' and 'Leaving' for submakes */
796b0c40a00SSimon J. Gerraty 	bool enterFlag;
797956e45f6SSimon J. Gerraty 
7989f45a3c8SSimon J. Gerraty 	/*
7999f45a3c8SSimon J. Gerraty 	 * -X: if true, do not export variables set on the command line to
8009f45a3c8SSimon J. Gerraty 	 * the environment.
8019f45a3c8SSimon J. Gerraty 	 */
802b0c40a00SSimon J. Gerraty 	bool varNoExportEnv;
803956e45f6SSimon J. Gerraty 
8049f45a3c8SSimon J. Gerraty 	/*
8059f45a3c8SSimon J. Gerraty 	 * The target names specified on the command line. Used to resolve
8069f45a3c8SSimon J. Gerraty 	 * .if make(...) statements.
8079f45a3c8SSimon J. Gerraty 	 */
80806b9b3e0SSimon J. Gerraty 	StringList create;
809956e45f6SSimon J. Gerraty 
810954401e6SSimon J. Gerraty 	/*
811954401e6SSimon J. Gerraty 	 * Randomize the order in which the targets from toBeMade are made,
812954401e6SSimon J. Gerraty 	 * to catch undeclared dependencies.
813954401e6SSimon J. Gerraty 	 */
814954401e6SSimon J. Gerraty 	bool randomizeTargets;
815956e45f6SSimon J. Gerraty } CmdOpts;
816956e45f6SSimon J. Gerraty 
817956e45f6SSimon J. Gerraty extern CmdOpts opts;
8188c973ee2SSimon J. Gerraty extern bool forceJobs;
8198c973ee2SSimon J. Gerraty extern char **environ;
8203955d011SMarcel Moolenaar 
8219f45a3c8SSimon J. Gerraty /* arch.c */
8229f45a3c8SSimon J. Gerraty void Arch_Init(void);
82322619282SSimon J. Gerraty #ifdef CLEANUP
8249f45a3c8SSimon J. Gerraty void Arch_End(void);
82522619282SSimon J. Gerraty #endif
8263955d011SMarcel Moolenaar 
8279f45a3c8SSimon J. Gerraty bool Arch_ParseArchive(char **, GNodeList *, GNode *);
8289f45a3c8SSimon J. Gerraty void Arch_Touch(GNode *);
8299f45a3c8SSimon J. Gerraty void Arch_TouchLib(GNode *);
830148ee845SSimon J. Gerraty void Arch_UpdateMTime(GNode *);
831148ee845SSimon J. Gerraty void Arch_UpdateMemberMTime(GNode *);
8329f45a3c8SSimon J. Gerraty void Arch_FindLib(GNode *, SearchPath *);
8339f45a3c8SSimon J. Gerraty bool Arch_LibOODate(GNode *) MAKE_ATTR_USE;
8349f45a3c8SSimon J. Gerraty bool Arch_IsLib(GNode *) MAKE_ATTR_USE;
8359f45a3c8SSimon J. Gerraty 
8369f45a3c8SSimon J. Gerraty /* compat.c */
8379f45a3c8SSimon J. Gerraty bool Compat_RunCommand(const char *, GNode *, StringListNode *);
838954401e6SSimon J. Gerraty void Compat_MakeAll(GNodeList *);
8399f45a3c8SSimon J. Gerraty void Compat_Make(GNode *, GNode *);
8409f45a3c8SSimon J. Gerraty 
8419f45a3c8SSimon J. Gerraty /* cond.c */
8424fde40d9SSimon J. Gerraty extern unsigned int cond_depth;
8439f45a3c8SSimon J. Gerraty CondResult Cond_EvalCondition(const char *) MAKE_ATTR_USE;
8449f45a3c8SSimon J. Gerraty CondResult Cond_EvalLine(const char *) MAKE_ATTR_USE;
845148ee845SSimon J. Gerraty Guard *Cond_ExtractGuard(const char *) MAKE_ATTR_USE;
8464fde40d9SSimon J. Gerraty void Cond_EndFile(void);
8479f45a3c8SSimon J. Gerraty 
8489f45a3c8SSimon J. Gerraty /* dir.c; see also dir.h */
8499f45a3c8SSimon J. Gerraty 
8509f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
8519f45a3c8SSimon J. Gerraty str_basename(const char *pathname)
8529f45a3c8SSimon J. Gerraty {
8539f45a3c8SSimon J. Gerraty 	const char *lastSlash = strrchr(pathname, '/');
8549f45a3c8SSimon J. Gerraty 	return lastSlash != NULL ? lastSlash + 1 : pathname;
8559f45a3c8SSimon J. Gerraty }
8569f45a3c8SSimon J. Gerraty 
8579f45a3c8SSimon J. Gerraty MAKE_INLINE SearchPath * MAKE_ATTR_USE
8589f45a3c8SSimon J. Gerraty SearchPath_New(void)
8599f45a3c8SSimon J. Gerraty {
8609f45a3c8SSimon J. Gerraty 	SearchPath *path = bmake_malloc(sizeof *path);
8619f45a3c8SSimon J. Gerraty 	Lst_Init(&path->dirs);
8629f45a3c8SSimon J. Gerraty 	return path;
8639f45a3c8SSimon J. Gerraty }
8649f45a3c8SSimon J. Gerraty 
8659f45a3c8SSimon J. Gerraty void SearchPath_Free(SearchPath *);
8669f45a3c8SSimon J. Gerraty 
8679f45a3c8SSimon J. Gerraty /* for.c */
8689f45a3c8SSimon J. Gerraty struct ForLoop;
8699f45a3c8SSimon J. Gerraty int For_Eval(const char *) MAKE_ATTR_USE;
8709f45a3c8SSimon J. Gerraty bool For_Accum(const char *, int *) MAKE_ATTR_USE;
8719f45a3c8SSimon J. Gerraty void For_Run(unsigned, unsigned);
8729f45a3c8SSimon J. Gerraty bool For_NextIteration(struct ForLoop *, Buffer *);
873148ee845SSimon J. Gerraty char *ForLoop_Details(const struct ForLoop *);
8749f45a3c8SSimon J. Gerraty void ForLoop_Free(struct ForLoop *);
8754fde40d9SSimon J. Gerraty void For_Break(struct ForLoop *);
8769f45a3c8SSimon J. Gerraty 
8779f45a3c8SSimon J. Gerraty /* job.c */
8789f45a3c8SSimon J. Gerraty void JobReapChild(pid_t, int, bool);
8799f45a3c8SSimon J. Gerraty 
88022619282SSimon J. Gerraty /* longer than this we use a temp file */
88122619282SSimon J. Gerraty #ifndef MAKE_CMDLEN_LIMIT
88222619282SSimon J. Gerraty # define MAKE_CMDLEN_LIMIT 1000
88322619282SSimon J. Gerraty #endif
8849f45a3c8SSimon J. Gerraty /* main.c */
8859f45a3c8SSimon J. Gerraty void Main_ParseArgLine(const char *);
88622619282SSimon J. Gerraty int Cmd_Argv(const char *, size_t, const char **, size_t, char *, size_t, bool, bool);
8879f45a3c8SSimon J. Gerraty char *Cmd_Exec(const char *, char **) MAKE_ATTR_USE;
8889f45a3c8SSimon J. Gerraty void Error(const char *, ...) MAKE_ATTR_PRINTFLIKE(1, 2);
8899f45a3c8SSimon J. Gerraty void Fatal(const char *, ...) MAKE_ATTR_PRINTFLIKE(1, 2) MAKE_ATTR_DEAD;
8909f45a3c8SSimon J. Gerraty void Punt(const char *, ...) MAKE_ATTR_PRINTFLIKE(1, 2) MAKE_ATTR_DEAD;
8919f45a3c8SSimon J. Gerraty void DieHorribly(void) MAKE_ATTR_DEAD;
8924fde40d9SSimon J. Gerraty int unlink_file(const char *) MAKE_ATTR_USE;
8939f45a3c8SSimon J. Gerraty void execDie(const char *, const char *);
8949f45a3c8SSimon J. Gerraty char *getTmpdir(void) MAKE_ATTR_USE;
8959f45a3c8SSimon J. Gerraty bool ParseBoolean(const char *, bool) MAKE_ATTR_USE;
8969f45a3c8SSimon J. Gerraty const char *cached_realpath(const char *, char *);
8979f45a3c8SSimon J. Gerraty bool GetBooleanExpr(const char *, bool);
8989f45a3c8SSimon J. Gerraty 
8999f45a3c8SSimon J. Gerraty /* parse.c */
90022619282SSimon J. Gerraty extern int parseErrors;
9019f45a3c8SSimon J. Gerraty void Parse_Init(void);
90222619282SSimon J. Gerraty #ifdef CLEANUP
9039f45a3c8SSimon J. Gerraty void Parse_End(void);
90422619282SSimon J. Gerraty #endif
9059f45a3c8SSimon J. Gerraty 
906954401e6SSimon J. Gerraty void PrintLocation(FILE *, bool, const GNode *);
9079f45a3c8SSimon J. Gerraty void PrintStackTrace(bool);
9089f45a3c8SSimon J. Gerraty void Parse_Error(ParseErrorLevel, const char *, ...) MAKE_ATTR_PRINTFLIKE(2, 3);
9099f45a3c8SSimon J. Gerraty bool Parse_VarAssign(const char *, bool, GNode *) MAKE_ATTR_USE;
9109f45a3c8SSimon J. Gerraty void Parse_File(const char *, int);
9119f45a3c8SSimon J. Gerraty void Parse_PushInput(const char *, unsigned, unsigned, Buffer,
9129f45a3c8SSimon J. Gerraty 		     struct ForLoop *);
9139f45a3c8SSimon J. Gerraty void Parse_MainName(GNodeList *);
9144fde40d9SSimon J. Gerraty unsigned int CurFile_CondMinDepth(void) MAKE_ATTR_USE;
915148ee845SSimon J. Gerraty void Parse_GuardElse(void);
916148ee845SSimon J. Gerraty void Parse_GuardEndif(void);
9179f45a3c8SSimon J. Gerraty 
9189f45a3c8SSimon J. Gerraty 
9199f45a3c8SSimon J. Gerraty /* suff.c */
9209f45a3c8SSimon J. Gerraty void Suff_Init(void);
92122619282SSimon J. Gerraty #ifdef CLEANUP
9229f45a3c8SSimon J. Gerraty void Suff_End(void);
92322619282SSimon J. Gerraty #endif
9249f45a3c8SSimon J. Gerraty 
9259f45a3c8SSimon J. Gerraty void Suff_ClearSuffixes(void);
9269f45a3c8SSimon J. Gerraty bool Suff_IsTransform(const char *) MAKE_ATTR_USE;
9279f45a3c8SSimon J. Gerraty GNode *Suff_AddTransform(const char *);
9289f45a3c8SSimon J. Gerraty void Suff_EndTransform(GNode *);
9299f45a3c8SSimon J. Gerraty void Suff_AddSuffix(const char *);
9309f45a3c8SSimon J. Gerraty SearchPath *Suff_GetPath(const char *) MAKE_ATTR_USE;
9319f45a3c8SSimon J. Gerraty void Suff_ExtendPaths(void);
9329f45a3c8SSimon J. Gerraty void Suff_AddInclude(const char *);
9339f45a3c8SSimon J. Gerraty void Suff_AddLib(const char *);
9349f45a3c8SSimon J. Gerraty void Suff_FindDeps(GNode *);
9359f45a3c8SSimon J. Gerraty SearchPath *Suff_FindPath(GNode *) MAKE_ATTR_USE;
9369f45a3c8SSimon J. Gerraty void Suff_SetNull(const char *);
9379f45a3c8SSimon J. Gerraty void Suff_PrintAll(void);
9389f45a3c8SSimon J. Gerraty char *Suff_NamesStr(void) MAKE_ATTR_USE;
9399f45a3c8SSimon J. Gerraty 
9409f45a3c8SSimon J. Gerraty /* targ.c */
9419f45a3c8SSimon J. Gerraty void Targ_Init(void);
9429f45a3c8SSimon J. Gerraty void Targ_End(void);
9439f45a3c8SSimon J. Gerraty 
9449f45a3c8SSimon J. Gerraty void Targ_Stats(void);
9459f45a3c8SSimon J. Gerraty GNodeList *Targ_List(void) MAKE_ATTR_USE;
9469f45a3c8SSimon J. Gerraty GNode *GNode_New(const char *) MAKE_ATTR_USE;
9479f45a3c8SSimon J. Gerraty GNode *Targ_FindNode(const char *) MAKE_ATTR_USE;
9489f45a3c8SSimon J. Gerraty GNode *Targ_GetNode(const char *) MAKE_ATTR_USE;
9499f45a3c8SSimon J. Gerraty GNode *Targ_NewInternalNode(const char *) MAKE_ATTR_USE;
9509f45a3c8SSimon J. Gerraty GNode *Targ_GetEndNode(void);
9519f45a3c8SSimon J. Gerraty void Targ_FindList(GNodeList *, StringList *);
9529f45a3c8SSimon J. Gerraty void Targ_PrintCmds(GNode *);
9539f45a3c8SSimon J. Gerraty void Targ_PrintNode(GNode *, int);
9549f45a3c8SSimon J. Gerraty void Targ_PrintNodes(GNodeList *, int);
9559f45a3c8SSimon J. Gerraty const char *Targ_FmtTime(time_t) MAKE_ATTR_USE;
9569f45a3c8SSimon J. Gerraty void Targ_PrintType(GNodeType);
9579f45a3c8SSimon J. Gerraty void Targ_PrintGraph(int);
9589f45a3c8SSimon J. Gerraty void Targ_Propagate(void);
9599f45a3c8SSimon J. Gerraty const char *GNodeMade_Name(GNodeMade) MAKE_ATTR_USE;
9608d5c8e21SSimon J. Gerraty #ifdef CLEANUP
9618d5c8e21SSimon J. Gerraty void Parse_RegisterCommand(char *);
9628d5c8e21SSimon J. Gerraty #else
9638d5c8e21SSimon J. Gerraty MAKE_INLINE
9648d5c8e21SSimon J. Gerraty void Parse_RegisterCommand(char *cmd MAKE_ATTR_UNUSED)
9658d5c8e21SSimon J. Gerraty {
9668d5c8e21SSimon J. Gerraty }
9678d5c8e21SSimon J. Gerraty #endif
9689f45a3c8SSimon J. Gerraty 
9699f45a3c8SSimon J. Gerraty /* var.c */
9709f45a3c8SSimon J. Gerraty 
9719f45a3c8SSimon J. Gerraty typedef enum VarEvalMode {
9729f45a3c8SSimon J. Gerraty 
9739f45a3c8SSimon J. Gerraty 	/*
9749f45a3c8SSimon J. Gerraty 	 * Only parse the expression but don't evaluate any part of it.
9759f45a3c8SSimon J. Gerraty 	 *
9769f45a3c8SSimon J. Gerraty 	 * TODO: Document what Var_Parse and Var_Subst return in this mode.
9779f45a3c8SSimon J. Gerraty 	 *  As of 2021-03-15, they return unspecified, inconsistent results.
9789f45a3c8SSimon J. Gerraty 	 */
9798d5c8e21SSimon J. Gerraty 	VARE_PARSE,
9809f45a3c8SSimon J. Gerraty 
9818c973ee2SSimon J. Gerraty 	/*
9828c973ee2SSimon J. Gerraty 	 * Parse text in which '${...}' and '$(...)' are not parsed as
9838c973ee2SSimon J. Gerraty 	 * subexpressions (with all their individual escaping rules) but
9848c973ee2SSimon J. Gerraty 	 * instead simply as text with balanced '${}' or '$()'.  Other '$'
9858c973ee2SSimon J. Gerraty 	 * are copied verbatim.
9868c973ee2SSimon J. Gerraty 	 */
9878c973ee2SSimon J. Gerraty 	VARE_PARSE_BALANCED,
9888c973ee2SSimon J. Gerraty 
9899f45a3c8SSimon J. Gerraty 	/* Parse and evaluate the expression. */
9908d5c8e21SSimon J. Gerraty 	VARE_EVAL,
9919f45a3c8SSimon J. Gerraty 
9929f45a3c8SSimon J. Gerraty 	/*
9939f45a3c8SSimon J. Gerraty 	 * Parse and evaluate the expression.  It is an error if a
9949f45a3c8SSimon J. Gerraty 	 * subexpression evaluates to undefined.
9959f45a3c8SSimon J. Gerraty 	 */
9966a7405f5SSimon J. Gerraty 	VARE_EVAL_DEFINED_LOUD,
9976a7405f5SSimon J. Gerraty 
9986a7405f5SSimon J. Gerraty 	/*
9996a7405f5SSimon J. Gerraty 	 * Parse and evaluate the expression.  It is a silent error if a
10006a7405f5SSimon J. Gerraty 	 * subexpression evaluates to undefined.
10016a7405f5SSimon J. Gerraty 	 */
10028d5c8e21SSimon J. Gerraty 	VARE_EVAL_DEFINED,
10039f45a3c8SSimon J. Gerraty 
10049f45a3c8SSimon J. Gerraty 	/*
10059f45a3c8SSimon J. Gerraty 	 * Parse and evaluate the expression.  Keep undefined variables as-is
10069f45a3c8SSimon J. Gerraty 	 * instead of expanding them to an empty string.
10079f45a3c8SSimon J. Gerraty 	 *
10089f45a3c8SSimon J. Gerraty 	 * Example for a ':=' assignment:
10099f45a3c8SSimon J. Gerraty 	 *	CFLAGS = $(.INCLUDES)
10109f45a3c8SSimon J. Gerraty 	 *	CFLAGS := -I.. $(CFLAGS)
10119f45a3c8SSimon J. Gerraty 	 *	# If .INCLUDES (an undocumented special variable, by the
10129f45a3c8SSimon J. Gerraty 	 *	# way) is still undefined, the updated CFLAGS becomes
10139f45a3c8SSimon J. Gerraty 	 *	# "-I.. $(.INCLUDES)".
10149f45a3c8SSimon J. Gerraty 	 */
10158d5c8e21SSimon J. Gerraty 	VARE_EVAL_KEEP_UNDEFINED,
10169f45a3c8SSimon J. Gerraty 
10179f45a3c8SSimon J. Gerraty 	/*
10189f45a3c8SSimon J. Gerraty 	 * Parse and evaluate the expression.  Keep '$$' as '$$' and preserve
10199f45a3c8SSimon J. Gerraty 	 * undefined subexpressions.
10209f45a3c8SSimon J. Gerraty 	 */
10218d5c8e21SSimon J. Gerraty 	VARE_EVAL_KEEP_DOLLAR_AND_UNDEFINED
10229f45a3c8SSimon J. Gerraty } VarEvalMode;
10239f45a3c8SSimon J. Gerraty 
10249f45a3c8SSimon J. Gerraty typedef enum VarSetFlags {
10259f45a3c8SSimon J. Gerraty 	VAR_SET_NONE		= 0,
10269f45a3c8SSimon J. Gerraty 
10279f45a3c8SSimon J. Gerraty 	/* do not export */
10289f45a3c8SSimon J. Gerraty 	VAR_SET_NO_EXPORT	= 1 << 0,
10299f45a3c8SSimon J. Gerraty 
10309f45a3c8SSimon J. Gerraty 	/*
10319f45a3c8SSimon J. Gerraty 	 * Make the variable read-only. No further modification is possible,
10328c973ee2SSimon J. Gerraty 	 * except for another call to Var_Set with the same flag. See the
10338c973ee2SSimon J. Gerraty 	 * special targets '.NOREADONLY' and '.READONLY'.
10349f45a3c8SSimon J. Gerraty 	 */
1035c9f4001fSSimon J. Gerraty 	VAR_SET_READONLY	= 1 << 1,
1036c9f4001fSSimon J. Gerraty 	VAR_SET_INTERNAL	= 1 << 2
10379f45a3c8SSimon J. Gerraty } VarSetFlags;
10389f45a3c8SSimon J. Gerraty 
10399f45a3c8SSimon J. Gerraty typedef enum VarExportMode {
10408d5c8e21SSimon J. Gerraty 	/* .export-all */
10418d5c8e21SSimon J. Gerraty 	VEM_ALL,
10429f45a3c8SSimon J. Gerraty 	/* .export-env */
10439f45a3c8SSimon J. Gerraty 	VEM_ENV,
10449f45a3c8SSimon J. Gerraty 	/* .export: Initial export or update an already exported variable. */
10459f45a3c8SSimon J. Gerraty 	VEM_PLAIN,
10469f45a3c8SSimon J. Gerraty 	/* .export-literal: Do not expand the variable value. */
10479f45a3c8SSimon J. Gerraty 	VEM_LITERAL
10489f45a3c8SSimon J. Gerraty } VarExportMode;
10499f45a3c8SSimon J. Gerraty 
10509f45a3c8SSimon J. Gerraty void Var_Delete(GNode *, const char *);
10518d5c8e21SSimon J. Gerraty #ifdef CLEANUP
10528d5c8e21SSimon J. Gerraty void Var_DeleteAll(GNode *scope);
10538d5c8e21SSimon J. Gerraty #endif
10549f45a3c8SSimon J. Gerraty void Var_Undef(const char *);
10559f45a3c8SSimon J. Gerraty void Var_Set(GNode *, const char *, const char *);
10569f45a3c8SSimon J. Gerraty void Var_SetExpand(GNode *, const char *, const char *);
10579f45a3c8SSimon J. Gerraty void Var_SetWithFlags(GNode *, const char *, const char *, VarSetFlags);
10589f45a3c8SSimon J. Gerraty void Var_Append(GNode *, const char *, const char *);
10599f45a3c8SSimon J. Gerraty void Var_AppendExpand(GNode *, const char *, const char *);
10609f45a3c8SSimon J. Gerraty bool Var_Exists(GNode *, const char *) MAKE_ATTR_USE;
10619f45a3c8SSimon J. Gerraty bool Var_ExistsExpand(GNode *, const char *) MAKE_ATTR_USE;
10629f45a3c8SSimon J. Gerraty FStr Var_Value(GNode *, const char *) MAKE_ATTR_USE;
10639f45a3c8SSimon J. Gerraty const char *GNode_ValueDirect(GNode *, const char *) MAKE_ATTR_USE;
10648c973ee2SSimon J. Gerraty FStr Var_Parse(const char **, GNode *, VarEvalMode);
10658c973ee2SSimon J. Gerraty char *Var_Subst(const char *, GNode *, VarEvalMode);
10668d5c8e21SSimon J. Gerraty char *Var_SubstInTarget(const char *, GNode *);
10679f45a3c8SSimon J. Gerraty void Var_Expand(FStr *, GNode *, VarEvalMode);
10689f45a3c8SSimon J. Gerraty void Var_Stats(void);
10699f45a3c8SSimon J. Gerraty void Var_Dump(GNode *);
1070c59c3bf3SSimon J. Gerraty void Var_ReexportVars(GNode *);
10719f45a3c8SSimon J. Gerraty void Var_Export(VarExportMode, const char *);
10729f45a3c8SSimon J. Gerraty void Var_ExportVars(const char *);
10739f45a3c8SSimon J. Gerraty void Var_UnExport(bool, const char *);
10744fde40d9SSimon J. Gerraty void Var_ReadOnly(const char *, bool);
10759f45a3c8SSimon J. Gerraty 
10769f45a3c8SSimon J. Gerraty void Global_Set(const char *, const char *);
10779f45a3c8SSimon J. Gerraty void Global_Append(const char *, const char *);
10789f45a3c8SSimon J. Gerraty void Global_Delete(const char *);
10794fde40d9SSimon J. Gerraty void Global_Set_ReadOnly(const char *, const char *);
10809f45a3c8SSimon J. Gerraty 
10816a7405f5SSimon J. Gerraty void EvalStack_PrintDetails(void);
1082548bfc56SSimon J. Gerraty 
10839f45a3c8SSimon J. Gerraty /* util.c */
10849f45a3c8SSimon J. Gerraty typedef void (*SignalProc)(int);
10859f45a3c8SSimon J. Gerraty SignalProc bmake_signal(int, SignalProc);
10869f45a3c8SSimon J. Gerraty 
10879f45a3c8SSimon J. Gerraty /* make.c */
1088e2eeea75SSimon J. Gerraty void GNode_UpdateYoungestChild(GNode *, GNode *);
10899f45a3c8SSimon J. Gerraty bool GNode_IsOODate(GNode *) MAKE_ATTR_USE;
1090956e45f6SSimon J. Gerraty void Make_ExpandUse(GNodeList *);
10919f45a3c8SSimon J. Gerraty time_t Make_Recheck(GNode *) MAKE_ATTR_USE;
10923955d011SMarcel Moolenaar void Make_HandleUse(GNode *, GNode *);
10933955d011SMarcel Moolenaar void Make_Update(GNode *);
1094b0c40a00SSimon J. Gerraty void GNode_SetLocalVars(GNode *);
1095b0c40a00SSimon J. Gerraty bool Make_Run(GNodeList *);
10969f45a3c8SSimon J. Gerraty bool shouldDieQuietly(GNode *, int) MAKE_ATTR_USE;
10973955d011SMarcel Moolenaar void PrintOnError(GNode *, const char *);
1098b0c40a00SSimon J. Gerraty void Main_ExportMAKEFLAGS(bool);
1099b0c40a00SSimon J. Gerraty bool Main_SetObjdir(bool, const char *, ...) MAKE_ATTR_PRINTFLIKE(2, 3);
11009f45a3c8SSimon J. Gerraty int mkTempFile(const char *, char *, size_t) MAKE_ATTR_USE;
1101d5e0a182SSimon J. Gerraty void AppendWords(StringList *, char *);
11022c3632d1SSimon J. Gerraty void GNode_FprintDetails(FILE *, const char *, const GNode *, const char *);
11039f45a3c8SSimon J. Gerraty bool GNode_ShouldExecute(GNode *gn) MAKE_ATTR_USE;
1104956e45f6SSimon J. Gerraty 
11054fde40d9SSimon J. Gerraty #ifndef HAVE_STRLCPY
11064fde40d9SSimon J. Gerraty size_t strlcpy(char *, const char *, size_t);
11074fde40d9SSimon J. Gerraty #endif
11084fde40d9SSimon J. Gerraty 
1109956e45f6SSimon J. Gerraty /* See if the node was seen on the left-hand side of a dependency operator. */
11109f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
1111956e45f6SSimon J. Gerraty GNode_IsTarget(const GNode *gn)
1112956e45f6SSimon J. Gerraty {
111312904384SSimon J. Gerraty 	return (gn->type & OP_OPMASK) != OP_NONE;
1114956e45f6SSimon J. Gerraty }
1115956e45f6SSimon J. Gerraty 
11169f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1117956e45f6SSimon J. Gerraty GNode_Path(const GNode *gn)
1118956e45f6SSimon J. Gerraty {
1119956e45f6SSimon J. Gerraty 	return gn->path != NULL ? gn->path : gn->name;
1120956e45f6SSimon J. Gerraty }
1121956e45f6SSimon J. Gerraty 
11229f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
112306b9b3e0SSimon J. Gerraty GNode_IsWaitingFor(const GNode *gn)
112406b9b3e0SSimon J. Gerraty {
112512904384SSimon J. Gerraty 	return gn->flags.remake && gn->made <= REQUESTED;
112606b9b3e0SSimon J. Gerraty }
112706b9b3e0SSimon J. Gerraty 
11289f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
112906b9b3e0SSimon J. Gerraty GNode_IsReady(const GNode *gn)
113006b9b3e0SSimon J. Gerraty {
113106b9b3e0SSimon J. Gerraty 	return gn->made > DEFERRED;
113206b9b3e0SSimon J. Gerraty }
113306b9b3e0SSimon J. Gerraty 
11349f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
113506b9b3e0SSimon J. Gerraty GNode_IsDone(const GNode *gn)
113606b9b3e0SSimon J. Gerraty {
113706b9b3e0SSimon J. Gerraty 	return gn->made >= MADE;
113806b9b3e0SSimon J. Gerraty }
113906b9b3e0SSimon J. Gerraty 
11409f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
114106b9b3e0SSimon J. Gerraty GNode_IsError(const GNode *gn)
114206b9b3e0SSimon J. Gerraty {
114306b9b3e0SSimon J. Gerraty 	return gn->made == ERROR || gn->made == ABORTED;
114406b9b3e0SSimon J. Gerraty }
114506b9b3e0SSimon J. Gerraty 
11469f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
11479f45a3c8SSimon J. Gerraty GNode_IsMainCandidate(const GNode *gn)
11489f45a3c8SSimon J. Gerraty {
11499f45a3c8SSimon J. Gerraty 	return (gn->type & (OP_NOTMAIN | OP_USE | OP_USEBEFORE |
11509f45a3c8SSimon J. Gerraty 			    OP_EXEC | OP_TRANSFORM)) == 0;
11519f45a3c8SSimon J. Gerraty }
11529f45a3c8SSimon J. Gerraty 
11539f45a3c8SSimon J. Gerraty /* Return whether the target file should be preserved on interrupt. */
11549f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
11559f45a3c8SSimon J. Gerraty GNode_IsPrecious(const GNode *gn)
11569f45a3c8SSimon J. Gerraty {
11579f45a3c8SSimon J. Gerraty 	/* XXX: Why are '::' targets precious? */
11589f45a3c8SSimon J. Gerraty 	return allPrecious || gn->type & (OP_PRECIOUS | OP_DOUBLEDEP);
11599f45a3c8SSimon J. Gerraty }
11609f45a3c8SSimon J. Gerraty 
11619f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1162dba7b0efSSimon J. Gerraty GNode_VarTarget(GNode *gn) { return GNode_ValueDirect(gn, TARGET); }
11639f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1164dba7b0efSSimon J. Gerraty GNode_VarOodate(GNode *gn) { return GNode_ValueDirect(gn, OODATE); }
11659f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1166dba7b0efSSimon J. Gerraty GNode_VarAllsrc(GNode *gn) { return GNode_ValueDirect(gn, ALLSRC); }
11679f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1168dba7b0efSSimon J. Gerraty GNode_VarImpsrc(GNode *gn) { return GNode_ValueDirect(gn, IMPSRC); }
11699f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1170dba7b0efSSimon J. Gerraty GNode_VarPrefix(GNode *gn) { return GNode_ValueDirect(gn, PREFIX); }
11719f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1172dba7b0efSSimon J. Gerraty GNode_VarArchive(GNode *gn) { return GNode_ValueDirect(gn, ARCHIVE); }
11739f45a3c8SSimon J. Gerraty MAKE_INLINE const char * MAKE_ATTR_USE
1174dba7b0efSSimon J. Gerraty GNode_VarMember(GNode *gn) { return GNode_ValueDirect(gn, MEMBER); }
11753955d011SMarcel Moolenaar 
11769f45a3c8SSimon J. Gerraty MAKE_INLINE void * MAKE_ATTR_USE
117712904384SSimon J. Gerraty UNCONST(const void *ptr)
117812904384SSimon J. Gerraty {
117912904384SSimon J. Gerraty 	void *ret;
118012904384SSimon J. Gerraty 	memcpy(&ret, &ptr, sizeof(ret));
118112904384SSimon J. Gerraty 	return ret;
118212904384SSimon J. Gerraty }
11833955d011SMarcel Moolenaar 
11844c620fe5SSimon J. Gerraty /* At least GNU/Hurd systems lack hardcoded MAXPATHLEN/PATH_MAX */
11854c620fe5SSimon J. Gerraty #ifdef HAVE_LIMITS_H
11864c620fe5SSimon J. Gerraty #include <limits.h>
11874c620fe5SSimon J. Gerraty #endif
11880dede8b0SSimon J. Gerraty #ifndef MAXPATHLEN
11890dede8b0SSimon J. Gerraty #define MAXPATHLEN	BMAKE_PATH_MAX
11900dede8b0SSimon J. Gerraty #endif
11914c620fe5SSimon J. Gerraty #ifndef PATH_MAX
11924c620fe5SSimon J. Gerraty #define PATH_MAX	MAXPATHLEN
11934c620fe5SSimon J. Gerraty #endif
11940dede8b0SSimon J. Gerraty 
1195c7019bf7SSimon J. Gerraty #if defined(SYSV)
1196c7019bf7SSimon J. Gerraty #define KILLPG(pid, sig) kill(-(pid), (sig))
1197c7019bf7SSimon J. Gerraty #else
1198c7019bf7SSimon J. Gerraty #define KILLPG(pid, sig) killpg((pid), (sig))
1199c7019bf7SSimon J. Gerraty #endif
1200c7019bf7SSimon J. Gerraty 
12019f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
1202e2eeea75SSimon J. Gerraty ch_isalnum(char ch) { return isalnum((unsigned char)ch) != 0; }
12039f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
1204e2eeea75SSimon J. Gerraty ch_isalpha(char ch) { return isalpha((unsigned char)ch) != 0; }
12059f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
1206e2eeea75SSimon J. Gerraty ch_isdigit(char ch) { return isdigit((unsigned char)ch) != 0; }
12079f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
12089f45a3c8SSimon J. Gerraty ch_islower(char ch) { return islower((unsigned char)ch) != 0; }
12099f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
1210e2eeea75SSimon J. Gerraty ch_isspace(char ch) { return isspace((unsigned char)ch) != 0; }
12119f45a3c8SSimon J. Gerraty MAKE_INLINE bool MAKE_ATTR_USE
1212e2eeea75SSimon J. Gerraty ch_isupper(char ch) { return isupper((unsigned char)ch) != 0; }
12139f45a3c8SSimon J. Gerraty MAKE_INLINE char MAKE_ATTR_USE
1214e2eeea75SSimon J. Gerraty ch_tolower(char ch) { return (char)tolower((unsigned char)ch); }
12159f45a3c8SSimon J. Gerraty MAKE_INLINE char MAKE_ATTR_USE
1216e2eeea75SSimon J. Gerraty ch_toupper(char ch) { return (char)toupper((unsigned char)ch); }
1217956e45f6SSimon J. Gerraty 
1218e2eeea75SSimon J. Gerraty MAKE_INLINE void
1219956e45f6SSimon J. Gerraty cpp_skip_whitespace(const char **pp)
1220956e45f6SSimon J. Gerraty {
1221956e45f6SSimon J. Gerraty 	while (ch_isspace(**pp))
1222956e45f6SSimon J. Gerraty 		(*pp)++;
1223956e45f6SSimon J. Gerraty }
1224956e45f6SSimon J. Gerraty 
1225e2eeea75SSimon J. Gerraty MAKE_INLINE void
1226e2eeea75SSimon J. Gerraty cpp_skip_hspace(const char **pp)
1227e2eeea75SSimon J. Gerraty {
1228e2eeea75SSimon J. Gerraty 	while (**pp == ' ' || **pp == '\t')
1229e2eeea75SSimon J. Gerraty 		(*pp)++;
1230e2eeea75SSimon J. Gerraty }
1231e2eeea75SSimon J. Gerraty 
12329f45a3c8SSimon J. Gerraty MAKE_INLINE bool
12339f45a3c8SSimon J. Gerraty cpp_skip_string(const char **pp, const char *s)
12349f45a3c8SSimon J. Gerraty {
12359f45a3c8SSimon J. Gerraty 	const char *p = *pp;
12369f45a3c8SSimon J. Gerraty 	while (*p == *s && *s != '\0')
12379f45a3c8SSimon J. Gerraty 		p++, s++;
12389f45a3c8SSimon J. Gerraty 	if (*s == '\0')
12399f45a3c8SSimon J. Gerraty 		*pp = p;
12409f45a3c8SSimon J. Gerraty 	return *s == '\0';
12419f45a3c8SSimon J. Gerraty }
12429f45a3c8SSimon J. Gerraty 
1243e2eeea75SSimon J. Gerraty MAKE_INLINE void
1244956e45f6SSimon J. Gerraty pp_skip_whitespace(char **pp)
1245956e45f6SSimon J. Gerraty {
1246956e45f6SSimon J. Gerraty 	while (ch_isspace(**pp))
1247956e45f6SSimon J. Gerraty 		(*pp)++;
1248956e45f6SSimon J. Gerraty }
1249956e45f6SSimon J. Gerraty 
1250e2eeea75SSimon J. Gerraty MAKE_INLINE void
1251e2eeea75SSimon J. Gerraty pp_skip_hspace(char **pp)
1252e2eeea75SSimon J. Gerraty {
1253e2eeea75SSimon J. Gerraty 	while (**pp == ' ' || **pp == '\t')
1254e2eeea75SSimon J. Gerraty 		(*pp)++;
1255e2eeea75SSimon J. Gerraty }
1256e2eeea75SSimon J. Gerraty 
125706b9b3e0SSimon J. Gerraty #if defined(lint)
125898875883SSimon J. Gerraty void do_not_define_rcsid(void); /* for lint */
125998875883SSimon J. Gerraty # define MAKE_RCSID(id) void do_not_define_rcsid(void)
126006b9b3e0SSimon J. Gerraty #elif defined(MAKE_NATIVE)
1261956e45f6SSimon J. Gerraty # include <sys/cdefs.h>
126212904384SSimon J. Gerraty # ifndef __IDSTRING
126312904384SSimon J. Gerraty #   define __IDSTRING(name,string) \
126412904384SSimon J. Gerraty 	static const char name[] MAKE_ATTR_UNUSED = string
126512904384SSimon J. Gerraty # endif
126612904384SSimon J. Gerraty # ifndef __RCSID
126712904384SSimon J. Gerraty #   define __RCSID(s) __IDSTRING(rcsid,s)
126812904384SSimon J. Gerraty # endif
126912904384SSimon J. Gerraty # ifndef __COPYRIGHT
127012904384SSimon J. Gerraty #   define __COPYRIGHT(s) __IDSTRING(copyright,s)
127112904384SSimon J. Gerraty # endif
1272956e45f6SSimon J. Gerraty # define MAKE_RCSID(id) __RCSID(id)
127306b9b3e0SSimon J. Gerraty #elif defined(MAKE_ALL_IN_ONE) && defined(__COUNTER__)
127406b9b3e0SSimon J. Gerraty # define MAKE_RCSID_CONCAT(x, y) CONCAT(x, y)
127506b9b3e0SSimon J. Gerraty # define MAKE_RCSID(id) static volatile char \
127606b9b3e0SSimon J. Gerraty 	MAKE_RCSID_CONCAT(rcsid_, __COUNTER__)[] = id
127706b9b3e0SSimon J. Gerraty #elif defined(MAKE_ALL_IN_ONE)
127898875883SSimon J. Gerraty # define MAKE_RCSID(id) void do_not_define_rcsid(void)
1279956e45f6SSimon J. Gerraty #else
1280956e45f6SSimon J. Gerraty # define MAKE_RCSID(id) static volatile char rcsid[] = id
1281956e45f6SSimon J. Gerraty #endif
1282956e45f6SSimon J. Gerraty 
12839f45a3c8SSimon J. Gerraty #endif
1284