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