// file : libbuild2/cc/common.hxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #ifndef LIBBUILD2_CC_COMMON_HXX #define LIBBUILD2_CC_COMMON_HXX #include <libbuild2/types.hxx> #include <libbuild2/utility.hxx> #include <libbuild2/context.hxx> #include <libbuild2/variable.hxx> #include <libbuild2/bin/target.hxx> #include <libbuild2/cc/types.hxx> #include <libbuild2/cc/guess.hxx> // compiler_id #include <libbuild2/cc/target.hxx> // h{} #include <libbuild2/cc/export.hxx> namespace build2 { namespace cc { // Data entries that define a concrete c-family module (e.g., c or cxx). // These classes are used as a virtual bases by the rules as well as the // modules. This way the member variables can be referenced as is, without // any extra decorations (in other words, it is a bunch of data members // that can be shared between several classes/instances). // struct config_data { lang x_lang; const char* x; // Module name ("c", "cxx"). const char* x_name; // Compiler name ("c", "c++"). const char* x_obj_name; // Same for Objective-X ("obj-c", "obj-c++"). const char* x_default; // Compiler default ("gcc", "g++"). const char* x_pext; // Preprocessed source extension (".i", ".ii"). const char* x_obj_pext; // Same for Objective-X (".mi", ".mii"). // Array of modules that can hint us the toolchain, terminate with // NULL. // const char* const* x_hinters; // We set this variable on the bmi*{} target to indicate whether it // belongs to a binless library. More specifically, it controls both // the production and consumption (linking) of the object file with // the following possible states: // // produce consume // true y y (binless normal or sidebuild) // false n n (binful sidebuild) // absent y n (binful normal) // const variable& b_binless; // bin.binless const variable& config_x; const variable& config_x_id; // <type>[-<variant>] const variable& config_x_version; const variable& config_x_target; const variable& config_x_std; const variable& config_x_poptions; const variable& config_x_coptions; const variable& config_x_loptions; const variable& config_x_aoptions; const variable& config_x_libs; const variable& config_x_internal_scope; const variable* config_x_translate_include; const variable& x_path; // Compiler process path. const variable& x_mode; // Compiler mode options. const variable& x_c_path; // Compiler path as configured. const variable& x_c_mode; // Compiler mode as configured. const variable& x_sys_lib_dirs; // System library search directories. const variable& x_sys_hdr_dirs; // System header search directories. const variable& x_std; const variable& x_poptions; const variable& x_coptions; const variable& x_loptions; const variable& x_aoptions; const variable& x_libs; const variable& x_internal_scope; const variable& x_internal_libs; const variable* x_translate_include; const variable& c_poptions; // cc.* const variable& c_coptions; const variable& c_loptions; const variable& c_aoptions; const variable& c_libs; const variable& x_export_poptions; const variable& x_export_coptions; const variable& x_export_loptions; const variable& x_export_libs; const variable& x_export_impl_libs; const variable& c_export_poptions; // cc.export.* const variable& c_export_coptions; const variable& c_export_loptions; const variable& c_export_libs; const variable& c_export_impl_libs; const variable& c_pkgconfig_include; const variable& c_pkgconfig_lib; const variable& x_stdlib; // x.stdlib const variable& c_runtime; // cc.runtime const variable& c_stdlib; // cc.stdlib const variable& c_type; // cc.type const variable& c_system; // cc.system const variable& c_module_name; // cc.module_name const variable& c_importable; // cc.importable const variable& c_reprocess; // cc.reprocess const variable& x_preprocessed; // x.preprocessed const variable* x_symexport; // x.features.symexport const variable& x_id; const variable& x_id_type; const variable& x_id_variant; const variable& x_class; // Note: must be adjacent (used as an array). // const variable* x_version; const variable* x_version_major; const variable* x_version_minor; const variable* x_version_patch; const variable* x_version_build; // Note: must be adjacent (used as an array). // const variable* x_variant_version; const variable* x_variant_version_major; const variable* x_variant_version_minor; const variable* x_variant_version_patch; const variable* x_variant_version_build; const variable& x_signature; const variable& x_checksum; const variable& x_pattern; const variable& x_target; const variable& x_target_cpu; const variable& x_target_vendor; const variable& x_target_system; const variable& x_target_version; const variable& x_target_class; }; struct data: config_data { string x_compile; // Rule names. string x_link; string x_install; // Cached values for some commonly-used variables/values. // const compiler_id& cid; // x.id compiler_type ctype; // x.id.type const string& cvariant; // x.id.variant compiler_class cclass; // x.class uint64_t cmaj; // x.version.major uint64_t cmin; // x.version.minor uint64_t cvmaj; // x.variant_version.major (0 if no variant) uint64_t cvmin; // x.variant_version.minor (0 if no variant) const process_path& cpath; // x.path const strings& cmode; // x.mode (options) const target_triplet& ctgt; // x.target const string& tsys; // x.target.system const string& tclass; // x.target.class const string& env_checksum; // config_module::env_checksum bool modules; // x.features.modules bool symexport; // x.features.symexport enum class internal_scope {current, base, root, bundle, strong, weak}; optional<internal_scope> iscope; // x.internal.scope const scope* iscope_current; const scope* effective_iscope (const scope& bs) const; const strings* c_ilibs; // cc.internal.libs const strings* x_ilibs; // x.internal.libs build2::cc::importable_headers* importable_headers; // The order of sys_*_dirs is the mode entries first, followed by the // extra entries (e.g., /usr/local/*), followed by the compiler built-in // entries. // // Note that even if we wanted to, we wouldn't be able to support extra // trailing (after built-in) directories since we would need a portable // equivalent of -idirafter for both headers and libraries. // const dir_paths& sys_lib_dirs; // x.sys_lib_dirs const dir_paths& sys_hdr_dirs; // x.sys_hdr_dirs const dir_paths* sys_mod_dirs; // compiler_info::sys_mod_dirs size_t sys_lib_dirs_mode; // Number of mode entries (0 if none). size_t sys_hdr_dirs_mode; size_t sys_mod_dirs_mode; size_t sys_lib_dirs_extra; // Number of extra entries (0 if none). size_t sys_hdr_dirs_extra; // Note that x_obj is patched in by the x.objx module. So it stays NULL // if Objective-X compilation is not enabled. Similarly for x_asp except // here we don't have duality and it's purely to signal (by the c.as-cpp // module) that it's enabled. // const target_type& x_src; // Source target type (c{}, cxx{}). const target_type* x_mod; // Module target type (mxx{}), if any. const target_type& x_inc; // Includable base target type (e.g., c_inc{}). const target_type* x_obj; // Objective-X target type (m{}, mm{}). const target_type* x_asp; // Assembler with CPP target type (S{}). // Check if an object (target, prerequisite, etc) is an Objective-X // source. // template <typename T> bool x_objective (const T& t) const { return x_obj != nullptr && t.is_a (*x_obj); } // Check if an object (target, prerequisite, etc) is an Assembler with // C preprocessor source. // template <typename T> bool x_assembler_cpp (const T& t) const { return x_asp != nullptr && t.is_a (*x_asp); } // Array of target types that are considered the X-language headers // (excluding h{} except for C). Keep them in the most likely to appear // order with the "real header" first and terminated with NULL. // const target_type* const* x_hdrs; // Check if an object (target, prerequisite, etc) is a header. // template <typename T> bool x_header (const T& t, bool c_hdr = true) const { for (const target_type* const* ht (x_hdrs); *ht != nullptr; ++ht) if (t.is_a (**ht)) return true; return c_hdr && t.is_a (h::static_type); } // Array of target types that can be #include'd. Used to reverse-lookup // extensions to target types. Keep them in the most likely to appear // order and terminate with NULL. // const target_type* const* x_incs; // Aggregate-like constructor with from-base support. // data (const config_data& cd, const char* compile, const char* link, const char* install, const compiler_id& ci, compiler_class cl, uint64_t mj, uint64_t mi, uint64_t vmj, uint64_t vmi, const process_path& path, const strings& mode, const target_triplet& tgt, const string& env_cs, bool fm, bool fs, optional<internal_scope> is, const scope* isc, const strings* cils, const strings* xils, const dir_paths& sld, const dir_paths& shd, const dir_paths* smd, size_t slm, size_t shm, size_t smm, size_t sle, size_t she, const target_type& src, const target_type* mod, const target_type& inc, const target_type* const* hdrs, const target_type* const* incs) : config_data (cd), x_compile (compile), x_link (link), x_install (install), cid (ci), ctype (ci.type), cvariant (ci.variant), cclass (cl), cmaj (mj), cmin (mi), cvmaj (vmj), cvmin (vmi), cpath (path), cmode (mode), ctgt (tgt), tsys (ctgt.system), tclass (ctgt.class_), env_checksum (env_cs), modules (fm), symexport (fs), iscope (is), iscope_current (isc), c_ilibs (cils), x_ilibs (xils), importable_headers (nullptr), sys_lib_dirs (sld), sys_hdr_dirs (shd), sys_mod_dirs (smd), sys_lib_dirs_mode (slm), sys_hdr_dirs_mode (shm), sys_mod_dirs_mode (smm), sys_lib_dirs_extra (sle), sys_hdr_dirs_extra (she), x_src (src), x_mod (mod), x_inc (inc), x_obj (nullptr), x_asp (nullptr), x_hdrs (hdrs), x_incs (incs) {} }; class LIBBUILD2_CC_SYMEXPORT common: public data { public: common (data&& d): data (move (d)) {} // Library handling. // public: struct library_cache_entry { optional<lorder> lo; string type; // name::type string value; // name::value reference_wrapper<const mtime_target> lib; const target* group; }; using library_cache = small_vector<library_cache_entry, 32>; // The prerequisite_target::include bit that indicates a library // member has been picked from the group. // static const uintptr_t include_group = 0x100; void process_libraries ( action, const scope&, optional<linfo>, const dir_paths&, const mtime_target&, bool, lflags, const function<bool (const target&, bool)>&, const function<bool (const target* const*, const small_vector<reference_wrapper<const string>, 2>&, lflags, const string*, bool)>&, const function<bool (const target&, const string&, bool, bool)>&, bool = false, bool = false, library_cache* = nullptr) const; void process_libraries_impl ( action, const scope&, optional<linfo>, const dir_paths&, const target*, const mtime_target&, bool, lflags, const function<bool (const target&, bool)>&, const function<bool (const target* const*, const small_vector<reference_wrapper<const string>, 2>&, lflags, const string*, bool)>&, const function<bool (const target&, const string&, bool, bool)>&, bool, bool, library_cache*, small_vector<const target*, 32>*, small_vector<const target*, 32>*) const; const target* search_library (action a, const dir_paths& sysd, optional<dir_paths>& usrd, const prerequisite& p) const { const target* r (p.target.load (memory_order_consume)); if (r == nullptr) { if ((r = search_library (a, sysd, usrd, p.key ())) != nullptr) { const target* e (nullptr); if (!p.target.compare_exchange_strong ( e, r, memory_order_release, memory_order_consume)) assert (e == r); } } return r; } public: pair<const mtime_target&, const target*> resolve_library (action, const scope&, const name&, const dir_path&, optional<linfo>, const dir_paths&, optional<dir_paths>&, library_cache* = nullptr) const; struct non_existent_library { const mtime_target& target; }; template <typename T> static ulock insert_library (context&, T*&, string, dir_path, const process_path&, optional<string>, bool, tracer&); target* search_library (optional<action>, const dir_paths&, optional<dir_paths>&, const prerequisite_key&, bool existing = false) const; const target* search_library_existing (action a, const dir_paths& sysd, optional<dir_paths>& usrd, const prerequisite_key& pk) const { return search_library (a, sysd, usrd, pk, true); } dir_paths extract_library_search_dirs (const scope&) const; // Alternative search logic for VC (msvc.cxx). // // The second half is false if we should poison the binless search via // the common .pc file. // pair<bin::liba*, bool> msvc_search_static (const process_path&, const dir_path&, const prerequisite_key&, bool existing) const; pair<bin::libs*, bool> msvc_search_shared (const process_path&, const dir_path&, const prerequisite_key&, bool existing) const; // The pkg-config file searching and loading (pkgconfig.cxx) // using pkgconfig_callback = function<bool (dir_path&& d)>; bool pkgconfig_derive (const dir_path&, const pkgconfig_callback&) const; pair<path, path> pkgconfig_search (const dir_path&, const optional<project_name>&, const string&, bool) const; void pkgconfig_load (optional<action>, const scope&, bin::lib&, bin::liba*, bin::libs*, const pair<path, path>&, const dir_path&, const dir_paths&, const dir_paths&, pair<bool, bool>) const; bool pkgconfig_load (optional<action>, const scope&, bin::lib&, bin::liba*, bin::libs*, const optional<project_name>&, const string&, const dir_path&, const dir_paths&, const dir_paths&, pair<bool, bool>) const; // Append compiler-specific diagnostics color options as necessary. // void append_diag_color_options (cstrings&) const; }; } } #include <libbuild2/cc/common.ixx> #include <libbuild2/cc/common.txx> #endif // LIBBUILD2_CC_COMMON_HXX