P4C
The P4 Compiler
Loading...
Searching...
No Matches
sideEffects.h
1/*
2 * Copyright 2016 VMware, Inc.
3 * SPDX-FileCopyrightText: 2016 VMware, Inc.
4 *
5 * SPDX-License-Identifier: Apache-2.0
6 */
7
8#ifndef FRONTENDS_P4_SIDEEFFECTS_H_
9#define FRONTENDS_P4_SIDEEFFECTS_H_
10
11/* makes explicit side effect ordering */
12
13#include "frontends/common/resolveReferences/referenceMap.h"
14#include "frontends/common/resolveReferences/resolveReferences.h"
15#include "frontends/p4/methodInstance.h"
16#include "frontends/p4/typeChecking/typeChecker.h"
17#include "ir/ir.h"
18
19namespace P4 {
20
22class HasTableApply : public Inspector, public ResolutionContext {
23 TypeMap *typeMap;
24
25 public:
26 const IR::P4Table *table;
27 const IR::MethodCallExpression *call;
28 explicit HasTableApply(TypeMap *typeMap) : typeMap(typeMap), table(nullptr), call(nullptr) {
29 CHECK_NULL(typeMap);
30 setName("HasTableApply");
31 }
32
33 void postorder(const IR::MethodCallExpression *expression) override {
34 auto mi = MethodInstance::resolve(expression, this, typeMap);
35 if (!mi->isApply()) return;
36 auto am = mi->to<P4::ApplyMethod>();
37 if (!am->object->is<IR::P4Table>()) return;
38 BUG_CHECK(table == nullptr, "%1% and %2%: multiple table applications in one expression",
39 table, am->object);
40 table = am->object->to<IR::P4Table>();
41 call = expression;
42 LOG3("Invoked table is " << dbp(table));
43 }
44};
45
53class SideEffects : public Inspector, public ResolutionContext {
54 private:
55 TypeMap *typeMap;
56
57 public:
59 const IR::Node *nodeWithSideEffect = nullptr;
60
62 unsigned sideEffectCount = 0;
63
64 void postorder(const IR::MethodCallExpression *mce) override {
65 if (mayHaveSideEffect(mce, this, typeMap)) {
68 }
69 }
70
71 void postorder(const IR::ConstructorCallExpression *cce) override {
74 }
75
78 explicit SideEffects(TypeMap *typeMap) : typeMap(typeMap) { setName("SideEffects"); }
79
81 static bool check(const IR::Expression *expression, const Visitor *calledBy,
82 TypeMap *typeMap = nullptr, const Visitor::Context *ctxt = nullptr) {
83 SideEffects se(typeMap);
84 se.setCalledBy(calledBy);
85 if (calledBy && !ctxt) ctxt = calledBy->getChildContext();
86 expression->apply(se, ctxt);
87 return se.nodeWithSideEffect != nullptr;
88 }
89
90 static bool mayHaveSideEffect(const IR::MethodCallExpression *mce, DeclarationLookup *refMap,
91 TypeMap *typeMap) {
92 // mce does not produce a side effect in few cases:
93 // * isValid()
94 // * function, extern function, or extern method with noSideEffects annotation
95 auto mi = MethodInstance::resolve(mce, refMap, typeMap, !typeMap);
96 if (!mi) return true;
97 if (const auto *ec = mi->to<P4::ExternCall>())
98 return !ec->method->hasAnnotation(IR::Annotation::noSideEffectsAnnotation);
99 if (const auto *ef = mi->to<P4::FunctionCall>())
100 return !ef->function->hasAnnotation(IR::Annotation::noSideEffectsAnnotation);
101 if (const auto *bim = mi->to<BuiltInMethod>()) return bim->name != IR::Type_Header::isValid;
102 return true;
103 }
104};
105
138class DoSimplifyExpressions : public Transform, P4WriteContext, public ResolutionContext {
139 MinimalNameGenerator nameGen;
140 TypeMap *typeMap;
141 // Expressions holding temporaries that are already added.
142 std::set<const IR::Expression *> *added;
143
144 IR::IndexedVector<IR::Declaration> toInsert; // temporaries
148 std::set<const IR::Expression *> temporaries;
149
150 cstring createTemporary(const IR::Type *type);
151 const IR::Expression *addAssignment(Util::SourceInfo srcInfo, cstring varName,
152 const IR::Expression *expression);
153 bool mayAlias(const IR::Expression *left, const IR::Expression *right,
154 const Visitor::Context *ctxt) const;
155 bool containsHeaderType(const IR::Type *type);
156
157 public:
158 DoSimplifyExpressions(TypeMap *typeMap, std::set<const IR::Expression *> *added)
159 : typeMap(typeMap), added(added) {
160 CHECK_NULL(typeMap);
161 setName("DoSimplifyExpressions");
162 }
163
164 const IR::Node *postorder(IR::Expression *expression) override;
165 const IR::Node *preorder(IR::StructExpression *expression) override;
166 const IR::Node *preorder(IR::ListExpression *expression) override;
167 const IR::Node *preorder(IR::Literal *expression) override;
168 const IR::Node *preorder(IR::ArrayIndex *expression) override;
169 const IR::Node *preorder(IR::Member *expression) override;
170 const IR::Node *preorder(IR::SelectExpression *expression) override;
171 const IR::Node *preorder(IR::Operation_Unary *expression) override;
172 const IR::Node *preorder(IR::Operation_Binary *expression) override;
173 const IR::Node *shortCircuit(IR::Operation_Binary *expression);
174 const IR::Node *preorder(IR::Mux *expression) override;
175 const IR::Node *preorder(IR::LAnd *expression) override;
176 const IR::Node *preorder(IR::LOr *expression) override;
177 const IR::Node *preorder(IR::MethodCallExpression *mce) override;
178
179 const IR::Node *preorder(IR::ConstructorCallExpression *cce) override {
180 prune();
181 return cce;
182 }
183 const IR::Node *preorder(IR::Property *prop) override {
184 prune();
185 return prop;
186 }
187 const IR::Node *preorder(IR::Annotation *anno) override {
188 prune();
189 return anno;
190 }
191
192 const IR::Node *postorder(IR::P4Parser *parser) override;
193 const IR::Node *postorder(IR::Function *function) override;
194 const IR::Node *postorder(IR::P4Control *control) override;
195 const IR::Node *postorder(IR::P4Action *action) override;
196 const IR::Node *postorder(IR::ParserState *state) override;
197 const IR::Node *postorder(IR::BaseAssignmentStatement *statement) override;
198 const IR::Node *postorder(IR::MethodCallStatement *statement) override;
199 const IR::Node *postorder(IR::ReturnStatement *statement) override;
200 const IR::Node *preorder(IR::SwitchStatement *statement) override;
201 const IR::Node *preorder(IR::IfStatement *statement) override;
202 const IR::Node *preorder(IR::ForStatement *statement) override;
203 const IR::Node *preorder(IR::ForInStatement *statement) override;
204
205 profile_t init_apply(const IR::Node *node) override;
206 void end_apply(const IR::Node *) override;
207};
208
210 public:
211 std::vector<const IR::Declaration_Variable *> declarations;
212 std::vector<const IR::BaseAssignmentStatement *> statements;
213};
214
253class KeySideEffect : public Transform, public ResolutionContext {
254 protected:
255 MinimalNameGenerator nameGen;
256 TypeMap *typeMap;
257 std::map<const IR::P4Table *, TableInsertions *> toInsert;
258 std::set<const IR::P4Table *> *invokedInKey;
259
260 public:
261 KeySideEffect(TypeMap *typeMap, std::set<const IR::P4Table *> *invokedInKey)
262 : typeMap(typeMap), invokedInKey(invokedInKey) {
263 CHECK_NULL(typeMap);
264 CHECK_NULL(invokedInKey);
265 setName("KeySideEffect");
266 }
267
268 profile_t init_apply(const IR::Node *node) override;
269
270 virtual const IR::Node *doStatement(const IR::Statement *statement,
271 const IR::Expression *expression,
272 const Visitor::Context *ctxt);
273
274 const IR::Node *preorder(IR::Key *key) override;
275
276 // These should be all kinds of statements that may contain a table apply
277 // after the program has been simplified
278 const IR::Node *postorder(IR::MethodCallStatement *statement) override {
279 return doStatement(statement, statement->methodCall, getContext());
280 }
281 const IR::Node *postorder(IR::IfStatement *statement) override {
282 return doStatement(statement, statement->condition, getContext());
283 }
284 const IR::Node *postorder(IR::SwitchStatement *statement) override {
285 return doStatement(statement, statement->expression, getContext());
286 }
287 const IR::Node *postorder(IR::BaseAssignmentStatement *statement) override {
288 return doStatement(statement, statement->right, getContext());
289 }
290 const IR::Node *postorder(IR::KeyElement *element) override;
291 const IR::Node *postorder(IR::P4Table *table) override;
292 const IR::Node *preorder(IR::P4Table *table) override;
293};
294
300class TablesInKeys : public Inspector {
301 TypeMap *typeMap;
302 std::set<const IR::P4Table *> *invokedInKey;
303
304 public:
305 TablesInKeys(TypeMap *typeMap, std::set<const IR::P4Table *> *invokedInKey)
306 : typeMap(typeMap), invokedInKey(invokedInKey) {
307 CHECK_NULL(invokedInKey);
308 setName("TableInKeys");
309 }
310 Visitor::profile_t init_apply(const IR::Node *node) override {
311 invokedInKey->clear();
312 return Inspector::init_apply(node);
313 }
314 void postorder(const IR::MethodCallExpression *mce) override {
315 if (!isInContext<IR::Key>()) return;
316 HasTableApply hta(typeMap);
317 hta.setCalledBy(this);
318 (void)mce->apply(hta, getContext());
319 if (hta.table != nullptr) {
320 LOG2("Table " << hta.table << " invoked in key of another table");
321 invokedInKey->emplace(hta.table);
322 }
323 }
324};
325
331class TablesInActions : public Inspector {
332 TypeMap *typeMap;
333
334 public:
335 explicit TablesInActions(TypeMap *typeMap) : typeMap(typeMap) {}
336 void postorder(const IR::MethodCallExpression *expression) override {
337 if (isInContext<IR::ActionList>()) {
338 HasTableApply hta(typeMap);
339 hta.setCalledBy(this);
340 (void)expression->apply(hta, getContext());
341 if (hta.table != nullptr) {
342 ::P4::error(ErrorType::ERR_UNSUPPORTED, "%1%: table invocation in action argument",
343 expression);
344 }
345 }
346 }
347};
348
349class SideEffectOrdering : public PassRepeated {
350 // Contains all tables that are invoked within key
351 // computations for other tables. The keys for these
352 // inner tables cannot be expanded until the keys of
353 // the caller tables have been expanded.
354 std::set<const IR::P4Table *> invokedInKey;
355 // Temporaries that were added
356 std::set<const IR::Expression *> added;
357
358 public:
359 SideEffectOrdering(TypeMap *typeMap, bool skipSideEffectOrdering,
360 TypeChecking *typeChecking = nullptr) {
361 if (!typeChecking) typeChecking = new TypeChecking(nullptr, typeMap);
362 if (!skipSideEffectOrdering) {
363 passes.push_back(new TypeChecking(nullptr, typeMap));
364 passes.push_back(new DoSimplifyExpressions(typeMap, &added));
365 passes.push_back(new ClearTypeMap(typeMap));
366 passes.push_back(typeChecking);
367 passes.push_back(new TablesInActions(typeMap));
368 passes.push_back(new TablesInKeys(typeMap, &invokedInKey));
369 passes.push_back(new KeySideEffect(typeMap, &invokedInKey));
370 }
371 setName("SideEffectOrdering");
372 }
373};
374
375} // namespace P4
376
377#endif /* FRONTENDS_P4_SIDEEFFECTS_H_ */
Definition methodInstance.h:129
Definition methodInstance.h:260
Definition typeChecker.h:32
Definition referenceMap.h:57
Convert expressions so that each expression contains at most one side effect.
Definition sideEffects.h:138
Definition methodInstance.h:149
Definition methodInstance.h:232
Checks to see whether an IR node includes a table.apply() sub-expression.
Definition sideEffects.h:22
Definition indexed_vector.h:31
Definition node.h:53
Definition visitor.h:418
Definition sideEffects.h:253
static MethodInstance * resolve(const IR::MethodCallExpression *mce, const DeclarationLookup *refMap, TypeMap *typeMap, bool useExpressionType=false, const Visitor::Context *ctxt=nullptr, bool incomplete=false)
Definition methodInstance.cpp:27
Definition referenceMap.h:36
Definition visitor.h:811
static bool mayHaveSideEffect(const IR::MethodCallExpression *mce, DeclarationLookup *refMap, TypeMap *typeMap)
Definition sideEffects.h:90
unsigned sideEffectCount
Number of side effects in this expression.
Definition sideEffects.h:62
static bool check(const IR::Expression *expression, const Visitor *calledBy, TypeMap *typeMap=nullptr, const Visitor::Context *ctxt=nullptr)
Definition sideEffects.h:81
const IR::Node * nodeWithSideEffect
Last visited side-effecting node. Null if no node has side effects.
Definition sideEffects.h:59
SideEffects(TypeMap *typeMap)
Definition sideEffects.h:78
Definition sideEffects.h:331
Definition sideEffects.h:300
Definition visitor.h:442
Definition typeChecker.h:55
Definition typeMap.h:32
Definition source_file.h:132
Definition visitor.h:78
Definition visitor.h:75
Definition cstring.h:85
TODO: this is not really specific to BMV2, it should reside somewhere else.
Definition applyOptionsPragmas.cpp:13
void error(const char *format, Args &&...args)
Report an error with the given message.
Definition lib/error.h:58
Definition sideEffects.h:209