From 22886f8a74b5925030889fed52af5a8add5617d7 Mon Sep 17 00:00:00 2001 From: Boris Kolpackov Date: Fri, 17 Mar 2006 14:24:20 +0000 Subject: [PATCH] Fixed Savannah bug #16053. --- ChangeLog | 17 ++++++++++- dep.h | 6 +++- file.c | 24 ++++++++++++---- implicit.c | 17 ++++------- main.c | 18 ++++-------- misc.c | 36 ++++++++++++++++++++--- read.c | 25 +++++----------- remake.c | 4 +-- rule.c | 9 ++---- tests/ChangeLog | 4 +++ tests/scripts/features/statipattrules | 41 +++++++++++++++++++++++++++ 11 files changed, 138 insertions(+), 63 deletions(-) diff --git a/ChangeLog b/ChangeLog index abe3b9b9..6fc7aef3 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,18 @@ +2006-03-17 Boris Kolpackov + + * dep.h (struct dep): Add the stem field. + * misc.c (alloc_dep, free_dep): New functions. + (copy_dep_chain): Copy stem. + (free_dep_chain): Use free_dep. + * read.c (record_files): Store stem in the dependency line. + * file.c (expand_deps): Use stem stored in the dependency line. Use + free_dep_chain instead of free_ns_chain. + * implicit.c (pattern_search): Use alloc_dep and free_dep. + * read.c (read_all_makefiles, eval_makefile, eval): Ditto. + * main.c (main, handle_non_switch_argument): Ditto. + * remake.c (check_dep): Ditto. + * rule.c (convert_suffix_rule, freerule): Ditto. + 2006-03-14 Paul D. Smith * expand.c (variable_append): Instead of appending everything then @@ -155,7 +170,7 @@ 2006-02-08 Boris Kolpackov - * job.h (struct child): Add dontcare bitfield. + * job.h (struct child): Add the dontcare bitfield. * job.c (new_job): Cache dontcare flag. * job.c (reap_children): Use cached dontcare flag instead of the one in struct file. Fixes Savannah bug #15641. diff --git a/dep.h b/dep.h index aafadb25..f0923cc2 100644 --- a/dep.h +++ b/dep.h @@ -28,7 +28,8 @@ Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */ /* Structure representing one dependency of a file. Each struct file's `deps' points to a chain of these, - chained through the `next'. + chained through the `next'. `stem' is the stem for this + dep line of static pattern rule or NULL. Note that the first two words of this match a struct nameseq. */ @@ -36,6 +37,7 @@ struct dep { struct dep *next; char *name; + char *stem; struct file *file; unsigned int changed : 8; unsigned int ignore_mtime : 1; @@ -72,6 +74,8 @@ extern struct nameseq *ar_glob PARAMS ((char *arname, char *member_pattern, unsi extern char *dep_name (); #endif +extern struct dep *alloc_dep PARAMS ((void)); +extern void free_dep PARAMS ((struct dep *d)); extern struct dep *copy_dep_chain PARAMS ((const struct dep *d)); extern void free_dep_chain PARAMS ((struct dep *d)); extern void free_ns_chain PARAMS ((struct nameseq *n)); diff --git a/file.c b/file.c index a0e82228..90dd1230 100644 --- a/file.c +++ b/file.c @@ -460,6 +460,7 @@ expand_deps (struct file *f) { struct dep *d; struct dep *old = f->deps; + char *file_stem = f->stem; unsigned int last_dep_has_cmds = f->updating; int initialized = 0; @@ -491,20 +492,29 @@ expand_deps (struct file *f) free (d->name); d->name = savestring (buffer, o - buffer); - d->staticpattern = 0; + d->staticpattern = 0; /* Clear staticpattern so that we don't + re-expand %s below. */ } /* We are going to do second expansion so initialize file variables - for the file. */ + for the file. Since the stem for static pattern rules comes from + individual dep lines, we will temporarily set f->stem to d->stem. + */ if (!initialized) { initialize_file_variables (f, 0); initialized = 1; } + if (d->stem != 0) + f->stem = d->stem; + set_file_variables (f); p = variable_expand_for_file (d->name, f); + + if (d->stem != 0) + f->stem = file_stem; } /* Parse the prerequisites. */ @@ -527,12 +537,12 @@ expand_deps (struct file *f) /* We have to handle empty stems specially, because that would be equivalent to $(patsubst %,dp->name,) which will always be empty. */ - if (f->stem[0] == '\0') + if (d->stem[0] == '\0') /* This needs memmove() in ISO C. */ bcopy (percent+1, percent, strlen (percent)); else { - char *o = patsubst_expand (buffer, f->stem, pattern, + char *o = patsubst_expand (buffer, d->stem, pattern, dp->name, pattern+1, percent+1); if (o == buffer) @@ -552,7 +562,9 @@ expand_deps (struct file *f) dp = new = new->next; else dp = dl->next = dp->next; - free ((char *)df); + /* @@ Are we leaking df->name here? */ + df->name = 0; + free_dep (df); continue; } } @@ -604,7 +616,7 @@ expand_deps (struct file *f) } } - free_ns_chain ((struct nameseq *) old); + free_dep_chain (old); } /* For each dependency of each file, make the `struct dep' point diff --git a/implicit.c b/implicit.c index 8085ea10..a5d37008 100644 --- a/implicit.c +++ b/implicit.c @@ -802,8 +802,7 @@ pattern_search (struct file *file, int archive, while (dep != 0) { struct dep *next = dep->next; - free (dep->name); - free ((char *)dep); + free_dep (dep); dep = next; } file->deps = 0; @@ -862,15 +861,12 @@ pattern_search (struct file *file, int archive, } } - dep = (struct dep *) xmalloc (sizeof (struct dep)); + dep = alloc_dep (); dep->ignore_mtime = d->ignore_mtime; - dep->staticpattern = 0; - dep->need_2nd_expansion = 0; s = d->name; /* Hijacking the name. */ d->name = 0; if (recursions == 0) { - dep->name = 0; dep->file = lookup_file (s); if (dep->file == 0) /* enter_file consumes S's storage. */ @@ -883,9 +879,8 @@ pattern_search (struct file *file, int archive, else { dep->name = s; - dep->file = 0; - dep->changed = 0; } + if (d->intermediate_file == 0 && tryrules[foundrule]->terminal) { /* If the file actually existed (was not an intermediate file), @@ -943,11 +938,9 @@ pattern_search (struct file *file, int archive, if (i != matches[foundrule]) { struct file *f; - struct dep *new = (struct dep *) xmalloc (sizeof (struct dep)); + struct dep *new = alloc_dep (); + /* GKM FIMXE: handle '|' here too */ - new->ignore_mtime = 0; - new->staticpattern = 0; - new->need_2nd_expansion = 0; new->name = p = (char *) xmalloc (rule->lens[i] + fullstemlen + 1); bcopy (rule->targets[i], p, rule->suffixes[i] - rule->targets[i] - 1); diff --git a/main.c b/main.c index a645e44c..53a2e2ea 100644 --- a/main.c +++ b/main.c @@ -1850,7 +1850,7 @@ main (int argc, char **argv, char **envp) last->next = d->next; /* Free the storage. */ - free ((char *) d); + free_dep (d); d = last == 0 ? read_makefiles : last->next; @@ -2172,12 +2172,7 @@ main (int argc, char **argv, char **envp) } } - goals = (struct dep *) xmalloc (sizeof (struct dep)); - goals->next = 0; - goals->name = 0; - goals->ignore_mtime = 0; - goals->staticpattern = 0; - goals->need_2nd_expansion = 0; + goals = alloc_dep (); goals->file = default_goal_file; } } @@ -2333,19 +2328,16 @@ handle_non_switch_argument (char *arg, int env) if (goals == 0) { - goals = (struct dep *) xmalloc (sizeof (struct dep)); + goals = alloc_dep (); lastgoal = goals; } else { - lastgoal->next = (struct dep *) xmalloc (sizeof (struct dep)); + lastgoal->next = alloc_dep (); lastgoal = lastgoal->next; } - lastgoal->name = 0; + lastgoal->file = f; - lastgoal->ignore_mtime = 0; - lastgoal->staticpattern = 0; - lastgoal->need_2nd_expansion = 0; { /* Add this target name to the MAKECMDGOALS variable. */ diff --git a/misc.c b/misc.c index 43df799a..082bcc67 100644 --- a/misc.c +++ b/misc.c @@ -479,6 +479,32 @@ find_next_token (char **ptr, unsigned int *lengthptr) return p; } + +/* Allocate a new `struct dep' with all fields initialized to 0. */ + +struct dep * +alloc_dep () +{ + struct dep *d = (struct dep *) xmalloc (sizeof (struct dep)); + bzero ((char *) d, sizeof (struct dep)); + return d; +} + + +/* Free `struct dep' along with `name' and `stem'. */ + +void +free_dep (struct dep *d) +{ + if (d->name != 0) + free (d->name); + + if (d->stem != 0) + free (d->stem); + + free ((char *)d); +} + /* Copy a chain of `struct dep', making a new chain with the same contents as the old one. */ @@ -493,8 +519,12 @@ copy_dep_chain (const struct dep *d) { c = (struct dep *) xmalloc (sizeof (struct dep)); bcopy ((char *) d, (char *) c, sizeof (struct dep)); + if (c->name != 0) c->name = xstrdup (c->name); + if (c->stem != 0) + c->stem = xstrdup (c->stem); + c->next = 0; if (firstnew == 0) firstnew = lastnew = c; @@ -516,14 +546,12 @@ free_dep_chain (struct dep *d) { struct dep *df = d; d = d->next; - - free (df->name); - free ((char *)df); + free_dep (df); } } /* Free a chain of `struct nameseq'. Each nameseq->name is freed - as well. Can be used on `struct dep' chains.*/ + as well. For `struct dep' chains use free_dep_chain. */ void free_ns_chain (struct nameseq *n) diff --git a/read.c b/read.c index 0c78c7f4..f6e89861 100644 --- a/read.c +++ b/read.c @@ -248,13 +248,9 @@ read_all_makefiles (char **makefiles) tail = tail->next; for (p = default_makefiles; *p != 0; ++p) { - struct dep *d = (struct dep *) xmalloc (sizeof (struct dep)); - d->name = 0; + struct dep *d = alloc_dep (); d->file = enter_file (*p); d->file->dontcare = 1; - d->ignore_mtime = 0; - d->staticpattern = 0; - d->need_2nd_expansion = 0; /* Tell update_goal_chain to bail out as soon as this file is made, and main not to die if we can't make this file. */ d->changed = RM_DONTCARE; @@ -366,18 +362,14 @@ eval_makefile (char *filename, int flags) } /* Add FILENAME to the chain of read makefiles. */ - deps = (struct dep *) xmalloc (sizeof (struct dep)); + deps = alloc_dep (); deps->next = read_makefiles; read_makefiles = deps; - deps->name = 0; deps->file = lookup_file (filename); if (deps->file == 0) deps->file = enter_file (xstrdup (filename)); filename = deps->file->name; deps->changed = flags; - deps->ignore_mtime = 0; - deps->staticpattern = 0; - deps->need_2nd_expansion = 0; if (flags & RM_DONTCARE) deps->file->dontcare = 1; @@ -1177,14 +1169,8 @@ eval (struct ebuffer *ebuf, int set_default) if (beg <= end && *beg != '\0') { /* Put all the prerequisites here; they'll be parsed later. */ - deps = (struct dep *) xmalloc (sizeof (struct dep)); - deps->next = 0; + deps = alloc_dep (); deps->name = savestring (beg, end - beg + 1); - deps->file = 0; - deps->changed = 0; - deps->ignore_mtime = 0; - deps->staticpattern = 0; - deps->need_2nd_expansion = 0; } else deps = 0; @@ -2111,7 +2097,10 @@ record_files (struct nameseq *filenames, char *pattern, char *pattern_percent, pattern_percent+1, percent+1); f->stem = savestring (buffer, o - buffer); if (this) - this->staticpattern = 1; + { + this->staticpattern = 1; + this->stem = xstrdup (f->stem); + } } /* Free name if not needed further. */ diff --git a/remake.c b/remake.c index da231b4a..2dfade3c 100644 --- a/remake.c +++ b/remake.c @@ -994,13 +994,13 @@ check_dep (struct file *file, unsigned int depth, if (lastd == 0) { file->deps = d->next; - free ((char *) d); + free_dep (d); d = file->deps; } else { lastd->next = d->next; - free ((char *) d); + free_dep (d); d = lastd->next; } continue; diff --git a/rule.c b/rule.c index 527b5b0c..e988db5b 100644 --- a/rule.c +++ b/rule.c @@ -201,12 +201,8 @@ convert_suffix_rule (char *target, char *source, struct commands *cmds) depname = xmalloc (1 + len + 1); depname[0] = '%'; bcopy (source, depname + 1, len + 1); - deps = (struct dep *) xmalloc (sizeof (struct dep)); - deps->next = 0; + deps = alloc_dep (); deps->name = depname; - deps->ignore_mtime = 0; - deps->staticpattern = 0; - deps->need_2nd_expansion = 0; } create_pattern_rule (names, percents, 0, deps, cmds, 0); @@ -428,7 +424,8 @@ freerule (struct rule *rule, struct rule *lastrule) t = dep->next; /* We might leak dep->name here, but I'm not sure how to fix this: I think that pointer might be shared (e.g., in the file hash?) */ - free ((char *) dep); + dep->name = 0; /* Make sure free_dep does not free name. */ + free_dep (dep); dep = t; } diff --git a/tests/ChangeLog b/tests/ChangeLog index 05dfaeca..67cf900e 100644 --- a/tests/ChangeLog +++ b/tests/ChangeLog @@ -1,3 +1,7 @@ +2006-03-17 Boris Kolpackov + + * scripts/features/statipattrules: Add tests for bug #16053. + 2006-03-09 Paul Smith * scripts/features/escape: Use "pre:" not "p:" to avoid conflicts diff --git a/tests/scripts/features/statipattrules b/tests/scripts/features/statipattrules index 429b56a1..3f363def 100644 --- a/tests/scripts/features/statipattrules +++ b/tests/scripts/features/statipattrules @@ -67,4 +67,45 @@ foo.baz: ;@: ', '', ''); + +# TEST #6: make sure the second stem does not overwrite the first +# perprerequisite's stem (Savannah bug #16053). +# +run_make_test(' +all.foo.bar: %.foo.bar: %.one + +all.foo.bar: %.bar: %.two + +all.foo.bar: + @echo $* + @echo $^ + +.DEFAULT:;@: +', +'', +'all.foo +all.one all.foo.two'); + + +# TEST #7: make sure the second stem does not overwrite the first +# perprerequisite's stem when second expansion is enabled +# (Savannah bug #16053). +# +run_make_test(' +.SECONDEXPANSION: + +all.foo.bar: %.foo.bar: %.one $$*-one + +all.foo.bar: %.bar: %.two $$*-two + +all.foo.bar: + @echo $* + @echo $^ + +.DEFAULT:;@: +', +'', +'all.foo +all.one all-one all.foo.two all.foo-two'); + 1;