8#ifndef FRONTENDS_P4_SIDEEFFECTS_H_
9#define FRONTENDS_P4_SIDEEFFECTS_H_
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"
22class HasTableApply :
public Inspector,
public ResolutionContext {
26 const IR::P4Table *table;
27 const IR::MethodCallExpression *call;
28 explicit HasTableApply(
TypeMap *typeMap) : typeMap(typeMap), table(
nullptr), call(
nullptr) {
30 setName(
"HasTableApply");
33 void postorder(
const IR::MethodCallExpression *expression)
override {
35 if (!mi->isApply())
return;
37 if (!am->object->is<IR::P4Table>())
return;
38 BUG_CHECK(table ==
nullptr,
"%1% and %2%: multiple table applications in one expression",
40 table = am->object->to<IR::P4Table>();
42 LOG3(
"Invoked table is " << dbp(table));
64 void postorder(
const IR::MethodCallExpression *mce)
override {
71 void postorder(
const IR::ConstructorCallExpression *cce)
override {
81 static bool check(
const IR::Expression *expression,
const Visitor *calledBy,
82 TypeMap *typeMap =
nullptr,
const Visitor::Context *ctxt =
nullptr) {
84 se.setCalledBy(calledBy);
85 if (calledBy && !ctxt) ctxt = calledBy->getChildContext();
86 expression->apply(se, ctxt);
98 return !ec->method->hasAnnotation(IR::Annotation::noSideEffectsAnnotation);
100 return !ef->function->hasAnnotation(IR::Annotation::noSideEffectsAnnotation);
101 if (
const auto *bim = mi->to<
BuiltInMethod>())
return bim->name != IR::Type_Header::isValid;
142 std::set<const IR::Expression *> *added;
148 std::set<const IR::Expression *> temporaries;
150 cstring createTemporary(
const IR::Type *type);
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);
158 DoSimplifyExpressions(
TypeMap *typeMap, std::set<const IR::Expression *> *added)
159 : typeMap(typeMap), added(added) {
161 setName(
"DoSimplifyExpressions");
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;
179 const IR::Node *preorder(IR::ConstructorCallExpression *cce)
override {
183 const IR::Node *preorder(IR::Property *prop)
override {
187 const IR::Node *preorder(IR::Annotation *anno)
override {
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;
206 void end_apply(
const IR::Node *)
override;
211 std::vector<const IR::Declaration_Variable *> declarations;
212 std::vector<const IR::BaseAssignmentStatement *> statements;
253class KeySideEffect :
public Transform,
public ResolutionContext {
257 std::map<const IR::P4Table *, TableInsertions *> toInsert;
258 std::set<const IR::P4Table *> *invokedInKey;
261 KeySideEffect(
TypeMap *typeMap, std::set<const IR::P4Table *> *invokedInKey)
262 : typeMap(typeMap), invokedInKey(invokedInKey) {
264 CHECK_NULL(invokedInKey);
265 setName(
"KeySideEffect");
270 virtual const IR::Node *doStatement(
const IR::Statement *statement,
271 const IR::Expression *expression,
272 const Visitor::Context *ctxt);
274 const IR::Node *preorder(IR::Key *key)
override;
278 const IR::Node *postorder(IR::MethodCallStatement *statement)
override {
279 return doStatement(statement, statement->methodCall, getContext());
281 const IR::Node *postorder(IR::IfStatement *statement)
override {
282 return doStatement(statement, statement->condition, getContext());
284 const IR::Node *postorder(IR::SwitchStatement *statement)
override {
285 return doStatement(statement, statement->expression, getContext());
287 const IR::Node *postorder(IR::BaseAssignmentStatement *statement)
override {
288 return doStatement(statement, statement->right, getContext());
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;
302 std::set<const IR::P4Table *> *invokedInKey;
305 TablesInKeys(
TypeMap *typeMap, std::set<const IR::P4Table *> *invokedInKey)
306 : typeMap(typeMap), invokedInKey(invokedInKey) {
307 CHECK_NULL(invokedInKey);
308 setName(
"TableInKeys");
311 invokedInKey->clear();
312 return Inspector::init_apply(node);
314 void postorder(
const IR::MethodCallExpression *mce)
override {
315 if (!isInContext<IR::Key>())
return;
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);
335 explicit TablesInActions(
TypeMap *typeMap) : typeMap(typeMap) {}
336 void postorder(
const IR::MethodCallExpression *expression)
override {
337 if (isInContext<IR::ActionList>()) {
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",
349class SideEffectOrdering :
public PassRepeated {
354 std::set<const IR::P4Table *> invokedInKey;
356 std::set<const IR::Expression *> added;
359 SideEffectOrdering(
TypeMap *typeMap,
bool skipSideEffectOrdering,
361 if (!typeChecking) typeChecking =
new TypeChecking(
nullptr, typeMap);
362 if (!skipSideEffectOrdering) {
366 passes.push_back(typeChecking);
368 passes.push_back(
new TablesInKeys(typeMap, &invokedInKey));
371 setName(
"SideEffectOrdering");
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 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
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 typeChecker.h:55
Definition source_file.h:132
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