// file : libbuild2/in/init.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include <libbuild2/in/init.hxx> #include <libbuild2/scope.hxx> #include <libbuild2/variable.hxx> #include <libbuild2/diagnostics.hxx> #include <libbuild2/in/rule.hxx> #include <libbuild2/in/target.hxx> using namespace std; namespace build2 { namespace in { static const rule rule_ ("in", "in"); bool base_init (scope& rs, scope&, const location&, bool first, bool, module_init_extra&) { tracer trace ("in::base_init"); l5 ([&]{trace << "for " << rs;}); assert (first); // Enter variables. // { // All the variables we enter are qualified so go straight for the // public variable pool. // auto& vp (rs.var_pool (true /* public */)); // Alternative variable substitution symbol with '$' being the // default. // vp.insert<string> ("in.symbol"); // Substitution mode. Valid values are 'strict' (default) and 'lax'. // In the strict mode every substitution symbol is expected to start a // substitution with the double symbol (e.g., $$) serving as an escape // sequence. // // In the lax mode a pair of substitution symbols is only treated as a // substitution if what's between them looks like a build2 variable // name (i.e., doesn't contain spaces, etc). Everything else, // including unterminated substitution symbols, is copied as is. Note // also that in this mode the double symbol is not treated as an // escape sequence. // // The lax mode is mostly useful when trying to reuse existing .in // files, for example, from autoconf. Note, however, that the lax mode // is still stricter than the autoconf's semantics which also leaves // unknown substitutions as is. // const variable& im (vp.insert<string> ("in.mode")); // Original name of this variable for backwards compatibility. // vp.insert_alias (im, "in.substitution"); // Substitution map. Substitutions can be specified as key-value pairs // rather than buildfile variables. This map is checked before the // variables. An absent value in key-value has the NULL semantics. // // This mechanism has two primary uses: Firstly, it allows us to have // substitution names that cannot be specified as buildfile variables. // For example, a name may start with an underscore and thus be // reserved or it may refer to one of the predefined variables such a // `include` or `extension` that may have a wrong visibility and/or // type. // // Secondly, this mechanism allows us to encapsulate a group of // substitutions and pass this group around as a single value. // vp.insert<map<string, optional<string>>> ("in.substitutions"); // Fallback value to use for NULL value substitutions. If unspecified, // NULL substitutions are an error. // vp.insert<string> ("in.null"); } // Register target types. // rs.insert_target_type<in> (); return true; } bool init (scope& rs, scope& bs, const location& loc, bool, bool, module_init_extra&) { tracer trace ("in::init"); l5 ([&]{trace << "for " << bs;}); // Load in.base. // load_module (rs, rs, "in.base", loc); // Register rules. // // There are rules that are "derived" from this generic in rule in // order to provide extended preprocessing functionality (see the // version module for an example). To make sure they are tried first // we register for path_target, not file, but in rule::match() we only // match if the target is a file. A bit of a hack. // bs.insert_rule<path_target> (perform_update_id, "in", rule_); bs.insert_rule<path_target> (perform_clean_id, "in", rule_); bs.insert_rule<path_target> (configure_update_id, "in", rule_); return true; } static const module_functions mod_functions[] = { // NOTE: don't forget to also update the documentation in init.hxx if // changing anything here. {"in.base", nullptr, base_init}, {"in", nullptr, init}, {nullptr, nullptr, nullptr} }; const module_functions* build2_in_load () { return mod_functions; } } }