P4C
The P4 Compiler
Loading...
Searching...
No Matches
namemap.h
1/*
2Copyright 2013-present Barefoot Networks, Inc.
3
4Licensed under the Apache License, Version 2.0 (the "License");
5you may not use this file except in compliance with the License.
6You may obtain a copy of the License at
7
8 http://www.apache.org/licenses/LICENSE-2.0
9
10Unless required by applicable law or agreed to in writing, software
11distributed under the License is distributed on an "AS IS" BASIS,
12WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13See the License for the specific language governing permissions and
14limitations under the License.
15*/
16
17#ifndef IR_NAMEMAP_H_
18#define IR_NAMEMAP_H_
19
20#include <map>
21
22#include "ir/node.h"
23#include "lib/cstring.h"
24#include "lib/enumerator.h"
25#include "lib/error.h"
26#include "lib/exceptions.h"
27#include "lib/map.h"
28
29class JSONLoader;
30
31namespace IR {
32
33template <class T, template <class K, class V, class COMP, class ALLOC> class MAP = std::map,
34 class COMP = std::less<cstring>,
35 class ALLOC = std::allocator<std::pair<const cstring, const T *>>>
36class NameMap : public Node {
37 typedef MAP<cstring, const T *, COMP, ALLOC> map_t;
38 map_t symbols;
39 /* if the object has a 'name' field, is it the same as name? */
40 template <class U>
41 auto match_name(cstring name, const U *obj) -> decltype(name == obj->name) {
42 return name == obj->name;
43 }
44 bool match_name(cstring, const void *) { return true; }
45 template <class U>
46 auto obj_name(const U *obj) -> decltype(obj->name) {
47 return obj->name;
48 }
49 cstring obj_name(const void *) { return nullptr; }
50
51 public:
52 NameMap() = default;
53 NameMap(const NameMap &) = default;
54 NameMap(NameMap &&) = default;
55 explicit NameMap(JSONLoader &);
56 NameMap &operator=(const NameMap &) = default;
57 NameMap &operator=(NameMap &&) = default;
58 typedef typename map_t::value_type value_type;
59 typedef typename map_t::iterator iterator;
60 typedef typename map_t::const_iterator const_iterator;
61 typedef typename map_t::reverse_iterator reverse_iterator;
62 typedef typename map_t::const_reverse_iterator const_reverse_iterator;
63
64 private:
65 struct elem_ref {
66 NameMap &self;
67 cstring name;
68 elem_ref(NameMap &s, cstring n) : self(s), name(n) {}
69 const T *operator=(const T *v) const {
70 if (!self.match_name(name, v)) BUG("Inserting into NameMap with incorrect name");
71 return self.symbols[name] = v;
72 }
73 operator const T *() const { return self.count(name) ? self.at(name) : nullptr; }
74 };
75
76 public:
77 const_iterator begin() const { return symbols.begin(); }
78 const_iterator end() const { return symbols.end(); }
79 iterator begin() { return symbols.begin(); }
80 iterator end() { return symbols.end(); }
81 // For multimaps
82 std::pair<const_iterator, const_iterator> equal_range(cstring key) const {
83 return symbols.equal_range(key);
84 }
85 const_reverse_iterator rbegin() const { return symbols.rbegin(); }
86 const_reverse_iterator rend() const { return symbols.rend(); }
87 reverse_iterator rbegin() { return symbols.rbegin(); }
88 reverse_iterator rend() { return symbols.rend(); }
89 size_t count(cstring name) const { return symbols.count(name); }
90 void clear() { symbols.clear(); }
91 size_t size() const { return symbols.size(); }
92 bool empty() const { return symbols.empty(); }
93 size_t erase(cstring n) { return symbols.erase(n); }
94 /* should be erase(const_iterator), but glibc++ std::list::erase is broken */
95 iterator erase(iterator p) { return symbols.erase(p); }
96 iterator erase(iterator f, iterator l) { return symbols.erase(f, l); }
97 const_iterator find(cstring name) const { return symbols.find(name); }
98 template <class U>
99 const U *get(cstring name) const {
100 for (auto it = symbols.find(name); it != symbols.end() && it->first == name; it++)
101 if (auto rv = it->second->template to<U>()) return rv;
102 return nullptr;
103 }
104 void add(cstring name, const T *n) {
105 if (!match_name(name, n)) BUG("Inserting into NameMap with incorrect name");
106 symbols.emplace(std::move(name), std::move(n));
107 }
108 // Expects to have a single node with each name.
109 // Only works for map and unordered_map
110 void addUnique(cstring name, const T *n) {
111 auto prev = symbols.find(name);
112 if (prev != symbols.end())
113 ::error(ErrorType::ERR_DUPLICATE, "%1%: duplicated name (%2% is previous instance)", n,
114 prev->second);
115 symbols.emplace(std::move(name), std::move(n));
116 }
117 // Expects to have a single node with each name.
118 // Only works for map and unordered_map
119 const T *getUnique(cstring name) const {
120 auto it = symbols.find(name);
121 if (it == symbols.end()) return nullptr;
122 return it->second;
123 }
124 elem_ref operator[](cstring name) { return elem_ref(*this, name); }
125 const T *operator[](cstring name) const { return count(name) ? at(name) : nullptr; }
126 const T *&at(cstring name) { return symbols.at(name); }
127 const T *const &at(cstring name) const { return symbols.at(name); }
128 void check_null() const {
129 for (auto &e : symbols) CHECK_NULL(e.second);
130 }
131
132 IRNODE_SUBCLASS(NameMap)
133 bool operator==(const Node &a) const override { return a.operator==(*this); }
134 bool operator==(const NameMap &a) const { return symbols == a.symbols; }
135 bool equiv(const Node &a_) const override {
136 if (static_cast<const Node *>(this) == &a_) return true;
137 if (this->typeId() != a_.typeId()) return false;
138 auto &a = static_cast<const NameMap<T, MAP, COMP, ALLOC> &>(a_);
139 if (size() != a.size()) return false;
140 auto it = a.begin();
141 for (auto &el : *this)
142 if (el.first != it->first || !el.second->equiv(*(it++)->second)) return false;
143 return true;
144 }
145 cstring node_type_name() const override { return "NameMap<" + T::static_type_name() + ">"; }
146 static cstring static_type_name() { return "NameMap<" + T::static_type_name() + ">"; }
147 void visit_children(Visitor &v) override;
148 void visit_children(Visitor &v) const override;
149 void toJSON(JSONGenerator &json) const override;
150 static NameMap<T, MAP, COMP, ALLOC> *fromJSON(JSONLoader &json);
151
152 Util::Enumerator<const T *> *valueEnumerator() const {
153 return Util::enumerate(Values(symbols));
154 }
155 template <typename S>
156 Util::Enumerator<const S *> *only() const {
157 return valueEnumerator()->template as<const S *>()->where(
158 [](const T *d) { return d != nullptr; });
159 }
160
161 DECLARE_TYPEINFO(NameMap, Node);
162};
163
164} // namespace IR
165
166#endif /* IR_NAMEMAP_H_ */
const T & as() const
Tries to convert the class to type T. A BUG occurs if the cast fails.
Definition castable.h:40
Definition namemap.h:36
Definition node.h:93
Definition json_generator.h:34
Definition json_loader.h:36
Type-erased Enumerator interface.
Definition enumerator.h:68
Definition visitor.h:73
Definition cstring.h:80
T * to() noexcept
Definition rtti.h:226
virtual TypeId typeId() const noexcept=0