// 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. // { auto& vp (rs.var_pool ()); // 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. // vp.insert<string> ("in.substitution"); } // 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; } } }