The recipe of a pattern rule with multiple targets gets executed once for each target if
a) the pattern isn't a prefix of the target, and b) the pattern stem contains a directory. The rule should only be executed once: Multiple target patterns in pattern rules are always treated as grouped targets (see Multiple Targets in a Rule) regardless of whether they use the : or &: separator. Here is an example: $ cat GNUmakefile <<END all: dir/axa dir/bxb dir/cxc: a%a b%b: c%c; echo "Executed only once!" END $ make echo "Executed only once!" Executed only once! echo "Executed only once!" Executed only once! The problem is that when populating also_make, the % is expanded incorrectly, and as a result the also made targets won't match to other targets. Specifically, the pattern is replaced with the entire stem, instead of with the file part of the stem and then prepending the directory to that. $ make -p ... dir/axa: dir/cxc ... # Also makes: bdir/xb ... Here the also_make should be "dir/bxb". Please find attached the reproducing makefile and a patch to fix it. -- Jarmo Jaakkola
all: dir/axa dir/bxb dir/cxc: a%a b%b: c%c; echo "Executed only once!"
>From 72bb1e8446a8043c0863778cba8c8a14ddcf9eca Mon Sep 17 00:00:00 2001 From: Jarmo Jaakkola <jarmo.jaakk...@roskakori.fi> Date: Thu, 16 Mar 2023 00:07:05 +0200 Subject: [PATCH] Fix also_make of non-prefix patterns In pattern rules with multiple targets where the pattern is *not* a prefix, i.e. "a% b%: c%; rule" vs. "%a %b: %c; rule" The stem was being expanded incorrectly when populating also_make. The pattern was replaced with the entire stem instead of prepending the directory part of the stem and replacing the pattern with only the file part of stem. E.g. given the rule above, for dir/atarget also_make would contain "bdir/target" instead of "dir/btarget" The incorrect names didn't match the intended targets and as a result the rule got executed for each target, instead of only once. In a parallel build they could even execute simultaneously. --- src/implicit.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/implicit.c b/src/implicit.c index 5fa30944..b3884dfa 100644 --- a/src/implicit.c +++ b/src/implicit.c @@ -1110,9 +1110,10 @@ pattern_search (struct file *file, int archive, struct dep *new = alloc_dep (); /* GKM FIMXE: handle '|' here too */ + p = mempcpy (p, file->stem, pathlen); p = mempcpy (p, rule->targets[ri], rule->suffixes[ri] - rule->targets[ri] - 1); - p = mempcpy (p, file->stem, fullstemlen); + p = mempcpy (p, file->stem + pathlen, stemlen); memcpy (p, rule->suffixes[ri], rule->lens[ri] - (rule->suffixes[ri] - rule->targets[ri])+1); new->name = strcache_add (nm); -- 2.30.2