// file : odb/pragma.hxx // copyright : Copyright (c) 2009-2013 Code Synthesis Tools CC // license : GNU GPL v3; see accompanying LICENSE file #ifndef ODB_PRAGMA_HXX #define ODB_PRAGMA_HXX #include #include #include #include #include #include // database #include #include #include struct virt_declaration { virt_declaration (location_t l, std::string const& n, int tc, tree t) : loc (l), name (n), tree_code (tc), type (t) {} location_t loc; std::string name; int tree_code; tree type; // Declaration's type. }; // Note that we consider virtual declarations with the same name but // different tree codes unequal. If that is too loose, then the // inserting code must do additional checks. // struct virt_declaration_set { typedef cutl::container::key key; typedef std::map map; typedef cutl::container::map_const_iterator const_iterator; std::pair insert (const virt_declaration& v) { std::pair r ( map_.insert (map::value_type (key (v.name, v.tree_code), v))); const_iterator i (r.first); if (r.second) r.first->first.assign (i->name, i->tree_code); return std::make_pair (i, r.second); } const_iterator find (std::string const& name, int tree_code) const { return map_.find (key (name, tree_code)); } const_iterator begin () const {return map_.begin ();} const_iterator end () const {return map_.end ();} private: map map_; }; // Map of scopes (e.g., class, namespace) to sets of virtual declarations. // typedef std::map virt_declarations; extern virt_declarations virt_declarations_; // Real or virtual declaration. If it is real, then it is a pointer to // the GCC tree node. Otherwise, it is a pointer to virt_declaration // from virt_declarations_ above. // struct declaration { declaration (): virt (false) {decl.real = 0;} declaration (tree d): virt (false) {decl.real = d;} declaration (virt_declaration const& d): virt (true) {decl.virt = &d;} bool virt; union { tree real; virt_declaration const* virt; } decl; int tree_code () const { return (virt ? decl.virt->tree_code : TREE_CODE (decl.real)); } typedef bool declaration::*bool_convertible; operator bool_convertible () const { return ptr () == 0 ? 0 : &declaration::virt; } public: bool operator== (declaration const& x) const { return virt == x.virt && ptr () == x.ptr (); } bool operator!= (declaration const& x) const { return !(*this == x); } bool operator< (declaration const& x) const { return virt < x.virt || (virt == x.virt && ptr () < x.ptr ()); } public: void const* ptr () const { return virt ? static_cast (decl.virt) : static_cast (decl.real); } }; inline bool operator== (declaration const& x, tree y) { return !x.virt && x.decl.real == y; } inline bool operator== (tree x, declaration const& y) { return !y.virt && y.decl.real == x; } struct pragma { // Check that the pragma is applicable to the declaration. Return true // on success, complain and return false otherwise. // typedef bool (*check_func) (declaration const& decl, std::string const& decl_name, std::string const& prag_name, location_t); // Add the pragma value to the context. // typedef void (*add_func) (cutl::compiler::context&, std::string const& key, cutl::container::any const& value, location_t); pragma (std::string const& pn, std::string const& cn, cutl::container::any const& v, location_t l, check_func c, add_func a) : pragma_name (pn), context_name (cn), value (v), loc (l), check (c), add (a) { } bool operator< (pragma const& y) const { if (add == 0) return context_name < y.context_name; else return context_name < y.context_name || (context_name == y.context_name && loc < y.loc); } std::string pragma_name; // Actual pragma name for diagnostics. std::string context_name; // Context entry name. cutl::container::any value; location_t loc; check_func check; add_func add; }; typedef std::vector pragma_list; // A set of pragmas. Insertion of a pragma with the same name and no // custom add function overrides the old value. // struct pragma_set: std::set { typedef std::set base; pragma& insert (pragma const& p) { std::pair r (base::insert (p)); pragma& x (const_cast (*r.first)); if (!r.second) x = p; return x; } template void insert (I begin, I end) { for (; begin != end; ++begin) insert (*begin); } }; // Position pragmas inside a class or namespace. The key for the // namespace case is the global_namespace node. // typedef std::map loc_pragmas; extern loc_pragmas loc_pragmas_; // Position pragmas for namespaces. Because re-opened namespaces do // not have any representation in the GCC tree, these are handled in // a special way. They are stored as a list of pragmas and their outer // namespaces. // struct ns_loc_pragma { typedef ::pragma pragma_type; ns_loc_pragma (tree n, pragma_type const& p): ns (n), pragma (p) {} tree ns; pragma_type pragma; }; typedef std::vector ns_loc_pragmas; extern ns_loc_pragmas ns_loc_pragmas_; // Pragmas associated with specific declarations (real or virtual). // typedef std::map decl_pragmas; extern decl_pragmas decl_pragmas_; // Database we are generating code for as well as multi-database support. // Used to ignore database-specific pragmas. // extern database pragma_db_; extern multi_database pragma_multi_; extern "C" void register_odb_pragmas (void*, void*); struct pragmas_failed {}; void post_process_pragmas (); #endif // ODB_PRAGMA_HXX