1 /* $NetBSD: suff.c,v 1.65 2008/12/13 15:19:29 dsl Exp $ */ 2 3 /* 4 * Copyright (c) 1988, 1989, 1990, 1993 5 * The Regents of the University of California. All rights reserved. 6 * 7 * This code is derived from software contributed to Berkeley by 8 * Adam de Boor. 9 * 10 * Redistribution and use in source and binary forms, with or without 11 * modification, are permitted provided that the following conditions 12 * are met: 13 * 1. Redistributions of source code must retain the above copyright 14 * notice, this list of conditions and the following disclaimer. 15 * 2. Redistributions in binary form must reproduce the above copyright 16 * notice, this list of conditions and the following disclaimer in the 17 * documentation and/or other materials provided with the distribution. 18 * 3. Neither the name of the University nor the names of its contributors 19 * may be used to endorse or promote products derived from this software 20 * without specific prior written permission. 21 * 22 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 23 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 24 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 25 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 26 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 27 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 28 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 29 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 31 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 32 * SUCH DAMAGE. 33 */ 34 35 /* 36 * Copyright (c) 1989 by Berkeley Softworks 37 * All rights reserved. 38 * 39 * This code is derived from software contributed to Berkeley by 40 * Adam de Boor. 41 * 42 * Redistribution and use in source and binary forms, with or without 43 * modification, are permitted provided that the following conditions 44 * are met: 45 * 1. Redistributions of source code must retain the above copyright 46 * notice, this list of conditions and the following disclaimer. 47 * 2. Redistributions in binary form must reproduce the above copyright 48 * notice, this list of conditions and the following disclaimer in the 49 * documentation and/or other materials provided with the distribution. 50 * 3. All advertising materials mentioning features or use of this software 51 * must display the following acknowledgement: 52 * This product includes software developed by the University of 53 * California, Berkeley and its contributors. 54 * 4. Neither the name of the University nor the names of its contributors 55 * may be used to endorse or promote products derived from this software 56 * without specific prior written permission. 57 * 58 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 59 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 60 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 61 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 62 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 63 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 64 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 65 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 66 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 67 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 68 * SUCH DAMAGE. 69 */ 70 71 #ifndef MAKE_NATIVE 72 static char rcsid[] = "$NetBSD: suff.c,v 1.65 2008/12/13 15:19:29 dsl Exp $"; 73 #else 74 #include <sys/cdefs.h> 75 #ifndef lint 76 #if 0 77 static char sccsid[] = "@(#)suff.c 8.4 (Berkeley) 3/21/94"; 78 #else 79 __RCSID("$NetBSD: suff.c,v 1.65 2008/12/13 15:19:29 dsl Exp $"); 80 #endif 81 #endif /* not lint */ 82 #endif 83 84 /*- 85 * suff.c -- 86 * Functions to maintain suffix lists and find implicit dependents 87 * using suffix transformation rules 88 * 89 * Interface: 90 * Suff_Init Initialize all things to do with suffixes. 91 * 92 * Suff_End Cleanup the module 93 * 94 * Suff_DoPaths This function is used to make life easier 95 * when searching for a file according to its 96 * suffix. It takes the global search path, 97 * as defined using the .PATH: target, and appends 98 * its directories to the path of each of the 99 * defined suffixes, as specified using 100 * .PATH<suffix>: targets. In addition, all 101 * directories given for suffixes labeled as 102 * include files or libraries, using the .INCLUDES 103 * or .LIBS targets, are played with using 104 * Dir_MakeFlags to create the .INCLUDES and 105 * .LIBS global variables. 106 * 107 * Suff_ClearSuffixes Clear out all the suffixes and defined 108 * transformations. 109 * 110 * Suff_IsTransform Return TRUE if the passed string is the lhs 111 * of a transformation rule. 112 * 113 * Suff_AddSuffix Add the passed string as another known suffix. 114 * 115 * Suff_GetPath Return the search path for the given suffix. 116 * 117 * Suff_AddInclude Mark the given suffix as denoting an include 118 * file. 119 * 120 * Suff_AddLib Mark the given suffix as denoting a library. 121 * 122 * Suff_AddTransform Add another transformation to the suffix 123 * graph. Returns GNode suitable for framing, I 124 * mean, tacking commands, attributes, etc. on. 125 * 126 * Suff_SetNull Define the suffix to consider the suffix of 127 * any file that doesn't have a known one. 128 * 129 * Suff_FindDeps Find implicit sources for and the location of 130 * a target based on its suffix. Returns the 131 * bottom-most node added to the graph or NULL 132 * if the target had no implicit sources. 133 * 134 * Suff_FindPath Return the appropriate path to search in 135 * order to find the node. 136 */ 137 138 #include <stdio.h> 139 #include "make.h" 140 #include "hash.h" 141 #include "dir.h" 142 143 static Lst sufflist; /* Lst of suffixes */ 144 #ifdef CLEANUP 145 static Lst suffClean; /* Lst of suffixes to be cleaned */ 146 #endif 147 static Lst srclist; /* Lst of sources */ 148 static Lst transforms; /* Lst of transformation rules */ 149 150 static int sNum = 0; /* Counter for assigning suffix numbers */ 151 152 /* 153 * Structure describing an individual suffix. 154 */ 155 typedef struct _Suff { 156 char *name; /* The suffix itself */ 157 int nameLen; /* Length of the suffix */ 158 short flags; /* Type of suffix */ 159 #define SUFF_INCLUDE 0x01 /* One which is #include'd */ 160 #define SUFF_LIBRARY 0x02 /* One which contains a library */ 161 #define SUFF_NULL 0x04 /* The empty suffix */ 162 Lst searchPath; /* The path along which files of this suffix 163 * may be found */ 164 int sNum; /* The suffix number */ 165 int refCount; /* Reference count of list membership */ 166 Lst parents; /* Suffixes we have a transformation to */ 167 Lst children; /* Suffixes we have a transformation from */ 168 Lst ref; /* List of lists this suffix is referenced */ 169 } Suff; 170 171 /* 172 * for SuffSuffIsSuffix 173 */ 174 typedef struct { 175 char *ename; /* The end of the name */ 176 int len; /* Length of the name */ 177 } SuffixCmpData; 178 179 /* 180 * Structure used in the search for implied sources. 181 */ 182 typedef struct _Src { 183 char *file; /* The file to look for */ 184 char *pref; /* Prefix from which file was formed */ 185 Suff *suff; /* The suffix on the file */ 186 struct _Src *parent; /* The Src for which this is a source */ 187 GNode *node; /* The node describing the file */ 188 int children; /* Count of existing children (so we don't free 189 * this thing too early or never nuke it) */ 190 #ifdef DEBUG_SRC 191 Lst cp; /* Debug; children list */ 192 #endif 193 } Src; 194 195 /* 196 * A structure for passing more than one argument to the Lst-library-invoked 197 * function... 198 */ 199 typedef struct { 200 Lst l; 201 Src *s; 202 } LstSrc; 203 204 typedef struct { 205 GNode **gn; 206 Suff *s; 207 Boolean r; 208 } GNodeSuff; 209 210 static Suff *suffNull; /* The NULL suffix for this run */ 211 static Suff *emptySuff; /* The empty suffix required for POSIX 212 * single-suffix transformation rules */ 213 214 215 static char *SuffStrIsPrefix(char *, char *); 216 static char *SuffSuffIsSuffix(Suff *, SuffixCmpData *); 217 static int SuffSuffIsSuffixP(ClientData, ClientData); 218 static int SuffSuffHasNameP(ClientData, ClientData); 219 static int SuffSuffIsPrefix(ClientData, ClientData); 220 static int SuffGNHasNameP(ClientData, ClientData); 221 static void SuffUnRef(ClientData, ClientData); 222 static void SuffFree(ClientData); 223 static void SuffInsert(Lst, Suff *); 224 static void SuffRemove(Lst, Suff *); 225 static Boolean SuffParseTransform(char *, Suff **, Suff **); 226 static int SuffRebuildGraph(ClientData, ClientData); 227 static int SuffScanTargets(ClientData, ClientData); 228 static int SuffAddSrc(ClientData, ClientData); 229 static int SuffRemoveSrc(Lst); 230 static void SuffAddLevel(Lst, Src *); 231 static Src *SuffFindThem(Lst, Lst); 232 static Src *SuffFindCmds(Src *, Lst); 233 static void SuffExpandChildren(LstNode, GNode *); 234 static void SuffExpandWildcards(LstNode, GNode *); 235 static Boolean SuffApplyTransform(GNode *, GNode *, Suff *, Suff *); 236 static void SuffFindDeps(GNode *, Lst); 237 static void SuffFindArchiveDeps(GNode *, Lst); 238 static void SuffFindNormalDeps(GNode *, Lst); 239 static int SuffPrintName(ClientData, ClientData); 240 static int SuffPrintSuff(ClientData, ClientData); 241 static int SuffPrintTrans(ClientData, ClientData); 242 243 /*************** Lst Predicates ****************/ 244 /*- 245 *----------------------------------------------------------------------- 246 * SuffStrIsPrefix -- 247 * See if pref is a prefix of str. 248 * 249 * Input: 250 * pref possible prefix 251 * str string to check 252 * 253 * Results: 254 * NULL if it ain't, pointer to character in str after prefix if so 255 * 256 * Side Effects: 257 * None 258 *----------------------------------------------------------------------- 259 */ 260 static char * 261 SuffStrIsPrefix(char *pref, char *str) 262 { 263 while (*str && *pref == *str) { 264 pref++; 265 str++; 266 } 267 268 return (*pref ? NULL : str); 269 } 270 271 /*- 272 *----------------------------------------------------------------------- 273 * SuffSuffIsSuffix -- 274 * See if suff is a suffix of str. sd->ename should point to THE END 275 * of the string to check. (THE END == the null byte) 276 * 277 * Input: 278 * s possible suffix 279 * sd string to examine 280 * 281 * Results: 282 * NULL if it ain't, pointer to character in str before suffix if 283 * it is. 284 * 285 * Side Effects: 286 * None 287 *----------------------------------------------------------------------- 288 */ 289 static char * 290 SuffSuffIsSuffix(Suff *s, SuffixCmpData *sd) 291 { 292 char *p1; /* Pointer into suffix name */ 293 char *p2; /* Pointer into string being examined */ 294 295 if (sd->len < s->nameLen) 296 return NULL; /* this string is shorter than the suffix */ 297 298 p1 = s->name + s->nameLen; 299 p2 = sd->ename; 300 301 while (p1 >= s->name && *p1 == *p2) { 302 p1--; 303 p2--; 304 } 305 306 return (p1 == s->name - 1 ? p2 : NULL); 307 } 308 309 /*- 310 *----------------------------------------------------------------------- 311 * SuffSuffIsSuffixP -- 312 * Predicate form of SuffSuffIsSuffix. Passed as the callback function 313 * to Lst_Find. 314 * 315 * Results: 316 * 0 if the suffix is the one desired, non-zero if not. 317 * 318 * Side Effects: 319 * None. 320 * 321 *----------------------------------------------------------------------- 322 */ 323 static int 324 SuffSuffIsSuffixP(ClientData s, ClientData sd) 325 { 326 return(!SuffSuffIsSuffix((Suff *)s, (SuffixCmpData *)sd)); 327 } 328 329 /*- 330 *----------------------------------------------------------------------- 331 * SuffSuffHasNameP -- 332 * Callback procedure for finding a suffix based on its name. Used by 333 * Suff_GetPath. 334 * 335 * Input: 336 * s Suffix to check 337 * sd Desired name 338 * 339 * Results: 340 * 0 if the suffix is of the given name. non-zero otherwise. 341 * 342 * Side Effects: 343 * None 344 *----------------------------------------------------------------------- 345 */ 346 static int 347 SuffSuffHasNameP(ClientData s, ClientData sname) 348 { 349 return (strcmp((char *)sname, ((Suff *)s)->name)); 350 } 351 352 /*- 353 *----------------------------------------------------------------------- 354 * SuffSuffIsPrefix -- 355 * See if the suffix described by s is a prefix of the string. Care 356 * must be taken when using this to search for transformations and 357 * what-not, since there could well be two suffixes, one of which 358 * is a prefix of the other... 359 * 360 * Input: 361 * s suffix to compare 362 * str string to examine 363 * 364 * Results: 365 * 0 if s is a prefix of str. non-zero otherwise 366 * 367 * Side Effects: 368 * None 369 *----------------------------------------------------------------------- 370 */ 371 static int 372 SuffSuffIsPrefix(ClientData s, ClientData str) 373 { 374 return (SuffStrIsPrefix(((Suff *)s)->name, (char *)str) == NULL ? 1 : 0); 375 } 376 377 /*- 378 *----------------------------------------------------------------------- 379 * SuffGNHasNameP -- 380 * See if the graph node has the desired name 381 * 382 * Input: 383 * gn current node we're looking at 384 * name name we're looking for 385 * 386 * Results: 387 * 0 if it does. non-zero if it doesn't 388 * 389 * Side Effects: 390 * None 391 *----------------------------------------------------------------------- 392 */ 393 static int 394 SuffGNHasNameP(ClientData gn, ClientData name) 395 { 396 return (strcmp((char *)name, ((GNode *)gn)->name)); 397 } 398 399 /*********** Maintenance Functions ************/ 400 401 static void 402 SuffUnRef(ClientData lp, ClientData sp) 403 { 404 Lst l = (Lst) lp; 405 406 LstNode ln = Lst_Member(l, sp); 407 if (ln != NULL) { 408 Lst_Remove(l, ln); 409 ((Suff *)sp)->refCount--; 410 } 411 } 412 413 /*- 414 *----------------------------------------------------------------------- 415 * SuffFree -- 416 * Free up all memory associated with the given suffix structure. 417 * 418 * Results: 419 * none 420 * 421 * Side Effects: 422 * the suffix entry is detroyed 423 *----------------------------------------------------------------------- 424 */ 425 static void 426 SuffFree(ClientData sp) 427 { 428 Suff *s = (Suff *)sp; 429 430 if (s == suffNull) 431 suffNull = NULL; 432 433 if (s == emptySuff) 434 emptySuff = NULL; 435 436 #ifdef notdef 437 /* We don't delete suffixes in order, so we cannot use this */ 438 if (s->refCount) 439 Punt("Internal error deleting suffix `%s' with refcount = %d", s->name, 440 s->refCount); 441 #endif 442 443 Lst_Destroy(s->ref, NULL); 444 Lst_Destroy(s->children, NULL); 445 Lst_Destroy(s->parents, NULL); 446 Lst_Destroy(s->searchPath, Dir_Destroy); 447 448 free(s->name); 449 free(s); 450 } 451 452 /*- 453 *----------------------------------------------------------------------- 454 * SuffRemove -- 455 * Remove the suffix into the list 456 * 457 * Results: 458 * None 459 * 460 * Side Effects: 461 * The reference count for the suffix is decremented and the 462 * suffix is possibly freed 463 *----------------------------------------------------------------------- 464 */ 465 static void 466 SuffRemove(Lst l, Suff *s) 467 { 468 SuffUnRef(l, s); 469 if (s->refCount == 0) { 470 SuffUnRef(sufflist, s); 471 SuffFree(s); 472 } 473 } 474 475 /*- 476 *----------------------------------------------------------------------- 477 * SuffInsert -- 478 * Insert the suffix into the list keeping the list ordered by suffix 479 * numbers. 480 * 481 * Input: 482 * l the list where in s should be inserted 483 * s the suffix to insert 484 * 485 * Results: 486 * None 487 * 488 * Side Effects: 489 * The reference count of the suffix is incremented 490 *----------------------------------------------------------------------- 491 */ 492 static void 493 SuffInsert(Lst l, Suff *s) 494 { 495 LstNode ln; /* current element in l we're examining */ 496 Suff *s2 = NULL; /* the suffix descriptor in this element */ 497 498 if (Lst_Open(l) == FAILURE) { 499 return; 500 } 501 while ((ln = Lst_Next(l)) != NULL) { 502 s2 = (Suff *)Lst_Datum(ln); 503 if (s2->sNum >= s->sNum) { 504 break; 505 } 506 } 507 508 Lst_Close(l); 509 if (DEBUG(SUFF)) { 510 fprintf(debug_file, "inserting %s(%d)...", s->name, s->sNum); 511 } 512 if (ln == NULL) { 513 if (DEBUG(SUFF)) { 514 fprintf(debug_file, "at end of list\n"); 515 } 516 (void)Lst_AtEnd(l, s); 517 s->refCount++; 518 (void)Lst_AtEnd(s->ref, l); 519 } else if (s2->sNum != s->sNum) { 520 if (DEBUG(SUFF)) { 521 fprintf(debug_file, "before %s(%d)\n", s2->name, s2->sNum); 522 } 523 (void)Lst_InsertBefore(l, ln, s); 524 s->refCount++; 525 (void)Lst_AtEnd(s->ref, l); 526 } else if (DEBUG(SUFF)) { 527 fprintf(debug_file, "already there\n"); 528 } 529 } 530 531 /*- 532 *----------------------------------------------------------------------- 533 * Suff_ClearSuffixes -- 534 * This is gross. Nuke the list of suffixes but keep all transformation 535 * rules around. The transformation graph is destroyed in this process, 536 * but we leave the list of rules so when a new graph is formed the rules 537 * will remain. 538 * This function is called from the parse module when a 539 * .SUFFIXES:\n line is encountered. 540 * 541 * Results: 542 * none 543 * 544 * Side Effects: 545 * the sufflist and its graph nodes are destroyed 546 *----------------------------------------------------------------------- 547 */ 548 void 549 Suff_ClearSuffixes(void) 550 { 551 #ifdef CLEANUP 552 Lst_Concat(suffClean, sufflist, LST_CONCLINK); 553 #endif 554 sufflist = Lst_Init(FALSE); 555 sNum = 0; 556 suffNull = emptySuff; 557 } 558 559 /*- 560 *----------------------------------------------------------------------- 561 * SuffParseTransform -- 562 * Parse a transformation string to find its two component suffixes. 563 * 564 * Input: 565 * str String being parsed 566 * srcPtr Place to store source of trans. 567 * targPtr Place to store target of trans. 568 * 569 * Results: 570 * TRUE if the string is a valid transformation and FALSE otherwise. 571 * 572 * Side Effects: 573 * The passed pointers are overwritten. 574 * 575 *----------------------------------------------------------------------- 576 */ 577 static Boolean 578 SuffParseTransform(char *str, Suff **srcPtr, Suff **targPtr) 579 { 580 LstNode srcLn; /* element in suffix list of trans source*/ 581 Suff *src; /* Source of transformation */ 582 LstNode targLn; /* element in suffix list of trans target*/ 583 char *str2; /* Extra pointer (maybe target suffix) */ 584 LstNode singleLn; /* element in suffix list of any suffix 585 * that exactly matches str */ 586 Suff *single = NULL;/* Source of possible transformation to 587 * null suffix */ 588 589 srcLn = NULL; 590 singleLn = NULL; 591 592 /* 593 * Loop looking first for a suffix that matches the start of the 594 * string and then for one that exactly matches the rest of it. If 595 * we can find two that meet these criteria, we've successfully 596 * parsed the string. 597 */ 598 for (;;) { 599 if (srcLn == NULL) { 600 srcLn = Lst_Find(sufflist, str, SuffSuffIsPrefix); 601 } else { 602 srcLn = Lst_FindFrom(sufflist, Lst_Succ(srcLn), str, 603 SuffSuffIsPrefix); 604 } 605 if (srcLn == NULL) { 606 /* 607 * Ran out of source suffixes -- no such rule 608 */ 609 if (singleLn != NULL) { 610 /* 611 * Not so fast Mr. Smith! There was a suffix that encompassed 612 * the entire string, so we assume it was a transformation 613 * to the null suffix (thank you POSIX). We still prefer to 614 * find a double rule over a singleton, hence we leave this 615 * check until the end. 616 * 617 * XXX: Use emptySuff over suffNull? 618 */ 619 *srcPtr = single; 620 *targPtr = suffNull; 621 return(TRUE); 622 } 623 return (FALSE); 624 } 625 src = (Suff *)Lst_Datum(srcLn); 626 str2 = str + src->nameLen; 627 if (*str2 == '\0') { 628 single = src; 629 singleLn = srcLn; 630 } else { 631 targLn = Lst_Find(sufflist, str2, SuffSuffHasNameP); 632 if (targLn != NULL) { 633 *srcPtr = src; 634 *targPtr = (Suff *)Lst_Datum(targLn); 635 return (TRUE); 636 } 637 } 638 } 639 } 640 641 /*- 642 *----------------------------------------------------------------------- 643 * Suff_IsTransform -- 644 * Return TRUE if the given string is a transformation rule 645 * 646 * 647 * Input: 648 * str string to check 649 * 650 * Results: 651 * TRUE if the string is a concatenation of two known suffixes. 652 * FALSE otherwise 653 * 654 * Side Effects: 655 * None 656 *----------------------------------------------------------------------- 657 */ 658 Boolean 659 Suff_IsTransform(char *str) 660 { 661 Suff *src, *targ; 662 663 return (SuffParseTransform(str, &src, &targ)); 664 } 665 666 /*- 667 *----------------------------------------------------------------------- 668 * Suff_AddTransform -- 669 * Add the transformation rule described by the line to the 670 * list of rules and place the transformation itself in the graph 671 * 672 * Input: 673 * line name of transformation to add 674 * 675 * Results: 676 * The node created for the transformation in the transforms list 677 * 678 * Side Effects: 679 * The node is placed on the end of the transforms Lst and links are 680 * made between the two suffixes mentioned in the target name 681 *----------------------------------------------------------------------- 682 */ 683 GNode * 684 Suff_AddTransform(char *line) 685 { 686 GNode *gn; /* GNode of transformation rule */ 687 Suff *s, /* source suffix */ 688 *t; /* target suffix */ 689 LstNode ln; /* Node for existing transformation */ 690 691 ln = Lst_Find(transforms, line, SuffGNHasNameP); 692 if (ln == NULL) { 693 /* 694 * Make a new graph node for the transformation. It will be filled in 695 * by the Parse module. 696 */ 697 gn = Targ_NewGN(line); 698 (void)Lst_AtEnd(transforms, gn); 699 } else { 700 /* 701 * New specification for transformation rule. Just nuke the old list 702 * of commands so they can be filled in again... We don't actually 703 * free the commands themselves, because a given command can be 704 * attached to several different transformations. 705 */ 706 gn = (GNode *)Lst_Datum(ln); 707 Lst_Destroy(gn->commands, NULL); 708 Lst_Destroy(gn->children, NULL); 709 gn->commands = Lst_Init(FALSE); 710 gn->children = Lst_Init(FALSE); 711 } 712 713 gn->type = OP_TRANSFORM; 714 715 (void)SuffParseTransform(line, &s, &t); 716 717 /* 718 * link the two together in the proper relationship and order 719 */ 720 if (DEBUG(SUFF)) { 721 fprintf(debug_file, "defining transformation from `%s' to `%s'\n", 722 s->name, t->name); 723 } 724 SuffInsert(t->children, s); 725 SuffInsert(s->parents, t); 726 727 return (gn); 728 } 729 730 /*- 731 *----------------------------------------------------------------------- 732 * Suff_EndTransform -- 733 * Handle the finish of a transformation definition, removing the 734 * transformation from the graph if it has neither commands nor 735 * sources. This is a callback procedure for the Parse module via 736 * Lst_ForEach 737 * 738 * Input: 739 * gnp Node for transformation 740 * dummy Node for transformation 741 * 742 * Results: 743 * === 0 744 * 745 * Side Effects: 746 * If the node has no commands or children, the children and parents 747 * lists of the affected suffixes are altered. 748 * 749 *----------------------------------------------------------------------- 750 */ 751 int 752 Suff_EndTransform(ClientData gnp, ClientData dummy) 753 { 754 GNode *gn = (GNode *)gnp; 755 756 if ((gn->type & OP_DOUBLEDEP) && !Lst_IsEmpty (gn->cohorts)) 757 gn = (GNode *)Lst_Datum(Lst_Last(gn->cohorts)); 758 if ((gn->type & OP_TRANSFORM) && Lst_IsEmpty(gn->commands) && 759 Lst_IsEmpty(gn->children)) 760 { 761 Suff *s, *t; 762 763 /* 764 * SuffParseTransform() may fail for special rules which are not 765 * actual transformation rules. (e.g. .DEFAULT) 766 */ 767 if (SuffParseTransform(gn->name, &s, &t)) { 768 Lst p; 769 770 if (DEBUG(SUFF)) { 771 fprintf(debug_file, "deleting transformation from `%s' to `%s'\n", 772 s->name, t->name); 773 } 774 775 /* 776 * Store s->parents because s could be deleted in SuffRemove 777 */ 778 p = s->parents; 779 780 /* 781 * Remove the source from the target's children list. We check for a 782 * nil return to handle a beanhead saying something like 783 * .c.o .c.o: 784 * 785 * We'll be called twice when the next target is seen, but .c and .o 786 * are only linked once... 787 */ 788 SuffRemove(t->children, s); 789 790 /* 791 * Remove the target from the source's parents list 792 */ 793 SuffRemove(p, t); 794 } 795 } else if ((gn->type & OP_TRANSFORM) && DEBUG(SUFF)) { 796 fprintf(debug_file, "transformation %s complete\n", gn->name); 797 } 798 799 return(dummy ? 0 : 0); 800 } 801 802 /*- 803 *----------------------------------------------------------------------- 804 * SuffRebuildGraph -- 805 * Called from Suff_AddSuffix via Lst_ForEach to search through the 806 * list of existing transformation rules and rebuild the transformation 807 * graph when it has been destroyed by Suff_ClearSuffixes. If the 808 * given rule is a transformation involving this suffix and another, 809 * existing suffix, the proper relationship is established between 810 * the two. 811 * 812 * Input: 813 * transformp Transformation to test 814 * sp Suffix to rebuild 815 * 816 * Results: 817 * Always 0. 818 * 819 * Side Effects: 820 * The appropriate links will be made between this suffix and 821 * others if transformation rules exist for it. 822 * 823 *----------------------------------------------------------------------- 824 */ 825 static int 826 SuffRebuildGraph(ClientData transformp, ClientData sp) 827 { 828 GNode *transform = (GNode *)transformp; 829 Suff *s = (Suff *)sp; 830 char *cp; 831 LstNode ln; 832 Suff *s2; 833 SuffixCmpData sd; 834 835 /* 836 * First see if it is a transformation from this suffix. 837 */ 838 cp = SuffStrIsPrefix(s->name, transform->name); 839 if (cp != NULL) { 840 ln = Lst_Find(sufflist, cp, SuffSuffHasNameP); 841 if (ln != NULL) { 842 /* 843 * Found target. Link in and return, since it can't be anything 844 * else. 845 */ 846 s2 = (Suff *)Lst_Datum(ln); 847 SuffInsert(s2->children, s); 848 SuffInsert(s->parents, s2); 849 return(0); 850 } 851 } 852 853 /* 854 * Not from, maybe to? 855 */ 856 sd.len = strlen(transform->name); 857 sd.ename = transform->name + sd.len; 858 cp = SuffSuffIsSuffix(s, &sd); 859 if (cp != NULL) { 860 /* 861 * Null-terminate the source suffix in order to find it. 862 */ 863 cp[1] = '\0'; 864 ln = Lst_Find(sufflist, transform->name, SuffSuffHasNameP); 865 /* 866 * Replace the start of the target suffix 867 */ 868 cp[1] = s->name[0]; 869 if (ln != NULL) { 870 /* 871 * Found it -- establish the proper relationship 872 */ 873 s2 = (Suff *)Lst_Datum(ln); 874 SuffInsert(s->children, s2); 875 SuffInsert(s2->parents, s); 876 } 877 } 878 return(0); 879 } 880 881 /*- 882 *----------------------------------------------------------------------- 883 * SuffScanTargets -- 884 * Called from Suff_AddSuffix via Lst_ForEach to search through the 885 * list of existing targets and find if any of the existing targets 886 * can be turned into a transformation rule. 887 * 888 * Results: 889 * 1 if a new main target has been selected, 0 otherwise. 890 * 891 * Side Effects: 892 * If such a target is found and the target is the current main 893 * target, the main target is set to NULL and the next target 894 * examined (if that exists) becomes the main target. 895 * 896 *----------------------------------------------------------------------- 897 */ 898 static int 899 SuffScanTargets(ClientData targetp, ClientData gsp) 900 { 901 GNode *target = (GNode *)targetp; 902 GNodeSuff *gs = (GNodeSuff *)gsp; 903 Suff *s, *t; 904 char *ptr; 905 906 if (*gs->gn == NULL && gs->r && (target->type & OP_NOTARGET) == 0) { 907 *gs->gn = target; 908 Targ_SetMain(target); 909 return 1; 910 } 911 912 if ((unsigned int)target->type == OP_TRANSFORM) 913 return 0; 914 915 if ((ptr = strstr(target->name, gs->s->name)) == NULL || 916 ptr == target->name) 917 return 0; 918 919 if (SuffParseTransform(target->name, &s, &t)) { 920 if (*gs->gn == target) { 921 gs->r = TRUE; 922 *gs->gn = NULL; 923 Targ_SetMain(NULL); 924 } 925 Lst_Destroy(target->children, NULL); 926 target->children = Lst_Init(FALSE); 927 target->type = OP_TRANSFORM; 928 /* 929 * link the two together in the proper relationship and order 930 */ 931 if (DEBUG(SUFF)) { 932 fprintf(debug_file, "defining transformation from `%s' to `%s'\n", 933 s->name, t->name); 934 } 935 SuffInsert(t->children, s); 936 SuffInsert(s->parents, t); 937 } 938 return 0; 939 } 940 941 /*- 942 *----------------------------------------------------------------------- 943 * Suff_AddSuffix -- 944 * Add the suffix in string to the end of the list of known suffixes. 945 * Should we restructure the suffix graph? Make doesn't... 946 * 947 * Input: 948 * str the name of the suffix to add 949 * 950 * Results: 951 * None 952 * 953 * Side Effects: 954 * A GNode is created for the suffix and a Suff structure is created and 955 * added to the suffixes list unless the suffix was already known. 956 * The mainNode passed can be modified if a target mutated into a 957 * transform and that target happened to be the main target. 958 *----------------------------------------------------------------------- 959 */ 960 void 961 Suff_AddSuffix(char *str, GNode **gn) 962 { 963 Suff *s; /* new suffix descriptor */ 964 LstNode ln; 965 GNodeSuff gs; 966 967 ln = Lst_Find(sufflist, str, SuffSuffHasNameP); 968 if (ln == NULL) { 969 s = bmake_malloc(sizeof(Suff)); 970 971 s->name = bmake_strdup(str); 972 s->nameLen = strlen(s->name); 973 s->searchPath = Lst_Init(FALSE); 974 s->children = Lst_Init(FALSE); 975 s->parents = Lst_Init(FALSE); 976 s->ref = Lst_Init(FALSE); 977 s->sNum = sNum++; 978 s->flags = 0; 979 s->refCount = 1; 980 981 (void)Lst_AtEnd(sufflist, s); 982 /* 983 * We also look at our existing targets list to see if adding 984 * this suffix will make one of our current targets mutate into 985 * a suffix rule. This is ugly, but other makes treat all targets 986 * that start with a . as suffix rules. 987 */ 988 gs.gn = gn; 989 gs.s = s; 990 gs.r = FALSE; 991 Lst_ForEach(Targ_List(), SuffScanTargets, &gs); 992 /* 993 * Look for any existing transformations from or to this suffix. 994 * XXX: Only do this after a Suff_ClearSuffixes? 995 */ 996 Lst_ForEach(transforms, SuffRebuildGraph, s); 997 } 998 } 999 1000 /*- 1001 *----------------------------------------------------------------------- 1002 * Suff_GetPath -- 1003 * Return the search path for the given suffix, if it's defined. 1004 * 1005 * Results: 1006 * The searchPath for the desired suffix or NULL if the suffix isn't 1007 * defined. 1008 * 1009 * Side Effects: 1010 * None 1011 *----------------------------------------------------------------------- 1012 */ 1013 Lst 1014 Suff_GetPath(char *sname) 1015 { 1016 LstNode ln; 1017 Suff *s; 1018 1019 ln = Lst_Find(sufflist, sname, SuffSuffHasNameP); 1020 if (ln == NULL) { 1021 return NULL; 1022 } else { 1023 s = (Suff *)Lst_Datum(ln); 1024 return (s->searchPath); 1025 } 1026 } 1027 1028 /*- 1029 *----------------------------------------------------------------------- 1030 * Suff_DoPaths -- 1031 * Extend the search paths for all suffixes to include the default 1032 * search path. 1033 * 1034 * Results: 1035 * None. 1036 * 1037 * Side Effects: 1038 * The searchPath field of all the suffixes is extended by the 1039 * directories in dirSearchPath. If paths were specified for the 1040 * ".h" suffix, the directories are stuffed into a global variable 1041 * called ".INCLUDES" with each directory preceded by a -I. The same 1042 * is done for the ".a" suffix, except the variable is called 1043 * ".LIBS" and the flag is -L. 1044 *----------------------------------------------------------------------- 1045 */ 1046 void 1047 Suff_DoPaths(void) 1048 { 1049 Suff *s; 1050 LstNode ln; 1051 char *ptr; 1052 Lst inIncludes; /* Cumulative .INCLUDES path */ 1053 Lst inLibs; /* Cumulative .LIBS path */ 1054 1055 if (Lst_Open(sufflist) == FAILURE) { 1056 return; 1057 } 1058 1059 inIncludes = Lst_Init(FALSE); 1060 inLibs = Lst_Init(FALSE); 1061 1062 while ((ln = Lst_Next(sufflist)) != NULL) { 1063 s = (Suff *)Lst_Datum(ln); 1064 if (!Lst_IsEmpty (s->searchPath)) { 1065 #ifdef INCLUDES 1066 if (s->flags & SUFF_INCLUDE) { 1067 Dir_Concat(inIncludes, s->searchPath); 1068 } 1069 #endif /* INCLUDES */ 1070 #ifdef LIBRARIES 1071 if (s->flags & SUFF_LIBRARY) { 1072 Dir_Concat(inLibs, s->searchPath); 1073 } 1074 #endif /* LIBRARIES */ 1075 Dir_Concat(s->searchPath, dirSearchPath); 1076 } else { 1077 Lst_Destroy(s->searchPath, Dir_Destroy); 1078 s->searchPath = Lst_Duplicate(dirSearchPath, Dir_CopyDir); 1079 } 1080 } 1081 1082 Var_Set(".INCLUDES", ptr = Dir_MakeFlags("-I", inIncludes), VAR_GLOBAL, 0); 1083 free(ptr); 1084 Var_Set(".LIBS", ptr = Dir_MakeFlags("-L", inLibs), VAR_GLOBAL, 0); 1085 free(ptr); 1086 1087 Lst_Destroy(inIncludes, Dir_Destroy); 1088 Lst_Destroy(inLibs, Dir_Destroy); 1089 1090 Lst_Close(sufflist); 1091 } 1092 1093 /*- 1094 *----------------------------------------------------------------------- 1095 * Suff_AddInclude -- 1096 * Add the given suffix as a type of file which gets included. 1097 * Called from the parse module when a .INCLUDES line is parsed. 1098 * The suffix must have already been defined. 1099 * 1100 * Input: 1101 * sname Name of the suffix to mark 1102 * 1103 * Results: 1104 * None. 1105 * 1106 * Side Effects: 1107 * The SUFF_INCLUDE bit is set in the suffix's flags field 1108 * 1109 *----------------------------------------------------------------------- 1110 */ 1111 void 1112 Suff_AddInclude(char *sname) 1113 { 1114 LstNode ln; 1115 Suff *s; 1116 1117 ln = Lst_Find(sufflist, sname, SuffSuffHasNameP); 1118 if (ln != NULL) { 1119 s = (Suff *)Lst_Datum(ln); 1120 s->flags |= SUFF_INCLUDE; 1121 } 1122 } 1123 1124 /*- 1125 *----------------------------------------------------------------------- 1126 * Suff_AddLib -- 1127 * Add the given suffix as a type of file which is a library. 1128 * Called from the parse module when parsing a .LIBS line. The 1129 * suffix must have been defined via .SUFFIXES before this is 1130 * called. 1131 * 1132 * Input: 1133 * sname Name of the suffix to mark 1134 * 1135 * Results: 1136 * None. 1137 * 1138 * Side Effects: 1139 * The SUFF_LIBRARY bit is set in the suffix's flags field 1140 * 1141 *----------------------------------------------------------------------- 1142 */ 1143 void 1144 Suff_AddLib(char *sname) 1145 { 1146 LstNode ln; 1147 Suff *s; 1148 1149 ln = Lst_Find(sufflist, sname, SuffSuffHasNameP); 1150 if (ln != NULL) { 1151 s = (Suff *)Lst_Datum(ln); 1152 s->flags |= SUFF_LIBRARY; 1153 } 1154 } 1155 1156 /********** Implicit Source Search Functions *********/ 1157 1158 /*- 1159 *----------------------------------------------------------------------- 1160 * SuffAddSrc -- 1161 * Add a suffix as a Src structure to the given list with its parent 1162 * being the given Src structure. If the suffix is the null suffix, 1163 * the prefix is used unaltered as the file name in the Src structure. 1164 * 1165 * Input: 1166 * sp suffix for which to create a Src structure 1167 * lsp list and parent for the new Src 1168 * 1169 * Results: 1170 * always returns 0 1171 * 1172 * Side Effects: 1173 * A Src structure is created and tacked onto the end of the list 1174 *----------------------------------------------------------------------- 1175 */ 1176 static int 1177 SuffAddSrc(ClientData sp, ClientData lsp) 1178 { 1179 Suff *s = (Suff *)sp; 1180 LstSrc *ls = (LstSrc *)lsp; 1181 Src *s2; /* new Src structure */ 1182 Src *targ; /* Target structure */ 1183 1184 targ = ls->s; 1185 1186 if ((s->flags & SUFF_NULL) && (*s->name != '\0')) { 1187 /* 1188 * If the suffix has been marked as the NULL suffix, also create a Src 1189 * structure for a file with no suffix attached. Two birds, and all 1190 * that... 1191 */ 1192 s2 = bmake_malloc(sizeof(Src)); 1193 s2->file = bmake_strdup(targ->pref); 1194 s2->pref = targ->pref; 1195 s2->parent = targ; 1196 s2->node = NULL; 1197 s2->suff = s; 1198 s->refCount++; 1199 s2->children = 0; 1200 targ->children += 1; 1201 (void)Lst_AtEnd(ls->l, s2); 1202 #ifdef DEBUG_SRC 1203 s2->cp = Lst_Init(FALSE); 1204 Lst_AtEnd(targ->cp, s2); 1205 fprintf(debug_file, "1 add %x %x to %x:", targ, s2, ls->l); 1206 Lst_ForEach(ls->l, PrintAddr, NULL); 1207 fprintf(debug_file, "\n"); 1208 #endif 1209 } 1210 s2 = bmake_malloc(sizeof(Src)); 1211 s2->file = str_concat(targ->pref, s->name, 0); 1212 s2->pref = targ->pref; 1213 s2->parent = targ; 1214 s2->node = NULL; 1215 s2->suff = s; 1216 s->refCount++; 1217 s2->children = 0; 1218 targ->children += 1; 1219 (void)Lst_AtEnd(ls->l, s2); 1220 #ifdef DEBUG_SRC 1221 s2->cp = Lst_Init(FALSE); 1222 Lst_AtEnd(targ->cp, s2); 1223 fprintf(debug_file, "2 add %x %x to %x:", targ, s2, ls->l); 1224 Lst_ForEach(ls->l, PrintAddr, NULL); 1225 fprintf(debug_file, "\n"); 1226 #endif 1227 1228 return(0); 1229 } 1230 1231 /*- 1232 *----------------------------------------------------------------------- 1233 * SuffAddLevel -- 1234 * Add all the children of targ as Src structures to the given list 1235 * 1236 * Input: 1237 * l list to which to add the new level 1238 * targ Src structure to use as the parent 1239 * 1240 * Results: 1241 * None 1242 * 1243 * Side Effects: 1244 * Lots of structures are created and added to the list 1245 *----------------------------------------------------------------------- 1246 */ 1247 static void 1248 SuffAddLevel(Lst l, Src *targ) 1249 { 1250 LstSrc ls; 1251 1252 ls.s = targ; 1253 ls.l = l; 1254 1255 Lst_ForEach(targ->suff->children, SuffAddSrc, &ls); 1256 } 1257 1258 /*- 1259 *---------------------------------------------------------------------- 1260 * SuffRemoveSrc -- 1261 * Free all src structures in list that don't have a reference count 1262 * 1263 * Results: 1264 * Ture if an src was removed 1265 * 1266 * Side Effects: 1267 * The memory is free'd. 1268 *---------------------------------------------------------------------- 1269 */ 1270 static int 1271 SuffRemoveSrc(Lst l) 1272 { 1273 LstNode ln; 1274 Src *s; 1275 int t = 0; 1276 1277 if (Lst_Open(l) == FAILURE) { 1278 return 0; 1279 } 1280 #ifdef DEBUG_SRC 1281 fprintf(debug_file, "cleaning %lx: ", (unsigned long) l); 1282 Lst_ForEach(l, PrintAddr, NULL); 1283 fprintf(debug_file, "\n"); 1284 #endif 1285 1286 1287 while ((ln = Lst_Next(l)) != NULL) { 1288 s = (Src *)Lst_Datum(ln); 1289 if (s->children == 0) { 1290 free(s->file); 1291 if (!s->parent) 1292 free(s->pref); 1293 else { 1294 #ifdef DEBUG_SRC 1295 LstNode ln = Lst_Member(s->parent->cp, s); 1296 if (ln != NULL) 1297 Lst_Remove(s->parent->cp, ln); 1298 #endif 1299 --s->parent->children; 1300 } 1301 #ifdef DEBUG_SRC 1302 fprintf(debug_file, "free: [l=%x] p=%x %d\n", l, s, s->children); 1303 Lst_Destroy(s->cp, NULL); 1304 #endif 1305 Lst_Remove(l, ln); 1306 free(s); 1307 t |= 1; 1308 Lst_Close(l); 1309 return TRUE; 1310 } 1311 #ifdef DEBUG_SRC 1312 else { 1313 fprintf(debug_file, "keep: [l=%x] p=%x %d: ", l, s, s->children); 1314 Lst_ForEach(s->cp, PrintAddr, NULL); 1315 fprintf(debug_file, "\n"); 1316 } 1317 #endif 1318 } 1319 1320 Lst_Close(l); 1321 1322 return t; 1323 } 1324 1325 /*- 1326 *----------------------------------------------------------------------- 1327 * SuffFindThem -- 1328 * Find the first existing file/target in the list srcs 1329 * 1330 * Input: 1331 * srcs list of Src structures to search through 1332 * 1333 * Results: 1334 * The lowest structure in the chain of transformations 1335 * 1336 * Side Effects: 1337 * None 1338 *----------------------------------------------------------------------- 1339 */ 1340 static Src * 1341 SuffFindThem(Lst srcs, Lst slst) 1342 { 1343 Src *s; /* current Src */ 1344 Src *rs; /* returned Src */ 1345 char *ptr; 1346 1347 rs = NULL; 1348 1349 while (!Lst_IsEmpty (srcs)) { 1350 s = (Src *)Lst_DeQueue(srcs); 1351 1352 if (DEBUG(SUFF)) { 1353 fprintf(debug_file, "\ttrying %s...", s->file); 1354 } 1355 1356 /* 1357 * A file is considered to exist if either a node exists in the 1358 * graph for it or the file actually exists. 1359 */ 1360 if (Targ_FindNode(s->file, TARG_NOCREATE) != NULL) { 1361 #ifdef DEBUG_SRC 1362 fprintf(debug_file, "remove %x from %x\n", s, srcs); 1363 #endif 1364 rs = s; 1365 break; 1366 } 1367 1368 if ((ptr = Dir_FindFile(s->file, s->suff->searchPath)) != NULL) { 1369 rs = s; 1370 #ifdef DEBUG_SRC 1371 fprintf(debug_file, "remove %x from %x\n", s, srcs); 1372 #endif 1373 free(ptr); 1374 break; 1375 } 1376 1377 if (DEBUG(SUFF)) { 1378 fprintf(debug_file, "not there\n"); 1379 } 1380 1381 SuffAddLevel(srcs, s); 1382 Lst_AtEnd(slst, s); 1383 } 1384 1385 if (DEBUG(SUFF) && rs) { 1386 fprintf(debug_file, "got it\n"); 1387 } 1388 return (rs); 1389 } 1390 1391 /*- 1392 *----------------------------------------------------------------------- 1393 * SuffFindCmds -- 1394 * See if any of the children of the target in the Src structure is 1395 * one from which the target can be transformed. If there is one, 1396 * a Src structure is put together for it and returned. 1397 * 1398 * Input: 1399 * targ Src structure to play with 1400 * 1401 * Results: 1402 * The Src structure of the "winning" child, or NULL if no such beast. 1403 * 1404 * Side Effects: 1405 * A Src structure may be allocated. 1406 * 1407 *----------------------------------------------------------------------- 1408 */ 1409 static Src * 1410 SuffFindCmds(Src *targ, Lst slst) 1411 { 1412 LstNode ln; /* General-purpose list node */ 1413 GNode *t, /* Target GNode */ 1414 *s; /* Source GNode */ 1415 int prefLen;/* The length of the defined prefix */ 1416 Suff *suff; /* Suffix on matching beastie */ 1417 Src *ret; /* Return value */ 1418 char *cp; 1419 1420 t = targ->node; 1421 (void)Lst_Open(t->children); 1422 prefLen = strlen(targ->pref); 1423 1424 for (;;) { 1425 ln = Lst_Next(t->children); 1426 if (ln == NULL) { 1427 Lst_Close(t->children); 1428 return NULL; 1429 } 1430 s = (GNode *)Lst_Datum(ln); 1431 1432 if (s->type & OP_OPTIONAL && Lst_IsEmpty(t->commands)) { 1433 /* 1434 * We haven't looked to see if .OPTIONAL files exist yet, so 1435 * don't use one as the implicit source. 1436 * This allows us to use .OPTIONAL in .depend files so make won't 1437 * complain "don't know how to make xxx.h' when a dependant file 1438 * has been moved/deleted. 1439 */ 1440 continue; 1441 } 1442 1443 cp = strrchr(s->name, '/'); 1444 if (cp == NULL) { 1445 cp = s->name; 1446 } else { 1447 cp++; 1448 } 1449 if (strncmp(cp, targ->pref, prefLen) != 0) 1450 continue; 1451 /* 1452 * The node matches the prefix ok, see if it has a known 1453 * suffix. 1454 */ 1455 ln = Lst_Find(sufflist, &cp[prefLen], 1456 SuffSuffHasNameP); 1457 if (ln == NULL) 1458 continue; 1459 /* 1460 * It even has a known suffix, see if there's a transformation 1461 * defined between the node's suffix and the target's suffix. 1462 * 1463 * XXX: Handle multi-stage transformations here, too. 1464 */ 1465 suff = (Suff *)Lst_Datum(ln); 1466 1467 if (Lst_Member(suff->parents, targ->suff) != NULL) 1468 break; 1469 } 1470 1471 /* 1472 * Hot Damn! Create a new Src structure to describe 1473 * this transformation (making sure to duplicate the 1474 * source node's name so Suff_FindDeps can free it 1475 * again (ick)), and return the new structure. 1476 */ 1477 ret = bmake_malloc(sizeof(Src)); 1478 ret->file = bmake_strdup(s->name); 1479 ret->pref = targ->pref; 1480 ret->suff = suff; 1481 suff->refCount++; 1482 ret->parent = targ; 1483 ret->node = s; 1484 ret->children = 0; 1485 targ->children += 1; 1486 #ifdef DEBUG_SRC 1487 ret->cp = Lst_Init(FALSE); 1488 fprintf(debug_file, "3 add %x %x\n", targ, ret); 1489 Lst_AtEnd(targ->cp, ret); 1490 #endif 1491 Lst_AtEnd(slst, ret); 1492 if (DEBUG(SUFF)) { 1493 fprintf(debug_file, "\tusing existing source %s\n", s->name); 1494 } 1495 return (ret); 1496 } 1497 1498 /*- 1499 *----------------------------------------------------------------------- 1500 * SuffExpandChildren -- 1501 * Expand the names of any children of a given node that contain 1502 * variable invocations or file wildcards into actual targets. 1503 * 1504 * Input: 1505 * cln Child to examine 1506 * pgn Parent node being processed 1507 * 1508 * Results: 1509 * === 0 (continue) 1510 * 1511 * Side Effects: 1512 * The expanded node is removed from the parent's list of children, 1513 * and the parent's unmade counter is decremented, but other nodes 1514 * may be added. 1515 * 1516 *----------------------------------------------------------------------- 1517 */ 1518 static void 1519 SuffExpandChildren(LstNode cln, GNode *pgn) 1520 { 1521 GNode *cgn = (GNode *)Lst_Datum(cln); 1522 GNode *gn; /* New source 8) */ 1523 char *cp; /* Expanded value */ 1524 1525 if (!Lst_IsEmpty(cgn->order_pred) || !Lst_IsEmpty(cgn->order_succ)) 1526 /* It is all too hard to process the result of .ORDER */ 1527 return; 1528 1529 if (cgn->type & OP_WAIT) 1530 /* Ignore these (& OP_PHONY ?) */ 1531 return; 1532 1533 /* 1534 * First do variable expansion -- this takes precedence over 1535 * wildcard expansion. If the result contains wildcards, they'll be gotten 1536 * to later since the resulting words are tacked on to the end of 1537 * the children list. 1538 */ 1539 if (strchr(cgn->name, '$') == NULL) { 1540 SuffExpandWildcards(cln, pgn); 1541 return; 1542 } 1543 1544 if (DEBUG(SUFF)) { 1545 fprintf(debug_file, "Expanding \"%s\"...", cgn->name); 1546 } 1547 cp = Var_Subst(NULL, cgn->name, pgn, TRUE); 1548 1549 if (cp != NULL) { 1550 Lst members = Lst_Init(FALSE); 1551 1552 if (cgn->type & OP_ARCHV) { 1553 /* 1554 * Node was an archive(member) target, so we want to call 1555 * on the Arch module to find the nodes for us, expanding 1556 * variables in the parent's context. 1557 */ 1558 char *sacrifice = cp; 1559 1560 (void)Arch_ParseArchive(&sacrifice, members, pgn); 1561 } else { 1562 /* 1563 * Break the result into a vector of strings whose nodes 1564 * we can find, then add those nodes to the members list. 1565 * Unfortunately, we can't use brk_string b/c it 1566 * doesn't understand about variable specifications with 1567 * spaces in them... 1568 */ 1569 char *start; 1570 char *initcp = cp; /* For freeing... */ 1571 1572 for (start = cp; *start == ' ' || *start == '\t'; start++) 1573 continue; 1574 for (cp = start; *cp != '\0'; cp++) { 1575 if (*cp == ' ' || *cp == '\t') { 1576 /* 1577 * White-space -- terminate element, find the node, 1578 * add it, skip any further spaces. 1579 */ 1580 *cp++ = '\0'; 1581 gn = Targ_FindNode(start, TARG_CREATE); 1582 (void)Lst_AtEnd(members, gn); 1583 while (*cp == ' ' || *cp == '\t') { 1584 cp++; 1585 } 1586 /* 1587 * Adjust cp for increment at start of loop, but 1588 * set start to first non-space. 1589 */ 1590 start = cp--; 1591 } else if (*cp == '$') { 1592 /* 1593 * Start of a variable spec -- contact variable module 1594 * to find the end so we can skip over it. 1595 */ 1596 char *junk; 1597 int len; 1598 void *freeIt; 1599 1600 junk = Var_Parse(cp, pgn, TRUE, &len, &freeIt); 1601 if (junk != var_Error) { 1602 cp += len - 1; 1603 } 1604 1605 if (freeIt) 1606 free(freeIt); 1607 } else if (*cp == '\\' && *cp != '\0') { 1608 /* 1609 * Escaped something -- skip over it 1610 */ 1611 cp++; 1612 } 1613 } 1614 1615 if (cp != start) { 1616 /* 1617 * Stuff left over -- add it to the list too 1618 */ 1619 gn = Targ_FindNode(start, TARG_CREATE); 1620 (void)Lst_AtEnd(members, gn); 1621 } 1622 /* 1623 * Point cp back at the beginning again so the variable value 1624 * can be freed. 1625 */ 1626 cp = initcp; 1627 } 1628 1629 /* 1630 * Add all elements of the members list to the parent node. 1631 */ 1632 while(!Lst_IsEmpty(members)) { 1633 gn = (GNode *)Lst_DeQueue(members); 1634 1635 if (DEBUG(SUFF)) { 1636 fprintf(debug_file, "%s...", gn->name); 1637 } 1638 /* Add gn to the parents child list before the original child */ 1639 (void)Lst_InsertBefore(pgn->children, cln, gn); 1640 (void)Lst_AtEnd(gn->parents, pgn); 1641 pgn->unmade++; 1642 /* Expand wildcards on new node */ 1643 SuffExpandWildcards(Lst_Prev(cln), pgn); 1644 } 1645 Lst_Destroy(members, NULL); 1646 1647 /* 1648 * Free the result 1649 */ 1650 free(cp); 1651 } 1652 if (DEBUG(SUFF)) { 1653 fprintf(debug_file, "\n"); 1654 } 1655 1656 /* 1657 * Now the source is expanded, remove it from the list of children to 1658 * keep it from being processed. 1659 */ 1660 pgn->unmade--; 1661 Lst_Remove(pgn->children, cln); 1662 Lst_Remove(cgn->parents, Lst_Member(cgn->parents, pgn)); 1663 } 1664 1665 static void 1666 SuffExpandWildcards(LstNode cln, GNode *pgn) 1667 { 1668 GNode *cgn = (GNode *)Lst_Datum(cln); 1669 GNode *gn; /* New source 8) */ 1670 char *cp; /* Expanded value */ 1671 Lst explist; /* List of expansions */ 1672 1673 if (!Dir_HasWildcards(cgn->name)) 1674 return; 1675 1676 /* 1677 * Expand the word along the chosen path 1678 */ 1679 explist = Lst_Init(FALSE); 1680 Dir_Expand(cgn->name, Suff_FindPath(cgn), explist); 1681 1682 while (!Lst_IsEmpty(explist)) { 1683 /* 1684 * Fetch next expansion off the list and find its GNode 1685 */ 1686 cp = (char *)Lst_DeQueue(explist); 1687 1688 if (DEBUG(SUFF)) { 1689 fprintf(debug_file, "%s...", cp); 1690 } 1691 gn = Targ_FindNode(cp, TARG_CREATE); 1692 1693 /* Add gn to the parents child list before the original child */ 1694 (void)Lst_InsertBefore(pgn->children, cln, gn); 1695 (void)Lst_AtEnd(gn->parents, pgn); 1696 pgn->unmade++; 1697 } 1698 1699 /* 1700 * Nuke what's left of the list 1701 */ 1702 Lst_Destroy(explist, NULL); 1703 1704 if (DEBUG(SUFF)) { 1705 fprintf(debug_file, "\n"); 1706 } 1707 1708 /* 1709 * Now the source is expanded, remove it from the list of children to 1710 * keep it from being processed. 1711 */ 1712 pgn->unmade--; 1713 Lst_Remove(pgn->children, cln); 1714 Lst_Remove(cgn->parents, Lst_Member(cgn->parents, pgn)); 1715 } 1716 1717 /*- 1718 *----------------------------------------------------------------------- 1719 * Suff_FindPath -- 1720 * Find a path along which to expand the node. 1721 * 1722 * If the word has a known suffix, use that path. 1723 * If it has no known suffix, use the default system search path. 1724 * 1725 * Input: 1726 * gn Node being examined 1727 * 1728 * Results: 1729 * The appropriate path to search for the GNode. 1730 * 1731 * Side Effects: 1732 * XXX: We could set the suffix here so that we don't have to scan 1733 * again. 1734 * 1735 *----------------------------------------------------------------------- 1736 */ 1737 Lst 1738 Suff_FindPath(GNode* gn) 1739 { 1740 Suff *suff = gn->suffix; 1741 1742 if (suff == NULL) { 1743 SuffixCmpData sd; /* Search string data */ 1744 LstNode ln; 1745 sd.len = strlen(gn->name); 1746 sd.ename = gn->name + sd.len; 1747 ln = Lst_Find(sufflist, &sd, SuffSuffIsSuffixP); 1748 1749 if (DEBUG(SUFF)) { 1750 fprintf(debug_file, "Wildcard expanding \"%s\"...", gn->name); 1751 } 1752 if (ln != NULL) 1753 suff = (Suff *)Lst_Datum(ln); 1754 /* XXX: Here we can save the suffix so we don't have to do this again */ 1755 } 1756 1757 if (suff != NULL) { 1758 if (DEBUG(SUFF)) { 1759 fprintf(debug_file, "suffix is \"%s\"...", suff->name); 1760 } 1761 return suff->searchPath; 1762 } else { 1763 /* 1764 * Use default search path 1765 */ 1766 return dirSearchPath; 1767 } 1768 } 1769 1770 /*- 1771 *----------------------------------------------------------------------- 1772 * SuffApplyTransform -- 1773 * Apply a transformation rule, given the source and target nodes 1774 * and suffixes. 1775 * 1776 * Input: 1777 * tGn Target node 1778 * sGn Source node 1779 * t Target suffix 1780 * s Source suffix 1781 * 1782 * Results: 1783 * TRUE if successful, FALSE if not. 1784 * 1785 * Side Effects: 1786 * The source and target are linked and the commands from the 1787 * transformation are added to the target node's commands list. 1788 * All attributes but OP_DEPMASK and OP_TRANSFORM are applied 1789 * to the target. The target also inherits all the sources for 1790 * the transformation rule. 1791 * 1792 *----------------------------------------------------------------------- 1793 */ 1794 static Boolean 1795 SuffApplyTransform(GNode *tGn, GNode *sGn, Suff *t, Suff *s) 1796 { 1797 LstNode ln, nln; /* General node */ 1798 char *tname; /* Name of transformation rule */ 1799 GNode *gn; /* Node for same */ 1800 1801 /* 1802 * Form the proper links between the target and source. 1803 */ 1804 (void)Lst_AtEnd(tGn->children, sGn); 1805 (void)Lst_AtEnd(sGn->parents, tGn); 1806 tGn->unmade += 1; 1807 1808 /* 1809 * Locate the transformation rule itself 1810 */ 1811 tname = str_concat(s->name, t->name, 0); 1812 ln = Lst_Find(transforms, tname, SuffGNHasNameP); 1813 free(tname); 1814 1815 if (ln == NULL) { 1816 /* 1817 * Not really such a transformation rule (can happen when we're 1818 * called to link an OP_MEMBER and OP_ARCHV node), so return 1819 * FALSE. 1820 */ 1821 return(FALSE); 1822 } 1823 1824 gn = (GNode *)Lst_Datum(ln); 1825 1826 if (DEBUG(SUFF)) { 1827 fprintf(debug_file, "\tapplying %s -> %s to \"%s\"\n", s->name, t->name, tGn->name); 1828 } 1829 1830 /* 1831 * Record last child for expansion purposes 1832 */ 1833 ln = Lst_Last(tGn->children); 1834 1835 /* 1836 * Pass the buck to Make_HandleUse to apply the rule 1837 */ 1838 (void)Make_HandleUse(gn, tGn); 1839 1840 /* 1841 * Deal with wildcards and variables in any acquired sources 1842 */ 1843 for (ln = Lst_Succ(ln); ln != NULL; ln = nln) { 1844 nln = Lst_Succ(ln); 1845 SuffExpandChildren(ln, tGn); 1846 } 1847 1848 /* 1849 * Keep track of another parent to which this beast is transformed so 1850 * the .IMPSRC variable can be set correctly for the parent. 1851 */ 1852 (void)Lst_AtEnd(sGn->iParents, tGn); 1853 1854 return(TRUE); 1855 } 1856 1857 1858 /*- 1859 *----------------------------------------------------------------------- 1860 * SuffFindArchiveDeps -- 1861 * Locate dependencies for an OP_ARCHV node. 1862 * 1863 * Input: 1864 * gn Node for which to locate dependencies 1865 * 1866 * Results: 1867 * None 1868 * 1869 * Side Effects: 1870 * Same as Suff_FindDeps 1871 * 1872 *----------------------------------------------------------------------- 1873 */ 1874 static void 1875 SuffFindArchiveDeps(GNode *gn, Lst slst) 1876 { 1877 char *eoarch; /* End of archive portion */ 1878 char *eoname; /* End of member portion */ 1879 GNode *mem; /* Node for member */ 1880 static const char *copy[] = { 1881 /* Variables to be copied from the member node */ 1882 TARGET, /* Must be first */ 1883 PREFIX, /* Must be second */ 1884 }; 1885 int i; /* Index into copy and vals */ 1886 Suff *ms; /* Suffix descriptor for member */ 1887 char *name; /* Start of member's name */ 1888 1889 /* 1890 * The node is an archive(member) pair. so we must find a 1891 * suffix for both of them. 1892 */ 1893 eoarch = strchr(gn->name, '('); 1894 eoname = strchr(eoarch, ')'); 1895 1896 *eoname = '\0'; /* Nuke parentheses during suffix search */ 1897 *eoarch = '\0'; /* So a suffix can be found */ 1898 1899 name = eoarch + 1; 1900 1901 /* 1902 * To simplify things, call Suff_FindDeps recursively on the member now, 1903 * so we can simply compare the member's .PREFIX and .TARGET variables 1904 * to locate its suffix. This allows us to figure out the suffix to 1905 * use for the archive without having to do a quadratic search over the 1906 * suffix list, backtracking for each one... 1907 */ 1908 mem = Targ_FindNode(name, TARG_CREATE); 1909 SuffFindDeps(mem, slst); 1910 1911 /* 1912 * Create the link between the two nodes right off 1913 */ 1914 (void)Lst_AtEnd(gn->children, mem); 1915 (void)Lst_AtEnd(mem->parents, gn); 1916 gn->unmade += 1; 1917 1918 /* 1919 * Copy in the variables from the member node to this one. 1920 */ 1921 for (i = (sizeof(copy)/sizeof(copy[0]))-1; i >= 0; i--) { 1922 char *p1; 1923 Var_Set(copy[i], Var_Value(copy[i], mem, &p1), gn, 0); 1924 if (p1) 1925 free(p1); 1926 1927 } 1928 1929 ms = mem->suffix; 1930 if (ms == NULL) { 1931 /* 1932 * Didn't know what it was -- use .NULL suffix if not in make mode 1933 */ 1934 if (DEBUG(SUFF)) { 1935 fprintf(debug_file, "using null suffix\n"); 1936 } 1937 ms = suffNull; 1938 } 1939 1940 1941 /* 1942 * Set the other two local variables required for this target. 1943 */ 1944 Var_Set(MEMBER, name, gn, 0); 1945 Var_Set(ARCHIVE, gn->name, gn, 0); 1946 1947 if (ms != NULL) { 1948 /* 1949 * Member has a known suffix, so look for a transformation rule from 1950 * it to a possible suffix of the archive. Rather than searching 1951 * through the entire list, we just look at suffixes to which the 1952 * member's suffix may be transformed... 1953 */ 1954 LstNode ln; 1955 SuffixCmpData sd; /* Search string data */ 1956 1957 /* 1958 * Use first matching suffix... 1959 */ 1960 sd.len = eoarch - gn->name; 1961 sd.ename = eoarch; 1962 ln = Lst_Find(ms->parents, &sd, SuffSuffIsSuffixP); 1963 1964 if (ln != NULL) { 1965 /* 1966 * Got one -- apply it 1967 */ 1968 if (!SuffApplyTransform(gn, mem, (Suff *)Lst_Datum(ln), ms) && 1969 DEBUG(SUFF)) 1970 { 1971 fprintf(debug_file, "\tNo transformation from %s -> %s\n", 1972 ms->name, ((Suff *)Lst_Datum(ln))->name); 1973 } 1974 } 1975 } 1976 1977 /* 1978 * Replace the opening and closing parens now we've no need of the separate 1979 * pieces. 1980 */ 1981 *eoarch = '('; *eoname = ')'; 1982 1983 /* 1984 * Pretend gn appeared to the left of a dependency operator so 1985 * the user needn't provide a transformation from the member to the 1986 * archive. 1987 */ 1988 if (OP_NOP(gn->type)) { 1989 gn->type |= OP_DEPENDS; 1990 } 1991 1992 /* 1993 * Flag the member as such so we remember to look in the archive for 1994 * its modification time. 1995 */ 1996 mem->type |= OP_MEMBER; 1997 } 1998 1999 /*- 2000 *----------------------------------------------------------------------- 2001 * SuffFindNormalDeps -- 2002 * Locate implicit dependencies for regular targets. 2003 * 2004 * Input: 2005 * gn Node for which to find sources 2006 * 2007 * Results: 2008 * None. 2009 * 2010 * Side Effects: 2011 * Same as Suff_FindDeps... 2012 * 2013 *----------------------------------------------------------------------- 2014 */ 2015 static void 2016 SuffFindNormalDeps(GNode *gn, Lst slst) 2017 { 2018 char *eoname; /* End of name */ 2019 char *sopref; /* Start of prefix */ 2020 LstNode ln, nln; /* Next suffix node to check */ 2021 Lst srcs; /* List of sources at which to look */ 2022 Lst targs; /* List of targets to which things can be 2023 * transformed. They all have the same file, 2024 * but different suff and pref fields */ 2025 Src *bottom; /* Start of found transformation path */ 2026 Src *src; /* General Src pointer */ 2027 char *pref; /* Prefix to use */ 2028 Src *targ; /* General Src target pointer */ 2029 SuffixCmpData sd; /* Search string data */ 2030 2031 2032 sd.len = strlen(gn->name); 2033 sd.ename = eoname = gn->name + sd.len; 2034 2035 sopref = gn->name; 2036 2037 /* 2038 * Begin at the beginning... 2039 */ 2040 ln = Lst_First(sufflist); 2041 srcs = Lst_Init(FALSE); 2042 targs = Lst_Init(FALSE); 2043 2044 /* 2045 * We're caught in a catch-22 here. On the one hand, we want to use any 2046 * transformation implied by the target's sources, but we can't examine 2047 * the sources until we've expanded any variables/wildcards they may hold, 2048 * and we can't do that until we've set up the target's local variables 2049 * and we can't do that until we know what the proper suffix for the 2050 * target is (in case there are two suffixes one of which is a suffix of 2051 * the other) and we can't know that until we've found its implied 2052 * source, which we may not want to use if there's an existing source 2053 * that implies a different transformation. 2054 * 2055 * In an attempt to get around this, which may not work all the time, 2056 * but should work most of the time, we look for implied sources first, 2057 * checking transformations to all possible suffixes of the target, 2058 * use what we find to set the target's local variables, expand the 2059 * children, then look for any overriding transformations they imply. 2060 * Should we find one, we discard the one we found before. 2061 */ 2062 2063 while (ln != NULL) { 2064 /* 2065 * Look for next possible suffix... 2066 */ 2067 ln = Lst_FindFrom(sufflist, ln, &sd, SuffSuffIsSuffixP); 2068 2069 if (ln != NULL) { 2070 int prefLen; /* Length of the prefix */ 2071 2072 /* 2073 * Allocate a Src structure to which things can be transformed 2074 */ 2075 targ = bmake_malloc(sizeof(Src)); 2076 targ->file = bmake_strdup(gn->name); 2077 targ->suff = (Suff *)Lst_Datum(ln); 2078 targ->suff->refCount++; 2079 targ->node = gn; 2080 targ->parent = NULL; 2081 targ->children = 0; 2082 #ifdef DEBUG_SRC 2083 targ->cp = Lst_Init(FALSE); 2084 #endif 2085 2086 /* 2087 * Allocate room for the prefix, whose end is found by subtracting 2088 * the length of the suffix from the end of the name. 2089 */ 2090 prefLen = (eoname - targ->suff->nameLen) - sopref; 2091 targ->pref = bmake_malloc(prefLen + 1); 2092 memcpy(targ->pref, sopref, prefLen); 2093 targ->pref[prefLen] = '\0'; 2094 2095 /* 2096 * Add nodes from which the target can be made 2097 */ 2098 SuffAddLevel(srcs, targ); 2099 2100 /* 2101 * Record the target so we can nuke it 2102 */ 2103 (void)Lst_AtEnd(targs, targ); 2104 2105 /* 2106 * Search from this suffix's successor... 2107 */ 2108 ln = Lst_Succ(ln); 2109 } 2110 } 2111 2112 /* 2113 * Handle target of unknown suffix... 2114 */ 2115 if (Lst_IsEmpty(targs) && suffNull != NULL) { 2116 if (DEBUG(SUFF)) { 2117 fprintf(debug_file, "\tNo known suffix on %s. Using .NULL suffix\n", gn->name); 2118 } 2119 2120 targ = bmake_malloc(sizeof(Src)); 2121 targ->file = bmake_strdup(gn->name); 2122 targ->suff = suffNull; 2123 targ->suff->refCount++; 2124 targ->node = gn; 2125 targ->parent = NULL; 2126 targ->children = 0; 2127 targ->pref = bmake_strdup(sopref); 2128 #ifdef DEBUG_SRC 2129 targ->cp = Lst_Init(FALSE); 2130 #endif 2131 2132 /* 2133 * Only use the default suffix rules if we don't have commands 2134 * defined for this gnode; traditional make programs used to 2135 * not define suffix rules if the gnode had children but we 2136 * don't do this anymore. 2137 */ 2138 if (Lst_IsEmpty(gn->commands)) 2139 SuffAddLevel(srcs, targ); 2140 else { 2141 if (DEBUG(SUFF)) 2142 fprintf(debug_file, "not "); 2143 } 2144 2145 if (DEBUG(SUFF)) 2146 fprintf(debug_file, "adding suffix rules\n"); 2147 2148 (void)Lst_AtEnd(targs, targ); 2149 } 2150 2151 /* 2152 * Using the list of possible sources built up from the target suffix(es), 2153 * try and find an existing file/target that matches. 2154 */ 2155 bottom = SuffFindThem(srcs, slst); 2156 2157 if (bottom == NULL) { 2158 /* 2159 * No known transformations -- use the first suffix found for setting 2160 * the local variables. 2161 */ 2162 if (!Lst_IsEmpty(targs)) { 2163 targ = (Src *)Lst_Datum(Lst_First(targs)); 2164 } else { 2165 targ = NULL; 2166 } 2167 } else { 2168 /* 2169 * Work up the transformation path to find the suffix of the 2170 * target to which the transformation was made. 2171 */ 2172 for (targ = bottom; targ->parent != NULL; targ = targ->parent) 2173 continue; 2174 } 2175 2176 Var_Set(TARGET, gn->path ? gn->path : gn->name, gn, 0); 2177 2178 pref = (targ != NULL) ? targ->pref : gn->name; 2179 Var_Set(PREFIX, pref, gn, 0); 2180 2181 /* 2182 * Now we've got the important local variables set, expand any sources 2183 * that still contain variables or wildcards in their names. 2184 */ 2185 for (ln = Lst_First(gn->children); ln != NULL; ln = nln) { 2186 nln = Lst_Succ(ln); 2187 SuffExpandChildren(ln, gn); 2188 } 2189 2190 if (targ == NULL) { 2191 if (DEBUG(SUFF)) { 2192 fprintf(debug_file, "\tNo valid suffix on %s\n", gn->name); 2193 } 2194 2195 sfnd_abort: 2196 /* 2197 * Deal with finding the thing on the default search path. We 2198 * always do that, not only if the node is only a source (not 2199 * on the lhs of a dependency operator or [XXX] it has neither 2200 * children or commands) as the old pmake did. 2201 */ 2202 if ((gn->type & (OP_PHONY|OP_NOPATH)) == 0) { 2203 free(gn->path); 2204 gn->path = Dir_FindFile(gn->name, 2205 (targ == NULL ? dirSearchPath : 2206 targ->suff->searchPath)); 2207 if (gn->path != NULL) { 2208 char *ptr; 2209 Var_Set(TARGET, gn->path, gn, 0); 2210 2211 if (targ != NULL) { 2212 /* 2213 * Suffix known for the thing -- trim the suffix off 2214 * the path to form the proper .PREFIX variable. 2215 */ 2216 int savep = strlen(gn->path) - targ->suff->nameLen; 2217 char savec; 2218 2219 if (gn->suffix) 2220 gn->suffix->refCount--; 2221 gn->suffix = targ->suff; 2222 gn->suffix->refCount++; 2223 2224 savec = gn->path[savep]; 2225 gn->path[savep] = '\0'; 2226 2227 if ((ptr = strrchr(gn->path, '/')) != NULL) 2228 ptr++; 2229 else 2230 ptr = gn->path; 2231 2232 Var_Set(PREFIX, ptr, gn, 0); 2233 2234 gn->path[savep] = savec; 2235 } else { 2236 /* 2237 * The .PREFIX gets the full path if the target has 2238 * no known suffix. 2239 */ 2240 if (gn->suffix) 2241 gn->suffix->refCount--; 2242 gn->suffix = NULL; 2243 2244 if ((ptr = strrchr(gn->path, '/')) != NULL) 2245 ptr++; 2246 else 2247 ptr = gn->path; 2248 2249 Var_Set(PREFIX, ptr, gn, 0); 2250 } 2251 } 2252 } 2253 2254 goto sfnd_return; 2255 } 2256 2257 /* 2258 * If the suffix indicates that the target is a library, mark that in 2259 * the node's type field. 2260 */ 2261 if (targ->suff->flags & SUFF_LIBRARY) { 2262 gn->type |= OP_LIB; 2263 } 2264 2265 /* 2266 * Check for overriding transformation rule implied by sources 2267 */ 2268 if (!Lst_IsEmpty(gn->children)) { 2269 src = SuffFindCmds(targ, slst); 2270 2271 if (src != NULL) { 2272 /* 2273 * Free up all the Src structures in the transformation path 2274 * up to, but not including, the parent node. 2275 */ 2276 while (bottom && bottom->parent != NULL) { 2277 if (Lst_Member(slst, bottom) == NULL) { 2278 Lst_AtEnd(slst, bottom); 2279 } 2280 bottom = bottom->parent; 2281 } 2282 bottom = src; 2283 } 2284 } 2285 2286 if (bottom == NULL) { 2287 /* 2288 * No idea from where it can come -- return now. 2289 */ 2290 goto sfnd_abort; 2291 } 2292 2293 /* 2294 * We now have a list of Src structures headed by 'bottom' and linked via 2295 * their 'parent' pointers. What we do next is create links between 2296 * source and target nodes (which may or may not have been created) 2297 * and set the necessary local variables in each target. The 2298 * commands for each target are set from the commands of the 2299 * transformation rule used to get from the src suffix to the targ 2300 * suffix. Note that this causes the commands list of the original 2301 * node, gn, to be replaced by the commands of the final 2302 * transformation rule. Also, the unmade field of gn is incremented. 2303 * Etc. 2304 */ 2305 if (bottom->node == NULL) { 2306 bottom->node = Targ_FindNode(bottom->file, TARG_CREATE); 2307 } 2308 2309 for (src = bottom; src->parent != NULL; src = src->parent) { 2310 targ = src->parent; 2311 2312 if (src->node->suffix) 2313 src->node->suffix->refCount--; 2314 src->node->suffix = src->suff; 2315 src->node->suffix->refCount++; 2316 2317 if (targ->node == NULL) { 2318 targ->node = Targ_FindNode(targ->file, TARG_CREATE); 2319 } 2320 2321 SuffApplyTransform(targ->node, src->node, 2322 targ->suff, src->suff); 2323 2324 if (targ->node != gn) { 2325 /* 2326 * Finish off the dependency-search process for any nodes 2327 * between bottom and gn (no point in questing around the 2328 * filesystem for their implicit source when it's already 2329 * known). Note that the node can't have any sources that 2330 * need expanding, since SuffFindThem will stop on an existing 2331 * node, so all we need to do is set the standard and System V 2332 * variables. 2333 */ 2334 targ->node->type |= OP_DEPS_FOUND; 2335 2336 Var_Set(PREFIX, targ->pref, targ->node, 0); 2337 2338 Var_Set(TARGET, targ->node->name, targ->node, 0); 2339 } 2340 } 2341 2342 if (gn->suffix) 2343 gn->suffix->refCount--; 2344 gn->suffix = src->suff; 2345 gn->suffix->refCount++; 2346 2347 /* 2348 * Nuke the transformation path and the Src structures left over in the 2349 * two lists. 2350 */ 2351 sfnd_return: 2352 if (bottom) 2353 if (Lst_Member(slst, bottom) == NULL) 2354 Lst_AtEnd(slst, bottom); 2355 2356 while (SuffRemoveSrc(srcs) || SuffRemoveSrc(targs)) 2357 continue; 2358 2359 Lst_Concat(slst, srcs, LST_CONCLINK); 2360 Lst_Concat(slst, targs, LST_CONCLINK); 2361 } 2362 2363 2364 /*- 2365 *----------------------------------------------------------------------- 2366 * Suff_FindDeps -- 2367 * Find implicit sources for the target described by the graph node 2368 * gn 2369 * 2370 * Results: 2371 * Nothing. 2372 * 2373 * Side Effects: 2374 * Nodes are added to the graph below the passed-in node. The nodes 2375 * are marked to have their IMPSRC variable filled in. The 2376 * PREFIX variable is set for the given node and all its 2377 * implied children. 2378 * 2379 * Notes: 2380 * The path found by this target is the shortest path in the 2381 * transformation graph, which may pass through non-existent targets, 2382 * to an existing target. The search continues on all paths from the 2383 * root suffix until a file is found. I.e. if there's a path 2384 * .o -> .c -> .l -> .l,v from the root and the .l,v file exists but 2385 * the .c and .l files don't, the search will branch out in 2386 * all directions from .o and again from all the nodes on the 2387 * next level until the .l,v node is encountered. 2388 * 2389 *----------------------------------------------------------------------- 2390 */ 2391 2392 void 2393 Suff_FindDeps(GNode *gn) 2394 { 2395 2396 SuffFindDeps(gn, srclist); 2397 while (SuffRemoveSrc(srclist)) 2398 continue; 2399 } 2400 2401 2402 /* 2403 * Input: 2404 * gn node we're dealing with 2405 * 2406 */ 2407 static void 2408 SuffFindDeps(GNode *gn, Lst slst) 2409 { 2410 if (gn->type & (OP_DEPS_FOUND|OP_PHONY)) { 2411 /* 2412 * If dependencies already found, no need to do it again... 2413 * If this is a .PHONY target, we do not apply suffix rules. 2414 */ 2415 return; 2416 } else { 2417 gn->type |= OP_DEPS_FOUND; 2418 } 2419 2420 if (DEBUG(SUFF)) { 2421 fprintf(debug_file, "SuffFindDeps (%s)\n", gn->name); 2422 } 2423 2424 if (gn->type & OP_ARCHV) { 2425 SuffFindArchiveDeps(gn, slst); 2426 } else if (gn->type & OP_LIB) { 2427 /* 2428 * If the node is a library, it is the arch module's job to find it 2429 * and set the TARGET variable accordingly. We merely provide the 2430 * search path, assuming all libraries end in ".a" (if the suffix 2431 * hasn't been defined, there's nothing we can do for it, so we just 2432 * set the TARGET variable to the node's name in order to give it a 2433 * value). 2434 */ 2435 LstNode ln; 2436 Suff *s; 2437 2438 ln = Lst_Find(sufflist, UNCONST(LIBSUFF), 2439 SuffSuffHasNameP); 2440 if (gn->suffix) 2441 gn->suffix->refCount--; 2442 if (ln != NULL) { 2443 gn->suffix = s = (Suff *)Lst_Datum(ln); 2444 gn->suffix->refCount++; 2445 Arch_FindLib(gn, s->searchPath); 2446 } else { 2447 gn->suffix = NULL; 2448 Var_Set(TARGET, gn->name, gn, 0); 2449 } 2450 /* 2451 * Because a library (-lfoo) target doesn't follow the standard 2452 * filesystem conventions, we don't set the regular variables for 2453 * the thing. .PREFIX is simply made empty... 2454 */ 2455 Var_Set(PREFIX, "", gn, 0); 2456 } else { 2457 SuffFindNormalDeps(gn, slst); 2458 } 2459 } 2460 2461 /*- 2462 *----------------------------------------------------------------------- 2463 * Suff_SetNull -- 2464 * Define which suffix is the null suffix. 2465 * 2466 * Input: 2467 * name Name of null suffix 2468 * 2469 * Results: 2470 * None. 2471 * 2472 * Side Effects: 2473 * 'suffNull' is altered. 2474 * 2475 * Notes: 2476 * Need to handle the changing of the null suffix gracefully so the 2477 * old transformation rules don't just go away. 2478 * 2479 *----------------------------------------------------------------------- 2480 */ 2481 void 2482 Suff_SetNull(char *name) 2483 { 2484 Suff *s; 2485 LstNode ln; 2486 2487 ln = Lst_Find(sufflist, name, SuffSuffHasNameP); 2488 if (ln != NULL) { 2489 s = (Suff *)Lst_Datum(ln); 2490 if (suffNull != NULL) { 2491 suffNull->flags &= ~SUFF_NULL; 2492 } 2493 s->flags |= SUFF_NULL; 2494 /* 2495 * XXX: Here's where the transformation mangling would take place 2496 */ 2497 suffNull = s; 2498 } else { 2499 Parse_Error(PARSE_WARNING, "Desired null suffix %s not defined.", 2500 name); 2501 } 2502 } 2503 2504 /*- 2505 *----------------------------------------------------------------------- 2506 * Suff_Init -- 2507 * Initialize suffixes module 2508 * 2509 * Results: 2510 * None 2511 * 2512 * Side Effects: 2513 * Many 2514 *----------------------------------------------------------------------- 2515 */ 2516 void 2517 Suff_Init(void) 2518 { 2519 sufflist = Lst_Init(FALSE); 2520 #ifdef CLEANUP 2521 suffClean = Lst_Init(FALSE); 2522 #endif 2523 srclist = Lst_Init(FALSE); 2524 transforms = Lst_Init(FALSE); 2525 2526 sNum = 0; 2527 /* 2528 * Create null suffix for single-suffix rules (POSIX). The thing doesn't 2529 * actually go on the suffix list or everyone will think that's its 2530 * suffix. 2531 */ 2532 emptySuff = suffNull = bmake_malloc(sizeof(Suff)); 2533 2534 suffNull->name = bmake_strdup(""); 2535 suffNull->nameLen = 0; 2536 suffNull->searchPath = Lst_Init(FALSE); 2537 Dir_Concat(suffNull->searchPath, dirSearchPath); 2538 suffNull->children = Lst_Init(FALSE); 2539 suffNull->parents = Lst_Init(FALSE); 2540 suffNull->ref = Lst_Init(FALSE); 2541 suffNull->sNum = sNum++; 2542 suffNull->flags = SUFF_NULL; 2543 suffNull->refCount = 1; 2544 2545 } 2546 2547 2548 /*- 2549 *---------------------------------------------------------------------- 2550 * Suff_End -- 2551 * Cleanup the this module 2552 * 2553 * Results: 2554 * None 2555 * 2556 * Side Effects: 2557 * The memory is free'd. 2558 *---------------------------------------------------------------------- 2559 */ 2560 2561 void 2562 Suff_End(void) 2563 { 2564 #ifdef CLEANUP 2565 Lst_Destroy(sufflist, SuffFree); 2566 Lst_Destroy(suffClean, SuffFree); 2567 if (suffNull) 2568 SuffFree(suffNull); 2569 Lst_Destroy(srclist, NULL); 2570 Lst_Destroy(transforms, NULL); 2571 #endif 2572 } 2573 2574 2575 /********************* DEBUGGING FUNCTIONS **********************/ 2576 2577 static int SuffPrintName(ClientData s, ClientData dummy) 2578 { 2579 fprintf(debug_file, "%s ", ((Suff *)s)->name); 2580 return (dummy ? 0 : 0); 2581 } 2582 2583 static int 2584 SuffPrintSuff(ClientData sp, ClientData dummy) 2585 { 2586 Suff *s = (Suff *)sp; 2587 int flags; 2588 int flag; 2589 2590 fprintf(debug_file, "# `%s' [%d] ", s->name, s->refCount); 2591 2592 flags = s->flags; 2593 if (flags) { 2594 fputs(" (", debug_file); 2595 while (flags) { 2596 flag = 1 << (ffs(flags) - 1); 2597 flags &= ~flag; 2598 switch (flag) { 2599 case SUFF_NULL: 2600 fprintf(debug_file, "NULL"); 2601 break; 2602 case SUFF_INCLUDE: 2603 fprintf(debug_file, "INCLUDE"); 2604 break; 2605 case SUFF_LIBRARY: 2606 fprintf(debug_file, "LIBRARY"); 2607 break; 2608 } 2609 fputc(flags ? '|' : ')', debug_file); 2610 } 2611 } 2612 fputc('\n', debug_file); 2613 fprintf(debug_file, "#\tTo: "); 2614 Lst_ForEach(s->parents, SuffPrintName, NULL); 2615 fputc('\n', debug_file); 2616 fprintf(debug_file, "#\tFrom: "); 2617 Lst_ForEach(s->children, SuffPrintName, NULL); 2618 fputc('\n', debug_file); 2619 fprintf(debug_file, "#\tSearch Path: "); 2620 Dir_PrintPath(s->searchPath); 2621 fputc('\n', debug_file); 2622 return (dummy ? 0 : 0); 2623 } 2624 2625 static int 2626 SuffPrintTrans(ClientData tp, ClientData dummy) 2627 { 2628 GNode *t = (GNode *)tp; 2629 2630 fprintf(debug_file, "%-16s: ", t->name); 2631 Targ_PrintType(t->type); 2632 fputc('\n', debug_file); 2633 Lst_ForEach(t->commands, Targ_PrintCmd, NULL); 2634 fputc('\n', debug_file); 2635 return(dummy ? 0 : 0); 2636 } 2637 2638 void 2639 Suff_PrintAll(void) 2640 { 2641 fprintf(debug_file, "#*** Suffixes:\n"); 2642 Lst_ForEach(sufflist, SuffPrintSuff, NULL); 2643 2644 fprintf(debug_file, "#*** Transformations:\n"); 2645 Lst_ForEach(transforms, SuffPrintTrans, NULL); 2646 } 2647