P4C
The P4 Compiler
Loading...
Searching...
No Matches
dpdkArch.h
1/*
2Copyright 2020 Intel Corp.
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 BACKENDS_DPDK_DPDKARCH_H_
18#define BACKENDS_DPDK_DPDKARCH_H_
19
20#include "constants.h"
21#include "dpdkProgramStructure.h"
22#include "dpdkUtils.h"
23#include "frontends/common/resolveReferences/resolveReferences.h"
24#include "frontends/p4/evaluator/evaluator.h"
25#include "frontends/p4/sideEffects.h"
26#include "frontends/p4/typeMap.h"
27#include "lib/error.h"
28#include "lib/ordered_map.h"
29#include "midend/flattenInterfaceStructs.h"
30#include "midend/removeLeftSlices.h"
31
32namespace DPDK {
33
34using namespace P4::literals;
35
36cstring TypeStruct2Name(const cstring *s);
37bool isSimpleExpression(const IR::Expression *e);
38bool isNonConstantSimpleExpression(const IR::Expression *e);
39void expressionUnrollSanityCheck(const IR::Expression *e);
40
41using UserMeta = std::set<cstring>;
42
43class CollectMetadataHeaderInfo;
44
45/* According to the implementation of DPDK backend, for a control block, there
46 * are only two parameters: header and metadata. Therefore, first we need to
47 * rewrite the declaration of PSA architecture included in psa.p4 in order to
48 * pass the type checking. In addition, this pass changes the definition of
49 * P4Control and P4Parser(parameter list) in the P4 program provided by the
50 * user.
51 *
52 * This pass also modifies all metadata references and header reference. For
53 * metadata, struct_name.field_name -> m.struct_name_field_name. For header
54 * headers.header_name.field_name -> h.header_name.field_name The parameter
55 * named for header and metadata are also updated to "h" and "m" respectively.
56 */
58 P4::ReferenceMap *refMap;
59 DpdkProgramStructure *structure;
60
61 const IR::Type_Control *rewriteControlType(const IR::Type_Control *, cstring);
62 const IR::Type_Parser *rewriteParserType(const IR::Type_Parser *, cstring);
63 const IR::Type_Control *rewriteDeparserType(const IR::Type_Control *, cstring);
64 const IR::Node *postorder(IR::Type_Control *c) override;
65 const IR::Node *postorder(IR::Type_Parser *p) override;
66 const IR::Node *preorder(IR::Member *m) override;
67 const IR::Node *preorder(IR::PathExpression *pe) override;
68
69 public:
71 : refMap(refMap), structure(structure) {
72 CHECK_NULL(structure);
73 }
74};
75
110 std::unordered_map<const IR::P4Program *, IR::IndexedVector<IR::Node>> newHeaderMap;
111 std::unordered_map<const IR::P4Parser *, IR::IndexedVector<IR::Declaration>> newLocalVarMap;
112 std::unordered_map<const IR::AssignmentStatement *, IR::IndexedVector<IR::StatOrDecl>>
113 newStatMap;
114
115 public:
116 void insertHeader(const IR::P4Program *p, const IR::Type_Header *h) {
117 if (newHeaderMap.count(p)) {
118 newHeaderMap.at(p).push_back(h);
119 } else {
120 newHeaderMap.emplace(p, IR::IndexedVector<IR::Node>(h));
121 }
122 LOG5("Program: " << dbp(p));
123 LOG2("Adding new header:" << std::endl << " " << h);
124 }
125 IR::IndexedVector<IR::Node> *getHeaders(const IR::P4Program *p) {
126 if (newHeaderMap.count(p)) {
127 return new IR::IndexedVector<IR::Node>(newHeaderMap.at(p));
128 }
129 return nullptr;
130 }
131 void insertVar(const IR::P4Parser *p, const IR::Declaration_Variable *v) {
132 if (newLocalVarMap.count(p)) {
133 newLocalVarMap.at(p).push_back(v);
134 } else {
135 newLocalVarMap.emplace(p, IR::IndexedVector<IR::Declaration>(v));
136 }
137 LOG5("Parser: " << dbp(p));
138 LOG2("Adding new local variable:" << std::endl << " " << v);
139 }
140 IR::IndexedVector<IR::Declaration> *getVars(const IR::P4Parser *p) {
141 if (newLocalVarMap.count(p)) {
142 return new IR::IndexedVector<IR::Declaration>(newLocalVarMap.at(p));
143 }
144 return nullptr;
145 }
146 void insertStatements(const IR::AssignmentStatement *as,
148 BUG_CHECK(newStatMap.count(as) == 0,
149 "Unexpectedly converting statement %1% multiple times!", as);
150 newStatMap.emplace(as, *vec);
151 LOG5("AssignmentStatement: " << dbp(as));
152 LOG2("Adding new statements:");
153 for (auto s : *vec) {
154 LOG2(" " << s);
155 }
156 }
157 IR::IndexedVector<IR::StatOrDecl> *getStatements(const IR::AssignmentStatement *as) {
158 if (newStatMap.count(as)) {
159 return new IR::IndexedVector<IR::StatOrDecl>(newStatMap.at(as));
160 }
161 return nullptr;
162 }
163 };
164
165 ReplacementMap repl;
166
167 class Collect : public Inspector {
168 P4::ReferenceMap *refMap;
169 P4::TypeMap *typeMap;
170 ReplacementMap *repl;
171
172 public:
173 Collect(P4::ReferenceMap *refMap, P4::TypeMap *typeMap, ReplacementMap *repl)
174 : refMap(refMap), typeMap(typeMap), repl(repl) {}
175 void postorder(const IR::AssignmentStatement *statement) override;
176 };
177
178 class Replace : public Transform {
179 DpdkProgramStructure *structure;
180 ReplacementMap *repl;
181
182 public:
184 : structure(structure), repl(repl) {}
185 const IR::Node *postorder(IR::AssignmentStatement *as) override;
186 const IR::Node *postorder(IR::Type_Struct *s) override;
187 const IR::Node *postorder(IR::P4Parser *parser) override;
188 };
189
191 passes.push_back(new P4::TypeChecking(refMap, typeMap));
192 passes.push_back(new Collect(refMap, typeMap, &repl));
193 passes.push_back(new Replace(s, &repl));
194 passes.push_back(new P4::ClearTypeMap(typeMap));
195 }
196};
197
198// This Pass collects infomation about the name of all metadata and header
199// And it collects every field of metadata and renames all fields with a prefix
200// according to the metadata struct name. Eventually, the reference of a fields
201// will become m.$(struct_name)_$(field_name).
203 DpdkProgramStructure *structure;
204
205 void pushMetadata(const IR::Parameter *p);
206 void pushMetadata(const IR::ParameterList *, std::list<int> indices);
207
208 public:
209 explicit CollectMetadataHeaderInfo(DpdkProgramStructure *structure) : structure(structure) {}
210 bool preorder(const IR::P4Program *p) override;
211 bool preorder(const IR::Type_Struct *s) override;
212};
213
214// Previously, we have collected the information about how the single metadata
215// struct looks like in CollectMetadataHeaderInfo. This pass finds a suitable
216// place to inject this struct.
218 DpdkProgramStructure *structure;
219
220 public:
221 explicit InjectJumboStruct(DpdkProgramStructure *structure) : structure(structure) {}
222 const IR::Node *preorder(IR::Type_Struct *s) override;
223};
224
225// This pass injects metadata field which is used as port for 'tx' instruction
226// into the single metadata struct.
227// This pass has to be applied after CollectMetadataHeaderInfo fills
228// local_metadata_type field in DpdkProgramStructure which is passed to the constructor.
230 DpdkProgramStructure *structure;
231
232 public:
233 explicit InjectFixedMetadataField(DpdkProgramStructure *structure) : structure(structure) {}
234 const IR::Node *preorder(IR::Type_Struct *s) override;
235};
236
241 DpdkProgramStructure *structure;
242
244
245 public:
246 explicit AlignHdrMetaField(DpdkProgramStructure *structure) : structure(structure) {
247 CHECK_NULL(structure);
248 }
249 const IR::Node *preorder(IR::Type_StructLike *st) override;
250 const IR::Node *preorder(IR::Member *m) override;
251};
252
253struct ByteAlignment : public PassManager {
254 P4::TypeMap *typeMap;
255 P4::ReferenceMap *refMap;
256 DpdkProgramStructure *structure;
257
258 public:
260 : typeMap(typeMap), refMap(refMap), structure(structure) {
261 CHECK_NULL(structure);
262 passes.push_back(new AlignHdrMetaField(structure));
263 passes.push_back(new P4::ClearTypeMap(typeMap));
264 passes.push_back(new P4::TypeChecking(refMap, typeMap, true));
265 /* DoRemoveLeftSlices pass converts the slice Member (LHS in assn stm)
266 resulting from above Pass into shift operation */
267 passes.push_back(new P4::DoRemoveLeftSlices(typeMap));
268 passes.push_back(new P4::ClearTypeMap(typeMap));
269 passes.push_back(new P4::TypeChecking(refMap, typeMap, true));
270 }
271};
272
274 public:
275 const IR::Node *postorder(IR::Type_Struct *st) override;
276};
277
278struct fieldInfo {
279 unsigned fieldWidth;
280 fieldInfo() { fieldWidth = 0; }
281};
282
283// This class is helpful for StatementUnroll and IfStatementUnroll. Since dpdk
284// asm is not able to process complex expression, e.g., a = b + c * d. We need
285// break it down. Therefore, we need some temporary variables to hold the
286// intermediate values. And this class is helpful to inject the declarations of
287// temporary value into P4Control and P4Parser.
289 std::map<const IR::Node *, IR::IndexedVector<IR::Declaration> *> decl_map;
290
291 public:
292 // push the declaration to the right code block.
293 void collect(const IR::P4Control *control, const IR::P4Parser *parser,
294 const IR::Declaration *decl) {
295 IR::IndexedVector<IR::Declaration> *decls = nullptr;
296 if (parser) {
297 auto res = decl_map.find(parser);
298 if (res != decl_map.end()) {
299 decls = res->second;
300 } else {
302 decl_map.emplace(parser, decls);
303 }
304 } else if (control) {
305 auto res = decl_map.find(control);
306 if (res != decl_map.end()) {
307 decls = res->second;
308 } else {
310 decl_map.emplace(control, decls);
311 }
312 }
313 BUG_CHECK(decls != nullptr, "decls cannot be null");
314 decls->push_back(decl);
315 }
316 IR::Node *inject_control(const IR::Node *orig, IR::P4Control *control) {
317 auto res = decl_map.find(orig);
318 if (res == decl_map.end()) {
319 return control;
320 }
321 control->controlLocals.prepend(*res->second);
322 return control;
323 }
324 IR::Node *inject_parser(const IR::Node *orig, IR::P4Parser *parser) {
325 auto res = decl_map.find(orig);
326 if (res == decl_map.end()) {
327 return parser;
328 }
329 parser->parserLocals.prepend(*res->second);
330 return parser;
331 }
332};
333
334/* This pass breaks complex expressions down, since dpdk asm cannot describe
335 * complex expression. This pass is not complete. MethodCallStatement should be
336 * unrolled as well. Note that IfStatement should not be unrolled here, as we
337 * have a separate pass for it, because IfStatement does not want to unroll
338 * logical expression(dpdk asm has conditional jmp for these cases)
339 */
341 private:
342 P4::ReferenceMap *refMap;
343 DpdkProgramStructure *structure;
344 DeclarationInjector injector;
345
346 public:
348 : refMap(refMap), structure(structure) {}
349 const IR::Node *preorder(IR::AssignmentStatement *a) override;
350 const IR::Node *postorder(IR::P4Control *a) override;
351 const IR::Node *postorder(IR::P4Parser *a) override;
352};
353
354/* This pass helps StatementUnroll to unroll expressions inside statements.
355 * For example, if an AssignmentStatement looks like this: a = b + c * d
356 * StatementUnroll's AssignmentStatement preorder will call ExpressionUnroll
357 * twice for BinaryExpression's left(b) and right(c * d). For left, since it is
358 * a simple expression, ExpressionUnroll will set root to PathExpression(b) and
359 * the decl and stmt is empty. For right, ExpressionUnroll will set root to
360 * PathExpression(tmp), decl contains tmp's declaration and stmt contains:
361 * tmp = c * d, which will be injected in front of the AssignmentStatement.
362 */
364 P4::ReferenceMap *refMap;
365
366 public:
369 IR::PathExpression *root;
370 ExpressionUnroll(P4::ReferenceMap *refMap, DpdkProgramStructure *) : refMap(refMap) {
371 setName("ExpressionUnroll");
372 }
373 bool preorder(const IR::Operation_Unary *a) override;
374 bool preorder(const IR::Operation_Binary *a) override;
375 bool preorder(const IR::MethodCallExpression *a) override;
376 bool preorder(const IR::Member *a) override;
377 bool preorder(const IR::PathExpression *a) override;
378 bool preorder(const IR::Constant *a) override;
379 bool preorder(const IR::BoolLiteral *a) override;
380};
381
382// This pass is similiar to StatementUnroll pass, the difference is that this
383// pass will call LogicalExpressionUnroll to unroll the expression, which treat
384// logical expression differently.
386 private:
387 P4::ReferenceMap *refMap;
388 DeclarationInjector injector;
389
390 public:
391 explicit IfStatementUnroll(P4::ReferenceMap *refMap) : refMap(refMap) {
392 setName("IfStatementUnroll");
393 }
394 const IR::Node *postorder(IR::SwitchStatement *a) override;
395 const IR::Node *postorder(IR::IfStatement *a) override;
396 const IR::Node *postorder(IR::P4Control *a) override;
397 const IR::Node *postorder(IR::P4Parser *a) override;
398};
399
400/* Assume one logical expression looks like this: a && (b + c > d), this pass
401 * will unroll the expression to {tmp = b + c; if(a && (tmp > d))}. Logical
402 * calculation will be unroll in a dedicated pass.
403 */
405 P4::ReferenceMap *refMap;
406
407 public:
410 IR::Expression *root;
411 static bool is_logical(const IR::Operation_Binary *bin) {
412 if (bin->is<IR::LAnd>() || bin->is<IR::LOr>() || bin->is<IR::Leq>() || bin->is<IR::Equ>() ||
413 bin->is<IR::Neq>() || bin->is<IR::Grt>() || bin->is<IR::Lss>() || bin->is<IR::Geq>() ||
414 bin->is<IR::Leq>())
415 return true;
416 else
417 return false;
418 }
419
420 explicit LogicalExpressionUnroll(P4::ReferenceMap *refMap) : refMap(refMap) {
421 visitDagOnce = false;
422 }
423 bool preorder(const IR::Operation_Unary *a) override;
424 bool preorder(const IR::Operation_Binary *a) override;
425 bool preorder(const IR::MethodCallExpression *a) override;
426 bool preorder(const IR::Member *a) override;
427 bool preorder(const IR::PathExpression *a) override;
428 bool preorder(const IR::Constant *a) override;
429 bool preorder(const IR::BoolLiteral *a) override;
430};
431
432// According to dpdk spec, Binary Operation will only have two parameters, which
433// looks like: a = a + b. Therefore, this pass transform all AssignStatement
434// that has Binary_Operation to become two-parameter form.
436 DeclarationInjector injector;
437 P4::ReferenceMap *refMap;
438
439 public:
440 explicit ConvertBinaryOperationTo2Params(P4::ReferenceMap *refMap) : refMap(refMap) {}
441 const IR::Node *postorder(IR::AssignmentStatement *a) override;
442 const IR::Node *postorder(IR::P4Control *a) override;
443 const IR::Node *postorder(IR::P4Parser *a) override;
444};
445
446// Since in dpdk asm, there is no local variable declaration, we need to collect
447// all local variables and inject them into the metadata struct.
448// Local variables which are of header types are injected into headers struct
449// instead of metadata struct, so that they can be instantiated as headers in the
450// resulting dpdk asm file.
453 P4::ReferenceMap *refMap;
454 P4::TypeMap *typeMap;
455 DpdkProgramStructure *structure;
456
457 void insert(const cstring prefix, const IR::IndexedVector<IR::Declaration> *locals) {
458 for (auto d : *locals) {
459 if (auto dv = d->to<IR::Declaration_Variable>()) {
460 const cstring name = refMap->newName(prefix + "_" + dv->name.name);
461 localsMap.emplace(dv, name);
462 } else if (!d->is<IR::P4Action>() && !d->is<IR::P4Table>() &&
463 !d->is<IR::Declaration_Instance>()) {
464 BUG("%1%: Unhandled declaration type", d);
465 }
466 }
467 }
468
469 public:
471 DpdkProgramStructure *structure)
472 : refMap(refMap), typeMap(typeMap), structure(structure) {}
473 const IR::Node *preorder(IR::P4Program *p) override;
474 const IR::Node *postorder(IR::Type_Struct *s) override;
475 const IR::Node *postorder(IR::Member *m) override;
476 const IR::Node *postorder(IR::PathExpression *path) override;
477 const IR::Node *postorder(IR::P4Control *c) override;
478 const IR::Node *postorder(IR::P4Parser *p) override;
479};
480
481// According to dpdk spec, action parameters should prepend a p. In order to
482// respect this, we need at first make all action parameter lists into separate
483// structs and declare that struct in the P4 program. Then we modify the action
484// parameter list. Eventually, it will only contain one parameter `t`, which is a
485// struct containing all parameters previously defined. Next, we prepend t. in
486// front of action parameters. Please note that it is possible that the user
487// defines a struct paremeter himself or define multiple struct parameters in
488// action parameterlist. Current implementation does not support this.
490 P4::TypeMap *typeMap;
491 P4::ReferenceMap *refMap;
492 DpdkProgramStructure *structure;
493
494 public:
496 DpdkProgramStructure *structure)
497 : typeMap(typeMap), refMap(refMap), structure(structure) {}
498 const IR::Node *postorder(IR::P4Action *a) override;
499 const IR::Node *postorder(IR::P4Program *s) override;
500 const IR::Node *preorder(IR::PathExpression *path) override;
501 const IR::Node *preorder(IR::MethodCallExpression *) override;
502};
503
504/* This class is used to process the default action
505 and store the parameter list for each table.
506 Later, this infomation is passed and saved in table
507 properties and then used for generating instruction
508 for default action in each table.
509*/
510class DefActionValue : public Inspector {
511 P4::TypeMap *typeMap;
512 P4::ReferenceMap *refMap;
513 DpdkProgramStructure *structure;
514
515 public:
517 : typeMap(typeMap), refMap(refMap), structure(structure) {}
518 void postorder(const IR::P4Table *t) override;
519};
520
521// dpdk does not support ternary operator so we need to translate ternary operator
522// to corresponding if else statement
523// Taken from frontend pass DoSimplifyExpressions in sideEffects.h
525 P4::TypeMap *typeMap;
526 P4::ReferenceMap *refMap;
527 IR::IndexedVector<IR::Declaration> toInsert; // temporaries
529
530 cstring createTemporary(const IR::Type *type);
531 const IR::Expression *addAssignment(Util::SourceInfo srcInfo, cstring varName,
532 const IR::Expression *expression);
533
534 public:
536 : typeMap(typeMap), refMap(refMap) {}
537 const IR::Node *preorder(IR::Mux *expression) override;
538 const IR::Node *postorder(IR::P4Parser *parser) override;
539 const IR::Node *postorder(IR::Function *function) override;
540 const IR::Node *postorder(IR::P4Control *control) override;
541 const IR::Node *postorder(IR::P4Action *action) override;
542 const IR::Node *postorder(IR::AssignmentStatement *statement) override;
543};
544
545// For dpdk asm, there is not object-oriented. Therefore, we cannot define a
546// checksum in dpdk asm. And dpdk asm only provides ckadd(checksum add) and
547// cksub(checksum sub). So we need to define a explicit state for each checksum
548// declaration. Essentially, this state will be declared in header struct and
549// initilized to 0. And for cksum.add(x), it will be translated to ckadd state
550// x. For dst = cksum.get(), it will be translated to mov dst state. This pass
551// collects checksum instances and index them.
553 P4::TypeMap *typeMap;
554 DpdkProgramStructure *structure;
555 std::vector<cstring> *csum_vec;
556 int index = 0;
557
558 public:
560 std::vector<cstring> *csum_vec)
561 : typeMap(typeMap), structure(structure), csum_vec(csum_vec) {}
562 bool preorder(const IR::Declaration_Instance *d) override {
563 auto type = typeMap->getType(d, true);
564 if (auto extn = type->to<IR::Type_Extern>()) {
565 if (extn->name == "InternetChecksum") {
566 std::ostringstream s;
567 s << "state_" << index++;
568 csum_vec->push_back(s.str());
569 structure->csum_map.emplace(d, s.str());
570 }
571 }
572 return false;
573 }
574};
575
576// This pass will inject checksum states into header. The reason why we inject
577// state into header instead of metadata is due to the implementation of dpdk
578// side(a question related to endianness)
580 DpdkProgramStructure *structure;
581 std::vector<cstring> *csum_vec;
582
583 public:
585 std::vector<cstring> *csum_vec)
586 : structure(structure), csum_vec(csum_vec) {}
587
588 const IR::Node *postorder(IR::P4Program *p) override {
589 auto new_objs = new IR::Vector<IR::Node>;
590 bool inserted = false;
591 for (auto obj : p->objects) {
592 if (obj->to<IR::Type_Header>() && !inserted) {
593 inserted = true;
594 if (csum_vec->size() > 0) {
595 auto fields = new IR::IndexedVector<IR::StructField>;
596 for (auto fld : *csum_vec) {
597 fields->push_back(
598 new IR::StructField(IR::ID(fld), IR::Type::Bits::get(16, false)));
599 }
600 new_objs->push_back(new IR::Type_Header(IR::ID("cksum_state_t"), *fields));
601 }
602 }
603 new_objs->push_back(obj);
604 }
605 p->objects = *new_objs;
606 return p;
607 }
608
609 const IR::Node *postorder(IR::Type_Struct *s) override {
610 if (s->name.name == structure->header_type) {
611 if (structure->csum_map.size() > 0)
612 s->fields.push_back(new IR::StructField(
613 IR::ID("cksum_state"), new IR::Type_Name(IR::ID("cksum_state_t"))));
614 }
615 return s;
616 }
617};
618
620 std::vector<cstring> csum_vec;
621
622 public:
624 passes.push_back(new CollectInternetChecksumInstance(typeMap, structure, &csum_vec));
625 passes.push_back(new InjectInternetChecksumIntermediateValue(structure, &csum_vec));
626 }
627};
628
629/* This pass collects PSA extern meter, counter and register declaration instances and
630 push them to a vector for emitting to the .spec file later */
632 DpdkProgramStructure *structure;
633
634 public:
635 explicit CollectExternDeclaration(DpdkProgramStructure *structure) : structure(structure) {}
636 bool preorder(const IR::Declaration_Instance *d) override {
637 if (auto type = d->type->to<IR::Type_Name>()) {
638 auto externTypeName = type->path->name.name;
639 if (externTypeName == "DirectMeter") {
640 if (d->arguments->size() != 1) {
641 ::error(ErrorType::ERR_EXPECTED,
642 "%1%: expected type of meter as the only argument", d);
643 } else {
644 /* Check if the Direct meter is of PACKETS (0) type */
645 if (d->arguments->at(0)->expression->to<IR::Constant>()->asUnsigned() == 0)
646 warn(ErrorType::WARN_UNSUPPORTED,
647 "%1%: Packet metering is not supported."
648 " Falling back to byte metering.",
649 d);
650 }
651 } else {
652 // unsupported extern type
653 return false;
654 }
655 structure->externDecls.push_back(d);
656 } else if (auto type = d->type->to<IR::Type_Specialized>()) {
657 auto externTypeName = type->baseType->path->name.name;
658 if (externTypeName == "Meter") {
659 if (d->arguments->size() != 2) {
660 ::error(ErrorType::ERR_EXPECTED,
661 "%1%: expected number of meters and type of meter as arguments", d);
662 } else {
663 /* Check if the meter is of PACKETS (0) type */
664 if (d->arguments->at(1)->expression->to<IR::Constant>()->asUnsigned() == 0)
665 warn(ErrorType::WARN_UNSUPPORTED,
666 "%1%: Packet metering is not supported."
667 " Falling back to byte metering.",
668 d);
669 }
670 } else if (externTypeName == "Counter") {
671 if (d->arguments->size() != 2) {
672 ::error(ErrorType::ERR_EXPECTED,
673 "%1%: expected number of counters and type of counter as arguments", d);
674 }
675 } else if (externTypeName == "DirectCounter") {
676 if (d->arguments->size() != 1) {
677 ::error(ErrorType::ERR_EXPECTED,
678 "%1%: expected type of counter as the only argument", d);
679 }
680 } else if (externTypeName == "Register") {
681 if (d->arguments->size() != 1 && d->arguments->size() != 2) {
682 ::error(ErrorType::ERR_EXPECTED,
683 "%1%: expected size and optionally init_val as arguments", d);
684 }
685 } else if (externTypeName == "Hash") {
686 if (d->arguments->size() != 1) {
687 ::error(ErrorType::ERR_EXPECTED,
688 "%1%: expected hash algorithm as the only argument", d);
689 }
690 } else {
691 // unsupported extern type
692 return false;
693 }
694 structure->externDecls.push_back(d);
695 }
696 return false;
697 }
698};
699
700// This pass is preparing logical expression for following branching statement
701// optimization. This pass breaks parenthesis looks liks this: (a && b) && c.
702// After this pass, the expression looks like this: a && b && c. (The AST is
703// different).
705 public:
706 const IR::Node *postorder(IR::LAnd *land) {
707 if (auto land2 = land->left->to<IR::LAnd>()) {
708 auto sub = new IR::LAnd(land2->right, land->right);
709 return new IR::LAnd(land2->left, sub);
710 } else if (!land->left->is<IR::LOr>() && !land->left->is<IR::Equ>() &&
711 !land->left->is<IR::Neq>() && !land->left->is<IR::Leq>() &&
712 !land->left->is<IR::Geq>() && !land->left->is<IR::Lss>() &&
713 !land->left->is<IR::Grt>() && !land->left->is<IR::MethodCallExpression>() &&
714 !land->left->is<IR::PathExpression>() && !land->left->is<IR::Member>()) {
715 BUG("Logical Expression Unroll pass failed");
716 }
717 return land;
718 }
719 const IR::Node *postorder(IR::LOr *lor) {
720 if (auto lor2 = lor->left->to<IR::LOr>()) {
721 auto sub = new IR::LOr(lor2->right, lor->right);
722 return new IR::LOr(lor2->left, sub);
723 } else if (!lor->left->is<IR::LAnd>() && !lor->left->is<IR::Equ>() &&
724 !lor->left->is<IR::Neq>() && !lor->left->is<IR::Lss>() &&
725 !lor->left->is<IR::Grt>() && !lor->left->is<IR::MethodCallExpression>() &&
726 !lor->left->is<IR::PathExpression>() && !lor->left->is<IR::Member>()) {
727 BUG("Logical Expression Unroll pass failed");
728 }
729 return lor;
730 }
731};
732
733// This pass will swap the simple expression to the front of an logical
734// expression. Note that even for a subexpression of a logical expression, we
735// will swap it as well. For example, a && ((b && c) || d), will become
736// a && (d || (b && c))
738 bool is_simple(const IR::Node *n) {
739 if (n->is<IR::Equ>() || n->is<IR::Neq>() || n->is<IR::Lss>() || n->is<IR::Grt>() ||
740 n->is<IR::Geq>() || n->is<IR::Leq>() || n->is<IR::MethodCallExpression>() ||
741 n->is<IR::PathExpression>() || n->is<IR::Member>()) {
742 return true;
743 } else if (!n->is<IR::LAnd>() && !n->is<IR::LOr>()) {
744 BUG("Logical Expression Unroll pass failed");
745 } else {
746 return false;
747 }
748 }
749
750 public:
751 const IR::Node *postorder(IR::LAnd *land) {
752 if (!is_simple(land->left) && is_simple(land->right)) {
753 return new IR::LAnd(land->right, land->left);
754 } else if (!is_simple(land->left)) {
755 if (auto land2 = land->right->to<IR::LAnd>()) {
756 if (is_simple(land2->left)) {
757 auto sub = new IR::LAnd(land->left, land2->right);
758 return new IR::LAnd(land2->left, sub);
759 }
760 }
761 }
762 return land;
763 }
764 const IR::Node *postorder(IR::LOr *lor) {
765 if (!is_simple(lor->left) && is_simple(lor->right)) {
766 return new IR::LOr(lor->right, lor->left);
767 } else if (!is_simple(lor->left)) {
768 if (auto lor2 = lor->right->to<IR::LOr>()) {
769 if (is_simple(lor2->left)) {
770 auto sub = new IR::LOr(lor->left, lor2->right);
771 return new IR::LOr(lor2->left, sub);
772 }
773 }
774 }
775 return lor;
776 }
777};
778
779// This passmanager togethor transform logical expression into a form that
780// the simple expression will go to the front of the expression. And for
781// expression at the same level(the same level is that expressions that are
782// connected directly by && or ||) should be traversed from left to right
783// (a && b) && c is not a valid expression here.
785 public:
788 passes.push_back(r);
790 passes.push_back(r);
791 }
792};
793
794// This Pass collects infomation about the table keys for each table. This information
795// is later used for generating the context JSON output for use by the control plane
796// software.
798 DpdkProgramStructure *structure;
799
800 public:
801 explicit CollectTableInfo(DpdkProgramStructure *structure) : structure(structure) {
802 setName("CollectTableInfo");
803 }
804 bool preorder(const IR::Key *key) override;
805};
806
807// This pass transforms the tables such that all the Match keys are part of the same
808// header/metadata struct. If the match keys are from different headers, this pass creates
809// mirror copies of the struct field into the metadata struct and updates the table to use
810// the metadata copy.
811// This pass must be called right before CollectLocalVariables pass as the temporary
812// variables created for holding copy of the table keys are inserted to Metadata by
813// CollectLocalVariables pass.
815 int offsetInMetadata;
816 int size;
817};
818
819struct keyInfo {
820 int numElements;
821 int numExistingMetaFields;
822 bool isLearner;
823 bool isExact;
824 int size;
825 std::vector<struct keyElementInfo *> elements;
826};
827
830 DpdkProgramStructure *structure;
831 bool metaCopyNeeded = false;
832
833 public:
834 CopyMatchKeysToSingleStruct(P4::TypeMap *typeMap, std::set<const IR::P4Table *> *invokedInKey,
835 DpdkProgramStructure *structure)
836 : P4::KeySideEffect(typeMap, invokedInKey), structure(structure) {
837 setName("CopyMatchKeysToSingleStruct");
838 }
839
840 const IR::Node *preorder(IR::Key *key) override;
841 const IR::Node *postorder(IR::KeyElement *element) override;
842 const IR::Node *doStatement(const IR::Statement *statement, const IR::Expression *expression,
843 const Visitor::Context *ctxt) override;
844 struct keyInfo *getKeyInfo(IR::Key *keys);
845 cstring getTableKeyName(const IR::Expression *e);
846 int getFieldSizeBits(const IR::Type *field_type);
847 bool isLearnerTable(const IR::P4Table *t);
848};
849
851 // Map which holds the switch expression variable and constant tuple per switch statement for
852 // each action.
853 // action_name: {<switch_var, constant_value>, <switch_var1, constant_value1>, ...}
854 std::map<cstring, std::vector<std::tuple<cstring, IR::Constant *>>> actionCaseMap;
855
856 public:
857 // Fill Switch Action Map
858 void addToSwitchMap(cstring actionName, cstring switchExprTmp, IR::Constant *caseLabelValue) {
859 actionCaseMap[actionName].push_back(std::make_tuple(switchExprTmp, caseLabelValue));
860 }
861
862 const IR::Node *setSwitchVarInAction(IR::P4Action *action) {
863 if (actionCaseMap.count(action->name.name)) {
864 // Insert assignment statements for each switch statement which uses this action name as
865 // switch case.
866 auto acm = actionCaseMap[action->name.name];
867 auto body = new IR::BlockStatement(action->body->srcInfo);
868 for (auto pair : acm) {
869 auto assn = new IR::AssignmentStatement(
870 new IR::PathExpression(IR::ID(get<0>(pair))), get<1>(pair));
871 body->push_back(assn);
872 }
873 for (auto s : action->body->components) body->push_back(s);
874 action->body = body;
875 actionCaseMap.erase(action->name.name);
876 }
877 return action;
878 }
879};
880
885 cstring switchExprTmp;
886 DeclarationInjector injector;
887
888 public:
889 enum class TableImplementation { DEFAULT, ACTION_PROFILE, ACTION_SELECTOR };
890 P4::ReferenceMap *refMap;
891 P4::TypeMap *typeMap;
892 DpdkProgramStructure *structure;
893 SwitchHandler &sw;
894 TableImplementation implementation;
895 std::set<cstring> match_tables;
896 std::map<cstring, cstring> group_tables;
897 std::map<cstring, cstring> member_tables;
898 std::map<cstring, cstring> member_ids;
899 std::map<cstring, cstring> group_ids;
900
902 DpdkProgramStructure *structure, SwitchHandler &sw)
903 : refMap(refMap), typeMap(typeMap), structure(structure), sw(sw) {
904 implementation = TableImplementation::DEFAULT;
905 }
906
907 const IR::Node *postorder(IR::MethodCallStatement *) override;
908 const IR::Node *postorder(IR::IfStatement *) override;
909 const IR::Node *postorder(IR::SwitchStatement *) override;
910 const IR::Node *postorder(IR::P4Control *) override;
911 std::tuple<const IR::P4Table *, cstring, cstring> create_match_table(
912 const IR::P4Table * /* tbl */);
913 const IR::P4Action *create_action(cstring /* actionName */, cstring /* id */, cstring);
914 const IR::P4Table *create_member_table(const IR::P4Table *, cstring, cstring);
915 const IR::P4Table *create_group_table(const IR::P4Table *, cstring, cstring, cstring, unsigned,
916 unsigned);
917 IR::Expression *initializeMemberAndGroupId(cstring tableName,
919};
920
928 public:
930 DpdkProgramStructure *structure, SwitchHandler &sw)
931 : SplitP4TableCommon(refMap, typeMap, structure, sw) {
932 implementation = TableImplementation::ACTION_SELECTOR;
933 }
934 const IR::Node *postorder(IR::P4Table *tbl) override;
935};
936
943 public:
945 DpdkProgramStructure *structure, SwitchHandler &sw)
946 : SplitP4TableCommon(refMap, typeMap, structure, sw) {
947 implementation = TableImplementation::ACTION_PROFILE;
948 }
949 const IR::Node *postorder(IR::P4Table *tbl) override;
950};
951
972 SwitchHandler &sw;
973
974 public:
975 explicit UpdateActionForSwitch(SwitchHandler &sw) : sw(sw) { setName("UpdateActionForSwitch"); }
976 const IR::Node *postorder(IR::P4Action *action) { return sw.setSwitchVarInAction(action); }
977};
981 SwitchHandler sw;
982
983 public:
985 DpdkProgramStructure *structure) {
986 passes.emplace_back(new P4::TypeChecking(refMap, typeMap));
987 passes.emplace_back(new SplitActionSelectorTable(refMap, typeMap, structure, sw));
988 passes.emplace_back(new UpdateActionForSwitch(sw));
989 passes.push_back(new P4::ClearTypeMap(typeMap));
990 passes.emplace_back(new P4::TypeChecking(refMap, typeMap, true));
991 passes.emplace_back(new SplitActionProfileTable(refMap, typeMap, structure, sw));
992 passes.emplace_back(new UpdateActionForSwitch(sw));
993 passes.push_back(new P4::ClearTypeMap(typeMap));
994 passes.emplace_back(new P4::TypeChecking(refMap, typeMap, true));
995 }
996};
997
998/* Collect size information from the owner table for direct counter and meter extern objects
999 * and validate some of the constraints on usage of Direct Meter and Direct Counter extern
1000 * methods */
1002 P4::ReferenceMap *refMap;
1003 P4::TypeMap *typeMap;
1004 DpdkProgramStructure *structure;
1005 // To validate presence of specified method call for instancename within given action
1006 cstring method;
1007 cstring instancename;
1008 // To validate that same method call for different direct meter/counter instance does not exist
1009 // in any action
1010 cstring oneInstance;
1011 bool methodCallFound;
1012 int getTableSize(const IR::P4Table *tbl);
1013 bool ifMethodFound(const IR::P4Action *a, cstring method,
1014 cstring instancename = cstring::empty);
1015 void checkMethodCallInAction(const P4::ExternMethod *);
1016
1017 public:
1018 static ordered_map<cstring, int> directMeterCounterSizeMap;
1020 DpdkProgramStructure *structure)
1021 : refMap(refMap), typeMap(typeMap), structure(structure) {
1022 setName("CollectDirectCounterMeter");
1023 visitDagOnce = false;
1024 method = cstring::empty;
1025 instancename = cstring::empty;
1026 oneInstance = cstring::empty;
1027 methodCallFound = false;
1028 }
1029
1030 bool preorder(const IR::MethodCallStatement *mcs) override;
1031 bool preorder(const IR::AssignmentStatement *assn) override;
1032 bool preorder(const IR::P4Action *a) override;
1033 bool preorder(const IR::P4Table *t) override;
1034};
1035
1037 P4::ReferenceMap *refMap;
1038 P4::TypeMap *typeMap;
1039 DpdkProgramStructure *structure;
1040 void validateMethodInvocation(P4::ExternMethod *);
1041
1042 public:
1044 DpdkProgramStructure *structure)
1045 : refMap(refMap), typeMap(typeMap), structure(structure) {}
1046
1047 void postorder(const IR::AssignmentStatement *) override;
1048 void postorder(const IR::MethodCallStatement *) override;
1049};
1050
1052 P4::ReferenceMap *refMap;
1053 P4::TypeMap *typeMap;
1054 DpdkProgramStructure *structure;
1055
1056 public:
1058 DpdkProgramStructure *structure)
1059 : refMap(refMap), typeMap(typeMap), structure(structure) {}
1060
1061 void postorder(const IR::P4Table *t) override;
1062 void postorder(const IR::MethodCallStatement *) override;
1063};
1064
1066 P4::ReferenceMap *refMap;
1067 P4::TypeMap *typeMap;
1068 DpdkProgramStructure *structure;
1069
1070 public:
1072 DpdkProgramStructure *structure)
1073 : refMap(refMap), typeMap(typeMap), structure(structure) {}
1074
1075 void postorder(const IR::MethodCallStatement *) override;
1076 cstring getDefActionName(const IR::P4Table *t) {
1077 auto act = t->getDefaultAction();
1078 BUG_CHECK(act != nullptr, "%1%: default action does not exist", t);
1079 if (auto mc = act->to<IR::MethodCallExpression>()) {
1080 auto method = mc->method->to<IR::PathExpression>();
1081 return method->path->name;
1082 }
1083 return NULL;
1084 }
1085};
1086
1087// Dpdk does not allow operations (arithmetic, logical, bitwise, relational etc) on operands
1088// greater than 64-bit.
1090 public:
1091 ValidateOperandSize() { setName("ValidateOperandSize"); }
1092 void isValidOperandSize(const IR::Expression *e) {
1093 if (auto t = e->type->to<IR::Type_Bits>()) {
1094 if (t->width_bits() > dpdk_max_operand_size) {
1095 ::error(ErrorType::ERR_UNSUPPORTED_ON_TARGET, "Unsupported bitwidth %1% in %2%",
1096 t->width_bits(), e);
1097 return;
1098 }
1099 }
1100 }
1101
1102 void postorder(const IR::Operation_Binary *binop) override {
1103 isValidOperandSize(binop->left);
1104 isValidOperandSize(binop->right);
1105 }
1106
1107 // Reject all operations except typecast if the operand size is beyond the supported limit
1108 void postorder(const IR::Operation_Unary *unop) override {
1109 if (unop->is<IR::Cast>()) return;
1110 isValidOperandSize(unop->expr);
1111 }
1112
1113 void postorder(const IR::Operation_Ternary *top) override {
1114 isValidOperandSize(top->e0);
1115 isValidOperandSize(top->e1);
1116 isValidOperandSize(top->e2);
1117 }
1118};
1119
1120class CollectErrors : public Inspector {
1121 DpdkProgramStructure *structure;
1122
1123 public:
1124 explicit CollectErrors(DpdkProgramStructure *structure) : structure(structure) {
1125 CHECK_NULL(structure);
1126 }
1127 void postorder(const IR::Type_Error *error) override {
1128 int id = 0;
1129 for (auto err : error->members) {
1130 if (structure->error_map.count(err->name.name) == 0) {
1131 structure->error_map.emplace(err->name.name, id++);
1132 }
1133 }
1134 }
1135};
1136
1162 P4::TypeMap *typeMap;
1172
1173 public:
1174 explicit ElimHeaderCopy(P4::TypeMap *typeMap) : typeMap{typeMap} {}
1175 bool isHeader(const IR::Expression *e);
1176 const IR::Node *preorder(IR::AssignmentStatement *as) override;
1177 const IR::Node *preorder(IR::MethodCallStatement *mcs) override;
1178 const IR::Node *postorder(IR::Member *m) override;
1179};
1180
1182 public:
1184 passes.push_back(new P4::ClearTypeMap(typeMap));
1185 passes.push_back(new P4::ResolveReferences(refMap));
1186 passes.push_back(new P4::TypeInference(typeMap, false));
1187 passes.push_back(new P4::TypeChecking(refMap, typeMap, true));
1188 passes.push_back(new ElimHeaderCopy(typeMap));
1189 }
1190};
1191
1193// If one operand is >64-bit and other is <= 64-bit, the smaller operand should be a header field
1194// to maintain the endianness for copy. This pass detects if these conditions are satisfied or not.
1196 bool &is_all_arg_header_fields;
1197
1198 public:
1199 explicit HaveNonHeaderLargeOperandAssignment(bool &is_all_arg_header_fields)
1200 : is_all_arg_header_fields(is_all_arg_header_fields) {}
1201 bool preorder(const IR::AssignmentStatement *assn) override {
1202 if (!is_all_arg_header_fields) return false;
1203 if ((isLargeFieldOperand(assn->left) && !isLargeFieldOperand(assn->right) &&
1204 !isInsideHeader(assn->right)) ||
1205 (isLargeFieldOperand(assn->left) && assn->right->is<IR::Constant>()) ||
1206 (!isLargeFieldOperand(assn->left) && isLargeFieldOperand(assn->right) &&
1207 !isInsideHeader(assn->left))) {
1208 is_all_arg_header_fields &= false;
1209 return false;
1210 }
1211 return false;
1212 }
1213};
1214
1218 P4::TypeMap *typeMap;
1219 bool &is_all_arg_header_fields;
1220
1221 public:
1222 HaveNonHeaderChecksumArgs(P4::TypeMap *typeMap, bool &is_all_arg_header_fields)
1223 : typeMap(typeMap), is_all_arg_header_fields(is_all_arg_header_fields) {}
1224 bool preorder(const IR::MethodCallExpression *mce) override {
1225 if (!is_all_arg_header_fields) return false;
1226 if (auto *m = mce->method->to<IR::Member>()) {
1227 if (auto *type = typeMap->getType(m->expr)->to<IR::Type_Extern>()) {
1228 if (type->name == "InternetChecksum") {
1229 if (m->member == "add" || m->member == "subtract") {
1230 for (auto arg : *mce->arguments) {
1231 if (auto se = arg->expression->to<IR::StructExpression>()) {
1232 for (auto c : se->components) {
1233 if (auto m0 = c->expression->to<IR::Member>()) {
1234 if (!typeMap->getType(m0->expr, true)
1235 ->is<IR::Type_Header>()) {
1236 is_all_arg_header_fields = false;
1237 return false;
1238 }
1239 } else {
1240 is_all_arg_header_fields = false;
1241 return false;
1242 }
1243 }
1244 } else if (arg->expression->to<IR::Constant>()) {
1245 is_all_arg_header_fields = false;
1246 return false;
1247 } else if (auto m = arg->expression->to<IR::Member>()) {
1248 if (!(typeMap->getType(m->expr, true)->is<IR::Type_Header>() ||
1249 typeMap->getType(m, true)->is<IR::Type_Header>())) {
1250 is_all_arg_header_fields = false;
1251 return false;
1252 }
1253 }
1254 }
1255 }
1256 }
1257 }
1258 }
1259 return false;
1260 }
1261};
1262
1270 P4::ReferenceMap *refMap;
1271 P4::TypeMap *typeMap;
1272 bool &is_all_args_header;
1273 IR::Vector<IR::Node> allTypeDecls;
1274
1275 public:
1276 static cstring pseudoHeaderInstanceName;
1277 static cstring pseudoHeaderTypeName;
1279 bool &is_all_args_header)
1280 : refMap(refMap), typeMap(typeMap), is_all_args_header(is_all_args_header) {
1281 pseudoHeaderInstanceName = refMap->newName("dpdk_pseudo_header");
1282 pseudoHeaderTypeName = refMap->newName("dpdk_pseudo_header_t");
1283 (void)this->typeMap;
1284 (void)this->refMap;
1285 }
1286
1287 const IR::Node *preorder(IR::P4Program *program) override;
1288 const IR::Node *preorder(IR::Type_Struct *st) override;
1289};
1290
1305
1307 P4::ReferenceMap *refMap;
1308 P4::TypeMap *typeMap;
1309 bool &is_all_args_header;
1310 IR::Vector<IR::Node> newStructTypes;
1311
1312 public:
1313 static std::vector<std::pair<cstring, const IR::Type *>> pseudoFieldNameType;
1315 bool &is_all_args_header)
1316 : refMap(refMap), typeMap(typeMap), is_all_args_header(is_all_args_header) {}
1317 std::pair<IR::AssignmentStatement *, IR::Member *> addAssignmentStmt(const IR::Expression *ne);
1318
1319 const IR::Node *postorder(IR::P4Program *p) override {
1320 if (newStructTypes.size() > 0) {
1321 IR::Vector<IR::Node> allTypeDecls;
1322 allTypeDecls.append(newStructTypes);
1323 allTypeDecls.append(p->objects);
1324 p->objects = allTypeDecls;
1325 }
1326 return p;
1327 }
1328 const IR::Node *postorder(IR::MethodCallStatement *statement) override;
1329 const IR::Node *postorder(IR::AssignmentStatement *statement) override;
1330};
1331
1335 P4::ReferenceMap *refMap;
1336 P4::TypeMap *typeMap;
1337 bool &is_all_args_header;
1338
1339 public:
1341 bool &is_all_args_header)
1342 : refMap(refMap), typeMap(typeMap), is_all_args_header(is_all_args_header) {
1343 (void)this->typeMap;
1344 (void)this->refMap;
1345 }
1346 const IR::Node *preorder(IR::Type_Header *h) override;
1347};
1348
1350 P4::ReferenceMap *refMap;
1351 P4::TypeMap *typeMap;
1352 bool &is_all_args_header;
1353
1354 public:
1356 bool &is_all_args_header_fields)
1357 : refMap(refMap), typeMap(typeMap), is_all_args_header(is_all_args_header_fields) {
1358 passes.push_back(new HaveNonHeaderChecksumArgs(typeMap, is_all_args_header));
1359 passes.push_back(new HaveNonHeaderLargeOperandAssignment(is_all_args_header));
1360 passes.push_back(new DpdkAddPseudoHeaderDecl(refMap, typeMap, is_all_args_header));
1361 passes.push_back(new P4::ClearTypeMap(typeMap));
1362 passes.push_back(new P4::TypeChecking(refMap, typeMap));
1363 passes.push_back(
1364 new MoveNonHeaderFieldsToPseudoHeader(refMap, typeMap, is_all_args_header));
1365 passes.push_back(new AddFieldsToPseudoHeader(refMap, typeMap, is_all_args_header));
1366 passes.push_back(new P4::ClearTypeMap(typeMap));
1367 passes.push_back(new P4::TypeChecking(refMap, typeMap));
1368 }
1369};
1370
1372 public:
1373 DpdkArchFirst() { setName("DpdkArchFirst"); }
1374};
1375
1377 public:
1378 DpdkArchLast() { setName("DpdkArchLast"); }
1379};
1380
1382 public:
1384 DpdkProgramStructure *structure) {
1385 auto *evaluator = new P4::EvaluatorPass(refMap, typeMap);
1386 auto *parseDpdk = new ParseDpdkArchitecture(structure);
1387 passes.push_back(evaluator);
1388 passes.push_back(new VisitFunctor([evaluator, parseDpdk]() {
1389 auto toplevel = evaluator->getToplevelBlock();
1390 auto main = toplevel->getMain();
1391 if (main == nullptr) {
1392 ::error(ErrorType::ERR_NOT_FOUND,
1393 "Could not locate top-level block; is there a %1% module?",
1394 IR::P4Program::main);
1395 return;
1396 }
1397 main->apply(*parseDpdk);
1398 }));
1399 }
1400};
1401
1402// Add collected local struct variable decls as a field in metadata
1403// struct
1405 P4::TypeMap *typeMap;
1406
1407 public:
1408 explicit MoveCollectedStructLocalVariableToMetadata(P4::TypeMap *typeMap) : typeMap(typeMap) {}
1409 const IR::Node *preorder(IR::Type_Struct *s) override;
1410 const IR::Node *postorder(IR::P4Control *c) override;
1411 const IR::Node *postorder(IR::P4Program *p) override;
1412 const IR::Node *postorder(IR::P4Parser *c) override;
1413};
1414
1415// Collect all local struct variable and metadata struct type
1417 P4::ReferenceMap *refMap;
1418 P4::TypeMap *typeMap;
1419
1420 public:
1422 : refMap(refMap), typeMap(typeMap) {}
1423 const IR::Node *postorder(IR::P4Parser *c) override;
1424 const IR::Node *preorder(IR::PathExpression *path) override;
1425 static const IR::Type_Struct *metadataStrct;
1426 static std::map<cstring, const IR::Type *> fieldNameType;
1427 static IR::Vector<IR::Node> type_tobe_moved_at_top;
1428};
1429
1430// Collect all local struct decls and move it to metadata struct and finally
1431// flatten the metadata struct.
1433 public:
1435 passes.push_back(new P4::ClearTypeMap(typeMap));
1436 passes.push_back(new P4::ResolveReferences(refMap));
1437 passes.push_back(new P4::TypeInference(typeMap, false));
1438 passes.push_back(new P4::TypeChecking(refMap, typeMap, true));
1439 passes.push_back(new CollectStructLocalVariables(refMap, typeMap));
1440 passes.push_back(new MoveCollectedStructLocalVariableToMetadata(typeMap));
1441 passes.push_back(new P4::ClearTypeMap(typeMap));
1442 passes.push_back(new P4::ResolveReferences(refMap));
1443 passes.push_back(new P4::TypeInference(typeMap, false));
1444 passes.push_back(new P4::TypeChecking(refMap, typeMap, true));
1445 passes.push_back(new P4::FlattenInterfaceStructs(refMap, typeMap));
1446 }
1447};
1448
1449/* Helper class to detect use of IPSec accelerator */
1451 bool &is_ipsec_used;
1452 int &sa_id_width;
1453 P4::ReferenceMap *refMap;
1454 P4::TypeMap *typeMap;
1455 DpdkProgramStructure *structure;
1456
1457 public:
1458 CollectIPSecInfo(bool &is_ipsec_used, int &sa_id_width, P4::ReferenceMap *refMap,
1459 P4::TypeMap *typeMap, DpdkProgramStructure *structure)
1460 : is_ipsec_used(is_ipsec_used),
1461 sa_id_width(sa_id_width),
1462 refMap(refMap),
1463 typeMap(typeMap),
1464 structure(structure) {}
1465 bool preorder(const IR::MethodCallStatement *mcs) override {
1466 auto mi = P4::MethodInstance::resolve(mcs, refMap, typeMap);
1467 if (auto a = mi->to<P4::ExternMethod>()) {
1468 if (a->originalExternType->getName().name == "ipsec_accelerator") {
1469 if (structure->isPSA()) {
1470 ::error(ErrorType::ERR_MODEL, "%1% is not available for PSA programs",
1471 a->originalExternType->getName().name);
1472 return false;
1473 }
1474 if (a->method->getName().name == "enable") {
1475 is_ipsec_used = true;
1476 } else if (a->method->getName().name == "set_sa_index") {
1477 auto typeArgs = a->expr->typeArguments;
1478 if (typeArgs->size() != 1) {
1479 ::error(ErrorType::ERR_MODEL, "Unexpected number of type arguments for %1%",
1480 a->method->name);
1481 return false;
1482 }
1483 auto width = typeArgs->at(0);
1484 if (!width->is<IR::Type_Bits>()) {
1485 ::error(ErrorType::ERR_MODEL, "Unexpected width type %1% for sa_index",
1486 width);
1487 return false;
1488 }
1489 sa_id_width = width->to<IR::Type_Bits>()->width_bits();
1490 }
1491 }
1492 }
1493 return false;
1494 }
1495};
1496
1497/* DPDK uses some fixed registers to hold the ipsec inbound/outbound input and output ports and a
1498 * pseudo compiler inserted header which shall be emitted in front of all headers. This class helps
1499 * insert required registers and a pseudo header for enabling IPSec encryption and decryption. It
1500 * also handles setting of output port in the deparser.
1501 */
1503 P4::ReferenceMap *refMap;
1504 DpdkProgramStructure *structure;
1505 bool &is_ipsec_used;
1506 int &sa_id_width;
1507 cstring newHeaderName = "platform_hdr_t"_cs;
1508 IR::Type_Header *ipsecHeader = nullptr;
1509 std::vector<cstring> registerInstanceNames = {
1510 "ipsec_port_out_inbound"_cs, "ipsec_port_out_outbound"_cs, "ipsec_port_in_inbound"_cs,
1511 "ipsec_port_in_outbound"_cs};
1512
1513 public:
1515 bool &is_ipsec_used, int &sa_id_width)
1516 : refMap(refMap),
1517 structure(structure),
1518 is_ipsec_used(is_ipsec_used),
1519 sa_id_width(sa_id_width) {
1520 setName("InsertReqDeclForIPSec");
1521 }
1522
1523 const IR::Node *preorder(IR::P4Program *program) override;
1524 const IR::Node *preorder(IR::Type_Struct *s) override;
1525 const IR::Node *preorder(IR::P4Control *c) override;
1526 IR::IndexedVector<IR::StatOrDecl> *addRegDeclInstance(std::vector<cstring> portRegs);
1527};
1528
1530 P4::ReferenceMap *refMap;
1531 P4::TypeMap *typeMap;
1532 DpdkProgramStructure *structure;
1533 bool is_ipsec_used = false;
1534 int sa_id_width = 32;
1535
1536 public:
1538 : refMap(refMap), typeMap(typeMap), structure(structure) {
1539 passes.push_back(
1540 new CollectIPSecInfo(is_ipsec_used, sa_id_width, refMap, typeMap, structure));
1541 passes.push_back(new InsertReqDeclForIPSec(refMap, structure, is_ipsec_used, sa_id_width));
1542 passes.push_back(new P4::ClearTypeMap(typeMap));
1543 passes.push_back(new P4::ResolveReferences(refMap));
1544 passes.push_back(new P4::TypeInference(typeMap, false));
1545 }
1546};
1547
1548} // namespace DPDK
1549#endif /* BACKENDS_DPDK_DPDKARCH_H_ */
This pass finally adds all the collected fields to pseudo header add collected pseudo header fields i...
Definition dpdkArch.h:1334
Definition dpdkArch.h:240
Definition dpdkArch.h:1051
Definition dpdkArch.h:1001
Definition dpdkArch.h:1120
Definition dpdkArch.h:631
Definition dpdkArch.h:1450
Definition dpdkArch.h:552
Definition dpdkArch.h:1432
Definition dpdkArch.h:451
Definition dpdkArch.h:202
Definition dpdkArch.h:1381
Definition dpdkArch.h:1416
Definition dpdkArch.h:797
bool preorder(const IR::Key *key) override
Definition dpdkArch.cpp:647
Definition dpdkArch.h:980
Definition dpdkArch.h:435
Definition dpdkArch.h:619
Definition dpdkArch.h:784
Definition dpdkArch.h:167
Definition dpdkArch.h:178
Definition dpdkArch.h:57
Definition dpdkArch.h:828
Definition dpdkArch.h:288
Definition dpdkArch.h:510
void postorder(const IR::P4Table *t) override
Definition dpdkArch.cpp:1228
Definition dpdkArch.h:524
This pass adds a pseudo header declaration, it will be used as container of operands where dpdk instr...
Definition dpdkArch.h:1269
Definition dpdkArch.h:1371
Definition dpdkArch.h:1376
Definition dpdkArch.h:1161
Definition dpdkArch.h:1181
Definition dpdkArch.h:363
Definition dpdkArch.h:1217
This pass checks whether an assignment statement has large operands (>64-bit).
Definition dpdkArch.h:1195
Definition dpdkArch.h:385
Definition dpdkArch.h:229
Definition dpdkArch.h:217
Definition dpdkArch.h:1502
IR::IndexedVector< IR::StatOrDecl > * addRegDeclInstance(std::vector< cstring > portRegs)
Create and add register declaration instance to program.
Definition dpdkArch.cpp:3221
Definition dpdkArch.h:404
bool preorder(const IR::Operation_Unary *a) override
Definition dpdkArch.cpp:910
This pass identifies and collects statement which requires it's operand to be in a header and also in...
Definition dpdkArch.h:1306
Definition dpdkArch.h:489
Definition dpdkArch.h:273
const IR::Node * postorder(IR::Type_Struct *st) override
Definition dpdkArch.cpp:623
Definition dpdkArch.h:942
Definition dpdkArch.h:927
Definition dpdkArch.h:884
std::tuple< const IR::P4Table *, cstring, cstring > create_match_table(const IR::P4Table *)
Definition dpdkArch.cpp:1785
IR::Expression * initializeMemberAndGroupId(cstring tableName, IR::IndexedVector< IR::StatOrDecl > *decls)
Definition dpdkArch.cpp:2134
Definition dpdkArch.h:340
Definition dpdkArch.h:850
Definition dpdkArch.h:971
Definition dpdkArch.h:1065
Definition dpdkArch.h:1036
Definition dpdkArch.h:1089
Definition node.h:50
Definition node.h:93
Definition vector.h:56
Definition visitor.h:396
Definition typeChecker.h:32
Definition removeLeftSlices.h:35
Definition evaluator.h:114
Definition methodInstance.h:149
Definition flattenInterfaceStructs.h:253
Definition sideEffects.h:295
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
Class used to encode maps from paths to declarations.
Definition referenceMap.h:66
cstring newName(std::string_view base) override
Generate a name from base that fresh for the program.
Definition referenceMap.cpp:96
Definition resolveReferences.h:121
Definition typeChecker.h:55
Definition typeChecker.h:78
Definition typeMap.h:41
Definition dpdkProgramStructure.h:117
Definition pass_manager.h:38
Definition pass_manager.h:143
Definition visitor.h:420
Definition source_file.h:123
Definition pass_manager.h:182
Definition cstring.h:80
Definition ordered_map.h:30
Definition dpdk/backend.cpp:36
void expressionUnrollSanityCheck(const IR::Expression *e)
Definition dpdkArch.cpp:47
Definition dpdkArch.h:814
Definition dpdkArch.h:819
Definition cstring.h:76
Definition dpdkArch.h:253
Definition dpdkArch.h:108
Definition dpdkArch.h:1349
Definition dpdkArch.h:1529
Definition dpdkArch.h:278
Collect information related to P4 programs targeting dpdk.
Definition dpdkProgramStructure.h:14
bool isPSA(void)
Predicate that states whether architecture is PSA or not.
Definition dpdkProgramStructure.h:87
Definition id.h:28
bool is() const noexcept
Definition rtti.h:216
Definition visitor.h:45