P4C
The P4 Compiler
Loading...
Searching...
No Matches
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_VECTOR_H_
18#define IR_VECTOR_H_
19
20#include "ir/node.h"
21#include "lib/enumerator.h"
22#include "lib/null.h"
23#include "lib/safe_vector.h"
24
25class JSONLoader;
26
27namespace IR {
28
29// Specialization of vector which
30// - only stores const IR::Node* objects inside (T should derive from Node)
31// - inherits from IR::Node itself
32class VectorBase : public Node {
33 public:
34 typedef const Node *const *iterator;
35 virtual iterator VectorBase_begin() const = 0;
36 virtual iterator VectorBase_end() const = 0;
37 virtual size_t size() const = 0;
38 virtual bool empty() const = 0;
39 iterator begin() const { return VectorBase_begin(); }
40 iterator end() const { return VectorBase_end(); }
41 VectorBase() = default;
42 VectorBase(const VectorBase &) = default;
43 VectorBase(VectorBase &&) = default;
44 VectorBase &operator=(const VectorBase &) = default;
45 VectorBase &operator=(VectorBase &&) = default;
46
47 protected:
48 explicit VectorBase(JSONLoader &json) : Node(json) {}
49
50 DECLARE_TYPEINFO_WITH_TYPEID(VectorBase, NodeKind::VectorBase, Node);
51};
52
53// This class should only be used in the IR.
54// User-level code should use regular std::vector
55template <class T>
56class Vector : public VectorBase {
58
59 public:
60 typedef const T *value_type;
61 Vector() = default;
62 Vector(const Vector &) = default;
63 Vector(Vector &&) = default;
64 explicit Vector(JSONLoader &json);
65 Vector &operator=(const Vector &) = default;
66 Vector &operator=(Vector &&) = default;
67 explicit Vector(const T *a) { vec.emplace_back(std::move(a)); }
68 explicit Vector(const safe_vector<const T *> &a) { vec.insert(vec.end(), a.begin(), a.end()); }
69 Vector(const std::initializer_list<const T *> &a) : vec(a) {}
70 static Vector<T> *fromJSON(JSONLoader &json);
71 typedef typename safe_vector<const T *>::iterator iterator;
72 typedef typename safe_vector<const T *>::const_iterator const_iterator;
73 iterator begin() { return vec.begin(); }
74 const_iterator begin() const { return vec.begin(); }
75 VectorBase::iterator VectorBase_begin() const override {
76 /* DANGER -- works as long as IR::Node is the first ultimate base class of T */
77 return reinterpret_cast<VectorBase::iterator>(&vec[0]);
78 }
79 iterator end() { return vec.end(); }
80 const_iterator end() const { return vec.end(); }
81 VectorBase::iterator VectorBase_end() const override {
82 /* DANGER -- works as long as IR::Node is the first ultimate base class of T */
83 return reinterpret_cast<VectorBase::iterator>(&vec[0] + vec.size());
84 }
85 std::reverse_iterator<iterator> rbegin() { return vec.rbegin(); }
86 std::reverse_iterator<const_iterator> rbegin() const { return vec.rbegin(); }
87 std::reverse_iterator<iterator> rend() { return vec.rend(); }
88 std::reverse_iterator<const_iterator> rend() const { return vec.rend(); }
89 size_t size() const override { return vec.size(); }
90 void resize(size_t sz) { vec.resize(sz); }
91 bool empty() const override { return vec.empty(); }
92 const T *const &front() const { return vec.front(); }
93 const T *&front() { return vec.front(); }
94 void clear() { vec.clear(); }
95 iterator erase(iterator i) { return vec.erase(i); }
96 iterator erase(iterator s, iterator e) { return vec.erase(s, e); }
97 template <typename ForwardIter>
98 iterator insert(iterator i, ForwardIter b, ForwardIter e) {
99 /* FIXME -- GCC prior to 4.9 is broken and the insert routine returns void
100 * FIXME -- rather than an iterator. So we recalculate it from an index */
101 int index = i - vec.begin();
102 vec.insert(i, b, e);
103 return vec.begin() + index;
104 }
105
106 template <typename Container>
107 iterator append(const Container &toAppend) {
108 return insert(end(), toAppend.begin(), toAppend.end());
109 }
110 template <typename Container>
111 iterator prepend(const Container &toAppend) {
112 return insert(begin(), toAppend.begin(), toAppend.end());
113 }
114
121 void pushBackOrAppend(const IR::Node *item) {
122 if (item == nullptr) return;
123 if (auto *itemAsVector = item->to<IR::Vector<T>>()) {
124 append(*itemAsVector);
125 return;
126 }
127 BUG_CHECK(item->is<T>(), "Unexpected vector element: %1%", item);
128 push_back(item->to<T>());
129 }
130
131 iterator insert(iterator i, const T *v) {
132 /* FIXME -- GCC prior to 4.9 is broken and the insert routine returns void
133 * FIXME -- rather than an iterator. So we recalculate it from an index */
134 int index = i - vec.begin();
135 vec.insert(i, v);
136 return vec.begin() + index;
137 }
138 iterator insert(iterator i, size_t n, const T *v) {
139 /* FIXME -- GCC prior to 4.9 is broken and the insert routine returns void
140 * FIXME -- rather than an iterator. So we recalculate it from an index */
141 int index = i - vec.begin();
142 vec.insert(i, n, v);
143 return vec.begin() + index;
144 }
145
146 const T *const &operator[](size_t idx) const { return vec[idx]; }
147 const T *&operator[](size_t idx) { return vec[idx]; }
148 const T *const &at(size_t idx) const { return vec.at(idx); }
149 const T *&at(size_t idx) { return vec.at(idx); }
150 template <class... Args>
151 void emplace_back(Args &&...args) {
152 vec.emplace_back(new T(std::forward<Args>(args)...));
153 }
154 void push_back(T *a) { vec.push_back(a); }
155 void push_back(const T *a) { vec.push_back(a); }
156 void pop_back() { vec.pop_back(); }
157 const T *const &back() const { return vec.back(); }
158 const T *&back() { return vec.back(); }
159 template <class U>
160 void push_back(U &a) {
161 vec.push_back(a);
162 }
163 void check_null() const {
164 for (auto e : vec) CHECK_NULL(e);
165 }
166
167 IRNODE_SUBCLASS(Vector)
168 IRNODE_DECLARE_APPLY_OVERLOAD(Vector)
169 bool operator==(const Node &a) const override { return a == *this; }
170 bool operator==(const Vector &a) const override { return vec == a.vec; }
171 /* DANGER -- if you get an error on the above line
172 * operator== ... marked ‘override’, but does not override
173 * that mean you're trying to create an instantiation of IR::Vector that
174 * does not appear anywhere in any .def file, which won't work.
175 * To make double-dispatch comparisons work, the IR generator must know
176 * about ALL instantiations of IR class templates, which it does by scanning
177 * all the .def files for instantiations. This could in theory be fixed by
178 * having the IR generator scan all C++ header and source files for
179 * instantiations, but that is currently not done.
180 *
181 * To avoid this problem, you need to have your code ONLY use instantiations
182 * of IR::Vector that appear somewhere in a .def file -- you can usually make
183 * it work by using an instantiation with an (abstract) base class rather
184 * than a concrete class, as most of those appear in .def files. */
185 bool equiv(const Node &a_) const override {
186 if (static_cast<const Node *>(this) == &a_) return true;
187 if (this->typeId() != a_.typeId()) return false;
188 auto &a = static_cast<const Vector<T> &>(a_);
189 if (size() != a.size()) return false;
190 auto it = a.begin();
191 for (auto *el : *this)
192 if (!el->equiv(**it++)) return false;
193 return true;
194 }
195 cstring node_type_name() const override { return "Vector<" + T::static_type_name() + ">"; }
196 static cstring static_type_name() { return "Vector<" + T::static_type_name() + ">"; }
197 void visit_children(Visitor &v) override;
198 void visit_children(Visitor &v) const override;
199 virtual void parallel_visit_children(Visitor &v);
200 virtual void parallel_visit_children(Visitor &v) const;
201 void toJSON(JSONGenerator &json) const override;
202 Util::Enumerator<const T *> *getEnumerator() const { return Util::enumerate(vec); }
203 template <typename S>
204 Util::Enumerator<const S *> *only() const {
205 return getEnumerator()->template as<const S *>()->where(
206 [](const T *d) { return d != nullptr; });
207 }
208
209 DECLARE_TYPEINFO_WITH_DISCRIMINATOR(Vector<T>, NodeDiscriminator::VectorT, T, VectorBase);
210};
211
212} // namespace IR
213
214// XXX(seth): We use this namespace to hide our get() overloads from ADL. GCC
215// 4.8 has a bug which causes these overloads to be considered when get() is
216// called on a type in the global namespace, even if the number of arguments
217// doesn't match up, which can trigger template instantiations that cause
218// errors.
219namespace GetImpl {
220
221template <class T, class U>
222const T *get(const IR::Vector<T> &vec, U name) {
223 for (auto el : vec)
224 if (el->name == name) return el;
225 return nullptr;
226}
227template <class T, class U>
228const T *get(const IR::Vector<T> *vec, U name) {
229 if (vec)
230 for (auto el : *vec)
231 if (el->name == name) return el;
232 return nullptr;
233}
234
235} // namespace GetImpl
236using namespace GetImpl; // NOLINT(build/namespaces)
237
238#endif /* IR_VECTOR_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 node.h:93
Definition vector.h:32
Definition vector.h:56
void pushBackOrAppend(const IR::Node *item)
Definition vector.h:121
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
T * to() noexcept
Definition rtti.h:226
virtual TypeId typeId() const noexcept=0
bool is() const noexcept
Definition rtti.h:216