aboutsummaryrefslogtreecommitdiff
path: root/build2/algorithm.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'build2/algorithm.cxx')
-rw-r--r--build2/algorithm.cxx107
1 files changed, 81 insertions, 26 deletions
diff --git a/build2/algorithm.cxx b/build2/algorithm.cxx
index bcdccf6..2e3fe83 100644
--- a/build2/algorithm.cxx
+++ b/build2/algorithm.cxx
@@ -206,33 +206,34 @@ namespace build2
sched.resume (task_count);
}
- target_lock
- add_adhoc_member (action a,
- target& t,
+ target&
+ add_adhoc_member (target& t,
const target_type& tt,
const dir_path& dir,
const dir_path& out,
- const string& n)
+ string n)
{
+ tracer trace ("add_adhoc_member");
+
const_ptr<target>* mp (&t.member);
for (; *mp != nullptr && !(*mp)->is_a (tt); mp = &(*mp)->member) ;
- const target& m (*mp != nullptr // Might already be there.
- ? **mp
- : search (t, tt, dir, out, n));
-
- target_lock l (lock (a, m));
- assert (l.target != nullptr); // Someone messing with ad hoc members?
-
+ target& m (*mp != nullptr // Might already be there.
+ ? **mp
+ : targets.insert (tt,
+ dir,
+ out,
+ move (n),
+ nullopt /* ext */,
+ true /* implied */,
+ trace).first);
if (*mp == nullptr)
{
- *mp = l.target;
- l.target->group = &t;
+ *mp = &m;
+ m.group = &t;
}
- else
- assert ((*mp)->dir == dir && (*mp)->name == n); // Basic sanity check.
- return l;
+ return m;
};
// Return the matching rule or NULL if no match and try_match is true.
@@ -522,22 +523,62 @@ namespace build2
? optional<scheduler::work_queue> (scheduler::work_none)
: nullopt));
- if (l.target == nullptr)
- {
- // Already applied, executed, or busy.
- //
- if (l.offset >= target::offset_busy)
- return make_pair (true, target_state::busy);
-
- // Fall through.
- }
- else
+ if (l.target != nullptr)
{
assert (l.offset < target::offset_applied); // Shouldn't lock otherwise.
if (try_match && l.offset == target::offset_tried)
return make_pair (false, target_state::unknown);
+ // Handle matching ad hoc group member.
+ //
+ if (ct.adhoc_member ())
+ {
+ const target& g (*ct.group);
+
+ // It feels natural to "convert" this call to the one for the group,
+ // including the try_match and async parts. However, that async part
+ // is tricky: we will be called again to finish the match (the else-
+ // block below) where we need to perform the equivalent conversion.
+ // Semantically, we want to achieve the following:
+ //
+ // match (a, g); | match_async (a, g);
+ // | match (a, g);
+ // match_recipe (l, group_recipe); | match_recipe (l, group_recipe);
+ //
+ // We also have to "unstack" this lock to avoid racing with stack
+ // modifications by the asynchronous match (see below). Since an ad
+ // hoc member doesn't have any prerequisites of its own (and thus
+ // cannot depend on the group), skipping this link during the cycle
+ // detection feels harmless (note that the other way around is
+ // possible and still works).
+ //
+ l.unstack ();
+
+
+#if 0 // The same story as with the lock stack. I think the only way to
+ // make this work is to somehow pass the member down to match_impl().
+ auto df = make_diag_frame (
+ [a, &ct](const diag_record& dr)
+ {
+ if (verb != 0)
+ dr << info << "while matching rule group to " << diag_do (a, ct);
+ });
+#endif
+
+ auto r (match (a, g, start_count, task_count, try_match));
+
+ if (r.first)
+ {
+ match_inc_dependens (a, g);
+ match_recipe (l, group_recipe);
+ }
+ else
+ l.offset = target::offset_tried;
+
+ return r; // Group state.
+ }
+
if (task_count == nullptr)
return match_impl (l, false /* step */, try_match);
@@ -579,6 +620,20 @@ namespace build2
// Matched synchronously, fall through.
}
+ else
+ {
+ // Already applied, executed, or busy.
+ //
+ if (l.offset >= target::offset_busy)
+ return make_pair (true, target_state::busy);
+
+ // Handle matching ad hoc group member (the finish part; see above).
+ //
+ if (ct.adhoc_member ())
+ return match (a, *ct.group, 0, nullptr);
+
+ // Fall through.
+ }
return ct.try_matched_state (a, false);
}