xref: /freebsd/sbin/rcorder/rcorder.c (revision e6bfd18d21b225af6a0ed67ceeaf1293b7b9eba5)
1 # if 0
2 /*	$NetBSD: rcorder.c,v 1.7 2000/08/04 07:33:55 enami Exp $	*/
3 #endif
4 
5 /*-
6  * SPDX-License-Identifier: BSD-2-Clause
7  *
8  * Copyright (c) 1998, 1999 Matthew R. Green
9  * All rights reserved.
10  * Copyright (c) 1998
11  * 	Perry E. Metzger.  All rights reserved.
12  * Copyright (c) 2020
13  *     Boris N. Lytochkin. All rights reserved.
14  *
15  * Redistribution and use in source and binary forms, with or without
16  * modification, are permitted provided that the following conditions
17  * are met:
18  * 1. Redistributions of source code must retain the above copyright
19  *    notice, this list of conditions and the following disclaimer.
20  * 2. Redistributions in binary form must reproduce the above copyright
21  *    notice, this list of conditions and the following disclaimer in the
22  *    documentation and/or other materials provided with the distribution.
23  * 3. All advertising materials mentioning features or use of this software
24  *    must display the following acknowledgement:
25  *	This product includes software developed for the NetBSD Project
26  *	by Perry E. Metzger.
27  * 4. The name of the author may not be used to endorse or promote products
28  *    derived from this software without specific prior written permission.
29  *
30  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
31  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
32  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
33  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
34  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
35  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
36  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
37  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
38  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
39  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40  */
41 
42 #include <sys/types.h>
43 __FBSDID("$FreeBSD$");
44 
45 #include <sys/stat.h>
46 
47 #include <err.h>
48 #include <stdio.h>
49 #include <libutil.h>
50 #include <stdlib.h>
51 #include <string.h>
52 #include <unistd.h>
53 #include <libgen.h>
54 #include <stdbool.h>
55 
56 #include "ealloc.h"
57 #include "sprite.h"
58 #include "hash.h"
59 
60 #ifdef DEBUG
61 static int debug = 0;
62 # define	DPRINTF(args) if (debug) { fflush(stdout); fprintf args; }
63 #else
64 # define	DPRINTF(args)
65 #endif
66 
67 #define REQUIRE_STR	"# REQUIRE:"
68 #define REQUIRE_LEN	(sizeof(REQUIRE_STR) - 1)
69 #define REQUIRES_STR	"# REQUIRES:"
70 #define REQUIRES_LEN	(sizeof(REQUIRES_STR) - 1)
71 #define PROVIDE_STR	"# PROVIDE:"
72 #define PROVIDE_LEN	(sizeof(PROVIDE_STR) - 1)
73 #define PROVIDES_STR	"# PROVIDES:"
74 #define PROVIDES_LEN	(sizeof(PROVIDES_STR) - 1)
75 #define BEFORE_STR	"# BEFORE:"
76 #define BEFORE_LEN	(sizeof(BEFORE_STR) - 1)
77 #define KEYWORD_STR	"# KEYWORD:"
78 #define KEYWORD_LEN	(sizeof(KEYWORD_STR) - 1)
79 #define KEYWORDS_STR	"# KEYWORDS:"
80 #define KEYWORDS_LEN	(sizeof(KEYWORDS_STR) - 1)
81 
82 #define	FAKE_PROV_NAME	"fake_prov_"
83 
84 static int exit_code;
85 static int file_count;
86 static char **file_list;
87 
88 #define TRUE 1
89 #define FALSE 0
90 typedef bool flag;
91 #define SET TRUE
92 #define RESET FALSE
93 
94 static flag do_graphviz = false;
95 static flag do_parallel = false;
96 
97 static Hash_Table provide_hash_s, *provide_hash;
98 
99 typedef struct provnode provnode;
100 typedef struct filenode filenode;
101 typedef struct f_provnode f_provnode;
102 typedef struct f_reqnode f_reqnode;
103 typedef struct strnodelist strnodelist;
104 
105 struct provnode {
106 	flag		head;
107 	flag		in_progress;
108 	int		sequence;
109 	filenode	*fnode;
110 	provnode	*next, *last;
111 };
112 
113 struct f_provnode {
114 	provnode	*pnode;
115 	Hash_Entry	*entry;
116 	f_provnode	*next;
117 };
118 
119 struct f_reqnode {
120 	Hash_Entry	*entry;
121 	f_reqnode	*next;
122 };
123 
124 struct strnodelist {
125 	filenode	*node;
126 	strnodelist	*next;
127 	char		s[1];
128 };
129 
130 struct filenode {
131 	char		*filename;
132 	flag		in_progress;
133 	filenode	*next, *last;
134 	f_reqnode	*req_list;
135 	f_provnode	*prov_list;
136 	strnodelist	*keyword_list;
137 	int		issues_count;
138 	int		sequence;
139 };
140 
141 static filenode fn_head_s, *fn_head, **fn_seqlist;
142 static int max_sequence = 0;
143 
144 static strnodelist *bl_list;
145 static strnodelist *keep_list;
146 static strnodelist *skip_list;
147 
148 static void do_file(filenode *fnode, strnodelist *);
149 static void strnode_add(strnodelist **, char *, filenode *);
150 static int skip_ok(filenode *fnode);
151 static int keep_ok(filenode *fnode);
152 static char *generate_loop_for_req(strnodelist *, provnode *, filenode *);
153 static void satisfy_req(f_reqnode *rnode, filenode *fnode, strnodelist *);
154 static void crunch_file(char *);
155 static void parse_require(filenode *, char *);
156 static void parse_provide(filenode *, char *);
157 static void parse_before(filenode *, char *);
158 static void parse_keywords(filenode *, char *);
159 static filenode *filenode_new(char *);
160 static void add_require(filenode *, char *);
161 static void add_provide(filenode *, char *);
162 static void add_before(filenode *, char *);
163 static void add_keyword(filenode *, char *);
164 static void insert_before(void);
165 static Hash_Entry *make_fake_provision(filenode *);
166 static void crunch_all_files(void);
167 static void initialize(void);
168 static void generate_graphviz_header(void);
169 static void generate_graphviz_footer(void);
170 static void generate_graphviz_file_links(Hash_Entry *, filenode *);
171 static void generate_graphviz_providers(void);
172 static inline int is_fake_prov(const char *);
173 static int sequence_cmp(const void *, const void *);
174 static void generate_ordering(void);
175 
176 int
177 main(int argc, char *argv[])
178 {
179 	int ch;
180 
181 	while ((ch = getopt(argc, argv, "dgk:ps:")) != -1)
182 		switch (ch) {
183 		case 'd':
184 #ifdef DEBUG
185 			debug = 1;
186 #else
187 			warnx("debugging not compiled in, -d ignored");
188 #endif
189 			break;
190 		case 'g':
191 			do_graphviz = true;
192 			break;
193 		case 'k':
194 			strnode_add(&keep_list, optarg, 0);
195 			break;
196 		case 'p':
197 			do_parallel = true;
198 			break;
199 		case 's':
200 			strnode_add(&skip_list, optarg, 0);
201 			break;
202 		default:
203 			/* XXX should crunch it? */
204 			break;
205 		}
206 	argc -= optind;
207 	argv += optind;
208 
209 	file_count = argc;
210 	file_list = argv;
211 
212 	DPRINTF((stderr, "parse_args\n"));
213 	initialize();
214 	DPRINTF((stderr, "initialize\n"));
215 	generate_graphviz_header();
216 	crunch_all_files();
217 	DPRINTF((stderr, "crunch_all_files\n"));
218 	generate_graphviz_providers();
219 	generate_ordering();
220 	DPRINTF((stderr, "generate_ordering\n"));
221 	generate_graphviz_footer();
222 
223 	exit(exit_code);
224 }
225 
226 /*
227  * initialise various variables.
228  */
229 static void
230 initialize(void)
231 {
232 
233 	fn_head = &fn_head_s;
234 
235 	provide_hash = &provide_hash_s;
236 	Hash_InitTable(provide_hash, file_count);
237 }
238 
239 /* generic function to insert a new strnodelist element */
240 static void
241 strnode_add(strnodelist **listp, char *s, filenode *fnode)
242 {
243 	strnodelist *ent;
244 
245 	ent = emalloc(sizeof *ent + strlen(s));
246 	ent->node = fnode;
247 	strcpy(ent->s, s);
248 	ent->next = *listp;
249 	*listp = ent;
250 }
251 
252 /*
253  * below are the functions that deal with creating the lists
254  * from the filename's given dependencies and provisions
255  * in each of these files.  no ordering or checking is done here.
256  */
257 
258 /*
259  * we have a new filename, create a new filenode structure.
260  * fill in the bits, and put it in the filenode linked list
261  */
262 static filenode *
263 filenode_new(char *filename)
264 {
265 	filenode *temp;
266 
267 	temp = emalloc(sizeof(*temp));
268 	memset(temp, 0, sizeof(*temp));
269 	temp->filename = estrdup(filename);
270 	temp->req_list = NULL;
271 	temp->prov_list = NULL;
272 	temp->keyword_list = NULL;
273 	temp->in_progress = RESET;
274 	/*
275 	 * link the filenode into the list of filenodes.
276 	 * note that the double linking means we can delete a
277 	 * filenode without searching for where it belongs.
278 	 */
279 	temp->next = fn_head->next;
280 	if (temp->next != NULL)
281 		temp->next->last = temp;
282 	temp->last = fn_head;
283 	fn_head->next = temp;
284 	return (temp);
285 }
286 
287 /*
288  * add a requirement to a filenode.
289  */
290 static void
291 add_require(filenode *fnode, char *s)
292 {
293 	Hash_Entry *entry;
294 	f_reqnode *rnode;
295 	int new;
296 
297 	entry = Hash_CreateEntry(provide_hash, s, &new);
298 	if (new)
299 		Hash_SetValue(entry, NULL);
300 	rnode = emalloc(sizeof(*rnode));
301 	rnode->entry = entry;
302 	rnode->next = fnode->req_list;
303 	fnode->req_list = rnode;
304 }
305 
306 /*
307  * add a provision to a filenode.  if this provision doesn't
308  * have a head node, create one here.
309  */
310 static void
311 add_provide(filenode *fnode, char *s)
312 {
313 	Hash_Entry *entry;
314 	f_provnode *f_pnode;
315 	provnode *pnode, *head;
316 	int new;
317 
318 	entry = Hash_CreateEntry(provide_hash, s, &new);
319 	head = Hash_GetValue(entry);
320 
321 	/* create a head node if necessary. */
322 	if (head == NULL) {
323 		head = emalloc(sizeof(*head));
324 		head->head = SET;
325 		head->in_progress = RESET;
326 		head->fnode = NULL;
327 		head->sequence = 0;
328 		head->last = head->next = NULL;
329 		Hash_SetValue(entry, head);
330 	}
331 #if 0
332 	/*
333 	 * Don't warn about this.  We want to be able to support
334 	 * scripts that do two complex things:
335 	 *
336 	 *	- Two independent scripts which both provide the
337 	 *	  same thing.  Both scripts must be executed in
338 	 *	  any order to meet the barrier.  An example:
339 	 *
340 	 *		Script 1:
341 	 *
342 	 *			PROVIDE: mail
343 	 *			REQUIRE: LOGIN
344 	 *
345 	 *		Script 2:
346 	 *
347 	 *			PROVIDE: mail
348 	 *			REQUIRE: LOGIN
349 	 *
350 	 * 	- Two interdependent scripts which both provide the
351 	 *	  same thing.  Both scripts must be executed in
352 	 *	  graph order to meet the barrier.  An example:
353 	 *
354 	 *		Script 1:
355 	 *
356 	 *			PROVIDE: nameservice dnscache
357 	 *			REQUIRE: SERVERS
358 	 *
359 	 *		Script 2:
360 	 *
361 	 *			PROVIDE: nameservice nscd
362 	 *			REQUIRE: dnscache
363 	 */
364 	else if (new == 0) {
365 		warnx("file `%s' provides `%s'.", fnode->filename, s);
366 		warnx("\tpreviously seen in `%s'.",
367 		    head->next->fnode->filename);
368 	}
369 #endif
370 
371 	pnode = emalloc(sizeof(*pnode));
372 	pnode->head = RESET;
373 	pnode->in_progress = RESET;
374 	pnode->fnode = fnode;
375 	pnode->next = head->next;
376 	pnode->last = head;
377 	head->next = pnode;
378 	if (pnode->next != NULL)
379 		pnode->next->last = pnode;
380 
381 	f_pnode = emalloc(sizeof(*f_pnode));
382 	f_pnode->pnode = pnode;
383 	f_pnode->entry = entry;
384 	f_pnode->next = fnode->prov_list;
385 	fnode->prov_list = f_pnode;
386 }
387 
388 /*
389  * put the BEFORE: lines to a list and handle them later.
390  */
391 static void
392 add_before(filenode *fnode, char *s)
393 {
394 	strnodelist *bf_ent;
395 
396 	bf_ent = emalloc(sizeof *bf_ent + strlen(s));
397 	bf_ent->node = fnode;
398 	strcpy(bf_ent->s, s);
399 	bf_ent->next = bl_list;
400 	bl_list = bf_ent;
401 }
402 
403 /*
404  * add a key to a filenode.
405  */
406 static void
407 add_keyword(filenode *fnode, char *s)
408 {
409 
410 	strnode_add(&fnode->keyword_list, s, fnode);
411 }
412 
413 /*
414  * loop over the rest of a REQUIRE line, giving each word to
415  * add_require() to do the real work.
416  */
417 static void
418 parse_require(filenode *node, char *buffer)
419 {
420 	char *s;
421 
422 	while ((s = strsep(&buffer, " \t\n")) != NULL)
423 		if (*s != '\0')
424 			add_require(node, s);
425 }
426 
427 /*
428  * loop over the rest of a PROVIDE line, giving each word to
429  * add_provide() to do the real work.
430  */
431 static void
432 parse_provide(filenode *node, char *buffer)
433 {
434 	char *s;
435 
436 	while ((s = strsep(&buffer, " \t\n")) != NULL)
437 		if (*s != '\0')
438 			add_provide(node, s);
439 }
440 
441 /*
442  * loop over the rest of a BEFORE line, giving each word to
443  * add_before() to do the real work.
444  */
445 static void
446 parse_before(filenode *node, char *buffer)
447 {
448 	char *s;
449 
450 	while ((s = strsep(&buffer, " \t\n")) != NULL)
451 		if (*s != '\0')
452 			add_before(node, s);
453 }
454 
455 /*
456  * loop over the rest of a KEYWORD line, giving each word to
457  * add_keyword() to do the real work.
458  */
459 static void
460 parse_keywords(filenode *node, char *buffer)
461 {
462 	char *s;
463 
464 	while ((s = strsep(&buffer, " \t\n")) != NULL)
465 		if (*s != '\0')
466 			add_keyword(node, s);
467 }
468 
469 /*
470  * given a file name, create a filenode for it, read in lines looking
471  * for provision and requirement lines, building the graphs as needed.
472  */
473 static void
474 crunch_file(char *filename)
475 {
476 	FILE *fp;
477 	char *buf;
478 	int require_flag, provide_flag, before_flag, keywords_flag;
479 	enum { BEFORE_PARSING, PARSING, PARSING_DONE } state;
480 	filenode *node;
481 	char delims[3] = { '\\', '\\', '\0' };
482 	struct stat st;
483 
484 	if ((fp = fopen(filename, "r")) == NULL) {
485 		warn("could not open %s", filename);
486 		return;
487 	}
488 
489 	if (fstat(fileno(fp), &st) == -1) {
490 		warn("could not stat %s", filename);
491 		fclose(fp);
492 		return;
493 	}
494 
495 	if (!S_ISREG(st.st_mode)) {
496 #if 0
497 		warnx("%s is not a file", filename);
498 #endif
499 		fclose(fp);
500 		return;
501 	}
502 
503 	node = filenode_new(filename);
504 
505 	/*
506 	 * we don't care about length, line number, don't want # for comments,
507 	 * and have no flags.
508 	 */
509 	for (state = BEFORE_PARSING; state != PARSING_DONE &&
510 	    (buf = fparseln(fp, NULL, NULL, delims, 0)) != NULL; free(buf)) {
511 		require_flag = provide_flag = before_flag = keywords_flag = 0;
512 		if (strncmp(REQUIRE_STR, buf, REQUIRE_LEN) == 0)
513 			require_flag = REQUIRE_LEN;
514 		else if (strncmp(REQUIRES_STR, buf, REQUIRES_LEN) == 0)
515 			require_flag = REQUIRES_LEN;
516 		else if (strncmp(PROVIDE_STR, buf, PROVIDE_LEN) == 0)
517 			provide_flag = PROVIDE_LEN;
518 		else if (strncmp(PROVIDES_STR, buf, PROVIDES_LEN) == 0)
519 			provide_flag = PROVIDES_LEN;
520 		else if (strncmp(BEFORE_STR, buf, BEFORE_LEN) == 0)
521 			before_flag = BEFORE_LEN;
522 		else if (strncmp(KEYWORD_STR, buf, KEYWORD_LEN) == 0)
523 			keywords_flag = KEYWORD_LEN;
524 		else if (strncmp(KEYWORDS_STR, buf, KEYWORDS_LEN) == 0)
525 			keywords_flag = KEYWORDS_LEN;
526 		else {
527 			if (state == PARSING)
528 				state = PARSING_DONE;
529 			continue;
530 		}
531 
532 		state = PARSING;
533 		if (require_flag)
534 			parse_require(node, buf + require_flag);
535 		else if (provide_flag)
536 			parse_provide(node, buf + provide_flag);
537 		else if (before_flag)
538 			parse_before(node, buf + before_flag);
539 		else if (keywords_flag)
540 			parse_keywords(node, buf + keywords_flag);
541 	}
542 	fclose(fp);
543 }
544 
545 static Hash_Entry *
546 make_fake_provision(filenode *node)
547 {
548 	Hash_Entry *entry;
549 	f_provnode *f_pnode;
550 	provnode *head, *pnode;
551 	static	int i = 0;
552 	int	new;
553 	char buffer[30];
554 
555 	do {
556 		snprintf(buffer, sizeof buffer, FAKE_PROV_NAME "%08d", i++);
557 		entry = Hash_CreateEntry(provide_hash, buffer, &new);
558 	} while (new == 0);
559 	head = emalloc(sizeof(*head));
560 	head->head = SET;
561 	head->in_progress = RESET;
562 	head->fnode = NULL;
563 	head->last = head->next = NULL;
564 	Hash_SetValue(entry, head);
565 
566 	pnode = emalloc(sizeof(*pnode));
567 	pnode->head = RESET;
568 	pnode->in_progress = RESET;
569 	pnode->fnode = node;
570 	pnode->next = head->next;
571 	pnode->last = head;
572 	head->next = pnode;
573 	if (pnode->next != NULL)
574 		pnode->next->last = pnode;
575 
576 	f_pnode = emalloc(sizeof(*f_pnode));
577 	f_pnode->entry = entry;
578 	f_pnode->pnode = pnode;
579 	f_pnode->next = node->prov_list;
580 	node->prov_list = f_pnode;
581 
582 	return (entry);
583 }
584 
585 /*
586  * go through the BEFORE list, inserting requirements into the graph(s)
587  * as required.  in the before list, for each entry B, we have a file F
588  * and a string S.  we create a "fake" provision (P) that F provides.
589  * for each entry in the provision list for S, add a requirement to
590  * that provisions filenode for P.
591  */
592 static void
593 insert_before(void)
594 {
595 	Hash_Entry *entry, *fake_prov_entry;
596 	provnode *pnode;
597 	f_reqnode *rnode;
598 	strnodelist *bl;
599 	int new;
600 
601 	while (bl_list != NULL) {
602 		bl = bl_list->next;
603 
604 		fake_prov_entry = make_fake_provision(bl_list->node);
605 
606 		entry = Hash_CreateEntry(provide_hash, bl_list->s, &new);
607 		if (new == 1)
608 			warnx("file `%s' is before unknown provision `%s'", bl_list->node->filename, bl_list->s);
609 
610 		if (new == 1 && do_graphviz == true)
611 			generate_graphviz_file_links(
612 			    Hash_FindEntry(provide_hash, bl_list->s),
613 			    bl_list->node);
614 
615 		for (pnode = Hash_GetValue(entry); pnode; pnode = pnode->next) {
616 			if (pnode->head)
617 				continue;
618 
619 			rnode = emalloc(sizeof(*rnode));
620 			rnode->entry = fake_prov_entry;
621 			rnode->next = pnode->fnode->req_list;
622 			pnode->fnode->req_list = rnode;
623 		}
624 
625 		free(bl_list);
626 		bl_list = bl;
627 	}
628 }
629 
630 /*
631  * loop over all the files calling crunch_file() on them to do the
632  * real work.  after we have built all the nodes, insert the BEFORE:
633  * lines into graph(s).
634  */
635 static void
636 crunch_all_files(void)
637 {
638 	int i;
639 
640 	for (i = 0; i < file_count; i++)
641 		crunch_file(file_list[i]);
642 	insert_before();
643 }
644 
645 static inline int
646 is_fake_prov(const char *name)
647 {
648 
649 	return (name == strstr(name, FAKE_PROV_NAME));
650 }
651 
652 /* loop though provide list of vnode drawing all non-fake dependencies */
653 static void
654 generate_graphviz_file_links(Hash_Entry *entry, filenode *fnode)
655 {
656 	char *dep_name, *fname;
657 	provnode *head;
658 	f_provnode *fpnode, *rfpnode;
659 	int is_before = 0;
660 
661 	dep_name = Hash_GetKey(entry);
662 	if (is_fake_prov(dep_name))
663 		is_before = 1;
664 	head = Hash_GetValue(entry);
665 
666 	for (fpnode = fnode->prov_list; fpnode && fpnode->entry;
667 	    fpnode = fpnode->next) {
668 		fname = Hash_GetKey(fpnode->entry);
669 		if (is_fake_prov(fname))
670 			continue;
671 		rfpnode = NULL;
672 		do {
673 			if (rfpnode)
674 				dep_name = Hash_GetKey(rfpnode->entry);
675 			else
676 				dep_name = Hash_GetKey(entry);
677 
678 			if (!is_fake_prov(dep_name)) {
679 				printf("\"%s\" -> \"%s\" [%s%s];\n",
680 				    fname, dep_name,
681 				    /* edge style */
682 				    (is_before ? "style=dashed" : "style=solid"),
683 				    /* circular dep? */
684 				    ((head == NULL ||
685 				    (head->next && head->in_progress == SET)) ?
686 				    ", color=red, penwidth=4" : ""));
687 				if (rfpnode == NULL)
688 					break;
689 			}
690 			/* dependency is solved already */
691 			if (head == NULL || head->next == NULL)
692 				break;
693 
694 			if (rfpnode == NULL)
695 				rfpnode = head->next->fnode->prov_list;
696 			else
697 				rfpnode = rfpnode->next;
698 		} while (rfpnode);
699 	}
700 }
701 
702 /*
703  * Walk the stack, find the looping point and generate traceback.
704  * NULL is returned on failure, otherwize pointer to a buffer holding
705  * text representation is returned, caller must run free(3) for the
706  * pointer.
707  */
708 static char *
709 generate_loop_for_req(strnodelist *stack_tail, provnode *head,
710     filenode *fnode)
711 {
712 	provnode *pnode;
713 	strnodelist *stack_ptr, *loop_entry;
714 	char *buf, **revstack;
715 	size_t bufsize;
716 	int i, stack_depth;
717 
718 	loop_entry = NULL;
719 	/* fast forward stack to the component that is required now */
720 	for (pnode = head->next; pnode; pnode = pnode->next) {
721 		if (loop_entry)
722 			break;
723 		stack_depth = 0;
724 		for (stack_ptr = stack_tail; stack_ptr;
725 		    stack_ptr = stack_ptr->next) {
726 			stack_depth++;
727 			if (stack_ptr->node == pnode->fnode) {
728 				loop_entry = stack_ptr;
729 				break;
730 			}
731 		}
732 	}
733 
734 	if (loop_entry == NULL)
735 		return (NULL);
736 
737 	stack_depth += 2; /* fnode + loop_entry */
738 	revstack = emalloc(sizeof(char *) * stack_depth);
739 	bzero(revstack, (sizeof(char *) * stack_depth));
740 
741 	/* reverse stack and estimate buffer size to allocate */
742 	bufsize = 1; /* tralining \0 */
743 
744 	revstack[stack_depth - 1] = loop_entry->node->filename;
745 	bufsize += strlen(revstack[stack_depth - 1]);
746 
747 	revstack[stack_depth - 2] = fnode->filename;
748 	bufsize += strlen(revstack[stack_depth - 2]);
749 	fnode->issues_count++;
750 
751 	stack_ptr = stack_tail;
752 	for (i = stack_depth - 3; i >= 0; i--) {
753 		revstack[i] = stack_ptr->node->filename;
754 		stack_ptr->node->issues_count++;
755 		stack_ptr = stack_ptr->next;
756 		bufsize += strlen(revstack[i]);
757 	}
758 	bufsize += strlen(" -> ") * (stack_depth - 1);
759 
760 	buf = emalloc(bufsize);
761 	bzero(buf, bufsize);
762 
763 	for (i = 0; i < stack_depth; i++) {
764 		strlcat(buf, revstack[i], bufsize);
765 		if (i < stack_depth - 1)
766 			strlcat(buf, " -> ", bufsize);
767 	}
768 
769 	free(revstack);
770 	return (buf);
771 }
772 /*
773  * below are the functions that traverse the graphs we have built
774  * finding out the desired ordering, printing each file in turn.
775  * if missing requirements, or cyclic graphs are detected, a
776  * warning will be issued, and we will continue on..
777  */
778 
779 /*
780  * given a requirement node (in a filename) we attempt to satisfy it.
781  * we do some sanity checking first, to ensure that we have providers,
782  * aren't already satisfied and aren't already being satisfied (ie,
783  * cyclic).  if we pass all this, we loop over the provision list
784  * calling do_file() (enter recursion) for each filenode in this
785  * provision.
786  */
787 static void
788 satisfy_req(f_reqnode *rnode, filenode *fnode, strnodelist *stack_ptr)
789 {
790 	Hash_Entry *entry;
791 	provnode *head;
792 	strnodelist stack_item;
793 	char *buf;
794 
795 	entry = rnode->entry;
796 	head = Hash_GetValue(entry);
797 
798 	if (do_graphviz == true)
799 		generate_graphviz_file_links(entry, fnode);
800 
801 	if (head == NULL) {
802 		warnx("requirement `%s' in file `%s' has no providers.",
803 		    Hash_GetKey(entry), fnode->filename);
804 		exit_code = 1;
805 		return;
806 	}
807 
808 	/* return if the requirement is already satisfied. */
809 	if (head->next == NULL)
810 		return;
811 
812 	/*
813 	 * if list is marked as in progress,
814 	 *	print that there is a circular dependency on it and abort
815 	 */
816 	if (head->in_progress == SET) {
817 		exit_code = 1;
818 		buf = generate_loop_for_req(stack_ptr, head,
819 		    fnode);
820 
821 		if (buf == NULL) {
822 			warnx("Circular dependency on provision `%s' in "
823 			    "file `%s' (tracing has failed).",
824 			    Hash_GetKey(entry), fnode->filename);
825 			return;
826 		}
827 
828 		warnx("Circular dependency on provision `%s': %s.",
829 		    Hash_GetKey(entry), buf);
830 		free(buf);
831 		return;
832 	}
833 
834 	head->in_progress = SET;
835 
836 	stack_item.next = stack_ptr;
837 	stack_item.node = fnode;
838 
839 	/*
840 	 * while provision_list is not empty
841 	 *	do_file(first_member_of(provision_list));
842 	 */
843 	while (head->next != NULL)
844 		do_file(head->next->fnode, &stack_item);
845 }
846 
847 static int
848 skip_ok(filenode *fnode)
849 {
850 	strnodelist *s;
851 	strnodelist *k;
852 
853 	for (s = skip_list; s; s = s->next)
854 		for (k = fnode->keyword_list; k; k = k->next)
855 			if (strcmp(k->s, s->s) == 0)
856 				return (0);
857 
858 	return (1);
859 }
860 
861 static int
862 keep_ok(filenode *fnode)
863 {
864 	strnodelist *s;
865 	strnodelist *k;
866 
867 	for (s = keep_list; s; s = s->next)
868 		for (k = fnode->keyword_list; k; k = k->next)
869 			if (strcmp(k->s, s->s) == 0)
870 				return (1);
871 
872 	/* an empty keep_list means every one */
873 	return (!keep_list);
874 }
875 
876 /*
877  * given a filenode, we ensure we are not a cyclic graph.  if this
878  * is ok, we loop over the filenodes requirements, calling satisfy_req()
879  * for each of them.. once we have done this, remove this filenode
880  * from each provision table, as we are now done.
881  *
882  * NOTE: do_file() is called recursively from several places and cannot
883  * safely free() anything related to items that may be recursed on.
884  * Circular dependencies will cause problems if we do.
885  */
886 static void
887 do_file(filenode *fnode, strnodelist *stack_ptr)
888 {
889 	f_reqnode *r;
890 	f_provnode *p, *p_tmp;
891 	provnode *pnode, *head;
892 	int was_set;
893 	char *dep_name;
894 
895 	DPRINTF((stderr, "do_file on %s.\n", fnode->filename));
896 
897 	/*
898 	 * if fnode is marked as in progress,
899 	 *	 print that fnode; is circularly depended upon and abort.
900 	 */
901 	if (fnode->in_progress == SET) {
902 		warnx("Circular dependency on file `%s'.",
903 			fnode->filename);
904 		was_set = exit_code = 1;
905 	} else
906 		was_set = 0;
907 
908 	/* mark fnode */
909 	fnode->in_progress = SET;
910 
911 	/*
912 	 * for each requirement of fnode -> r
913 	 *	satisfy_req(r, filename)
914 	 */
915 	r = fnode->req_list;
916 	fnode->sequence = 0;
917 	while (r != NULL) {
918 		satisfy_req(r, fnode, stack_ptr);
919 		/* find sequence number where all requirements are satisfied */
920 		head = Hash_GetValue(r->entry);
921 		if (head && head->sequence > fnode->sequence)
922 			fnode->sequence = head->sequence;
923 		r = r->next;
924 	}
925 	fnode->req_list = NULL;
926 	fnode->sequence++;
927 
928 	/* if we've seen issues with this file - put it to the tail */
929 	if (fnode->issues_count)
930 		fnode->sequence = max_sequence + 1;
931 
932 	if (max_sequence < fnode->sequence)
933 		max_sequence = fnode->sequence;
934 
935 	/*
936 	 * for each provision of fnode -> p
937 	 *	remove fnode from provision list for p in hash table
938 	 */
939 	p = fnode->prov_list;
940 	while (p != NULL) {
941 		/* mark all troublemakers on graphviz */
942 		if (do_graphviz == true && fnode->issues_count) {
943 			dep_name = Hash_GetKey(p->entry);
944 			if (!is_fake_prov(dep_name))
945 				printf("\"%s\" [ color=red, penwidth=4 ];\n",
946 				    dep_name);
947 		}
948 
949 		/* update sequence when provided requirements are satisfied */
950 		head = Hash_GetValue(p->entry);
951 		if (head->sequence < fnode->sequence)
952 			head->sequence = fnode->sequence;
953 
954 		p_tmp = p;
955 		pnode = p->pnode;
956 		if (pnode->next != NULL) {
957 			pnode->next->last = pnode->last;
958 		}
959 		if (pnode->last != NULL) {
960 			pnode->last->next = pnode->next;
961 		}
962 		free(pnode);
963 		p = p->next;
964 		free(p_tmp);
965 	}
966 	fnode->prov_list = NULL;
967 
968 	/* do_it(fnode) */
969 	DPRINTF((stderr, "next do: "));
970 
971 	/* if we were already in progress, don't print again */
972 	if (do_graphviz != true && was_set == 0 && skip_ok(fnode) &&
973 	    keep_ok(fnode)) {
974 		*fn_seqlist = fnode;
975 		fn_seqlist++;
976 	}
977 
978 	if (fnode->next != NULL) {
979 		fnode->next->last = fnode->last;
980 	}
981 	if (fnode->last != NULL) {
982 		fnode->last->next = fnode->next;
983 	}
984 
985 	if (fnode->issues_count)
986 		warnx("`%s' was seen in circular dependencies for %d times.",
987 		    fnode->filename, fnode->issues_count);
988 
989 	DPRINTF((stderr, "nuking %s\n", fnode->filename));
990 }
991 
992 static void
993 generate_graphviz_header(void)
994 {
995 
996 	if (do_graphviz != true)
997 		return;
998 
999 	printf("digraph rcorder {\n"
1000 	    "rankdir=\"BT\";\n"
1001 	    "node [style=rounded, shape=record];\n"
1002 	    "graph [overlap = false];\n");
1003 }
1004 
1005 static void
1006 generate_graphviz_footer(void)
1007 {
1008 
1009 	if (do_graphviz == true)
1010 		printf("}\n");
1011 }
1012 
1013 static void
1014 generate_graphviz_providers(void)
1015 {
1016 	Hash_Entry *entry;
1017 	Hash_Search psearch;
1018 	provnode *head, *pnode;
1019 	char *dep_name;
1020 
1021 	if (do_graphviz != true)
1022 		return;
1023 
1024 	entry = Hash_EnumFirst(provide_hash, &psearch);
1025 	if (entry == NULL)
1026 		return;
1027 
1028 	do {
1029 		dep_name = Hash_GetKey(entry);
1030 		if (is_fake_prov(dep_name))
1031 			continue;
1032 		head = Hash_GetValue(entry);
1033 		/* no providers for this requirement */
1034 		if (head == NULL || head->next == NULL) {
1035 			printf("\"%s\" [label=\"{ %s | ENOENT }\", "
1036 			    "style=\"rounded,filled\", color=red];\n",
1037 			    dep_name, dep_name);
1038 			continue;
1039 		}
1040 		/* one PROVIDE word for one file that matches */
1041 		if (head->next->next == NULL &&
1042 		    strcmp(dep_name,
1043 		    basename(head->next->fnode->filename)) == 0) {
1044 		        continue;
1045 		}
1046 		printf("\"%s\" [label=\"{ %s | ", dep_name, dep_name);
1047 		for (pnode = head->next; pnode; pnode = pnode->next)
1048 			printf("%s\\n", basename(pnode->fnode->filename));
1049 
1050 		printf("}\"];\n");
1051 	} while (NULL != (entry = Hash_EnumNext(&psearch)));
1052 }
1053 
1054 static int
1055 sequence_cmp(const void *a, const void *b)
1056 {
1057 	const filenode *fna = *((const filenode * const *)a);
1058 	const filenode *fnb = *((const filenode * const *)b);
1059 	int left, right;
1060 
1061 	/* push phantom files to the end */
1062 	if (fna == NULL || fnb == NULL)
1063 		return ((fna < fnb) - (fna > fnb));
1064 
1065 	left =  fna->sequence;
1066 	right = fnb->sequence;
1067 
1068 	return ((left > right) - (left < right));
1069 }
1070 
1071 static void
1072 generate_ordering(void)
1073 {
1074 	filenode **seqlist, **psl;
1075 	int last_seq = 0;
1076 
1077 	/* Prepare order buffer, use an additional one as a list terminator */
1078 	seqlist = emalloc(sizeof(filenode *) * (file_count + 1));
1079 	bzero(seqlist, sizeof(filenode *) * (file_count + 1));
1080 	fn_seqlist = seqlist;
1081 
1082 	/*
1083 	 * while there remain undone files{f},
1084 	 *	pick an arbitrary f, and do_file(f)
1085 	 * Note that the first file in the file list is perfectly
1086 	 * arbitrary, and easy to find, so we use that.
1087 	 */
1088 
1089 	/*
1090 	 * N.B.: the file nodes "self delete" after they execute, so
1091 	 * after each iteration of the loop, the head will be pointing
1092 	 * to something totally different. The loop ends up being
1093 	 * executed only once for every strongly connected set of
1094 	 * nodes.
1095 	 */
1096 	while (fn_head->next != NULL) {
1097 		DPRINTF((stderr, "generate on %s\n", fn_head->next->filename));
1098 		do_file(fn_head->next, NULL);
1099 	}
1100 
1101 	/* Sort filenode list based on sequence */
1102 	qsort(seqlist, file_count, sizeof(filenode *), sequence_cmp);
1103 
1104 	for (psl = seqlist; *psl; psl++) {
1105 		printf("%s%s",
1106 		    (last_seq == 0 ? "" :
1107 		    (do_parallel != true || last_seq != (*psl)->sequence) ?
1108 		    "\n" : " "),
1109 		(*psl)->filename);
1110 		last_seq = (*psl)->sequence;
1111 		free((*psl)->filename);
1112 		free(*psl);
1113 	}
1114 	if (last_seq)
1115 		printf("\n");
1116 
1117 	free(seqlist);
1118 }
1119