xref: /freebsd/contrib/bmake/targ.c (revision 226192822cddc30cacecd55bccb48f39c653058c)
1*22619282SSimon J. Gerraty /*	$NetBSD: targ.c,v 1.184 2024/07/07 09:54:12 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 
353955d011SMarcel Moolenaar /*
363955d011SMarcel Moolenaar  * Copyright (c) 1989 by Berkeley Softworks
373955d011SMarcel Moolenaar  * All rights reserved.
383955d011SMarcel Moolenaar  *
393955d011SMarcel Moolenaar  * This code is derived from software contributed to Berkeley by
403955d011SMarcel Moolenaar  * Adam de Boor.
413955d011SMarcel Moolenaar  *
423955d011SMarcel Moolenaar  * Redistribution and use in source and binary forms, with or without
433955d011SMarcel Moolenaar  * modification, are permitted provided that the following conditions
443955d011SMarcel Moolenaar  * are met:
453955d011SMarcel Moolenaar  * 1. Redistributions of source code must retain the above copyright
463955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer.
473955d011SMarcel Moolenaar  * 2. Redistributions in binary form must reproduce the above copyright
483955d011SMarcel Moolenaar  *    notice, this list of conditions and the following disclaimer in the
493955d011SMarcel Moolenaar  *    documentation and/or other materials provided with the distribution.
503955d011SMarcel Moolenaar  * 3. All advertising materials mentioning features or use of this software
513955d011SMarcel Moolenaar  *    must display the following acknowledgement:
523955d011SMarcel Moolenaar  *	This product includes software developed by the University of
533955d011SMarcel Moolenaar  *	California, Berkeley and its contributors.
543955d011SMarcel Moolenaar  * 4. Neither the name of the University nor the names of its contributors
553955d011SMarcel Moolenaar  *    may be used to endorse or promote products derived from this software
563955d011SMarcel Moolenaar  *    without specific prior written permission.
573955d011SMarcel Moolenaar  *
583955d011SMarcel Moolenaar  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
593955d011SMarcel Moolenaar  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
603955d011SMarcel Moolenaar  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
613955d011SMarcel Moolenaar  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
623955d011SMarcel Moolenaar  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
633955d011SMarcel Moolenaar  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
643955d011SMarcel Moolenaar  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
653955d011SMarcel Moolenaar  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
663955d011SMarcel Moolenaar  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
673955d011SMarcel Moolenaar  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
683955d011SMarcel Moolenaar  * SUCH DAMAGE.
693955d011SMarcel Moolenaar  */
703955d011SMarcel Moolenaar 
71e2eeea75SSimon J. Gerraty /*
72e2eeea75SSimon J. Gerraty  * Maintaining the targets and sources, which are both implemented as GNode.
733955d011SMarcel Moolenaar  *
743955d011SMarcel Moolenaar  * Interface:
75e2eeea75SSimon J. Gerraty  *	Targ_Init	Initialize the module.
763955d011SMarcel Moolenaar  *
77e2eeea75SSimon J. Gerraty  *	Targ_End	Clean up the module.
783955d011SMarcel Moolenaar  *
793955d011SMarcel Moolenaar  *	Targ_List	Return the list of all targets so far.
803955d011SMarcel Moolenaar  *
819f45a3c8SSimon J. Gerraty  *	GNode_New	Create a new GNode with the given name, don't add it
829f45a3c8SSimon J. Gerraty  *			to allNodes.
833955d011SMarcel Moolenaar  *
84956e45f6SSimon J. Gerraty  *	Targ_FindNode	Find the node, or return NULL.
85956e45f6SSimon J. Gerraty  *
86956e45f6SSimon J. Gerraty  *	Targ_GetNode	Find the node, or create it.
87956e45f6SSimon J. Gerraty  *
88956e45f6SSimon J. Gerraty  *	Targ_NewInternalNode
89956e45f6SSimon J. Gerraty  *			Create an internal node.
903955d011SMarcel Moolenaar  *
913955d011SMarcel Moolenaar  *	Targ_FindList	Given a list of names, find nodes for all
92956e45f6SSimon J. Gerraty  *			of them, creating them as necessary.
933955d011SMarcel Moolenaar  *
94956e45f6SSimon J. Gerraty  *	Targ_Propagate	Propagate information between related nodes.
95956e45f6SSimon J. Gerraty  *			Should be called after the makefiles are parsed
96956e45f6SSimon J. Gerraty  *			but before any action is taken.
973955d011SMarcel Moolenaar  *
983955d011SMarcel Moolenaar  * Debugging:
99956e45f6SSimon J. Gerraty  *	Targ_PrintGraph
10006b9b3e0SSimon J. Gerraty  *			Print out the entire graph, all variables and
1019f45a3c8SSimon J. Gerraty  *			statistics for the directory cache.
1023955d011SMarcel Moolenaar  */
1033955d011SMarcel Moolenaar 
1043955d011SMarcel Moolenaar #include <time.h>
1053955d011SMarcel Moolenaar 
1063955d011SMarcel Moolenaar #include "make.h"
1073955d011SMarcel Moolenaar #include "dir.h"
1083955d011SMarcel Moolenaar 
109956e45f6SSimon J. Gerraty /*	"@(#)targ.c	8.2 (Berkeley) 3/19/94"	*/
110*22619282SSimon J. Gerraty MAKE_RCSID("$NetBSD: targ.c,v 1.184 2024/07/07 09:54:12 rillig Exp $");
1113955d011SMarcel Moolenaar 
11206b9b3e0SSimon J. Gerraty /*
11306b9b3e0SSimon J. Gerraty  * All target nodes that appeared on the left-hand side of one of the
11406b9b3e0SSimon J. Gerraty  * dependency operators ':', '::', '!'.
11506b9b3e0SSimon J. Gerraty  */
11606b9b3e0SSimon J. Gerraty static GNodeList allTargets = LST_INIT;
117e2eeea75SSimon J. Gerraty static HashTable allTargetsByName;
118956e45f6SSimon J. Gerraty 
1193955d011SMarcel Moolenaar #ifdef CLEANUP
12006b9b3e0SSimon J. Gerraty static GNodeList allNodes = LST_INIT;
121e2eeea75SSimon J. Gerraty 
122548bfc56SSimon J. Gerraty static void GNode_Free(GNode *);
1233955d011SMarcel Moolenaar #endif
1243955d011SMarcel Moolenaar 
1253955d011SMarcel Moolenaar void
Targ_Init(void)1263955d011SMarcel Moolenaar Targ_Init(void)
1273955d011SMarcel Moolenaar {
128e2eeea75SSimon J. Gerraty 	HashTable_Init(&allTargetsByName);
129*22619282SSimon J. Gerraty 	SCOPE_INTERNAL = GNode_New("Internal");
130*22619282SSimon J. Gerraty 	SCOPE_GLOBAL = GNode_New("Global");
131*22619282SSimon J. Gerraty 	SCOPE_CMDLINE = GNode_New("Command");
1323955d011SMarcel Moolenaar }
1333955d011SMarcel Moolenaar 
134*22619282SSimon J. Gerraty #ifdef CLEANUP
1353955d011SMarcel Moolenaar void
Targ_End(void)1363955d011SMarcel Moolenaar Targ_End(void)
1373955d011SMarcel Moolenaar {
138548bfc56SSimon J. Gerraty 	GNodeListNode *ln;
139*22619282SSimon J. Gerraty 
14006b9b3e0SSimon J. Gerraty 	Lst_Done(&allTargets);
141e2eeea75SSimon J. Gerraty 	HashTable_Done(&allTargetsByName);
142548bfc56SSimon J. Gerraty 	for (ln = allNodes.first; ln != NULL; ln = ln->next)
143548bfc56SSimon J. Gerraty 		GNode_Free(ln->datum);
144548bfc56SSimon J. Gerraty 	Lst_Done(&allNodes);
1453955d011SMarcel Moolenaar }
146*22619282SSimon J. Gerraty #endif
1473955d011SMarcel Moolenaar 
1482c3632d1SSimon J. Gerraty void
Targ_Stats(void)1492c3632d1SSimon J. Gerraty Targ_Stats(void)
1502c3632d1SSimon J. Gerraty {
151e2eeea75SSimon J. Gerraty 	HashTable_DebugStats(&allTargetsByName, "targets");
1522c3632d1SSimon J. Gerraty }
1532c3632d1SSimon J. Gerraty 
154e2eeea75SSimon J. Gerraty /*
155e2eeea75SSimon J. Gerraty  * Return the list of all targets, which are all nodes that appear on the
156e2eeea75SSimon J. Gerraty  * left-hand side of a dependency declaration such as "target: source".
157e2eeea75SSimon J. Gerraty  * The returned list does not contain pure sources.
158e2eeea75SSimon J. Gerraty  */
159956e45f6SSimon J. Gerraty GNodeList *
Targ_List(void)1603955d011SMarcel Moolenaar Targ_List(void)
1613955d011SMarcel Moolenaar {
16206b9b3e0SSimon J. Gerraty 	return &allTargets;
1633955d011SMarcel Moolenaar }
1643955d011SMarcel Moolenaar 
16506b9b3e0SSimon J. Gerraty /*
16606b9b3e0SSimon J. Gerraty  * Create a new graph node, but don't register it anywhere.
1673955d011SMarcel Moolenaar  *
1684fde40d9SSimon J. Gerraty  * Graph nodes that occur on the left-hand side of a dependency line such
169e2eeea75SSimon J. Gerraty  * as "target: source" are called targets.  XXX: In some cases (like the
1704fde40d9SSimon J. Gerraty  * .ALLTARGETS variable), other nodes are called targets as well, even if
1714fde40d9SSimon J. Gerraty  * they never occur on the left-hand side of a dependency line.
172e2eeea75SSimon J. Gerraty  *
173e2eeea75SSimon J. Gerraty  * Typical names for graph nodes are:
1744fde40d9SSimon J. Gerraty  *	"src.c"		an ordinary file
1754fde40d9SSimon J. Gerraty  *	"clean"		a .PHONY target
1764fde40d9SSimon J. Gerraty  *	".END"		a special hook target
1774fde40d9SSimon J. Gerraty  *	"-lm"		a library
1784fde40d9SSimon J. Gerraty  *	"libm.a(sin.o)"	an archive member
1793955d011SMarcel Moolenaar  */
1803955d011SMarcel Moolenaar GNode *
GNode_New(const char * name)181e2eeea75SSimon J. Gerraty GNode_New(const char *name)
1823955d011SMarcel Moolenaar {
1833955d011SMarcel Moolenaar 	GNode *gn;
1843955d011SMarcel Moolenaar 
185e2eeea75SSimon J. Gerraty 	gn = bmake_malloc(sizeof *gn);
1863955d011SMarcel Moolenaar 	gn->name = bmake_strdup(name);
1873955d011SMarcel Moolenaar 	gn->uname = NULL;
1883955d011SMarcel Moolenaar 	gn->path = NULL;
18906b9b3e0SSimon J. Gerraty 	gn->type = name[0] == '-' && name[1] == 'l' ? OP_LIB : OP_NONE;
19012904384SSimon J. Gerraty 	memset(&gn->flags, 0, sizeof(gn->flags));
191e2eeea75SSimon J. Gerraty 	gn->made = UNMADE;
192e2eeea75SSimon J. Gerraty 	gn->unmade = 0;
1933955d011SMarcel Moolenaar 	gn->mtime = 0;
194956e45f6SSimon J. Gerraty 	gn->youngestChild = NULL;
19506b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->implicitParents);
19606b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->parents);
19706b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->children);
19806b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->order_pred);
19906b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->order_succ);
20006b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->cohorts);
201e2eeea75SSimon J. Gerraty 	gn->cohort_num[0] = '\0';
202e2eeea75SSimon J. Gerraty 	gn->unmade_cohorts = 0;
203e2eeea75SSimon J. Gerraty 	gn->centurion = NULL;
204e2eeea75SSimon J. Gerraty 	gn->checked_seqno = 0;
20506b9b3e0SSimon J. Gerraty 	HashTable_Init(&gn->vars);
20606b9b3e0SSimon J. Gerraty 	Lst_Init(&gn->commands);
2073955d011SMarcel Moolenaar 	gn->suffix = NULL;
2083955d011SMarcel Moolenaar 	gn->fname = NULL;
2092c3632d1SSimon J. Gerraty 	gn->lineno = 0;
210c59c3bf3SSimon J. Gerraty 	gn->exit_status = 0;
2113955d011SMarcel Moolenaar 
2123955d011SMarcel Moolenaar #ifdef CLEANUP
21306b9b3e0SSimon J. Gerraty 	Lst_Append(&allNodes, gn);
2143955d011SMarcel Moolenaar #endif
2153955d011SMarcel Moolenaar 
2163841c287SSimon J. Gerraty 	return gn;
2173955d011SMarcel Moolenaar }
2183955d011SMarcel Moolenaar 
2193955d011SMarcel Moolenaar #ifdef CLEANUP
2203955d011SMarcel Moolenaar static void
GNode_Free(GNode * gn)221548bfc56SSimon J. Gerraty GNode_Free(GNode *gn)
2223955d011SMarcel Moolenaar {
2238d5c8e21SSimon J. Gerraty 	Var_DeleteAll(gn);
2248d5c8e21SSimon J. Gerraty 
2253955d011SMarcel Moolenaar 	free(gn->name);
2263955d011SMarcel Moolenaar 	free(gn->uname);
2273955d011SMarcel Moolenaar 	free(gn->path);
22806b9b3e0SSimon J. Gerraty 
22906b9b3e0SSimon J. Gerraty 	/* Don't free gn->youngestChild since it is not owned by this node. */
23006b9b3e0SSimon J. Gerraty 
23106b9b3e0SSimon J. Gerraty 	/*
23206b9b3e0SSimon J. Gerraty 	 * In the following lists, only free the list nodes, but not the
23306b9b3e0SSimon J. Gerraty 	 * GNodes in them since these are not owned by this node.
23406b9b3e0SSimon J. Gerraty 	 */
23506b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->implicitParents);
23606b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->parents);
23706b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->children);
23806b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->order_pred);
23906b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->order_succ);
24006b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->cohorts);
24106b9b3e0SSimon J. Gerraty 
24206b9b3e0SSimon J. Gerraty 	HashTable_Done(&gn->vars);
24306b9b3e0SSimon J. Gerraty 
24406b9b3e0SSimon J. Gerraty 	/*
24506b9b3e0SSimon J. Gerraty 	 * Do not free the commands themselves, as they may be shared with
24606b9b3e0SSimon J. Gerraty 	 * other nodes.
24706b9b3e0SSimon J. Gerraty 	 */
24806b9b3e0SSimon J. Gerraty 	Lst_Done(&gn->commands);
24906b9b3e0SSimon J. Gerraty 
25006b9b3e0SSimon J. Gerraty 	/*
25106b9b3e0SSimon J. Gerraty 	 * gn->suffix is not owned by this node.
25206b9b3e0SSimon J. Gerraty 	 *
25306b9b3e0SSimon J. Gerraty 	 * XXX: gn->suffix should be unreferenced here.  This requires a
25406b9b3e0SSimon J. Gerraty 	 * thorough check that the reference counting is done correctly in
25506b9b3e0SSimon J. Gerraty 	 * all places, otherwise a suffix might be freed too early.
25606b9b3e0SSimon J. Gerraty 	 */
2573955d011SMarcel Moolenaar 
2583955d011SMarcel Moolenaar 	free(gn);
2593955d011SMarcel Moolenaar }
2603955d011SMarcel Moolenaar #endif
2613955d011SMarcel Moolenaar 
262956e45f6SSimon J. Gerraty /* Get the existing global node, or return NULL. */
2633955d011SMarcel Moolenaar GNode *
Targ_FindNode(const char * name)264956e45f6SSimon J. Gerraty Targ_FindNode(const char *name)
2653955d011SMarcel Moolenaar {
266e2eeea75SSimon J. Gerraty 	return HashTable_FindValue(&allTargetsByName, name);
2673955d011SMarcel Moolenaar }
2683955d011SMarcel Moolenaar 
269956e45f6SSimon J. Gerraty /* Get the existing global node, or create it. */
270956e45f6SSimon J. Gerraty GNode *
Targ_GetNode(const char * name)271956e45f6SSimon J. Gerraty Targ_GetNode(const char *name)
272956e45f6SSimon J. Gerraty {
273b0c40a00SSimon J. Gerraty 	bool isNew;
274e2eeea75SSimon J. Gerraty 	HashEntry *he = HashTable_CreateEntry(&allTargetsByName, name, &isNew);
2753955d011SMarcel Moolenaar 	if (!isNew)
276956e45f6SSimon J. Gerraty 		return HashEntry_Get(he);
277956e45f6SSimon J. Gerraty 
278956e45f6SSimon J. Gerraty 	{
279956e45f6SSimon J. Gerraty 		GNode *gn = Targ_NewInternalNode(name);
280956e45f6SSimon J. Gerraty 		HashEntry_Set(he, gn);
281956e45f6SSimon J. Gerraty 		return gn;
282956e45f6SSimon J. Gerraty 	}
2833955d011SMarcel Moolenaar }
2843955d011SMarcel Moolenaar 
285e2eeea75SSimon J. Gerraty /*
286e2eeea75SSimon J. Gerraty  * Create a node, register it in .ALLTARGETS but don't store it in the
287956e45f6SSimon J. Gerraty  * table of global nodes.  This means it cannot be found by name.
288956e45f6SSimon J. Gerraty  *
289e2eeea75SSimon J. Gerraty  * This is used for internal nodes, such as cohorts or .WAIT nodes.
290e2eeea75SSimon J. Gerraty  */
291956e45f6SSimon J. Gerraty GNode *
Targ_NewInternalNode(const char * name)292956e45f6SSimon J. Gerraty Targ_NewInternalNode(const char *name)
293956e45f6SSimon J. Gerraty {
294e2eeea75SSimon J. Gerraty 	GNode *gn = GNode_New(name);
295dba7b0efSSimon J. Gerraty 	Global_Append(".ALLTARGETS", name);
29606b9b3e0SSimon J. Gerraty 	Lst_Append(&allTargets, gn);
29706b9b3e0SSimon J. Gerraty 	DEBUG1(TARG, "Adding \"%s\" to all targets.\n", gn->name);
2983955d011SMarcel Moolenaar 	if (doing_depend)
29912904384SSimon J. Gerraty 		gn->flags.fromDepend = true;
3003955d011SMarcel Moolenaar 	return gn;
3013955d011SMarcel Moolenaar }
3023955d011SMarcel Moolenaar 
303e2eeea75SSimon J. Gerraty /*
304e2eeea75SSimon J. Gerraty  * Return the .END node, which contains the commands to be run when
305e2eeea75SSimon J. Gerraty  * everything else has been made.
306e2eeea75SSimon J. Gerraty  */
30706b9b3e0SSimon J. Gerraty GNode *
Targ_GetEndNode(void)30806b9b3e0SSimon J. Gerraty Targ_GetEndNode(void)
3093955d011SMarcel Moolenaar {
31006b9b3e0SSimon J. Gerraty 	/*
31106b9b3e0SSimon J. Gerraty 	 * Save the node locally to avoid having to search for it all
31206b9b3e0SSimon J. Gerraty 	 * the time.
31306b9b3e0SSimon J. Gerraty 	 */
314956e45f6SSimon J. Gerraty 	static GNode *endNode = NULL;
31506b9b3e0SSimon J. Gerraty 
316956e45f6SSimon J. Gerraty 	if (endNode == NULL) {
317956e45f6SSimon J. Gerraty 		endNode = Targ_GetNode(".END");
318956e45f6SSimon J. Gerraty 		endNode->type = OP_SPECIAL;
319956e45f6SSimon J. Gerraty 	}
320956e45f6SSimon J. Gerraty 	return endNode;
321956e45f6SSimon J. Gerraty }
3223955d011SMarcel Moolenaar 
32306b9b3e0SSimon J. Gerraty /* Add the named nodes to the list, creating them as necessary. */
32406b9b3e0SSimon J. Gerraty void
Targ_FindList(GNodeList * gns,StringList * names)32506b9b3e0SSimon J. Gerraty Targ_FindList(GNodeList *gns, StringList *names)
326956e45f6SSimon J. Gerraty {
327956e45f6SSimon J. Gerraty 	StringListNode *ln;
32806b9b3e0SSimon J. Gerraty 
329956e45f6SSimon J. Gerraty 	for (ln = names->first; ln != NULL; ln = ln->next) {
330956e45f6SSimon J. Gerraty 		const char *name = ln->datum;
331956e45f6SSimon J. Gerraty 		GNode *gn = Targ_GetNode(name);
33206b9b3e0SSimon J. Gerraty 		Lst_Append(gns, gn);
3333955d011SMarcel Moolenaar 	}
3343955d011SMarcel Moolenaar }
3353955d011SMarcel Moolenaar 
336956e45f6SSimon J. Gerraty static void
PrintNodeNames(GNodeList * gnodes)337956e45f6SSimon J. Gerraty PrintNodeNames(GNodeList *gnodes)
3383955d011SMarcel Moolenaar {
33906b9b3e0SSimon J. Gerraty 	GNodeListNode *ln;
3403955d011SMarcel Moolenaar 
34106b9b3e0SSimon J. Gerraty 	for (ln = gnodes->first; ln != NULL; ln = ln->next) {
34206b9b3e0SSimon J. Gerraty 		GNode *gn = ln->datum;
343956e45f6SSimon J. Gerraty 		debug_printf(" %s%s", gn->name, gn->cohort_num);
344956e45f6SSimon J. Gerraty 	}
3453955d011SMarcel Moolenaar }
3463955d011SMarcel Moolenaar 
347956e45f6SSimon J. Gerraty static void
PrintNodeNamesLine(const char * label,GNodeList * gnodes)348956e45f6SSimon J. Gerraty PrintNodeNamesLine(const char *label, GNodeList *gnodes)
3493955d011SMarcel Moolenaar {
350956e45f6SSimon J. Gerraty 	if (Lst_IsEmpty(gnodes))
351956e45f6SSimon J. Gerraty 		return;
352956e45f6SSimon J. Gerraty 	debug_printf("# %s:", label);
353956e45f6SSimon J. Gerraty 	PrintNodeNames(gnodes);
354956e45f6SSimon J. Gerraty 	debug_printf("\n");
355956e45f6SSimon J. Gerraty }
356956e45f6SSimon J. Gerraty 
357956e45f6SSimon J. Gerraty void
Targ_PrintCmds(GNode * gn)358956e45f6SSimon J. Gerraty Targ_PrintCmds(GNode *gn)
359956e45f6SSimon J. Gerraty {
360956e45f6SSimon J. Gerraty 	StringListNode *ln;
36106b9b3e0SSimon J. Gerraty 
36206b9b3e0SSimon J. Gerraty 	for (ln = gn->commands.first; ln != NULL; ln = ln->next) {
363956e45f6SSimon J. Gerraty 		const char *cmd = ln->datum;
364956e45f6SSimon J. Gerraty 		debug_printf("\t%s\n", cmd);
365956e45f6SSimon J. Gerraty 	}
3663955d011SMarcel Moolenaar }
3673955d011SMarcel Moolenaar 
36806b9b3e0SSimon J. Gerraty /*
36906b9b3e0SSimon J. Gerraty  * Format a modification time in some reasonable way and return it.
37006b9b3e0SSimon J. Gerraty  * The formatted time is placed in a static area, so it is overwritten
37106b9b3e0SSimon J. Gerraty  * with each call.
37206b9b3e0SSimon J. Gerraty  */
37306b9b3e0SSimon J. Gerraty const char *
Targ_FmtTime(time_t tm)3743955d011SMarcel Moolenaar Targ_FmtTime(time_t tm)
3753955d011SMarcel Moolenaar {
3763955d011SMarcel Moolenaar 	static char buf[128];
3773955d011SMarcel Moolenaar 
37806b9b3e0SSimon J. Gerraty 	struct tm *parts = localtime(&tm);
379b0c40a00SSimon J. Gerraty 	(void)strftime(buf, sizeof buf, "%H:%M:%S %b %d, %Y", parts);
3803841c287SSimon J. Gerraty 	return buf;
3813955d011SMarcel Moolenaar }
3823955d011SMarcel Moolenaar 
3832c3632d1SSimon J. Gerraty /* Print out a type field giving only those attributes the user can set. */
3843955d011SMarcel Moolenaar void
Targ_PrintType(GNodeType type)38512904384SSimon J. Gerraty Targ_PrintType(GNodeType type)
3863955d011SMarcel Moolenaar {
38712904384SSimon J. Gerraty 	static const struct {
38812904384SSimon J. Gerraty 		GNodeType bit;
38912904384SSimon J. Gerraty 		bool internal;
39012904384SSimon J. Gerraty 		const char name[10];
39112904384SSimon J. Gerraty 	} names[] = {
39212904384SSimon J. Gerraty 		{ OP_MEMBER,	true,	"MEMBER"	},
39312904384SSimon J. Gerraty 		{ OP_LIB,	true,	"LIB"		},
39412904384SSimon J. Gerraty 		{ OP_ARCHV,	true,	"ARCHV"		},
39512904384SSimon J. Gerraty 		{ OP_PHONY,	true,	"PHONY"		},
39612904384SSimon J. Gerraty 		{ OP_NOTMAIN,	false,	"NOTMAIN"	},
39712904384SSimon J. Gerraty 		{ OP_INVISIBLE,	false,	"INVISIBLE"	},
39812904384SSimon J. Gerraty 		{ OP_MADE,	true,	"MADE"		},
39912904384SSimon J. Gerraty 		{ OP_JOIN,	false,	"JOIN"		},
40012904384SSimon J. Gerraty 		{ OP_MAKE,	false,	"MAKE"		},
40112904384SSimon J. Gerraty 		{ OP_SILENT,	false,	"SILENT"	},
40212904384SSimon J. Gerraty 		{ OP_PRECIOUS,	false,	"PRECIOUS"	},
40312904384SSimon J. Gerraty 		{ OP_IGNORE,	false,	"IGNORE"	},
40412904384SSimon J. Gerraty 		{ OP_EXEC,	false,	"EXEC"		},
40512904384SSimon J. Gerraty 		{ OP_USE,	false,	"USE"		},
4062f2a5ecdSSimon J. Gerraty 		{ OP_USEBEFORE,	false,	"USEBEFORE"	},
40712904384SSimon J. Gerraty 		{ OP_OPTIONAL,	false,	"OPTIONAL"	},
40812904384SSimon J. Gerraty 	};
40912904384SSimon J. Gerraty 	size_t i;
4103955d011SMarcel Moolenaar 
41112904384SSimon J. Gerraty 	for (i = 0; i < sizeof(names) / sizeof(names[0]); i++) {
41212904384SSimon J. Gerraty 		if (type & names[i].bit) {
41312904384SSimon J. Gerraty 			if (names[i].internal)
41412904384SSimon J. Gerraty 				DEBUG1(TARG, " .%s", names[i].name);
41512904384SSimon J. Gerraty 			else
41612904384SSimon J. Gerraty 				debug_printf(" .%s", names[i].name);
4173955d011SMarcel Moolenaar 		}
4183955d011SMarcel Moolenaar 	}
4193955d011SMarcel Moolenaar }
4203955d011SMarcel Moolenaar 
421dba7b0efSSimon J. Gerraty const char *
GNodeMade_Name(GNodeMade made)422dba7b0efSSimon J. Gerraty GNodeMade_Name(GNodeMade made)
4233955d011SMarcel Moolenaar {
4243955d011SMarcel Moolenaar 	switch (made) {
4253955d011SMarcel Moolenaar 	case UNMADE:    return "unmade";
4263955d011SMarcel Moolenaar 	case DEFERRED:  return "deferred";
4273955d011SMarcel Moolenaar 	case REQUESTED: return "requested";
4283955d011SMarcel Moolenaar 	case BEINGMADE: return "being made";
4293955d011SMarcel Moolenaar 	case MADE:      return "made";
4303955d011SMarcel Moolenaar 	case UPTODATE:  return "up-to-date";
4313955d011SMarcel Moolenaar 	case ERROR:     return "error when made";
4323955d011SMarcel Moolenaar 	case ABORTED:   return "aborted";
4333955d011SMarcel Moolenaar 	default:        return "unknown enum_made value";
4343955d011SMarcel Moolenaar 	}
4353955d011SMarcel Moolenaar }
4363955d011SMarcel Moolenaar 
437956e45f6SSimon J. Gerraty static const char *
GNode_OpName(const GNode * gn)438956e45f6SSimon J. Gerraty GNode_OpName(const GNode *gn)
4393955d011SMarcel Moolenaar {
440956e45f6SSimon J. Gerraty 	switch (gn->type & OP_OPMASK) {
441956e45f6SSimon J. Gerraty 	case OP_DEPENDS:
442956e45f6SSimon J. Gerraty 		return ":";
443956e45f6SSimon J. Gerraty 	case OP_FORCE:
444956e45f6SSimon J. Gerraty 		return "!";
445956e45f6SSimon J. Gerraty 	case OP_DOUBLEDEP:
446956e45f6SSimon J. Gerraty 		return "::";
447956e45f6SSimon J. Gerraty 	}
448956e45f6SSimon J. Gerraty 	return "";
449956e45f6SSimon J. Gerraty }
4503955d011SMarcel Moolenaar 
45112904384SSimon J. Gerraty static bool
GNodeFlags_IsNone(GNodeFlags flags)45212904384SSimon J. Gerraty GNodeFlags_IsNone(GNodeFlags flags)
45312904384SSimon J. Gerraty {
45412904384SSimon J. Gerraty 	return !flags.remake
45512904384SSimon J. Gerraty 	       && !flags.childMade
45612904384SSimon J. Gerraty 	       && !flags.force
45712904384SSimon J. Gerraty 	       && !flags.doneWait
45812904384SSimon J. Gerraty 	       && !flags.doneOrder
45912904384SSimon J. Gerraty 	       && !flags.fromDepend
46012904384SSimon J. Gerraty 	       && !flags.doneAllsrc
46112904384SSimon J. Gerraty 	       && !flags.cycle
46212904384SSimon J. Gerraty 	       && !flags.doneCycle;
46312904384SSimon J. Gerraty }
46412904384SSimon J. Gerraty 
465956e45f6SSimon J. Gerraty /* Print the contents of a node. */
466956e45f6SSimon J. Gerraty void
Targ_PrintNode(GNode * gn,int pass)467956e45f6SSimon J. Gerraty Targ_PrintNode(GNode *gn, int pass)
468956e45f6SSimon J. Gerraty {
469956e45f6SSimon J. Gerraty 	debug_printf("# %s%s", gn->name, gn->cohort_num);
470956e45f6SSimon J. Gerraty 	GNode_FprintDetails(opts.debug_file, ", ", gn, "\n");
47112904384SSimon J. Gerraty 	if (GNodeFlags_IsNone(gn->flags))
472956e45f6SSimon J. Gerraty 		return;
4733955d011SMarcel Moolenaar 
47406b9b3e0SSimon J. Gerraty 	if (!GNode_IsTarget(gn))
47506b9b3e0SSimon J. Gerraty 		return;
47606b9b3e0SSimon J. Gerraty 
477956e45f6SSimon J. Gerraty 	debug_printf("#\n");
4789f45a3c8SSimon J. Gerraty 	if (gn == mainNode)
479956e45f6SSimon J. Gerraty 		debug_printf("# *** MAIN TARGET ***\n");
48006b9b3e0SSimon J. Gerraty 
4813955d011SMarcel Moolenaar 	if (pass >= 2) {
48206b9b3e0SSimon J. Gerraty 		if (gn->unmade > 0)
483956e45f6SSimon J. Gerraty 			debug_printf("# %d unmade children\n", gn->unmade);
48406b9b3e0SSimon J. Gerraty 		else
485956e45f6SSimon J. Gerraty 			debug_printf("# No unmade children\n");
4863955d011SMarcel Moolenaar 		if (!(gn->type & (OP_JOIN | OP_USE | OP_USEBEFORE | OP_EXEC))) {
4873955d011SMarcel Moolenaar 			if (gn->mtime != 0) {
488956e45f6SSimon J. Gerraty 				debug_printf("# last modified %s: %s\n",
4893955d011SMarcel Moolenaar 				    Targ_FmtTime(gn->mtime),
490dba7b0efSSimon J. Gerraty 				    GNodeMade_Name(gn->made));
4913955d011SMarcel Moolenaar 			} else if (gn->made != UNMADE) {
49206b9b3e0SSimon J. Gerraty 				debug_printf("# nonexistent (maybe): %s\n",
493dba7b0efSSimon J. Gerraty 				    GNodeMade_Name(gn->made));
49406b9b3e0SSimon J. Gerraty 			} else
495956e45f6SSimon J. Gerraty 				debug_printf("# unmade\n");
4963955d011SMarcel Moolenaar 		}
49706b9b3e0SSimon J. Gerraty 		PrintNodeNamesLine("implicit parents", &gn->implicitParents);
4983955d011SMarcel Moolenaar 	} else {
49906b9b3e0SSimon J. Gerraty 		if (gn->unmade != 0)
500956e45f6SSimon J. Gerraty 			debug_printf("# %d unmade children\n", gn->unmade);
5013955d011SMarcel Moolenaar 	}
50206b9b3e0SSimon J. Gerraty 
50306b9b3e0SSimon J. Gerraty 	PrintNodeNamesLine("parents", &gn->parents);
50406b9b3e0SSimon J. Gerraty 	PrintNodeNamesLine("order_pred", &gn->order_pred);
50506b9b3e0SSimon J. Gerraty 	PrintNodeNamesLine("order_succ", &gn->order_succ);
5063955d011SMarcel Moolenaar 
507956e45f6SSimon J. Gerraty 	debug_printf("%-16s%s", gn->name, GNode_OpName(gn));
5083955d011SMarcel Moolenaar 	Targ_PrintType(gn->type);
50906b9b3e0SSimon J. Gerraty 	PrintNodeNames(&gn->children);
510956e45f6SSimon J. Gerraty 	debug_printf("\n");
511956e45f6SSimon J. Gerraty 	Targ_PrintCmds(gn);
512956e45f6SSimon J. Gerraty 	debug_printf("\n\n");
51306b9b3e0SSimon J. Gerraty 	if (gn->type & OP_DOUBLEDEP)
51406b9b3e0SSimon J. Gerraty 		Targ_PrintNodes(&gn->cohorts, pass);
5153955d011SMarcel Moolenaar }
5163955d011SMarcel Moolenaar 
517956e45f6SSimon J. Gerraty void
Targ_PrintNodes(GNodeList * gnodes,int pass)518956e45f6SSimon J. Gerraty Targ_PrintNodes(GNodeList *gnodes, int pass)
5193955d011SMarcel Moolenaar {
520956e45f6SSimon J. Gerraty 	GNodeListNode *ln;
52106b9b3e0SSimon J. Gerraty 
522956e45f6SSimon J. Gerraty 	for (ln = gnodes->first; ln != NULL; ln = ln->next)
523956e45f6SSimon J. Gerraty 		Targ_PrintNode(ln->datum, pass);
524956e45f6SSimon J. Gerraty }
5253955d011SMarcel Moolenaar 
526956e45f6SSimon J. Gerraty static void
PrintOnlySources(void)527956e45f6SSimon J. Gerraty PrintOnlySources(void)
528956e45f6SSimon J. Gerraty {
529956e45f6SSimon J. Gerraty 	GNodeListNode *ln;
530956e45f6SSimon J. Gerraty 
53106b9b3e0SSimon J. Gerraty 	for (ln = allTargets.first; ln != NULL; ln = ln->next) {
532956e45f6SSimon J. Gerraty 		GNode *gn = ln->datum;
533956e45f6SSimon J. Gerraty 		if (GNode_IsTarget(gn))
534956e45f6SSimon J. Gerraty 			continue;
535956e45f6SSimon J. Gerraty 
536956e45f6SSimon J. Gerraty 		debug_printf("#\t%s [%s]", gn->name, GNode_Path(gn));
5373955d011SMarcel Moolenaar 		Targ_PrintType(gn->type);
538956e45f6SSimon J. Gerraty 		debug_printf("\n");
539956e45f6SSimon J. Gerraty 	}
5403955d011SMarcel Moolenaar }
5413955d011SMarcel Moolenaar 
54206b9b3e0SSimon J. Gerraty /*
54306b9b3e0SSimon J. Gerraty  * Input:
5442c3632d1SSimon J. Gerraty  *	pass		1 => before processing
5452c3632d1SSimon J. Gerraty  *			2 => after processing
5462c3632d1SSimon J. Gerraty  *			3 => after processing, an error occurred
5473955d011SMarcel Moolenaar  */
5483955d011SMarcel Moolenaar void
Targ_PrintGraph(int pass)5493955d011SMarcel Moolenaar Targ_PrintGraph(int pass)
5503955d011SMarcel Moolenaar {
551956e45f6SSimon J. Gerraty 	debug_printf("#*** Input graph:\n");
55206b9b3e0SSimon J. Gerraty 	Targ_PrintNodes(&allTargets, pass);
553e2eeea75SSimon J. Gerraty 	debug_printf("\n");
554e2eeea75SSimon J. Gerraty 	debug_printf("\n");
555e2eeea75SSimon J. Gerraty 
556e2eeea75SSimon J. Gerraty 	debug_printf("#\n");
557e2eeea75SSimon J. Gerraty 	debug_printf("#   Files that are only sources:\n");
558956e45f6SSimon J. Gerraty 	PrintOnlySources();
559e2eeea75SSimon J. Gerraty 
560956e45f6SSimon J. Gerraty 	debug_printf("#*** Global Variables:\n");
561dba7b0efSSimon J. Gerraty 	Var_Dump(SCOPE_GLOBAL);
562e2eeea75SSimon J. Gerraty 
563956e45f6SSimon J. Gerraty 	debug_printf("#*** Command-line Variables:\n");
564dba7b0efSSimon J. Gerraty 	Var_Dump(SCOPE_CMDLINE);
565e2eeea75SSimon J. Gerraty 
566956e45f6SSimon J. Gerraty 	debug_printf("\n");
5673955d011SMarcel Moolenaar 	Dir_PrintDirectories();
568956e45f6SSimon J. Gerraty 	debug_printf("\n");
569e2eeea75SSimon J. Gerraty 
5703955d011SMarcel Moolenaar 	Suff_PrintAll();
5713955d011SMarcel Moolenaar }
5723955d011SMarcel Moolenaar 
57306b9b3e0SSimon J. Gerraty /*
57406b9b3e0SSimon J. Gerraty  * Propagate some type information to cohort nodes (those from the '::'
5752c3632d1SSimon J. Gerraty  * dependency operator).
5763955d011SMarcel Moolenaar  *
5772c3632d1SSimon J. Gerraty  * Should be called after the makefiles are parsed but before any action is
57806b9b3e0SSimon J. Gerraty  * taken.
57906b9b3e0SSimon J. Gerraty  */
5803955d011SMarcel Moolenaar void
Targ_Propagate(void)5813955d011SMarcel Moolenaar Targ_Propagate(void)
5823955d011SMarcel Moolenaar {
583956e45f6SSimon J. Gerraty 	GNodeListNode *ln, *cln;
5842c3632d1SSimon J. Gerraty 
58506b9b3e0SSimon J. Gerraty 	for (ln = allTargets.first; ln != NULL; ln = ln->next) {
586956e45f6SSimon J. Gerraty 		GNode *gn = ln->datum;
587956e45f6SSimon J. Gerraty 		GNodeType type = gn->type;
5882c3632d1SSimon J. Gerraty 
589956e45f6SSimon J. Gerraty 		if (!(type & OP_DOUBLEDEP))
5902c3632d1SSimon J. Gerraty 			continue;
5912c3632d1SSimon J. Gerraty 
59206b9b3e0SSimon J. Gerraty 		for (cln = gn->cohorts.first; cln != NULL; cln = cln->next) {
593956e45f6SSimon J. Gerraty 			GNode *cohort = cln->datum;
5942c3632d1SSimon J. Gerraty 
5954fde40d9SSimon J. Gerraty 			cohort->type |= type & (unsigned)~OP_OPMASK;
5962c3632d1SSimon J. Gerraty 		}
5972c3632d1SSimon J. Gerraty 	}
5983955d011SMarcel Moolenaar }
599