1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
|
// file : libbuild2/config/utility.txx -*- C++ -*-
// license : MIT; see accompanying LICENSE file
namespace build2
{
namespace config
{
template <typename T>
pair<lookup, bool>
lookup_config_impl (scope& rs,
const variable& var,
T&& def_val,
uint64_t sflags,
bool def_ovr)
{
// Note: see also the other lookup_config() implementation if changing
// anything here.
save_variable (rs, var, sflags);
pair<lookup, size_t> org (rs.find_original (var));
bool n (false); // New flag.
lookup l (org.first);
// The interaction with command line overrides can get tricky. For
// example, the override to default value could make (non-recursive)
// command line override in the outer scope no longer apply. So what we
// are going to do is first ignore overrides and perform the normal
// logic on the original. Then we apply the overrides on the result.
//
// Note that this is not exactly the "lookup and set to default if
// undefined" semantics in case there is no original but there is an
// override. In this case we will set original to default and then apply
// the override, which could be append or non-recursive (as mentioned
// above). It does, however, feel like taking into account the default
// in such cases is the correct semantics since append is meant as an
// addition to something existing and non-recursive override is only
// meant to override at the level it was specified. Though it won't be
// surprising at all if we end up with some counter-intuitive behavior
// here.
//
// Actually, the above analysis is not the full picture: if we have one
// of those overrides (append, non-recursive) in the outer project, then
// the lookup_config() call at that level will set the corresponding
// variable on that scope and we will see it as "original-defined" from
// our scope. Of course if there is no call to lookup_config() for this
// variable in the outer scope, then we won't see anything but then our
// behavior in this case seems correct: since that value is not part of
// the configuration (and won't be saved), then we should stick to our
// default. In other words, we should only inherit the value if it is
// actually recognized as a configuration value by the outer project.
//
// So, to summarize the current understanding, while our semantics is
// not exactly "lookup and set to default if undefined" in some obscure
// corner cases, it seem to be the correct/preferred one.
//
if (!l.defined () || (def_ovr && !l.belongs (rs)))
{
value& v (rs.assign (var) = std::forward<T> (def_val)); // VC14
v.extra = true; // Default value flag.
n = (sflags & save_default_commented) == 0; // Absence means default.
l = lookup (v, var, rs);
org = make_pair (l, 1); // Lookup depth is 1 since it's in rs.vars.
}
// Treat an inherited value that was set to default as new.
//
else if (l->extra)
n = (sflags & save_default_commented) == 0; // Absence means default.
if (var.overrides != nullptr)
{
pair<lookup, size_t> ovr (rs.find_override (var, move (org)));
if (l != ovr.first) // Overriden?
{
// Override is always treated as new.
//
n = true;
l = move (ovr.first);
}
}
return pair<lookup, bool> (l, n);
}
}
}
|