// file : build/name -*- C++ -*- // copyright : Copyright (c) 2014-2015 Code Synthesis Ltd // license : MIT; see accompanying LICENSE file #ifndef BUILD_NAME #define BUILD_NAME #include <string> #include <vector> #include <iosfwd> #include <utility> // move() #include <butl/path> // Note: include <build/types> instead of this file directly. // namespace build { using butl::dir_path; // A name is what we operate on by default. Depending on the context, // it can be interpreted as a target or prerequisite name. A name // without a type and directory can be used to represent any text. // A name with directory and empty value represents a directory. // A name may also be project-qualified. If the project name is // empty, then it means the name is in a project other than our // own (e.g., it is installed). // // If pair is not '\0', then this name and the next in the list // form a pair. // struct name { name () = default; explicit name (std::string v): value (std::move (v)) {} name& operator= (std::string v) {return *this = name (std::move (v));} explicit name (dir_path d): dir (std::move (d)) {} name& operator= (dir_path d) {return *this = name (std::move (d));} name (std::string t, std::string v) : type (std::move (t)), value (std::move (v)) {} name (dir_path d, std::string t, std::string v) : dir (std::move (d)), type (std::move (t)), value (std::move (v)) {} // The first argument should be from project_name_pool. // name (const std::string* p, dir_path d, std::string t, std::string v) : proj (p), dir (std::move (d)), type (std::move (t)), value (std::move (v)) {} bool qualified () const {return proj != nullptr;} bool unqualified () const {return proj == nullptr;} bool typed () const {return !type.empty ();} bool untyped () const {return type.empty ();} bool empty () const {return dir.empty () && value.empty ();} // Note that strictly speaking the following tests should be // orthogonal to qualification. However, the vast majority of // cases where we expect a simple or directory name, we also // expect it to be unqualified. // // Note also that empty name is simple but not a directory. // bool simple () const {return unqualified () && untyped () && dir.empty ();} bool directory () const {return unqualified () && untyped () && !dir.empty () && value.empty ();} const std::string* proj = nullptr; // Points to project_name_pool. dir_path dir; std::string type; std::string value; char pair = '\0'; // Pair symbol, if any. }; inline bool operator== (const name& x, const name& y) { return x.proj == y.proj && // Pooled, so can just compare pointers. x.type == y.type && x.dir == y.dir && x.value == y.value; } inline bool operator!= (const name& x, const name& y) {return !(x == y);} typedef std::vector<name> names; std::ostream& operator<< (std::ostream&, const name&); std::ostream& operator<< (std::ostream&, const names&); } #endif // BUILD_NAME