P4C
The P4 Compiler
Loading...
Searching...
No Matches
indexed_vector.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_INDEXED_VECTOR_H_
18#define IR_INDEXED_VECTOR_H_
19
20#include "ir/declaration.h"
21#include "ir/vector.h"
22#include "lib/enumerator.h"
23#include "lib/error.h"
24#include "lib/map.h"
25#include "lib/null.h"
26#include "lib/safe_vector.h"
27#include "lib/string_map.h"
28
29class JSONLoader;
30
31namespace IR {
32
37template <class T>
38class IndexedVector : public Vector<T> {
40 bool invalid = false; // set when an error occurs; then we don't
41 // expect the validity check to succeed.
42
43 void insertInMap(const T *a) {
44 if (a == nullptr || !a->template is<IDeclaration>()) return;
45 auto decl = a->template to<IDeclaration>();
46 auto name = decl->getName().name;
47 auto [it, inserted] = declarations.emplace(name, decl);
48 if (!inserted) {
49 invalid = true;
50 ::error(ErrorType::ERR_DUPLICATE, "%1%: Duplicates declaration %2%", a, it->second);
51 }
52 }
53 void removeFromMap(const T *a) {
54 if (a == nullptr) return;
55 auto decl = a->template to<IDeclaration>();
56 if (decl == nullptr) return;
57 cstring name = decl->getName().name;
58 auto it = declarations.find(name);
59 if (it == declarations.end()) BUG("%1% does not exist", a);
60 declarations.erase(it);
61 }
62
63 public:
64 using Vector<T>::begin;
65 using Vector<T>::end;
66
67 IndexedVector() = default;
68 IndexedVector(const IndexedVector &) = default;
69 IndexedVector(IndexedVector &&) = default;
70 IndexedVector(std::initializer_list<const T *> a) : Vector<T>(a) {
71 for (auto el : *this) insertInMap(el);
72 }
73 IndexedVector &operator=(const IndexedVector &) = default;
74 IndexedVector &operator=(IndexedVector &&) = default;
75 explicit IndexedVector(const T *a) { push_back(std::move(a)); }
76 explicit IndexedVector(const safe_vector<const T *> &a) {
77 insert(Vector<T>::end(), a.begin(), a.end());
78 }
79 explicit IndexedVector(const Vector<T> &a) { insert(Vector<T>::end(), a.begin(), a.end()); }
80 explicit IndexedVector(JSONLoader &json);
81
82 void clear() {
84 declarations.clear();
85 }
86 // TODO: Although this is not a const_iterator, it should NOT
87 // be used to modify the vector directly. I don't know
88 // how to enforce this property, though.
89 typedef typename Vector<T>::iterator iterator;
90
91 const IDeclaration *getDeclaration(cstring name) const {
92 auto it = declarations.find(name);
93 if (it == declarations.end()) return nullptr;
94 return it->second;
95 }
96 const IDeclaration *getDeclaration(std::string_view name) const {
97 auto it = declarations.find(name);
98 if (it == declarations.end()) return nullptr;
99 return it->second;
100 }
101 template <class U>
102 const U *getDeclaration(cstring name) const {
103 auto it = declarations.find(name);
104 if (it == declarations.end()) return nullptr;
105 return it->second->template to<U>();
106 }
107 template <class U>
108 const U *getDeclaration(std::string_view name) const {
109 auto it = declarations.find(name);
110 if (it == declarations.end()) return nullptr;
111 return it->second->template to<U>();
112 }
113 Util::Enumerator<const IDeclaration *> *getDeclarations() const {
114 return Util::enumerate(Values(declarations));
115 }
116 iterator erase(iterator i) {
117 removeFromMap(*i);
118 return Vector<T>::erase(i);
119 }
120 template <typename ForwardIter>
121 iterator insert(iterator i, ForwardIter b, ForwardIter e) {
122 for (auto it = b; it != e; ++it) insertInMap(*it);
123 return Vector<T>::insert(i, b, e);
124 }
125 iterator replace(iterator i, const T *v) {
126 removeFromMap(*i);
127 *i = v;
128 insertInMap(v);
129 return ++i;
130 }
131 template <typename Container>
132 iterator append(const Container &toAppend) {
133 return insert(Vector<T>::end(), toAppend.begin(), toAppend.end());
134 }
135 template <typename Container>
136 iterator prepend(const Container &toAppend) {
137 return insert(Vector<T>::begin(), toAppend.begin(), toAppend.end());
138 }
139 iterator insert(iterator i, const T *v) {
140 insertInMap(v);
141 return Vector<T>::insert(i, v);
142 }
143 template <class... Args>
144 void emplace_back(Args &&...args) {
145 auto el = new T(std::forward<Args>(args)...);
146 push_back(el);
147 }
148 bool removeByName(cstring name) {
149 for (auto it = begin(); it != end(); ++it) {
150 auto decl = (*it)->template to<IDeclaration>();
151 if (decl != nullptr && decl->getName() == name) {
152 erase(it);
153 return true;
154 }
155 }
156 return false;
157 }
158 void push_back(T *a) {
159 CHECK_NULL(a);
160 Vector<T>::push_back(a);
161 insertInMap(a);
162 }
163 void push_back(const T *a) {
164 CHECK_NULL(a);
165 Vector<T>::push_back(a);
166 insertInMap(a);
167 }
168 void pop_back() {
169 if (Vector<T>::empty()) BUG("pop_back from empty IndexedVector");
170 auto last = Vector<T>::back();
171 removeFromMap(last);
172 Vector<T>::pop_back();
173 }
174 template <class U>
175 void push_back(U &a) {
176 Vector<T>::push_back(a);
177 insertInMap(a);
178 }
179
180 IRNODE_SUBCLASS(IndexedVector)
181 IRNODE_DECLARE_APPLY_OVERLOAD(IndexedVector)
182 bool operator==(const Node &a) const override { return a == *this; }
183 bool operator==(const Vector<T> &a) const override { return a == *this; }
184 bool operator==(const IndexedVector &a) const override {
185 return Vector<T>::operator==(static_cast<const Vector<T> &>(a));
186 }
187 /* DANGER -- if you get an error on one of the above lines
188 * operator== ... marked ‘override’, but does not override
189 * that mean you're trying to create an instantiation of IR::IndexedVector
190 * that does not appear anywhere in any .def file, which won't work.
191 * To make double-dispatch comparisons work, the IR generator must know
192 * about ALL instantiations of IR class templates, which it does by scanning
193 * all the .def files for instantiations. This could in theory be fixed by
194 * having the IR generator scan all C++ header and source files for
195 * instantiations, but that is currently not done.
196 *
197 * To avoid this problem, you need to have your code ONLY use instantiations
198 * of IR::IndexedVector that appear somewhere in a .def file -- you can usually
199 * make it work by using an instantiation with an (abstract) base class rather
200 * than a concrete class, as most of those appear in .def files. */
201
202 cstring node_type_name() const override {
203 return "IndexedVector<" + T::static_type_name() + ">";
204 }
205 static cstring static_type_name() { return "IndexedVector<" + T::static_type_name() + ">"; }
206 void visit_children(Visitor &v) override;
207 void visit_children(Visitor &v) const override;
208
209 void toJSON(JSONGenerator &json) const override;
210 static IndexedVector<T> *fromJSON(JSONLoader &json);
211 void validate() const override {
212 if (invalid) return; // don't crash the compiler because an error happened
213 for (auto el : *this) {
214 auto decl = el->template to<IR::IDeclaration>();
215 if (!decl) continue;
216 auto it = declarations.find(decl->getName());
217 BUG_CHECK(it != declarations.end() && it->second->getNode() == el->getNode(),
218 "invalid element %1%", el);
219 }
220 }
221
222 DECLARE_TYPEINFO_WITH_DISCRIMINATOR(IndexedVector<T>, NodeDiscriminator::IndexedVectorT, T,
223 Vector<T>);
224};
225
226} // namespace IR
227
228#endif /* IR_INDEXED_VECTOR_H_ */
Definition vector.h:56
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
Definition safe_vector.h:25
Definition string_map.h:39
T * to() noexcept
Definition rtti.h:226
bool is() const noexcept
Definition rtti.h:216