// file : libbuild2/variable.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include #include // memcmp(), memcpy() #include #ifndef BUILD2_BOOTSTRAP # include # include #endif #include #include using namespace std; namespace build2 { // variable_visibility // string to_string (variable_visibility v) { string r; switch (v) { case variable_visibility::global: r = "global"; break; case variable_visibility::project: r = "project"; break; case variable_visibility::scope: r = "scope"; break; case variable_visibility::target: r = "target"; break; case variable_visibility::prereq: r = "prerequisite"; break; } return r; } // value // void value:: reset () { if (type == nullptr) as ().~names (); else if (type->dtor != nullptr) type->dtor (*this); null = true; } value:: value (value&& v) noexcept : type (v.type), null (v.null), extra (v.extra) { if (!null) { if (type == nullptr) new (&data_) names (move (v).as ()); else if (type->copy_ctor != nullptr) type->copy_ctor (*this, v, true); else memcpy (data_, v.data_, size_); // Copy as POD. } } value:: value (const value& v) : type (v.type), null (v.null), extra (v.extra) { if (!null) { if (type == nullptr) new (&data_) names (v.as ()); else if (type->copy_ctor != nullptr) type->copy_ctor (*this, v, false); else memcpy (data_, v.data_, size_); // Copy as POD. } } value& value:: operator= (value&& v) { if (this != &v) { // Prepare the receiving value. // if (type != v.type) { *this = nullptr; type = v.type; } // Now our types are the same. If the receiving value is NULL, then call // copy_ctor() instead of copy_assign(). // if (v) { if (type == nullptr) { if (null) new (&data_) names (move (v).as ()); else // Note: can throw (see small_vector for details). // as () = move (v).as (); } else if (auto f = null ? type->copy_ctor : type->copy_assign) f (*this, v, true); else memcpy (data_, v.data_, size_); // Assign as POD. null = v.null; } else *this = nullptr; } return *this; } value& value:: operator= (const value& v) { if (this != &v) { // Prepare the receiving value. // if (type != v.type) { *this = nullptr; type = v.type; } // Now our types are the same. If the receiving value is NULL, then call // copy_ctor() instead of copy_assign(). // if (v) { if (type == nullptr) { if (null) new (&data_) names (v.as ()); else as () = v.as (); } else if (auto f = null ? type->copy_ctor : type->copy_assign) f (*this, v, false); else memcpy (data_, v.data_, size_); // Assign as POD. null = v.null; } else *this = nullptr; } return *this; } void value:: assign (names&& ns, const variable* var) { assert (type == nullptr || type->assign != nullptr); if (type == nullptr) { if (null) new (&data_) names (move (ns)); else as () = move (ns); } else type->assign (*this, move (ns), var); null = false; } void value:: append (names&& ns, const variable* var) { if (type == nullptr) { if (null) new (&data_) names (move (ns)); else { names& p (as ()); if (p.empty ()) p = move (ns); else if (!ns.empty ()) { p.insert (p.end (), make_move_iterator (ns.begin ()), make_move_iterator (ns.end ())); } } } else { if (type->append == nullptr) { diag_record dr (fail); dr << "cannot append to " << type->name << " value"; if (var != nullptr) dr << " in variable " << var->name; } type->append (*this, move (ns), var); } null = false; } void value:: prepend (names&& ns, const variable* var) { if (type == nullptr) { if (null) new (&data_) names (move (ns)); else { names& p (as ()); if (p.empty ()) p = move (ns); else if (!ns.empty ()) { ns.insert (ns.end (), make_move_iterator (p.begin ()), make_move_iterator (p.end ())); p = move (ns); } } } else { if (type->prepend == nullptr) { diag_record dr (fail); dr << "cannot prepend to " << type->name << " value"; if (var != nullptr) dr << " in variable " << var->name; } type->prepend (*this, move (ns), var); } null = false; } bool operator== (const value& x, const value& y) { bool xn (x.null); bool yn (y.null); assert (x.type == y.type || (xn && x.type == nullptr) || (yn && y.type == nullptr)); if (xn || yn) return xn == yn; if (x.type == nullptr) return x.as () == y.as (); if (x.type->compare == nullptr) return memcmp (&x.data_, &y.data_, x.type->size) == 0; return x.type->compare (x, y) == 0; } bool operator< (const value& x, const value& y) { bool xn (x.null); bool yn (y.null); assert (x.type == y.type || (xn && x.type == nullptr) || (yn && y.type == nullptr)); // NULL value is always less than non-NULL. // if (xn || yn) return xn > yn; // !xn < !yn if (x.type == nullptr) return x.as () < y.as (); if (x.type->compare == nullptr) return memcmp (&x.data_, &y.data_, x.type->size) < 0; return x.type->compare (x, y) < 0; } bool operator> (const value& x, const value& y) { bool xn (x.null); bool yn (y.null); assert (x.type == y.type || (xn && x.type == nullptr) || (yn && y.type == nullptr)); // NULL value is always less than non-NULL. // if (xn || yn) return xn < yn; // !xn > !yn if (x.type == nullptr) return x.as () > y.as (); if (x.type->compare == nullptr) return memcmp (&x.data_, &y.data_, x.type->size) > 0; return x.type->compare (x, y) > 0; } void typify (value& v, const value_type& t, const variable* var, memory_order mo) { if (v.type == nullptr) { if (v) { // Note: the order in which we do things here is important. // names ns (move (v).as ()); v = nullptr; // Use value_type::assign directly to delay v.type change. // t.assign (v, move (ns), var); v.null = false; } else v.type = &t; v.type.store (&t, mo); } else if (v.type != &t) { diag_record dr (fail); dr << "type mismatch"; if (var != nullptr) dr << " in variable " << var->name; dr << info << "value type is " << v.type->name; dr << info << (var != nullptr && &t == var->type ? "variable" : "new") << " type is " << t.name; } } void typify_atomic (context& ctx, value& v, const value_type& t, const variable* var) { // Typification is kind of like caching so we reuse that mutex shard. // shared_mutex& m ( ctx.mutexes->variable_cache[ hash () (&v) % ctx.mutexes->variable_cache_size]); // Note: v.type is rechecked by typify() under lock. // ulock l (m); typify (v, t, var, memory_order_release); } void untypify (value& v, bool reduce) { if (v.type == nullptr) return; if (v.null) { v.type = nullptr; return; } names ns; names_view nv (v.type->reverse (v, ns, reduce)); if (nv.empty () || nv.data () == ns.data ()) { // If the data is in storage, then we are all set. // ns.resize (nv.size ()); // Just to be sure. } else { // If the data is somewhere in the value itself, then steal it. // auto b (const_cast (nv.data ())); ns.assign (make_move_iterator (b), make_move_iterator (b + nv.size ())); } v = nullptr; // Free old data. v.type = nullptr; // Change type. v.assign (move (ns), nullptr); // Assign new data. } [[noreturn]] void convert_throw (const value_type* from, const value_type& to) { string m ("invalid "); m += to.name; m += " value: "; if (from != nullptr) { m += "conversion from "; m += from->name; } else m += "null"; throw invalid_argument (move (m)); } // Throw invalid_argument for an invalid simple value. // [[noreturn]] static void throw_invalid_argument (const name& n, const name* r, const char* type, bool pair_ok = false) { string m; string t (type); // Note that the message should be suitable for appending "in variable X". // if (!pair_ok && r != nullptr) m = "pair in " + t + " value"; else if (n.pattern || (r != nullptr && r->pattern)) m = "pattern in " + t + " value"; else { m = "invalid " + t + " value "; if (n.simple ()) m += '\'' + n.value + '\''; else if (n.directory ()) m += '\'' + n.dir.representation () + '\''; else m += "name '" + to_string (n) + '\''; } throw invalid_argument (move (m)); } // names // const names& value_traits::empty_instance = empty_names; // bool value // bool value_traits:: convert (const name& n, const name* r) { if (r == nullptr && !n.pattern && n.simple ()) { const string& s (n.value); if (s == "true") return true; if (s == "false") return false; // Fall through. } throw_invalid_argument (n, r, "bool"); } const char* const value_traits::type_name = "bool"; const value_type value_traits::value_type { type_name, sizeof (bool), nullptr, // No base. false, // Not container. nullptr, // No element. nullptr, // No dtor (POD). nullptr, // No copy_ctor (POD). nullptr, // No copy_assign (POD). &simple_assign, &simple_append, &simple_append, // Prepend same as append. &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, nullptr, // Never empty. nullptr, // Subscript. nullptr // Iterate. }; // int64_t value // int64_t value_traits:: convert (const name& n, const name* r) { if (r == nullptr && !n.pattern && n.simple ()) { try { const string& v (n.value); if (!wspace (v[0])) { // Note that unlike uint64, we don't support hex notation for int64. // May throw invalid_argument or out_of_range. // size_t i; int64_t r (stoll (v, &i)); if (i == v.size ()) return r; // Fall through. } // Fall through. } catch (const std::exception&) { // Fall through. } } throw_invalid_argument (n, r, "int64"); } const char* const value_traits::type_name = "int64"; const value_type value_traits::value_type { type_name, sizeof (int64_t), nullptr, // No base. false, // Not container. nullptr, // No element. nullptr, // No dtor (POD). nullptr, // No copy_ctor (POD). nullptr, // No copy_assign (POD). &simple_assign, &simple_append, &simple_append, // Prepend same as append. &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, nullptr, // Never empty. nullptr, // Subscript. nullptr // Iterate. }; // uint64_t value // uint64_t value_traits:: convert (const name& n, const name* r) { if (r == nullptr && !n.pattern && n.simple ()) { try { const string& v (n.value); if (!wspace (v[0])) { // Note: see also similar code in to_json_value(). // int b (v[0] == '0' && (v[1] == 'x' || v[1] == 'X') ? 16 : 10); // May throw invalid_argument or out_of_range. // size_t i; uint64_t r (stoull (v, &i, b)); if (i == v.size ()) return r; // Fall through. } // Fall through. } catch (const std::exception&) { // Fall through. } } throw_invalid_argument (n, r, "uint64"); } const char* const value_traits::type_name = "uint64"; const value_type value_traits::value_type { type_name, sizeof (uint64_t), nullptr, // No base. false, // Not container. nullptr, // No element. nullptr, // No dtor (POD). nullptr, // No copy_ctor (POD). nullptr, // No copy_assign (POD). &simple_assign, &simple_append, &simple_append, // Prepend same as append. &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, nullptr, // Never empty. nullptr, // Subscript. nullptr // Iterate. }; // string value // string value_traits:: convert (name&& n, name* r) { // The goal is to reverse the name into its original representation. The // code is a bit convoluted because we try to avoid extra allocations for // the common cases (unqualified, unpaired simple name or directory). // // We can only convert project-qualified untyped names. // if (n.pattern || n.typed ()) throw_invalid_argument (n, nullptr, "string"); if (r != nullptr) { if (r->pattern || r->typed ()) throw_invalid_argument (*r, nullptr, "string"); } string s; if (n.simple (true)) s.swap (n.value); else { // Note that here we cannot assume what's in dir is really a // path (think s/foo/bar/) so we have to reverse it exactly. // s = move (n.dir).representation (); // Move out of path. if (!n.value.empty ()) s += n.value; // Separator is already there. } // Convert project qualification to its string representation. // if (n.qualified ()) { string p (move (*n.proj).string ()); p += '%'; p += s; p.swap (s); } // The same for the RHS of a pair, if we have one. // if (r != nullptr) { s += '@'; if (r->qualified ()) { s += r->proj->string (); s += '%'; } if (r->simple (true)) s += r->value; else { s += move (r->dir).representation (); if (!r->value.empty ()) s += r->value; } } return s; } const string& value_traits::empty_instance = empty_string; const char* const value_traits::type_name = "string"; const value_type value_traits::value_type { type_name, sizeof (string), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, &simple_append, &simple_prepend, &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // path value // path value_traits:: convert (name&& n, name* r) { if (r == nullptr && !n.pattern) { // A directory path is a path. // if (n.directory ()) return move (n.dir); if (n.simple ()) { try { return path (move (n.value)); } catch (invalid_path& e) { n.value = move (e.path); // Restore the name object for diagnostics. // Fall through. } } // Reassemble split dir/value. // if (n.untyped () && n.unqualified ()) { try { return n.dir / n.value; } catch (const invalid_path&) { // Fall through. } } // Fall through. } throw_invalid_argument (n, r, "path"); } const path& value_traits::empty_instance = empty_path; const char* const value_traits::type_name = "path"; const value_type value_traits::value_type { type_name, sizeof (path), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, &simple_append, &simple_prepend, &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // dir_path value // dir_path value_traits:: convert (name&& n, name* r) { if (r == nullptr && !n.pattern) { if (n.directory ()) return move (n.dir); if (n.simple ()) { try { return dir_path (move (n.value)); } catch (invalid_path& e) { n.value = move (e.path); // Restore the name object for diagnostics. // Fall through. } } // Reassemble split dir/value. // if (n.untyped () && n.unqualified ()) { try { n.dir /= n.value; return move (n.dir); } catch (const invalid_path&) { // Fall through. } } // Fall through. } throw_invalid_argument (n, r, "dir_path"); } const dir_path& value_traits::empty_instance = empty_dir_path; const char* const value_traits::type_name = "dir_path"; const value_type value_traits::value_type { type_name, sizeof (dir_path), &value_traits::value_type, // Base (assuming direct cast works for // both). false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, &simple_append, &simple_prepend, &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // abs_dir_path value // abs_dir_path value_traits:: convert (name&& n, name* r) { if (r == nullptr && !n.pattern && (n.simple () || n.directory ())) { try { dir_path d (n.simple () ? dir_path (move (n.value)) : move (n.dir)); if (!d.empty ()) { if (d.relative ()) d.complete (); d.normalize (true); // Actualize. } return abs_dir_path (move (d)); } catch (invalid_path& e) { // We moved from name so reconstruct the path. Let's always make it // simple since we may not be able to construct dir_path. Should be // good enough for diagnostics. // n.value = move (e.path); } } throw_invalid_argument (n, r, "abs_dir_path"); } const char* const value_traits::type_name = "abs_dir_path"; const value_type value_traits::value_type { type_name, sizeof (abs_dir_path), &value_traits::value_type, // Base (assuming direct cast works // for both). false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, &simple_append, nullptr, // No prepend. &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // name value // name value_traits:: convert (name&& n, name* r) { if (r == nullptr && !n.pattern) return move (n); throw_invalid_argument (n, r, "name"); } static names_view name_reverse (const value& v, names&, bool reduce) { const name& n (v.as ()); return reduce && n.empty () ? names_view (nullptr, 0) : names_view (&n, 1); } const char* const value_traits::type_name = "name"; const value_type value_traits::value_type { type_name, sizeof (name), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, nullptr, // Append not supported. nullptr, // Prepend not supported. &name_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // name_pair // name_pair value_traits:: convert (name&& n, name* r) { if (n.pattern || (r != nullptr && r->pattern)) throw_invalid_argument (n, r, "name_pair", true /* pair_ok */); n.pair = '\0'; // Keep "unpaired" in case r is empty. return name_pair (move (n), r != nullptr ? move (*r) : name ()); } static void name_pair_assign (value& v, names&& ns, const variable* var) { using traits = value_traits; size_t n (ns.size ()); if (n <= 2) { try { traits::assign ( v, (n == 0 ? name_pair () : traits::convert (move (ns[0]), n == 2 ? &ns[1] : nullptr))); return; } catch (const invalid_argument&) {} // Fall through. } diag_record dr (fail); dr << "invalid name_pair value '" << ns << "'"; if (var != nullptr) dr << " in variable " << var->name; } static names_view name_pair_reverse (const value& v, names& ns, bool reduce) { const name_pair& p (v.as ()); const name& f (p.first); const name& s (p.second); if (reduce && f.empty () && s.empty ()) return names_view (nullptr, 0); if (f.empty ()) return names_view (&s, 1); if (s.empty ()) return names_view (&f, 1); ns.push_back (f); ns.back ().pair = '@'; ns.push_back (s); return ns; } const char* const value_traits::type_name = "name_pair"; const value_type value_traits::value_type { type_name, sizeof (name_pair), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &name_pair_assign, nullptr, // Append not supported. nullptr, // Prepend not supported. &name_pair_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // process_path value // template static T process_path_convert (name&& n, name* r, const char* what) { if ( !n.pattern && n.untyped () && n.unqualified () && !n.empty () && (r == nullptr || (!r->pattern && r->untyped () && r->unqualified () && !r->empty ()))) { path rp (move (n.dir)); if (rp.empty ()) rp = path (move (n.value)); else rp /= n.value; path ep; if (r != nullptr) { ep = move (r->dir); if (ep.empty ()) ep = path (move (r->value)); else ep /= r->value; } T pp (nullptr, move (rp), move (ep)); pp.initial = pp.recall.string ().c_str (); return pp; } throw_invalid_argument (n, r, what, true /* pair_ok */); } process_path value_traits:: convert (name&& n, name* r) { return process_path_convert (move (n), r, "process_path"); } static void process_path_assign (value& v, names&& ns, const variable* var) { using traits = value_traits; size_t n (ns.size ()); if (n <= 2) { try { traits::assign ( v, (n == 0 ? process_path () : traits::convert (move (ns[0]), n == 2 ? &ns[1] : nullptr))); return; } catch (const invalid_argument&) {} // Fall through. } diag_record dr (fail); dr << "invalid process_path value '" << ns << "'"; if (var != nullptr) dr << " in variable " << var->name; } template static void process_path_copy_ctor (value& l, const value& r, bool m) { const auto& rhs (r.as ()); if (m) new (&l.data_) T (move (const_cast (rhs))); else { auto& lhs ( *new (&l.data_) T ( nullptr, path (rhs.recall), path (rhs.effect))); lhs.initial = lhs.recall.string ().c_str (); } } static void process_path_copy_assign (value& l, const value& r, bool m) { auto& lhs (l.as ()); const auto& rhs (r.as ()); if (m) lhs = move (const_cast (rhs)); else { lhs.recall = rhs.recall; lhs.effect = rhs.effect; lhs.initial = lhs.recall.string ().c_str (); } } static void process_path_reverse_impl (const process_path& x, names& s) { s.push_back (name (x.recall.directory (), string (), x.recall.leaf ().string ())); if (!x.effect.empty ()) { s.back ().pair = '@'; s.push_back (name (x.effect.directory (), string (), x.effect.leaf ().string ())); } } static names_view process_path_reverse (const value& v, names& s, bool) { const auto& x (v.as ()); // Note that strictly speaking process_path doesn't have empty // representation (see convert() above). Thus we always return reduced // representation. // if (!x.empty ()) { s.reserve (x.effect.empty () ? 1 : 2); process_path_reverse_impl (x, s); } return s; } const char* const value_traits::type_name = "process_path"; const value_type value_traits::value_type { type_name, sizeof (process_path), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &process_path_copy_ctor, &process_path_copy_assign, &process_path_assign, nullptr, // Append not supported. nullptr, // Prepend not supported. &process_path_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // process_path_ex value // process_path_ex value_traits:: convert (names&& ns) { if (ns.empty ()) return process_path_ex (); bool p (ns[0].pair); process_path_ex pp ( process_path_convert ( move (ns[0]), p ? &ns[1] : nullptr, "process_path_ex")); for (auto i (ns.begin () + (p ? 2 : 1)); i != ns.end (); ++i) { if (!i->pair) throw invalid_argument ("non-pair in process_path_ex value"); if (i->pattern || !i->simple ()) throw_invalid_argument (*i, nullptr, "process_path_ex"); const string& k ((i++)->value); // NOTE: see also find_end() below. // if (k == "name") { if (i->pattern || !i->simple ()) throw_invalid_argument (*i, nullptr, "process_path_ex name"); pp.name = move (i->value); } else if (k == "checksum") { if (i->pattern || !i->simple ()) throw_invalid_argument ( *i, nullptr, "process_path_ex executable checksum"); pp.checksum = move (i->value); } else if (k == "env-checksum") { if (i->pattern || !i->simple ()) throw_invalid_argument ( *i, nullptr, "process_path_ex environment checksum"); pp.env_checksum = move (i->value); } else throw invalid_argument ( "unknown key '" + k + "' in process_path_ex value"); } return pp; } names::iterator value_traits:: find_end (names& ns) { auto b (ns.begin ()), i (b), e (ns.end ()); for (i += i->pair ? 2 : 1; i != e && i->pair; i += 2) { if (!i->simple () || (i->value != "name" && i->value != "checksum" && i->value != "env-checksum")) break; } return i; } static void process_path_ex_assign (value& v, names&& ns, const variable* var) { using traits = value_traits; try { traits::assign (v, traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid process_path_ex value"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void process_path_ex_copy_ex (value& l, const value& r, bool m) { auto& lhs (l.as ()); if (m) { const auto& rhs (const_cast (r).as ()); lhs.name = move (rhs.name); lhs.checksum = move (rhs.checksum); lhs.env_checksum = move (rhs.env_checksum); } else { const auto& rhs (r.as ()); lhs.name = rhs.name; lhs.checksum = rhs.checksum; lhs.env_checksum = rhs.env_checksum; } } static void process_path_ex_copy_ctor (value& l, const value& r, bool m) { process_path_copy_ctor (l, r, m); if (!m) process_path_ex_copy_ex (l, r, false); } static void process_path_ex_copy_assign (value& l, const value& r, bool m) { process_path_copy_assign (l, r, m); process_path_ex_copy_ex (l, r, m); } static names_view process_path_ex_reverse (const value& v, names& s, bool) { const auto& x (v.as ()); // Note that process_path_ex only has reduced empty representation (see // convert() above). // if (!x.empty ()) { s.reserve ((x.effect.empty () ? 1 : 2) + (x.name ? 2 : 0) + (x.checksum ? 2 : 0) + (x.env_checksum ? 2 : 0)); process_path_reverse_impl (x, s); if (x.name) { s.push_back (name ("name")); s.back ().pair = '@'; s.push_back (name (*x.name)); } if (x.checksum) { s.push_back (name ("checksum")); s.back ().pair = '@'; s.push_back (name (*x.checksum)); } if (x.env_checksum) { s.push_back (name ("env-checksum")); s.back ().pair = '@'; s.push_back (name (*x.env_checksum)); } } return s; } const char* const value_traits::type_name = "process_path_ex"; const value_type value_traits::value_type { type_name, sizeof (process_path_ex), &value_traits< // Base (assuming direct cast works process_path>::value_type, // for both). false, // Not container. nullptr, // No element. &default_dtor, &process_path_ex_copy_ctor, &process_path_ex_copy_assign, &process_path_ex_assign, nullptr, // Append not supported. nullptr, // Prepend not supported. &process_path_ex_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, // For now compare as process_path. &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // target_triplet value // target_triplet value_traits:: convert (name&& n, name* r) { if (r == nullptr && !n.pattern && n.simple ()) { try { return n.empty () ? target_triplet () : target_triplet (n.value); } catch (const invalid_argument& e) { throw invalid_argument ( string ("invalid target_triplet value: ") + e.what ()); } } throw_invalid_argument (n, r, "target_triplet"); } const char* const value_traits::type_name = "target_triplet"; const value_type value_traits::value_type { type_name, sizeof (target_triplet), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, nullptr, // Append not supported. nullptr, // Prepend not supported. &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // project_name value // project_name value_traits:: convert (name&& n, name* r) { if (r == nullptr && !n.pattern && n.simple ()) { try { return n.empty () ? project_name () : project_name (move (n.value)); } catch (const invalid_argument& e) { throw invalid_argument ( string ("invalid project_name value: ") + e.what ()); } } throw_invalid_argument (n, r, "project_name"); } const project_name& value_traits::empty_instance = empty_project_name; const char* const value_traits::type_name = "project_name"; const value_type value_traits::value_type { type_name, sizeof (project_name), nullptr, // No base. false, // Not container. nullptr, // No element. &default_dtor, &default_copy_ctor, &default_copy_assign, &simple_assign, nullptr, // Append not supported. nullptr, // Prepend not supported. &simple_reverse, nullptr, // No cast (cast data_ directly). &simple_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // json // static string to_string_value (name& n, const char* what) { if (n.typed () || n.qualified () || n.pattern) throw_invalid_argument (n, nullptr, what); string s; if (n.simple ()) s.swap (n.value); else { // Note that here we cannot assume what's in dir is really a path (think // s/foo/bar/) so we have to reverse it exactly. // s = move (n.dir).representation (); // Move out of path. if (!n.value.empty ()) s += n.value; // Separator is already there. } return s; } static json_value to_json_value (name& n, const char* what) { if (n.typed () || n.qualified () || n.pattern) throw_invalid_argument (n, nullptr, what); string s; if (n.simple ()) s.swap (n.value); else { // Note that here we cannot assume what's in dir is really a path (think // s/foo/bar/) so we have to reverse it exactly. // s = move (n.dir).representation (); // Move out of path. if (!n.value.empty ()) s += n.value; // Separator is already there. // A path is always interpreted as a JSON string. // return json_value (move (s)); } bool f; if (s.empty ()) return json_value (string ()); if (s == "null") return json_value (); else if ((f = (s == "true")) || s == "false") return json_value (f); else if (s.find_first_not_of ( "0123456789", (f = (s[0] == '-')) ? 1 : 0) == string::npos) { name n (move (s)); return f ? json_value (value_traits::convert (n, nullptr)) : json_value (value_traits::convert (n, nullptr)); } // // Handle the hex notation similar to ::convert() (and JSON5). // else if (s[0] == '0' && (s[1] == 'x' || s[1] == 'X') && s.size () > 2 && s.find_first_not_of ("0123456789aAbBcCdDeEfF", 2) == string::npos) { return json_value ( value_traits::convert (name (move (s)), nullptr), true /* hex */); } else { // If this is not a JSON representation of string, array, or object, // then treat it as a string. // // Note that the special `"`, `{`, and `[` characters could be preceded // with whitespaces. Note: see similar test in json_object below. // size_t p (s.find_first_not_of (" \t\n\r")); if (p == string::npos || (s[p] != '"' && s[p] != '{' && s[p] != '[')) return json_value (move (s)); // Parse as valid JSON input text. // #ifndef BUILD2_BOOTSTRAP try { json_parser p (s, nullptr /* name */); return json_value (p); } catch (const invalid_json_input& e) { // Turned out printing line/column/offset can be misleading since we // could be parsing a single name from a potential list of names. // feels like without also printing the value this is of not much use. // #if 0 string m ("invalid json input at line "); m += to_string (e.line); m += ", column "; m += to_string (e.column); m += ", byte offset "; m += to_string (e.position); m += ": "; m += e.what (); #else string m ("invalid json input: "); m += e.what (); #endif throw invalid_argument (move (m)); } #else throw invalid_argument ("json parsing requested during bootstrap"); #endif } } json_value value_traits:: convert (name&& l, name* r) { // Here we expect either a simple value or a serialized representation. // if (r != nullptr) throw invalid_argument ("pair in json element value"); return to_json_value (l, "json element"); } json_value value_traits:: convert (names&& ns) { size_t n (ns.size ()); if (n == 0) { // Note: this is the ([json] ) case, not ([json] ""). See also the // relevant note in json_reverse() below. // return json_value (); // null } else if (n == 1) { return to_json_value (ns.front (), "json"); } else { if (ns.front ().pair) // object { json_value r (json_type::object); r.object.reserve (n / 2); for (auto i (ns.begin ()); i != ns.end (); ++i) { if (!i->pair) throw invalid_argument ( "expected pair in json member value '" + to_string (*i) + '\''); // Note that we could support JSON-quoted member names but it's // unclear why would someone want that (and if they do, they can // always specify JSON text instead). // // @@ The empty pair value ([json] one@ ) which is currently empty // string is inconsistent with empty value ([json] ) above which // is null. Maybe we could distinguish the one@ and one@"" cases // via type hints? // string n (to_string_value (*i, "json member name")); json_value v (to_json_value (*++i, "json member")); // Check for duplicates. One can use append/prepend to merge. // if (find_if (r.object.begin (), r.object.end (), [&n] (const json_member& m) { return m.name == n; }) != r.object.end ()) { throw invalid_argument ( "duplicate json object member '" + n + '\''); } r.object.push_back (json_member {move (n), move (v)}); } return r; } else // array { json_value r (json_type::array); r.array.reserve (n); for (name& n: ns) { if (n.pair) throw invalid_argument ( "unexpected pair in json array element value '" + to_string (n) + '\''); r.array.push_back (to_json_value (n, "json array element")); } return r; } } } static void json_assign (value& v, names&& ns, const variable* var) { using traits = value_traits; try { traits::assign (v, traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json value"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void json_append (value& v, names&& ns, const variable* var) { using traits = value_traits; try { traits::append (v, traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json value"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void json_prepend (value& v, names&& ns, const variable* var) { using traits = value_traits; try { traits::prepend (v, traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json value"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } name value_traits:: reverse (const json_value& v) { switch (v.type) { case json_type::null: { // Note that here we cannot return empty (e.g., to be consistent with // other places) because we treat empty name (as opposed to empty // names) as string, not null (see to_json_value() above). // // Thankfully this version of reverse() is only used when json_value // representation is needed as part of a container. Which means in // "consumption" contexts (e.g., result of subscript) null will still // decay to empty. // #if 1 return name ("null"); #else return name (); #endif } case json_type::boolean: { return name (v.boolean ? "true" : "false"); } case json_type::signed_number: { return value_traits::reverse (v.signed_number); } case json_type::unsigned_number: { return value_traits::reverse (v.unsigned_number); } case json_type::hexadecimal_number: { return name (to_string (v.unsigned_number, 16)); } case json_type::string: // // @@ Hm, it would be nice if this somehow got mapped to unquoted // string but still be round-trippable to JSON value. Perhaps via // the type hint idea? This is pretty bad. See also subscript we // hacked around this somewhat. // // Note that it may be tempting to fix this by only quoting strings // that would otherwise be mis-interpreted (null, true, all digits, // etc). But that would be worse: things would seem to work but // fall apart in the perhaps unlikely event of encountering one of // the problematic values. It is better to produce a consistent // result. // case json_type::array: case json_type::object: { // Serialize as JSON output text. // string o; #ifndef BUILD2_BOOTSTRAP try { // Disable pretty-printing so that the output is all on the same // line. While it's not going to be easy to read for larger JSON // outputs, it will fit better into the existing model where none of // the value representations use formatting newlines. If a pretty- // printed representation is required, then the $json.serialize() // function can be used to obtain it. // json_buffer_serializer s (o, 0 /* indentation */); v.serialize (s); } catch (const invalid_json_output& e) { // Note that while it feels like value_traits::reverse() should // throw invalid_argument, we don't currently handle it anywhere so // for now let's just fail. // // Note: the same diagnostics as in $json.serialize(). // diag_record dr; dr << fail << "invalid json value: " << e; if (e.event) dr << info << "while serializing " << to_string (*e.event); if (e.offset != string::npos) dr << info << "offending byte offset " << e.offset; } #else fail << "json serialization requested during bootstrap"; #endif return name (move (o)); } } assert (false); return name (); } static names_view json_reverse (const value& x, names& ns, bool reduce) { const json_value& v (x.as ()); // @@ Hm, it would be nice if JSON null somehow got mapped to [null]/empty // but still be round-trippable to JSON null. Perhaps via type hint? // // But won't `print ([json] null)` printing nothing be surprising. // Also, it's not clear that mapping JSON null to out [null] is a good // idea since our [null] means "no value" while JSON null means "null // value". // // Maybe the current semantics is the best: we map our [null] and empty // names to JSON null (naturally) but we always reverse JSON null to // the JSON "null" literal. Or maybe we could reverse it to null but // type-hint it that it's a spelling or [null]/empty. Quite fuzzy, // admittedly. In our model null values decay to empty so JSON null // decaying to "null" literal is strange. Let's try and see how it // goes. See also json_subscript_impl() below. // if (v.type != json_type::null || !reduce) ns.push_back (value_traits::reverse (v)); return ns; } static int json_compare (const value& l, const value& r) { return l.as ().compare (r.as ()); } // Return the value as well as the indication of whether the index/name is // in range. // static pair json_subscript_impl (const value& val, value* val_data, uint64_t i, const string& n, bool index) { const json_value& jv (val.as ()); json_value jr; if (index) { if (i >= (jv.type == json_type::array ? jv.array.size () : jv.type == json_type::object ? jv.object.size () : jv.type == json_type::null ? 0 : 1)) return make_pair (value (), false); switch (jv.type) { case json_type::boolean: case json_type::signed_number: case json_type::unsigned_number: case json_type::hexadecimal_number: case json_type::string: { // Steal the value if possible. // jr = (&val == val_data ? json_value (move (const_cast (jv))) : json_value (jv)); break; } case json_type::array: { // Steal the value if possible. // const json_value& r (jv.array[i]); jr = (&val == val_data ? json_value (move (const_cast (r))) : json_value (r)); break; } case json_type::object: { // Represent as an object with one member. // new (&jr.object) json_value::object_type (); jr.type = json_type::object; // Steal the member if possible. // const json_member& m (jv.object[i]); jr.object.push_back (&val == val_data ? json_member (move (const_cast (m))) : json_member (m)); break; } case json_type::null: assert (false); } } else { auto i (find_if (jv.object.begin (), jv.object.end (), [&n] (const json_member& m) { return m.name == n; })); if (i == jv.object.end ()) return make_pair (value (), false); // Steal the member value if possible. // jr = (&val == val_data ? json_value (move (const_cast (i->value))) : json_value (i->value)); } // @@ As a temporary work around for the lack of type hints (see // json_reverse() for background), reverse simple JSON values to the // corresponding fundamental type values. The thinking here is that // subscript (and iteration) is primarily meant for consumption (as // opposed to reverse() where it is used to build up values and thus // needs things to be fully reversible). Once we add type hints, then // this should become unnecessary and we should be able to just always // return json_value. See also $json.member_value() where we do the // same thing. // // @@ TODO: split this function into two (index/name) once get rid of this. // value r; switch (jr.type) { // Seeing that we are reversing for consumption, it feels natural to // reverse JSON null to our [null] rather than empty. This, in // particular, helps chained subscript. // #if 0 case json_type::null: r = value (names {}); break; #else case json_type::null: r = value (); break; #endif case json_type::boolean: r = value (jr.boolean); break; case json_type::signed_number: r = value (jr.signed_number); break; case json_type::unsigned_number: case json_type::hexadecimal_number: r = value (jr.unsigned_number); break; case json_type::string: r = value (move (jr.string)); break; case json_type::array: case json_type::object: r = value (move (jr)); break; } return make_pair (move (r), true); } static value json_subscript (const value& val, value* val_data, value&& sub, const location& sloc, const location& bloc) { const json_value* jv (val.null ? nullptr : &val.as ()); // For consistency with other places treat JSON null value as maybe // missing array/object. In particular, we don't want to fail trying to // lookup by-name on a null value which could have been an object. // if (jv != nullptr && jv->type == json_type::null) jv = nullptr; // Process subscript even if the value is null to make sure it is valid. // bool index; uint64_t i (0); string n; // Always interpret uint64-typed subscript as index even for objects. // This can be used to, for example, to iterate with an index over object // members. // if (!sub.null && sub.type == &value_traits::value_type) { i = sub.as (); index = true; } else { // How we interpret the subscript depends on the JSON value type. For // objects we treat it as a string (member name) and for everything else // as an index. // // What if the value is null and we don't have a JSON type? In this case // we treat as a string since a valid number is also a valid string. // try { if (jv == nullptr || jv->type == json_type::object) { n = convert (move (sub)); index = false; } else { i = convert (move (sub)); index = true; } } catch (const invalid_argument& e) { // We will likely be trying to interpret a member name as an integer // due to the incorrect value type so issue appropriate diagnostics. // diag_record dr; dr << fail (sloc) << "invalid json value subscript: " << e; if (jv != nullptr && jv->type != json_type::object) dr << info << "json value type is " << jv->type; dr << info (bloc) << "use the '\\[' escape sequence if this is a " << "wildcard pattern" << endf; } } value r (jv != nullptr ? json_subscript_impl (val, val_data, i, n, index).first : value ()); // Typify null values so that we get called for chained subscripts. // if (r.null) r.type = &value_traits::value_type; return r; } static bool json_iterate (const value& val, const function& f) { // Implement in terms of subscript for consistency (in particular, // iterating over simple values like number, string). // for (uint64_t i (0);; ++i) { pair e (json_subscript_impl (val, nullptr, i, {}, true)); if (!e.second) break; if (!f (move (e.first), i == 0)) return false; } return true; } const json_value value_traits::empty_instance; const char* const value_traits::type_name = "json"; // Note that whether the json value is a container or not depends on its // payload type. However, for our purposes it feels correct to assume it is // a container rather than not with itself as the element type (see // value_traits::{container, element_type} usage for details). // const value_type value_traits::value_type { type_name, sizeof (json_value), nullptr, // No base. true, // Container. &value_traits::value_type, // Element (itself). &default_dtor, &default_copy_ctor, &default_copy_assign, &json_assign, json_append, json_prepend, &json_reverse, nullptr, // No cast (cast data_ directly). &json_compare, &default_empty, &json_subscript, &json_iterate }; // json_array // json_array value_traits:: convert (names&& ns) { json_array r; size_t n (ns.size ()); if (n == 0) ; // Empty. else if (n == 1) { // Tricky: this can still be JSON input text that is an array. And if // it's not, then make it an element of an array. // // @@ Hm, this is confusing: [json_array] a = null ! Maybe not? But then // this won't work: [json_array] a = ([json_array] null). Maybe // distinguish in assign? // json_value v (to_json_value (ns.front (), "json")); if (v.type == json_type::array) r.array = move (v.array); else r.array.push_back (move (v)); } else { r.array.reserve (n); for (name& n: ns) { if (n.pair) throw invalid_argument ( "unexpected pair in json array element value '" + to_string (n) + '\''); r.array.push_back (to_json_value (n, "json array element")); } } return r; } static void json_array_assign (value& v, names&& ns, const variable* var) { using traits = value_traits; try { traits::assign (v, traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json array"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void json_array_append (value& v, names&& ns, const variable* var) { using val_traits = value_traits; using arr_traits = value_traits; try { arr_traits::append (v, val_traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json array"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void json_array_prepend (value& v, names&& ns, const variable* var) { using val_traits = value_traits; using arr_traits = value_traits; try { arr_traits::prepend (v, val_traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json array"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } const json_array value_traits::empty_instance; const char* const value_traits::type_name = "json_array"; const value_type value_traits::value_type { type_name, sizeof (json_array), &value_traits::value_type, // Base (assuming direct cast works // for both). true, // Container. &value_traits::value_type, // Element (json_value). &default_dtor, &default_copy_ctor, &default_copy_assign, &json_array_assign, &json_array_append, &json_array_prepend, &json_reverse, nullptr, // No cast (cast data_ directly). &json_compare, &default_empty, &json_subscript, &json_iterate }; // json_object // json_object value_traits:: convert (names&& ns) { json_object r; size_t n (ns.size ()); if (n == 0) ; // Empty. else if (n == 1) { // Tricky: this can still be JSON input text that is an object. So do // a similar check as in to_json_value() above. // name& n (ns.front ()); if (!n.simple () || n.pattern) throw_invalid_argument (n, nullptr, "json object"); string& s (n.value); size_t p (s.find_first_not_of (" \t\n\r")); if (p == string::npos || s[p] != '{') { // Unlike for array above, we cannot turn any value into a member. // throw invalid_argument ("expected json object instead of '" + s + '\''); } json_value v (to_json_value (ns.front (), "json object")); assert (v.type == json_type::object); r.object = move (v.object); } else { r.object.reserve (n / 2); for (auto i (ns.begin ()); i != ns.end (); ++i) { if (!i->pair) throw invalid_argument ( "expected pair in json member value '" + to_string (*i) + '\''); string n (to_string_value (*i, "json member name")); json_value v (to_json_value (*++i, "json member")); if (find_if (r.object.begin (), r.object.end (), [&n] (const json_member& m) { return m.name == n; }) != r.object.end ()) { throw invalid_argument ( "duplicate json object member '" + n + '\''); } r.object.push_back (json_member {move (n), move (v)}); } } return r; } static void json_object_assign (value& v, names&& ns, const variable* var) { using traits = value_traits; try { traits::assign (v, traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json object"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void json_object_append (value& v, names&& ns, const variable* var) { using val_traits = value_traits; using obj_traits = value_traits; try { obj_traits::append (v, val_traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json object"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } static void json_object_prepend (value& v, names&& ns, const variable* var) { using val_traits = value_traits; using obj_traits = value_traits; try { obj_traits::prepend (v, val_traits::convert (move (ns))); } catch (const invalid_argument& e) { // Note: ns is not guaranteed to be valid. // diag_record dr (fail); dr << "invalid json object"; if (var != nullptr) dr << " in variable " << var->name; dr << ": " << e; } } const json_object value_traits::empty_instance; const char* const value_traits::type_name = "json_object"; const value_type value_traits::value_type { type_name, sizeof (json_object), &value_traits::value_type, // Base (assuming direct cast works // for both). true, // Container. &value_traits::value_type, // Element (json_value). &default_dtor, &default_copy_ctor, &default_copy_assign, &json_object_assign, &json_object_append, &json_object_prepend, &json_reverse, nullptr, // No cast (cast data_ directly). &json_compare, &default_empty, &json_subscript, &json_iterate }; // cmdline // cmdline value_traits:: convert (names&& ns) { return cmdline (make_move_iterator (ns.begin ()), make_move_iterator (ns.end ())); } void value_traits:: assign (value& v, cmdline&& x) { if (v) v.as () = move (x); else new (&v.data_) cmdline (move (x)); } void value_traits:: append (value& v, cmdline&& x) { if (v) { cmdline& p (v.as ()); if (p.empty ()) p.swap (x); else p.insert (p.end (), make_move_iterator (x.begin ()), make_move_iterator (x.end ())); } else new (&v.data_) cmdline (move (x)); } void value_traits:: prepend (value& v, cmdline&& x) { if (v) { cmdline& p (v.as ()); if (!p.empty ()) x.insert (x.end (), make_move_iterator (p.begin ()), make_move_iterator (p.end ())); p.swap (x); } else new (&v.data_) cmdline (move (x)); } static void cmdline_assign (value& v, names&& ns, const variable*) { if (!v) { new (&v.data_) cmdline (); v.null = false; } v.as ().assign (make_move_iterator (ns.begin ()), make_move_iterator (ns.end ())); } static void cmdline_append (value& v, names&& ns, const variable*) { if (!v) { new (&v.data_) cmdline (); v.null = false; } auto& x (v.as ()); x.insert (x.end (), make_move_iterator (ns.begin ()), make_move_iterator (ns.end ())); } static void cmdline_prepend (value& v, names&& ns, const variable*) { if (!v) { new (&v.data_) cmdline (); v.null = false; } auto& x (v.as ()); x.insert (x.begin (), make_move_iterator (ns.begin ()), make_move_iterator (ns.end ())); } static names_view cmdline_reverse (const value& v, names&, bool) { const auto& x (v.as ()); return names_view (x.data (), x.size ()); } static int cmdline_compare (const value& l, const value& r) { return vector_compare (l, r); } const cmdline value_traits::empty_instance; const char* const value_traits::type_name = "cmdline"; const value_type value_traits::value_type { type_name, sizeof (cmdline), nullptr, // No base. true, // Container. &value_traits::value_type, // Element type. &default_dtor, &default_copy_ctor, &default_copy_assign, &cmdline_assign, &cmdline_append, &cmdline_prepend, &cmdline_reverse, nullptr, // No cast (cast data_ directly). &cmdline_compare, &default_empty, nullptr, // Subscript. nullptr // Iterate. }; // variable_pool // void variable_pool:: update (variable& var, const build2::value_type* t, const variable_visibility* v, const bool* o) const { assert (var.owner == this); if (outer_ != nullptr) { // Project-private variable. Assert visibility/overridability, the same // as in insert(). // assert ((o == nullptr || !*o) && (v == nullptr || *v >= variable_visibility::project)); } // Check overridability (all overrides, if any, should already have // been entered; see context ctor for details). // if (o != nullptr && var.overrides != nullptr && !*o) fail << "variable " << var.name << " cannot be overridden"; bool ut (t != nullptr && var.type != t); bool uv (v != nullptr && var.visibility != *v); // Variable should not be updated post-aliasing. // assert (var.aliases == &var || (!ut && !uv)); // Update type? // if (ut) { assert (var.type == nullptr); var.type = t; } // Change visibility? While this might at first seem like a bad idea, it // can happen that the variable lookup happens before any values were set // in which case the variable will be entered with the default (project) // visibility. // // For example, a buildfile, for some reason, could reference a target- // specific variable (say, test) before loading a module (say, test) that // sets this visibility. While strictly speaking we could have potentially // already made a lookup using the wrong visibility, presumably this // should be harmless. // // @@ But consider a situation where this test is set on scope prior to // loading the module: now this value will effectively be unreachable // without any diagnostics. So maybe we should try to clean this up. // But we will need proper diagnostics instead of assert (which means // we would probably need to track the location where the variable // was first entered). // // Note also that this won't work well for global visibility since we only // allow restrictions. The thinking is that global visibility is special // and requiring special arrangements (like variable patterns, similar to // how we've done it for config.*) is reasonable. In fact, it feels like // only the build system core should be allowed to use global visibility // (see the context ctor for details). // if (uv) { assert (*v > var.visibility); var.visibility = *v; } } static bool match_pattern (const string& n, const string& p, const string& s, bool multi) { size_t nn (n.size ()), pn (p.size ()), sn (s.size ()); if (nn < pn + sn + 1) return false; if (pn != 0) { if (n.compare (0, pn, p) != 0) return false; } if (sn != 0) { if (n.compare (nn - sn, sn, s) != 0) return false; } // Make sure the stem is a single name unless instructed otherwise. // return multi || string::traits_type::find (n.c_str () + pn, nn - pn - sn, '.') == nullptr; } static inline void merge_pattern (const variable_patterns::pattern& p, const build2::value_type*& t, const variable_visibility*& v, const bool*& o) { if (p.type) { if (t == nullptr) t = *p.type; else if (p.match) assert (t == *p.type); } if (p.visibility) { if (v == nullptr) v = &*p.visibility; else if (p.match) { // Allow the pattern to restrict but not relax. // if (*p.visibility > *v) v = &*p.visibility; else assert (*v == *p.visibility); } } if (p.overridable) { if (o == nullptr) o = &*p.overridable; else if (p.match) { // Allow the pattern to restrict but not relax. // if (*o) o = &*p.overridable; else assert (*o == *p.overridable); } } } pair variable_pool:: insert (string n, const build2::value_type* t, const variable_visibility* v, const bool* o, bool pat) { if (outer_ != nullptr) { // Project-private pool. // if (n.find ('.') != string::npos) // Qualified. return outer_->insert (move (n), t, v, o, pat); // Unqualified. // // The pool chaining semantics for insertion: first check the outer pool // then, if not found, insert in own pool. // if (const variable* var = outer_->find (n)) { // Verify type/visibility/overridability. // // Should we assert or fail? Currently the buildfile parser goes // through update() to set these so let's do assert for now. We also // require equality (these are a handful of special variables). // assert ((t == nullptr || t == var->type) && (v == nullptr || *v == var->visibility) && (o == nullptr || *o || var->overrides == nullptr)); return pair (const_cast (*var), false); } // Project-private variable. Assert visibility/overridability and fall // through. Again, we expect the buildfile parser to verify and diagnose // these. // // Note: similar code in update(). // assert ((o == nullptr || !*o) && (v == nullptr || *v >= variable_visibility::project)); } else if (shared_) { // Public pool. // // Make sure all the unqualified variables are pre-entered during // initialization. // assert (shared_->load_generation == 0 || n.find ('.') != string::npos); } assert (!shared_ || shared_->phase == run_phase::load); // Apply pattern. // using pattern = variable_patterns::pattern; const pattern* pa (nullptr); auto pt (t); auto pv (v); auto po (o); if (pat && patterns_ != nullptr) { if (n.find ('.') != string::npos) { // Reverse means from the "largest" (most specific). // for (const pattern& p: reverse_iterate (patterns_->patterns_)) { if (match_pattern (n, p.prefix, p.suffix, p.multi)) { merge_pattern (p, pt, pv, po); pa = &p; break; } } } } auto r ( insert ( variable { move (n), nullptr, nullptr, pt, nullptr, pv != nullptr ? *pv : variable_visibility::project})); variable& var (r.first->second); if (r.second) { var.owner = this; var.aliases = &var; } else // Note: overridden variable will always exist. { // This is tricky: if the pattern does not require a match, then we // should re-merge it with values that came from the variable. // bool vo; if (pa != nullptr && !pa->match) { pt = t != nullptr ? t : var.type; pv = v != nullptr ? v : &var.visibility; po = o != nullptr ? o : &(vo = true); merge_pattern (*pa, pt, pv, po); } if (po == nullptr) // NULL overridable falls back to false. po = &(vo = false); update (var, pt, pv, po); // Not changing the key. } return pair (var, r.second); } const variable& variable_pool:: insert_alias (const variable& var, string n) { if (outer_ != nullptr) { assert (n.find ('.') != string::npos); // Qualified. return outer_->insert_alias (var, move (n)); } assert (var.owner == this && var.aliases != nullptr && var.overrides == nullptr); variable& a (insert (move (n), var.type, &var.visibility, nullptr /* override */, false /* pattern */).first); assert (a.overrides == nullptr); if (a.aliases == &a) // Not aliased yet. { a.aliases = var.aliases; const_cast (var).aliases = &a; } else assert (a.alias (var)); // Make sure it is already an alias of var. return a; } void variable_patterns:: insert (const string& p, optional t, optional o, optional v, bool retro, bool match) { assert (!shared_ || shared_->phase == run_phase::load); size_t pn (p.size ()); size_t w (p.find ('*')); assert (w != string::npos); bool multi (w + 1 != pn && p[w + 1] == '*'); // Extract prefix and suffix. // string pfx, sfx; if (w != 0) { assert (p[w - 1] == '.' && w != 1); pfx.assign (p, 0, w); } w += multi ? 2 : 1; // First suffix character. size_t sn (pn - w); // Suffix length. if (sn != 0) { assert (p[w] == '.' && sn != 1); sfx.assign (p, w, sn); } auto i ( patterns_.insert ( pattern {move (pfx), move (sfx), multi, match, t, v, o})); // Apply retrospectively to existing variables. // if (retro && pool_ != nullptr) { for (auto& p: pool_->map_) { variable& var (p.second); if (match_pattern (var.name, i->prefix, i->suffix, i->multi)) { // Make sure that none of the existing more specific patterns // match. // auto j (i), e (patterns_.end ()); for (++j; j != e; ++j) { if (match_pattern (var.name, j->prefix, j->suffix, j->multi)) break; } if (j == e) pool_->update (var, t ? *t : nullptr, v ? &*v : nullptr, o ? &*o : nullptr); // Not changing the key. } } } } // variable_map // const variable_map empty_variable_map (variable_map::owner::empty); // Need scope/target definition thus not inline. // variable_map:: variable_map (const scope& s, bool shared) : shared_ (shared), owner_ (owner::scope), scope_ (&s), ctx (&s.ctx) { } variable_map:: variable_map (const target& t, bool shared) : shared_ (shared), owner_ (owner::target), target_ (&t), ctx (&t.ctx) { } variable_map:: variable_map (const prerequisite& p, bool shared) : shared_ (shared), owner_ (owner::prereq), prereq_ (&p), ctx (&p.scope.ctx) { } variable_map:: variable_map (variable_map&& v, const prerequisite& p, bool shared) : shared_ (shared), owner_ (owner::scope), prereq_ (&p), ctx (&p.scope.ctx), m_ (move (v.m_)) { } variable_map:: variable_map (const variable_map& v, const prerequisite& p, bool shared) : shared_ (shared), owner_ (owner::scope), prereq_ (&p), ctx (&p.scope.ctx), m_ (v.m_) { } lookup variable_map:: lookup (const string& name) const { lookup_type r; const scope* bs (owner_ == owner::scope ? scope_ : owner_ == owner::target ? &target_->base_scope () : owner_ == owner::prereq ? &prereq_->scope : nullptr); if (const variable* var = bs->var_pool ().find (name)) { auto p (lookup (*var)); r = lookup_type (p.first, &p.second, this); } return r; } auto variable_map:: lookup (const variable& var, bool typed, bool aliased) const -> pair { const variable* v (&var); const value_data* r (nullptr); do { // @@ Should we verify that there are no distinct values for aliases? // This can happen if the values were entered before the variables // were aliased. Possible but probably highly unlikely. // auto i (m_.find (*v)); if (i != m_.end ()) { r = &i->second; break; } if (aliased) v = v->aliases; } while (v != &var && v != nullptr); // Check if this is the first access after being assigned a type. // if (r != nullptr && typed && v->type != nullptr) typify (*r, *v); return pair ( r, r != nullptr ? *v : var); } auto variable_map:: lookup_to_modify (const variable& var, bool typed) -> pair { auto p (lookup (var, typed)); auto* r (const_cast (p.first)); if (r != nullptr) { r->extra = 0; r->version++; } return pair (r, p.second); } value& variable_map:: assign (const string& name) { assert (owner_ != owner::context); const scope* bs (owner_ == owner::scope ? scope_ : owner_ == owner::target ? &target_->base_scope () : owner_ == owner::prereq ? &prereq_->scope : nullptr); return insert (bs->var_pool ()[name]).first; } pair variable_map:: insert (const variable& var, bool typed, bool reset_extra) { assert (!shared_ || ctx->phase == run_phase::load); auto p (m_.emplace (var, value_data (typed ? var.type : nullptr))); value_data& r (p.first->second); if (!p.second) { if (reset_extra) r.extra = 0; // Check if this is the first access after being assigned a type. // // Note: we still need atomic in case this is not a shared state. // if (typed && var.type != nullptr) typify (r, var); } r.version++; return pair (r, p.second); } auto variable_map:: find (const string& name) const -> const_iterator { assert (owner_ != owner::context); const scope* bs (owner_ == owner::scope ? scope_ : owner_ == owner::target ? &target_->base_scope () : owner_ == owner::prereq ? &prereq_->scope : nullptr); const variable* var (bs->var_pool ().find (name)); return var != nullptr ? find (*var) : end (); } bool variable_map:: erase (const variable& var) { assert (!shared_ || ctx->phase == run_phase::load); return m_.erase (var) != 0; } variable_map::const_iterator variable_map:: erase (const_iterator i) { assert (!shared_ || ctx->phase == run_phase::load); return const_iterator (m_.erase (i), *this); } // variable_pattern_map // variable_map& variable_pattern_map:: insert (pattern_type type, string&& text) { // Note that this variable map is special and we use context as its owner // (see variable_map for details). // auto r (map_.emplace (pattern {type, false, move (text), {}}, variable_map (ctx, shared_))); // Compile the regex. // if (r.second && type == pattern_type::regex_pattern) { // On exception restore the text argument (so that it's available for // diagnostics) and remove the element from the map. // auto eg (make_exception_guard ( [&text, &r, this] () { text = r.first->first.text; map_.erase (r.first); })); const string& t (r.first->first.text); size_t n (t.size ()), p (t.rfind (t[0])); // Convert flags. // regex::flag_type f (regex::ECMAScript); for (size_t i (p + 1); i != n; ++i) { switch (t[i]) { case 'i': f |= regex::icase; break; case 'e': r.first->first.match_ext = true; break; } } // Skip leading delimiter as well as trailing delimiter and flags. // r.first->first.regex = regex (t.c_str () + 1, p - 1, f); } return r.first->second; } // variable_type_map // lookup variable_type_map:: find (const target_key& tk, const variable& var, optional& oname) const { // Compute and cache "effective" name that we will be matching. // // See also the additional match_ext logic below. // auto name = [&tk, &oname] () -> const string& { if (!oname) { oname = string (); tk.effective_name (*oname); } return oname->empty () ? *tk.name : *oname; }; // Search across target type hierarchy. // for (auto tt (tk.type); tt != nullptr; tt = tt->base) { auto i (map_.find (*tt)); if (i == end ()) continue; // Try to match the pattern, starting from the longest values. // const variable_pattern_map& m (i->second); for (auto j (m.rbegin ()); j != m.rend (); ++j) { using pattern = variable_pattern_map::pattern; using pattern_type = variable_pattern_map::pattern_type; const pattern& pat (j->first); bool r, e (false); if (pat.type == pattern_type::path) { r = pat.text == "*" || butl::path_match (name (), pat.text); } else { const string& n (name ()); // Deal with match_ext: first see if the extension would be added by // default. If not, then temporarily add it in oname and then clean // it up if there is no match (to prevent another pattern from using // it). While we may keep adding it if there are multiple patterns // with such a flag, we will at least reuse the buffer in oname. // e = pat.match_ext && tk.ext && !tk.ext->empty () && oname->empty (); if (e) { *oname = *tk.name; *oname += '.'; *oname += *tk.ext; } r = regex_match (e ? *oname : n, *pat.regex); } // Ok, this pattern matches. But is there a variable? // // Since we store append/prepend values untyped, instruct find() not // to automatically type it. And if it is assignment, then typify it // ourselves. // if (r) { const variable_map& vm (j->second); auto p (vm.lookup (var, false)); if (const variable_map::value_data* v = p.first) { // Check if this is the first access after being assigned a type. // if (v->extra == 0 && var.type != nullptr) vm.typify (*v, var); // Make sure the effective name is computed if this is // append/prepend (it is used as a cache key). // if (v->extra != 0 && !oname) name (); return lookup (*v, p.second, vm); } } if (e) oname->clear (); } } return lookup (); } template struct LIBBUILD2_DEFEXPORT value_traits; template struct LIBBUILD2_DEFEXPORT value_traits>; template struct LIBBUILD2_DEFEXPORT value_traits; template struct LIBBUILD2_DEFEXPORT value_traits; template struct LIBBUILD2_DEFEXPORT value_traits; template struct LIBBUILD2_DEFEXPORT value_traits; template struct LIBBUILD2_DEFEXPORT value_traits>>; template struct LIBBUILD2_DEFEXPORT value_traits>>>; template struct LIBBUILD2_DEFEXPORT value_traits>>>; template struct LIBBUILD2_DEFEXPORT value_traits, string>>>; template struct LIBBUILD2_DEFEXPORT value_traits, bool>>>; template struct LIBBUILD2_DEFEXPORT value_traits>; template struct LIBBUILD2_DEFEXPORT value_traits>; template struct LIBBUILD2_DEFEXPORT value_traits>; template struct LIBBUILD2_DEFEXPORT value_traits>; template struct LIBBUILD2_DEFEXPORT value_traits>>; template struct LIBBUILD2_DEFEXPORT value_traits, string>>; template struct LIBBUILD2_DEFEXPORT value_traits>>; template struct LIBBUILD2_DEFEXPORT value_traits>; }