From bfa6b0c470510847f832396856b34e85964bf2f0 Mon Sep 17 00:00:00 2001 From: "Komal, Jain" Date: Mon, 6 Jan 2025 07:23:15 +0530 Subject: [PATCH] Align key components to the nearest 8-bit size Signed-off-by: Komal, Jain --- backends/ebpf/codeGen.cpp | 120 ++++++++-- backends/ebpf/codeGen.h | 6 +- backends/ebpf/ebpfBackend.cpp | 2 +- backends/ebpf/ebpfParser.cpp | 10 +- backends/ebpf/ebpfParser.h | 1 + backends/ebpf/ebpfType.cpp | 62 ++++- backends/ebpf/ebpfType.h | 18 +- backends/ebpf/psa/backend.cpp | 2 +- backends/ebpf/target.h | 5 + backends/tc/CMakeLists.txt | 1 + backends/tc/backend.cpp | 2 +- backends/tc/ebpfCodeGen.cpp | 226 +++++++++++++++--- backends/tc/ebpfCodeGen.h | 2 + .../add_entry_1_example_control_blocks.c | 14 +- .../add_entry_1_example_parser.c | 2 +- .../add_entry_1_example_parser.h | 53 +++- .../add_entry_3_example_control_blocks.c | 14 +- .../add_entry_3_example_parser.c | 2 +- .../add_entry_3_example_parser.h | 53 +++- .../add_entry_example_parser.h | 49 +++- .../calculator_control_blocks.c | 36 +-- .../p4tc_samples_outputs/calculator_parser.c | 12 +- .../p4tc_samples_outputs/calculator_parser.h | 49 +++- .../checksum_control_blocks.c | 8 +- .../p4tc_samples_outputs/checksum_parser.h | 49 +++- .../const_entries_range_mask_parser.h | 45 ++++ ...default_action_example_01_control_blocks.c | 4 +- .../default_action_example_01_parser.c | 4 +- .../default_action_example_01_parser.h | 49 +++- .../default_action_example_control_blocks.c | 4 +- .../default_action_example_parser.c | 4 +- .../default_action_example_parser.h | 49 +++- ...ault_action_with_param_01_control_blocks.c | 4 +- .../default_action_with_param_01_parser.c | 4 +- .../default_action_with_param_01_parser.h | 49 +++- ...default_action_with_param_control_blocks.c | 4 +- .../default_action_with_param_parser.c | 4 +- .../default_action_with_param_parser.h | 49 +++- ...default_hit_const_example_control_blocks.c | 4 +- .../default_hit_const_example_parser.c | 4 +- .../default_hit_const_example_parser.h | 49 +++- .../digest_01_control_blocks.c | 8 +- .../p4tc_samples_outputs/digest_01_parser.h | 51 +++- .../digest_control_blocks.c | 10 +- testdata/p4tc_samples_outputs/digest_parser.h | 51 +++- .../digest_parser_meta_control_blocks.c | 8 +- .../digest_parser_meta_parser.h | 49 +++- .../direct_counter_example_control_blocks.c | 8 +- .../direct_counter_example_parser.h | 49 +++- .../direct_meter_color_parser.h | 49 +++- .../direct_meter_parser.h | 49 +++- .../drop_packet_example_control_blocks.c | 4 +- .../drop_packet_example_parser.c | 4 +- .../drop_packet_example_parser.h | 49 +++- .../global_action_example_01_control_blocks.c | 12 +- .../global_action_example_01_parser.h | 49 +++- .../global_action_example_02_control_blocks.c | 16 +- .../global_action_example_02_parser.h | 49 +++- testdata/p4tc_samples_outputs/hash1_parser.h | 49 +++- testdata/p4tc_samples_outputs/hash_parser.h | 49 +++- ...direct_counter_01_example_control_blocks.c | 8 +- .../indirect_counter_01_example_parser.h | 49 +++- .../internetchecksum_01_control_blocks.c | 4 +- .../internetchecksum_01_parser.h | 49 +++- .../ipip_control_blocks.c | 4 +- testdata/p4tc_samples_outputs/ipip_parser.h | 49 +++- .../is_host_port_control_blocks.c | 8 +- .../is_host_port_parser.h | 49 +++- .../is_net_port_control_blocks.c | 8 +- .../p4tc_samples_outputs/is_net_port_parser.h | 49 +++- .../matchtype_control_blocks.c | 4 +- .../p4tc_samples_outputs/matchtype_parser.c | 4 +- .../p4tc_samples_outputs/matchtype_parser.h | 49 +++- .../p4tc_samples_outputs/meter_color_parser.h | 49 +++- testdata/p4tc_samples_outputs/meter_parser.h | 49 +++- .../mix_matchtype_example_control_blocks.c | 4 +- .../mix_matchtype_example_parser.c | 4 +- .../mix_matchtype_example_parser.h | 49 +++- ...ultiple_tables_example_01_control_blocks.c | 4 +- .../multiple_tables_example_01_parser.c | 4 +- .../multiple_tables_example_01_parser.h | 49 +++- ...ultiple_tables_example_02_control_blocks.c | 4 +- .../multiple_tables_example_02_parser.c | 4 +- .../multiple_tables_example_02_parser.h | 49 +++- .../name_annotation_example_control_blocks.c | 4 +- .../name_annotation_example_parser.c | 4 +- .../name_annotation_example_parser.h | 49 +++- .../no_table_example_control_blocks.c | 4 +- .../no_table_example_parser.c | 4 +- .../no_table_example_parser.h | 49 +++- .../noaction_example_01_control_blocks.c | 4 +- .../noaction_example_01_parser.c | 4 +- .../noaction_example_01_parser.h | 49 +++- .../noaction_example_02_control_blocks.c | 4 +- .../noaction_example_02_parser.c | 4 +- .../noaction_example_02_parser.h | 49 +++- ...ummask_annotation_example_control_blocks.c | 4 +- .../nummask_annotation_example_parser.c | 4 +- .../nummask_annotation_example_parser.h | 49 +++- .../send_to_port_example_control_blocks.c | 4 +- .../send_to_port_example_parser.c | 4 +- .../send_to_port_example_parser.h | 49 +++- .../set_entry_timer_example_control_blocks.c | 4 +- .../set_entry_timer_example_parser.c | 4 +- .../set_entry_timer_example_parser.h | 49 +++- .../simple_exact_example_control_blocks.c | 8 +- .../simple_exact_example_parser.h | 49 +++- .../simple_extern_example_control_blocks.c | 8 +- .../simple_extern_example_parser.h | 49 +++- .../simple_lpm_example_control_blocks.c | 8 +- .../simple_lpm_example_parser.h | 49 +++- .../simple_ternary_example_control_blocks.c | 8 +- .../simple_ternary_example_parser.h | 49 +++- .../size_param_example_control_blocks.c | 4 +- .../size_param_example_parser.c | 4 +- .../size_param_example_parser.h | 49 +++- .../skb_meta_control_blocks.c | 12 +- .../p4tc_samples_outputs/skb_meta_parser.c | 4 +- .../p4tc_samples_outputs/skb_meta_parser.h | 49 +++- ...c_may_override_example_01_control_blocks.c | 4 +- .../tc_may_override_example_01_parser.c | 4 +- .../tc_may_override_example_01_parser.h | 49 +++- ...c_may_override_example_02_control_blocks.c | 4 +- .../tc_may_override_example_02_parser.c | 4 +- .../tc_may_override_example_02_parser.h | 49 +++- ...c_may_override_example_03_control_blocks.c | 4 +- .../tc_may_override_example_03_parser.c | 4 +- .../tc_may_override_example_03_parser.h | 49 +++- ...c_may_override_example_04_control_blocks.c | 4 +- .../tc_may_override_example_04_parser.c | 4 +- .../tc_may_override_example_04_parser.h | 49 +++- ...c_may_override_example_05_control_blocks.c | 4 +- .../tc_may_override_example_05_parser.c | 4 +- .../tc_may_override_example_05_parser.h | 49 +++- ...c_may_override_example_06_control_blocks.c | 4 +- .../tc_may_override_example_06_parser.c | 4 +- .../tc_may_override_example_06_parser.h | 49 +++- ...c_may_override_example_07_control_blocks.c | 4 +- .../tc_may_override_example_07_parser.c | 4 +- .../tc_may_override_example_07_parser.h | 49 +++- ...c_may_override_example_08_control_blocks.c | 4 +- .../tc_may_override_example_08_parser.c | 4 +- .../tc_may_override_example_08_parser.h | 49 +++- ...c_may_override_example_09_control_blocks.c | 4 +- .../tc_may_override_example_09_parser.c | 4 +- .../tc_may_override_example_09_parser.h | 49 +++- .../tc_type_annotation_example_parser.h | 49 +++- .../test_ipv6_example_control_blocks.c | 4 +- .../test_ipv6_example_parser.c | 2 +- .../test_ipv6_example_parser.h | 51 +++- 150 files changed, 3302 insertions(+), 410 deletions(-) diff --git a/backends/ebpf/codeGen.cpp b/backends/ebpf/codeGen.cpp index 8dc1086123b..76fe5f437d8 100644 --- a/backends/ebpf/codeGen.cpp +++ b/backends/ebpf/codeGen.cpp @@ -352,7 +352,6 @@ void CodeGenInspector::emitAssignStatement(const IR::Type *ltype, const IR::Expr width = scalar->implementationWidthInBits(); memcpy = !EBPFScalarType::generatesScalar(width); } - builder->emitIndent(); if (memcpy) { builder->append("__builtin_memcpy(&"); @@ -368,15 +367,15 @@ void CodeGenInspector::emitAssignStatement(const IR::Type *ltype, const IR::Expr visit(rexpr); builder->appendFormat(", %d)", scalar->bytesRequired()); } else { - if (lexpr != nullptr) { - visit(lexpr); - } else { - builder->append(lpath); - } - builder->append(" = "); if (builder->target->name == "P4TC") { - emitTCAssignmentEndianessConversion(lexpr, rexpr); + emitTCAssignmentEndianessConversion(ltype, lexpr, rexpr, lpath); } else { + if (lexpr != nullptr) { + visit(lexpr); + } else { + builder->append(lpath); + } + builder->append(" = "); visit(rexpr); } } @@ -493,7 +492,7 @@ void CodeGenInspector::emitAndConvertByteOrder(const IR::Expression *expr, cstri } unsigned shift = loadSize - widthToEmit; builder->appendFormat("%v(", emit); - visit(expr); + getBitAlignment(expr); if (shift != 0 && byte_order == "HOST") builder->appendFormat(" << %d", shift); builder->append(")"); } @@ -513,7 +512,7 @@ void CodeGenInspector::emitTCBinaryOperation(const IR::Operation_Binary *b, bool rByteOrder = tcTarget->getByteOrder(typeMap, action, rexpr); } if (lByteOrder == rByteOrder) { - visit(lexpr); + getBitAlignment(lexpr); if (isScalar) { builder->spc(); builder->append(stringop); @@ -522,7 +521,7 @@ void CodeGenInspector::emitTCBinaryOperation(const IR::Operation_Binary *b, bool builder->append(", &"); } if (!b->is()) expressionPrecedence = b->getPrecedence() + 1; - visit(rexpr); + getBitAlignment(rexpr); return; } if (lByteOrder == "NETWORK") { @@ -543,14 +542,14 @@ void CodeGenInspector::emitTCBinaryOperation(const IR::Operation_Binary *b, bool builder->append(", &"); } if (!b->is()) expressionPrecedence = b->getPrecedence() + 1; - visit(rexpr); + getBitAlignment(rexpr); return; } else if (rByteOrder == "NETWORK") { // ConvertRight auto ftype = typeMap->getType(rexpr); auto et = EBPFTypeFactory::instance->create(ftype); unsigned width = dynamic_cast(et)->widthInBits(); - visit(lexpr); + getBitAlignment(rexpr); if (isScalar) { builder->spc(); builder->append(stringop); @@ -568,8 +567,19 @@ void CodeGenInspector::emitTCBinaryOperation(const IR::Operation_Binary *b, bool return; } -void CodeGenInspector::emitTCAssignmentEndianessConversion(const IR::Expression *lexpr, - const IR::Expression *rexpr) { +void CodeGenInspector::emitTCAssignmentEndianessConversion(const IR::Type *ltype, + const IR::Expression *lexpr, + const IR::Expression *rexpr, + cstring lpath) { + bool needsBitAlignment = storeBitAlignment(ltype, lexpr, lpath); + if (!needsBitAlignment) { + if (lexpr != nullptr) { + visit(lexpr); + } else { + builder->append(lpath); + } + builder->append(" = "); + } auto action = findContext(); auto b = dynamic_cast(builder->target); cstring lByteOrder = "HOST"_cs, rByteOrder = "HOST"_cs; @@ -579,18 +589,14 @@ void CodeGenInspector::emitTCAssignmentEndianessConversion(const IR::Expression if (rexpr) { rByteOrder = b->getByteOrder(typeMap, action, rexpr); } - if (lByteOrder == rByteOrder) { - visit(rexpr); - return; - } auto ftype = typeMap->getType(rexpr); auto et = EBPFTypeFactory::instance->create(ftype); unsigned width = dynamic_cast(et)->widthInBits(); if (width <= 8) { visit(rexpr); - return; - } - if (rByteOrder == "NETWORK") { + } else if (lByteOrder == rByteOrder) { + getBitAlignment(rexpr); + } else if (rByteOrder == "NETWORK") { // If left side of assignment is not annotated field i.e host endian and right expression // is annotated field i.e network endian, we need to convert rexp to host order. // Example - @@ -598,8 +604,7 @@ void CodeGenInspector::emitTCAssignmentEndianessConversion(const IR::Expression // select_0 = bntoh(hdr.ipv4.diffserv) // emitAndConvertByteOrder(rexpr, "HOST"_cs); - } - if (lByteOrder == "NETWORK") { + } else if (lByteOrder == "NETWORK") { // If left side of assignment is annotated field i.e network endian, we need to convert // right expression to network order. // Example - @@ -608,8 +613,73 @@ void CodeGenInspector::emitTCAssignmentEndianessConversion(const IR::Expression // emitAndConvertByteOrder(rexpr, "NETWORK"_cs); } + if (needsBitAlignment) { + builder->append("))"); + } +} - return; +bool CodeGenInspector::storeBitAlignment(const IR::Type *ltype, const IR::Expression *lexpr, + cstring lpath) { + auto tcTarget = dynamic_cast(builder->target); + if (lexpr != nullptr) { + if (!(lexpr->is() || lexpr->is())) { + return false; + } + } + auto ebpfType = EBPFTypeFactory::instance->create(ltype); + EBPFScalarType *scalar = nullptr; + if (ebpfType->is()) { + scalar = ebpfType->to(); + bool primitive = tcTarget->isPrimitiveByteAligned(scalar->implementationWidthInBits()); + if (primitive) { + return false; + } else { + cstring storePrimitive = scalar->implementationWidthInBits() < 32 + ? "storePrimitive32"_cs + : "storePrimitive64"_cs; + builder->appendFormat("%v((u8 *)&", storePrimitive); + if (lexpr != nullptr) { + visit(lexpr); + } else { + builder->append(lpath); + } + builder->appendFormat(", %d, (", scalar->implementationWidthInBits()); + return true; + } + } + return false; +} + +void CodeGenInspector::getBitAlignment(const IR::Expression *expression) { + if (expression->is() || expression->is()) { + auto ftype = typeMap->getType(expression); + if (!ftype) { + visit(expression); + return; + } + auto tcTarget = dynamic_cast(builder->target); + auto ebpfType = EBPFTypeFactory::instance->create(ftype); + EBPFScalarType *scalar = nullptr; + if (ebpfType->is()) { + scalar = ebpfType->to(); + bool isPrimitive = + tcTarget->isPrimitiveByteAligned(scalar->implementationWidthInBits()); + if (!isPrimitive) { + cstring getPrimitive = scalar->implementationWidthInBits() < 32 + ? "getPrimitive32"_cs + : "getPrimitive64"_cs; + builder->appendFormat("%v((u8 *)", getPrimitive); + visit(expression); + builder->appendFormat(", %d)", scalar->implementationWidthInBits()); + } else { + visit(expression); + } + } else { + visit(expression); + } + } else { + visit(expression); + } } unsigned EBPFInitializerUtils::ebpfTypeWidth(P4::TypeMap *typeMap, const IR::Expression *expr) { diff --git a/backends/ebpf/codeGen.h b/backends/ebpf/codeGen.h index ebc4b8eb40b..a359f38ca0e 100644 --- a/backends/ebpf/codeGen.h +++ b/backends/ebpf/codeGen.h @@ -128,8 +128,10 @@ class CodeGenInspector : public Inspector { void widthCheck(const IR::Node *node) const; void emitAndConvertByteOrder(const IR::Expression *expr, cstring byte_order); void emitTCBinaryOperation(const IR::Operation_Binary *b, bool isScalar); - void emitTCAssignmentEndianessConversion(const IR::Expression *lexpr, - const IR::Expression *rexpr); + void emitTCAssignmentEndianessConversion(const IR::Type *ltype, const IR::Expression *lexpr, + const IR::Expression *rexpr, cstring lpath); + void getBitAlignment(const IR::Expression *expression); + bool storeBitAlignment(const IR::Type *ltype, const IR::Expression *lexpr, cstring lpath); }; class EBPFInitializerUtils { diff --git a/backends/ebpf/ebpfBackend.cpp b/backends/ebpf/ebpfBackend.cpp index 86477f45138..f9b074f0ed6 100644 --- a/backends/ebpf/ebpfBackend.cpp +++ b/backends/ebpf/ebpfBackend.cpp @@ -32,7 +32,7 @@ void emitFilterModel(const EbpfOptions &options, Target *target, const IR::Tople CodeBuilder c(target); CodeBuilder h(target); - EBPFTypeFactory::createFactory(typeMap); + EBPFTypeFactory::createFactory(typeMap, false); auto ebpfprog = new EBPFProgram(options, toplevel->getProgram(), refMap, typeMap, toplevel); if (!ebpfprog->build()) return; diff --git a/backends/ebpf/ebpfParser.cpp b/backends/ebpf/ebpfParser.cpp index 85eaaa1fbc0..9183300f7eb 100644 --- a/backends/ebpf/ebpfParser.cpp +++ b/backends/ebpf/ebpfParser.cpp @@ -185,17 +185,17 @@ bool StateTranslationVisitor::preorder(const IR::SelectExpression *expression) { BUG_CHECK(expression->select->components.size() == 1, "%1%: tuple not eliminated in select", expression->select); selectValue = state->parser->program->refMap->newName("select"); - auto type = state->parser->program->typeMap->getType(expression->select, true); - if (auto list = type->to()) { + selectType = state->parser->program->typeMap->getType(expression->select, true); + if (auto list = selectType->to()) { BUG_CHECK(list->components.size() == 1, "%1% list type with more than 1 element", list); - type = list->components.at(0); + selectType = list->components.at(0); } - auto etype = EBPFTypeFactory::instance->create(type); + auto etype = EBPFTypeFactory::instance->create(selectType); builder->emitIndent(); etype->declare(builder, selectValue, false); builder->endOfStatement(true); - emitAssignStatement(type, nullptr, selectValue, expression->select->components.at(0)); + emitAssignStatement(selectType, nullptr, selectValue, expression->select->components.at(0)); builder->newline(); // Init value_sets diff --git a/backends/ebpf/ebpfParser.h b/backends/ebpf/ebpfParser.h index 42b645a2e8d..a88debcf7f3 100644 --- a/backends/ebpf/ebpfParser.h +++ b/backends/ebpf/ebpfParser.h @@ -32,6 +32,7 @@ class StateTranslationVisitor : public CodeGenInspector { protected: /// Stores the result of evaluating the select argument. cstring selectValue; + const IR::Type *selectType; P4::P4CoreLibrary &p4lib; const EBPFParserState *state; diff --git a/backends/ebpf/ebpfType.cpp b/backends/ebpf/ebpfType.cpp index 10f4f7a26aa..8cb91d02dc4 100644 --- a/backends/ebpf/ebpfType.cpp +++ b/backends/ebpf/ebpfType.cpp @@ -19,6 +19,7 @@ limitations under the License. namespace P4::EBPF { EBPFTypeFactory *EBPFTypeFactory::instance; +bool EBPFTypeFactory::isTC; EBPFType *EBPFTypeFactory::create(const IR::Type *type) { CHECK_NULL(type); @@ -27,7 +28,11 @@ EBPFType *EBPFTypeFactory::create(const IR::Type *type) { if (type->is()) { result = new EBPFBoolType(); } else if (auto bt = type->to()) { - result = new EBPFScalarType(bt); + if (EBPFTypeFactory::isTC) { + result = new EBPFScalarTypePNA(bt); + } else { + result = new EBPFScalarType(bt); + } } else if (auto st = type->to()) { result = new EBPFStructType(st); } else if (auto tt = type->to()) { @@ -384,4 +389,59 @@ void EBPFMethodDeclaration::emit(CodeBuilder *builder) { builder->newline(); } +unsigned EBPFScalarTypePNA::alignment() const { + if (width <= 8) + return 1; + else if (width <= 16) + return 2; + else if (width <= 24) + return 1; // compiled as u8* + else if (width <= 32) + return 4; + else if (width <= 56) + return 1; // compiled as u8* + else if (width <= 64) + return 8; + else + // compiled as u8* + return 1; +} + +void EBPFScalarTypePNA::declare(CodeBuilder *builder, cstring id, bool asPointer) { + if (generatesScalar(width) && isPrimitiveByteAligned == true) { + emit(builder); + if (asPointer) builder->append("*"); + builder->spc(); + builder->append(id); + } else { + if (asPointer) + builder->appendFormat("u8* %s", id.c_str()); + else + builder->appendFormat("u8 %s[%d]", id.c_str(), bytesRequired()); + } +} + +void EBPFScalarTypePNA::declareInit(CodeBuilder *builder, cstring id, bool asPointer) { + if (generatesScalar(width) && isPrimitiveByteAligned == true) { + emit(builder); + if (asPointer) builder->append("*"); + builder->spc(); + id = id + cstring(" = 0"); + builder->append(id); + } else { + if (asPointer) + builder->appendFormat("u8* %s = NULL", id.c_str()); + else + builder->appendFormat("u8 %s[%d] = {0}", id.c_str(), bytesRequired()); + } +} + +void EBPFScalarTypePNA::emitInitializer(CodeBuilder *builder) { + if (generatesScalar(width) && isPrimitiveByteAligned == true) { + builder->append("0"); + } else { + builder->append("{ 0 }"); + } +} + } // namespace P4::EBPF diff --git a/backends/ebpf/ebpfType.h b/backends/ebpf/ebpfType.h index c3493b6affc..e160cb58906 100644 --- a/backends/ebpf/ebpfType.h +++ b/backends/ebpf/ebpfType.h @@ -62,8 +62,10 @@ class EBPFTypeFactory { public: static EBPFTypeFactory *instance; - static void createFactory(const P4::TypeMap *typeMap) { + static bool isTC; + static void createFactory(const P4::TypeMap *typeMap, bool TC) { EBPFTypeFactory::instance = new EBPFTypeFactory(typeMap); + EBPFTypeFactory::isTC = TC; } virtual EBPFType *create(const IR::Type *type); }; @@ -223,6 +225,20 @@ class EBPFMethodDeclaration : public EBPFObject { DECLARE_TYPEINFO(EBPFMethodDeclaration, EBPFObject); }; +class EBPFScalarTypePNA : public EBPFScalarType { + bool isPrimitiveByteAligned = false; + + public: + explicit EBPFScalarTypePNA(const IR::Type_Bits *bits) : EBPFScalarType(bits) { + isPrimitiveByteAligned = (width <= 8 || width <= 16 || (width > 24 && width <= 32) || + (width > 56 && width <= 64)); + } + unsigned alignment() const; + void declare(CodeBuilder *builder, cstring id, bool asPointer); + void declareInit(CodeBuilder *builder, cstring id, bool asPointer); + void emitInitializer(CodeBuilder *builder); +}; + } // namespace P4::EBPF #endif /* BACKENDS_EBPF_EBPFTYPE_H_ */ diff --git a/backends/ebpf/psa/backend.cpp b/backends/ebpf/psa/backend.cpp index 03293b15908..4fb9c01fc87 100644 --- a/backends/ebpf/psa/backend.cpp +++ b/backends/ebpf/psa/backend.cpp @@ -61,7 +61,7 @@ void PSASwitchBackend::convert(const IR::ToplevelBlock *tlb) { main->apply(*parsePsaArch); program = toplevel->getProgram(); - EBPFTypeFactory::createFactory(typeMap); + EBPFTypeFactory::createFactory(typeMap, false); auto *convertToEbpfPSA = new ConvertToEbpfPSA(options, refMap, typeMap); PassManager toEBPF = { new P4::DiscoverStructure(&structure), diff --git a/backends/ebpf/target.h b/backends/ebpf/target.h index 4a8bb813439..831a6e1c7b0 100644 --- a/backends/ebpf/target.h +++ b/backends/ebpf/target.h @@ -238,6 +238,11 @@ class P4TCTarget : public KernelSamplesTarget { } return "HOST"_cs; } + + bool isPrimitiveByteAligned(int width) const { + return (width <= 8 || width <= 16 || (width > 24 && width <= 32) || + (width > 56 && width <= 64)); + } }; /// Target XDP. diff --git a/backends/tc/CMakeLists.txt b/backends/tc/CMakeLists.txt index 92f308fea97..2e3ab1594f5 100644 --- a/backends/tc/CMakeLists.txt +++ b/backends/tc/CMakeLists.txt @@ -63,6 +63,7 @@ set(P4TC_BACKEND_HEADERS pnaProgramStructure.h tcAnnotations.h tcExterns.h + handleBitAlignment.h version.h ../ebpf/codeGen.h ../ebpf/ebpfBackend.h diff --git a/backends/tc/backend.cpp b/backends/tc/backend.cpp index 1d3f93461d7..11378ce51f2 100644 --- a/backends/tc/backend.cpp +++ b/backends/tc/backend.cpp @@ -101,7 +101,7 @@ bool Backend::ebpfCodeGen(P4::ReferenceMap *refMapEBPF, P4::TypeMap *typeMapEBPF main->apply(*parsePnaArch); program = top->getProgram(); - EBPF::EBPFTypeFactory::createFactory(typeMapEBPF); + EBPF::EBPFTypeFactory::createFactory(typeMapEBPF, true); auto convertToEbpf = new ConvertToEbpfPNA(ebpfOption, refMapEBPF, typeMapEBPF, tcIR); PassManager toEBPF = { new P4::DiscoverStructure(&structure), diff --git a/backends/tc/ebpfCodeGen.cpp b/backends/tc/ebpfCodeGen.cpp index ac4bded08ff..6bf784ef118 100644 --- a/backends/tc/ebpfCodeGen.cpp +++ b/backends/tc/ebpfCodeGen.cpp @@ -222,6 +222,60 @@ void PNAArchTC::emitInstances(EBPF::CodeBuilder *builder) const { builder->newline(); } +void PNAArchTC::emitGlobalFunctions(EBPF::CodeBuilder *builder) const { + const char *code = + "static inline u32 getPrimitive32(u8 *a, int size) {\n" + " if(size <=32 || size >=56) {\n" + " bpf_printk(\"Invalid size.\");\n" + " };\n" + " return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]);\n" + "}\n" + "static inline u64 getPrimitive64(u8 *a, int size) {\n" + " if(size <=32 || size >=56) {\n" + " bpf_printk(\"Invalid size.\");\n" + " };\n" + " if(size <= 40) {\n" + " return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | " + "(((u64)a[1]) << 8) | a[0]);\n" + " } else {\n" + " if(size <= 48) {\n" + " return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | " + "(((u64)a[2]) << 16) | (((u64)a[1]) << " + "8) | a[0]);\n" + " } else {\n" + " return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | " + "(((u64)a[3]) << 24) | (((u64)a[2]) << " + "16) | (((u64)a[1]) << 8) | a[0]);\n" + " }\n" + " }\n" + "}\n" + "static inline void storePrimitive32(u8 *a, int size, u32 value) {\n" + " if(size <=16 || size >=24) {\n" + " bpf_printk(\"Invalid size.\");\n" + " };\n" + " a[0] = (u8)(value);\n" + " a[1] = (u8)(value >> 8);\n" + " a[2] = (u8)(value >> 16);\n" + "}\n" + "static inline void storePrimitive64(u8 *a, int size, u64 value) {\n" + " if(size <=32 || size >=56) {\n" + " bpf_printk(\"Invalid size.\");\n" + " };\n" + " a[0] = (u8)(value);\n" + " a[1] = (u8)(value >> 8);\n" + " a[2] = (u8)(value >> 16);\n" + " a[3] = (u8)(value >> 24);\n" + " a[4] = (u8)(value >> 32);\n" + " if (size > 40) {\n" + " a[5] = (u8)(value >> 40);\n" + " }\n" + " if (size > 48) {\n" + " a[6] = (u8)(value >> 48);\n" + " }\n" + "}\n"; + builder->appendLine(code); +} + void PNAArchTC::emitParser(EBPF::CodeBuilder *builder) const { /** * Structure of a C Parser program for PNA @@ -258,6 +312,7 @@ void PNAArchTC::emitHeader(EBPF::CodeBuilder *builder) const { emitP4TCFilterFields(builder); // BPF map definitions. emitInstances(builder); + emitGlobalFunctions(builder); } // =====================TCIngressPipelinePNA============================= @@ -657,6 +712,8 @@ void PnaStateTranslationVisitor::compileExtractField(const IR::Expression *expr, noEndiannessConversion = true; } } + auto tcTarget = dynamic_cast(builder->target); + bool isPrimitive = tcTarget->isPrimitiveByteAligned(widthToExtract); msgStr = absl::StrFormat("Parser: extracting field %v", fieldName); builder->target->emitTraceMessage(builder, msgStr.c_str()); @@ -691,19 +748,38 @@ void PnaStateTranslationVisitor::compileExtractField(const IR::Expression *expr, builder->appendFormat(".%v, %v + BYTES(%v), %d)", fieldName, program->packetStartVar, program->offsetVar, widthToExtract / 8); } else { - visit(expr); - builder->appendFormat(".%v = (", fieldName); - type->emit(builder); - builder->appendFormat(")((%s(%v, BYTES(%v))", helper, program->packetStartVar, - program->offsetVar); - if (shift != 0) builder->appendFormat(" >> %d", shift); - builder->append(")"); - if (widthToExtract != loadSize) { - builder->append(" & EBPF_MASK("); + if (!isPrimitive) { + cstring storePrimitive = + widthToExtract < 32 ? "storePrimitive32"_cs : "storePrimitive64"_cs; + builder->appendFormat("%v((u8 *)&", storePrimitive); + visit(expr); + builder->appendFormat(".%v, %d, (", fieldName, widthToExtract); type->emit(builder); - builder->appendFormat(", %d)", widthToExtract); + builder->appendFormat(")((%s(%v, BYTES(%v))", helper, program->packetStartVar, + program->offsetVar); + if (shift != 0) builder->appendFormat(" >> %d", shift); + builder->append(")"); + if (widthToExtract != loadSize) { + builder->append(" & EBPF_MASK("); + type->emit(builder); + builder->appendFormat(", %d)", widthToExtract); + } + builder->append("))"); + } else { + visit(expr); + builder->appendFormat(".%v = (", fieldName); + type->emit(builder); + builder->appendFormat(")((%s(%v, BYTES(%v))", helper, program->packetStartVar, + program->offsetVar); + if (shift != 0) builder->appendFormat(" >> %d", shift); + builder->append(")"); + if (widthToExtract != loadSize) { + builder->append(" & EBPF_MASK("); + type->emit(builder); + builder->appendFormat(", %d)", widthToExtract); + } + builder->append(")"); } - builder->append(")"); } builder->endOfStatement(true); } else { @@ -832,6 +908,76 @@ void PnaStateTranslationVisitor::compileLookahead(const IR::Expression *destinat builder->blockEnd(true); } +bool PnaStateTranslationVisitor::preorder(const IR::SelectCase *selectCase) { + unsigned width = EBPF::EBPFInitializerUtils::ebpfTypeWidth(typeMap, selectCase->keyset); + bool scalar = EBPF::EBPFScalarType::generatesScalar(width); + auto ebpfType = EBPF::EBPFTypeFactory::instance->create(selectType); + unsigned selectWidth = 0; + if (ebpfType->is()) { + selectWidth = ebpfType->to()->implementationWidthInBits(); + } + auto tcTarget = dynamic_cast(builder->target); + bool isPrimitive = tcTarget->isPrimitiveByteAligned(selectWidth); + + builder->emitIndent(); + if (auto pe = selectCase->keyset->to()) { + builder->append("if ("); + cstring pvsName = pe->path->name.name; + auto pvs = state->parser->getValueSet(pvsName); + if (pvs) pvs->emitLookup(builder); + builder->append(" != NULL)"); + } else if (auto mask = selectCase->keyset->to()) { + if (scalar) { + if (!isPrimitive) { + cstring getPrimitive = selectWidth < 32 ? "getPrimitive32"_cs : "getPrimitive64"_cs; + builder->appendFormat("if ((%v((u8 *)%v, %d)", getPrimitive, selectValue, + selectWidth); + } else { + builder->appendFormat("if ((%v", selectValue); + } + builder->append(" & "); + visit(mask->right); + builder->append(") == ("); + visit(mask->left); + builder->append(" & "); + visit(mask->right); + builder->append("))"); + } else { + unsigned bytes = ROUNDUP(width, 8); + bool first = true; + cstring hex = EBPF::EBPFInitializerUtils::genHexStr( + mask->right->to()->value, width, mask->right); + cstring value = EBPF::EBPFInitializerUtils::genHexStr( + mask->left->to()->value, width, mask->left); + builder->append("if ("); + for (unsigned i = 0; i < bytes; ++i) { + if (!first) { + builder->append(" && "); + } + builder->appendFormat("((%v[%u] & 0x%v)", selectValue, i, hex.substr(2 * i, 2)); + builder->append(" == "); + builder->appendFormat("(%v & %v))", value.substr(2 * i, 2), hex.substr(2 * i, 2)); + first = false; + } + builder->append(") "); + } + } else { + if (!isPrimitive) { + cstring getPrimitive = selectWidth < 32 ? "getPrimitive32"_cs : "getPrimitive64"_cs; + builder->appendFormat("if (%v((u8 *)%v, %d)", getPrimitive, selectValue, selectWidth); + } else { + builder->appendFormat("if (%v", selectValue); + } + builder->append(" == "); + visit(selectCase->keyset); + builder->append(")"); + } + builder->append("goto "); + visit(selectCase->state); + builder->endOfStatement(true); + return false; +} + // =====================EBPFTablePNA============================= void EBPFTablePNA::emitKeyType(EBPF::CodeBuilder *builder) { builder->emitIndent(); @@ -1872,10 +2018,11 @@ void ControlBodyTranslatorPNA::processFunction(const P4::ExternFunction *functio actionName); } builder->emitIndent(); - builder->appendFormat("update_act_bpf_val->u.%v.%v = ", actionExtName, - param); - visit(components.at(index)->expression); - builder->endOfStatement(); + cstring value = "update_act_bpf_val->u."_cs + actionExtName + "."_cs + + param->toString(); + auto valuetype = typeMap->getType(param, true); + emitAssignStatement(valuetype, nullptr, value, + components.at(index)->expression); builder->newline(); } builder->emitIndent(); @@ -2038,10 +2185,13 @@ void ControlBodyTranslatorPNA::processApply(const P4::ApplyMethod *method) { } } + bool primitive = true; if (ebpfType->is()) { scalar = ebpfType->to(); unsigned width = scalar->implementationWidthInBits(); memcpy = !EBPF::EBPFScalarType::generatesScalar(width); + auto tcTarget = dynamic_cast(builder->target); + primitive = tcTarget->isPrimitiveByteAligned(scalar->implementationWidthInBits()); if (isKeyBigEndian == "NETWORK" && isDefnBigEndian == "HOST") { if (width <= 8) { @@ -2059,15 +2209,18 @@ void ControlBodyTranslatorPNA::processApply(const P4::ApplyMethod *method) { } builder->emitIndent(); - if (memcpy) { - builder->appendFormat("__builtin_memcpy(&(%s.%s[0]), &(", keyname.c_str(), - fieldName.c_str()); + if (!primitive) { + builder->appendFormat("__builtin_memcpy(&(%v.%v), &(", keyname, fieldName); + table->codeGen->visit(c->expression); + builder->appendFormat("), %d)", scalar->bytesRequired()); + } else if (memcpy) { + builder->appendFormat("__builtin_memcpy(&(%v.%v[0]), &(", keyname, fieldName); table->codeGen->visit(c->expression); builder->appendFormat("[0]), %d)", scalar->bytesRequired()); } else { - builder->appendFormat("%s.%s = ", keyname.c_str(), fieldName.c_str()); + builder->appendFormat("%v.%v = ", keyname, fieldName); if (isKeyBigEndian == "NETWORK" && isDefnBigEndian == "HOST") - builder->appendFormat("%s(", swap.c_str()); + builder->appendFormat("%v(", swap); table->codeGen->visit(c->expression); if (isKeyBigEndian == "NETWORK" && isDefnBigEndian == "HOST") builder->append(")"); } @@ -2454,6 +2607,8 @@ void DeparserHdrEmitTranslatorPNA::emitField(EBPF::CodeBuilder *builder, cstring unsigned widthToEmit = et->widthInBits(); unsigned emitSize = 0; cstring swap = ""_cs, msgStr; + auto tcTarget = dynamic_cast(builder->target); + bool isPrimitive = tcTarget->isPrimitiveByteAligned(widthToEmit); if (widthToEmit <= 64) { if (program->options.emitTraceMessages) { @@ -2490,14 +2645,29 @@ void DeparserHdrEmitTranslatorPNA::emitField(EBPF::CodeBuilder *builder, cstring widthToEmit < 8 ? (emitSize - alignment - widthToEmit) : (emitSize - widthToEmit); if (!swap.isNullOrEmpty() && !noEndiannessConversion) { - builder->emitIndent(); - visit(hdrExpr); - builder->appendFormat(".%v = %v(", field, swap); - visit(hdrExpr); - builder->appendFormat(".%v", field); - if (shift != 0) builder->appendFormat(" << %d", shift); - builder->append(")"); - builder->endOfStatement(true); + if (!isPrimitive) { + builder->emitIndent(); + cstring storePrimitive = + widthToEmit < 32 ? "storePrimitive32"_cs : "storePrimitive64"_cs; + cstring getPrimitive = widthToEmit < 32 ? "getPrimitive32"_cs : "getPrimitive64"_cs; + builder->appendFormat("%v((u8 *)&", storePrimitive); + visit(hdrExpr); + builder->appendFormat(".%v, %d, (%v(%v(", field, widthToEmit, swap, getPrimitive); + visit(hdrExpr); + builder->appendFormat(".%v, %d)", field, widthToEmit); + if (shift != 0) builder->appendFormat(" << %d", shift); + builder->append(")))"); + builder->endOfStatement(true); + } else { + builder->emitIndent(); + visit(hdrExpr); + builder->appendFormat(".%v = %v(", field, swap); + visit(hdrExpr); + builder->appendFormat(".%v", field); + if (shift != 0) builder->appendFormat(" << %d", shift); + builder->append(")"); + builder->endOfStatement(true); + } } unsigned bitsInFirstByte = widthToEmit % 8; if (bitsInFirstByte == 0) bitsInFirstByte = 8; diff --git a/backends/tc/ebpfCodeGen.h b/backends/tc/ebpfCodeGen.h index 4c44b97532b..08d90a22691 100644 --- a/backends/tc/ebpfCodeGen.h +++ b/backends/tc/ebpfCodeGen.h @@ -105,6 +105,7 @@ class PNAArchTC : public PNAEbpfGenerator { void emitParser(EBPF::CodeBuilder *builder) const override; void emitHeader(EBPF::CodeBuilder *builder) const override; void emitInstances(EBPF::CodeBuilder *builder) const override; + void emitGlobalFunctions(EBPF::CodeBuilder *builder) const; }; class TCIngressPipelinePNA : public EBPF::TCIngressPipeline { @@ -133,6 +134,7 @@ class PnaStateTranslationVisitor : public EBPF::PsaStateTranslationVisitor { void compileExtractField(const IR::Expression *expr, const IR::StructField *field, unsigned hdrOffsetBits, EBPF::EBPFType *type) override; void compileLookahead(const IR::Expression *destination) override; + bool preorder(const IR::SelectCase *selectCase) override; }; class EBPFPnaParser : public EBPF::EBPFPsaParser { diff --git a/testdata/p4tc_samples_outputs/add_entry_1_example_control_blocks.c b/testdata/p4tc_samples_outputs/add_entry_1_example_control_blocks.c index 6cbed581b2d..d07a46bb271 100644 --- a/testdata/p4tc_samples_outputs/add_entry_1_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/add_entry_1_example_control_blocks.c @@ -31,8 +31,8 @@ struct __attribute__((__packed__)) MainControlImpl_ipv4_tbl_1_value { struct { } MainControlImpl_next_hop; struct __attribute__((__packed__)) { - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } MainControlImpl_send_nh; struct { } MainControlImpl_dflt_route_drop; @@ -98,8 +98,8 @@ if (/* hdr->ipv4.isValid() */ /* add_entry(""send_nh"", {hdr->ethernet.dstAddr, hdr->ethernet.srcAddr}, 2) */ struct p4tc_table_entry_act_bpf update_act_bpf = {}; struct MainControlImpl_ipv4_tbl_1_value *update_act_bpf_val = (struct MainControlImpl_ipv4_tbl_1_value*) &update_act_bpf; - update_act_bpf_val->u.MainControlImpl_send_nh.dmac = hdr->ethernet.dstAddr; - update_act_bpf_val->u.MainControlImpl_send_nh.smac = hdr->ethernet.srcAddr; + storePrimitive64((u8 *)&update_act_bpf_val->u.MainControlImpl_send_nh.dmac, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&update_act_bpf_val->u.MainControlImpl_send_nh.smac, 48, (ntohll(getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48) << 16))); update_act_bpf_val->action = MAINCONTROLIMPL_IPV4_TBL_1_ACT_MAINCONTROLIMPL_SEND_NH; /* construct key */ @@ -119,8 +119,8 @@ if (/* hdr->ipv4.isValid() */ break; case MAINCONTROLIMPL_IPV4_TBL_1_ACT_MAINCONTROLIMPL_SEND_NH: { - hdr->ethernet.srcAddr = bpf_cpu_to_be64(value->u.MainControlImpl_send_nh.smac); - hdr->ethernet.dstAddr = ntohll(value->u.MainControlImpl_send_nh.dmac << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (bpf_cpu_to_be64(getPrimitive64((u8 *)value->u.MainControlImpl_send_nh.smac, 48)))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (ntohll(getPrimitive64((u8 *)value->u.MainControlImpl_send_nh.dmac, 48) << 16))); } break; case MAINCONTROLIMPL_IPV4_TBL_1_ACT_MAINCONTROLIMPL_DFLT_ROUTE_DROP: @@ -192,7 +192,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/add_entry_1_example_parser.c b/testdata/p4tc_samples_outputs/add_entry_1_example_parser.c index 0b1c93f47e3..a76620cd9d2 100644 --- a/testdata/p4tc_samples_outputs/add_entry_1_example_parser.c +++ b/testdata/p4tc_samples_outputs/add_entry_1_example_parser.c @@ -85,7 +85,7 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; __builtin_memcpy(&hdr->ethernet.srcAddr, pkt + BYTES(ebpf_packetOffsetInBits), 6); diff --git a/testdata/p4tc_samples_outputs/add_entry_1_example_parser.h b/testdata/p4tc_samples_outputs/add_entry_1_example_parser.h index 6e42437522c..584e28520e9 100644 --- a/testdata/p4tc_samples_outputs/add_entry_1_example_parser.h +++ b/testdata/p4tc_samples_outputs/add_entry_1_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -39,8 +39,8 @@ struct headers_t { struct ipv4_t ipv4; /* ipv4_t */ }; struct tuple_0 { - u64 f0; /* bit<48> */ - u64 f1; /* bit<48> */ + u8 f0[6]; /* bit<48> */ + u8 f1[6]; /* bit<48> */ }; struct hdr_md { @@ -65,3 +65,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/add_entry_3_example_control_blocks.c b/testdata/p4tc_samples_outputs/add_entry_3_example_control_blocks.c index 7e9355909bb..2cbc36a3af0 100644 --- a/testdata/p4tc_samples_outputs/add_entry_3_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/add_entry_3_example_control_blocks.c @@ -31,8 +31,8 @@ struct __attribute__((__packed__)) MainControlImpl_ipv4_tbl_1_value { struct { } MainControlImpl_next_hop; struct __attribute__((__packed__)) { - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } MainControlImpl_send_nh; struct { } MainControlImpl_dflt_route_drop; @@ -98,8 +98,8 @@ if (/* hdr->ipv4.isValid() */ /* add_entry(""send_nh"", {hdr->ethernet.dstAddr, hdr->ethernet.srcAddr}, 2) */ struct p4tc_table_entry_act_bpf update_act_bpf = {}; struct MainControlImpl_ipv4_tbl_1_value *update_act_bpf_val = (struct MainControlImpl_ipv4_tbl_1_value*) &update_act_bpf; - update_act_bpf_val->u.MainControlImpl_send_nh.dmac = hdr->ethernet.dstAddr; - update_act_bpf_val->u.MainControlImpl_send_nh.smac = hdr->ethernet.srcAddr; + storePrimitive64((u8 *)&update_act_bpf_val->u.MainControlImpl_send_nh.dmac, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&update_act_bpf_val->u.MainControlImpl_send_nh.smac, 48, (ntohll(getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48) << 16))); update_act_bpf_val->action = MAINCONTROLIMPL_IPV4_TBL_1_ACT_MAINCONTROLIMPL_SEND_NH; /* construct key */ @@ -119,8 +119,8 @@ if (/* hdr->ipv4.isValid() */ break; case MAINCONTROLIMPL_IPV4_TBL_1_ACT_MAINCONTROLIMPL_SEND_NH: { - hdr->ethernet.srcAddr = bpf_cpu_to_be64(value->u.MainControlImpl_send_nh.smac); - hdr->ethernet.dstAddr = ntohll(value->u.MainControlImpl_send_nh.dmac << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (bpf_cpu_to_be64(getPrimitive64((u8 *)value->u.MainControlImpl_send_nh.smac, 48)))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (ntohll(getPrimitive64((u8 *)value->u.MainControlImpl_send_nh.dmac, 48) << 16))); } break; case MAINCONTROLIMPL_IPV4_TBL_1_ACT_MAINCONTROLIMPL_DFLT_ROUTE_DROP: @@ -192,7 +192,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/add_entry_3_example_parser.c b/testdata/p4tc_samples_outputs/add_entry_3_example_parser.c index 84e09a9ebba..731f3573fff 100644 --- a/testdata/p4tc_samples_outputs/add_entry_3_example_parser.c +++ b/testdata/p4tc_samples_outputs/add_entry_3_example_parser.c @@ -85,7 +85,7 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; __builtin_memcpy(&hdr->ethernet.srcAddr, pkt + BYTES(ebpf_packetOffsetInBits), 6); diff --git a/testdata/p4tc_samples_outputs/add_entry_3_example_parser.h b/testdata/p4tc_samples_outputs/add_entry_3_example_parser.h index 6e42437522c..584e28520e9 100644 --- a/testdata/p4tc_samples_outputs/add_entry_3_example_parser.h +++ b/testdata/p4tc_samples_outputs/add_entry_3_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -39,8 +39,8 @@ struct headers_t { struct ipv4_t ipv4; /* ipv4_t */ }; struct tuple_0 { - u64 f0; /* bit<48> */ - u64 f1; /* bit<48> */ + u8 f0[6]; /* bit<48> */ + u8 f1[6]; /* bit<48> */ }; struct hdr_md { @@ -65,3 +65,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/add_entry_example_parser.h b/testdata/p4tc_samples_outputs/add_entry_example_parser.h index 929d5e054a7..4d41f26aa1c 100644 --- a/testdata/p4tc_samples_outputs/add_entry_example_parser.h +++ b/testdata/p4tc_samples_outputs/add_entry_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -65,3 +65,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/calculator_control_blocks.c b/testdata/p4tc_samples_outputs/calculator_control_blocks.c index 60547beff5f..0bcca9316eb 100644 --- a/testdata/p4tc_samples_outputs/calculator_control_blocks.c +++ b/testdata/p4tc_samples_outputs/calculator_control_blocks.c @@ -69,7 +69,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, meta = &(hdrMd->cpumap_usermeta); { u8 hit; - u64 tmp_5 = 0; + u8 tmp_5[6] = {0}; { if (/* hdr->p4calc.isValid() */ hdr->p4calc.ebpf_valid) { @@ -102,9 +102,9 @@ if (/* hdr->p4calc.isValid() */ case MAINCONTROLIMPL_CALCULATE_ACT_MAINCONTROLIMPL_OPERATION_ADD: { hdr->p4calc.res = (hdr->p4calc.operand_a + hdr->p4calc.operand_b); - tmp_5 = hdr->ethernet.dstAddr; - hdr->ethernet.dstAddr = hdr->ethernet.srcAddr; - hdr->ethernet.srcAddr = tmp_5; + storePrimitive64((u8 *)&tmp_5, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)tmp_5, 48))); /* send_to_port(skb->ifindex) */ compiler_meta__->drop = false; send_to_port(skb->ifindex); @@ -113,9 +113,9 @@ if (/* hdr->p4calc.isValid() */ case MAINCONTROLIMPL_CALCULATE_ACT_MAINCONTROLIMPL_OPERATION_SUB: { hdr->p4calc.res = (hdr->p4calc.operand_a - hdr->p4calc.operand_b); - tmp_5 = hdr->ethernet.dstAddr; - hdr->ethernet.dstAddr = hdr->ethernet.srcAddr; - hdr->ethernet.srcAddr = tmp_5; + storePrimitive64((u8 *)&tmp_5, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)tmp_5, 48))); /* send_to_port(skb->ifindex) */ compiler_meta__->drop = false; send_to_port(skb->ifindex); @@ -124,9 +124,9 @@ if (/* hdr->p4calc.isValid() */ case MAINCONTROLIMPL_CALCULATE_ACT_MAINCONTROLIMPL_OPERATION_AND: { hdr->p4calc.res = (hdr->p4calc.operand_a & hdr->p4calc.operand_b); - tmp_5 = hdr->ethernet.dstAddr; - hdr->ethernet.dstAddr = hdr->ethernet.srcAddr; - hdr->ethernet.srcAddr = tmp_5; + storePrimitive64((u8 *)&tmp_5, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)tmp_5, 48))); /* send_to_port(skb->ifindex) */ compiler_meta__->drop = false; send_to_port(skb->ifindex); @@ -135,9 +135,9 @@ if (/* hdr->p4calc.isValid() */ case MAINCONTROLIMPL_CALCULATE_ACT_MAINCONTROLIMPL_OPERATION_OR: { hdr->p4calc.res = (hdr->p4calc.operand_a | hdr->p4calc.operand_b); - tmp_5 = hdr->ethernet.dstAddr; - hdr->ethernet.dstAddr = hdr->ethernet.srcAddr; - hdr->ethernet.srcAddr = tmp_5; + storePrimitive64((u8 *)&tmp_5, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)tmp_5, 48))); /* send_to_port(skb->ifindex) */ compiler_meta__->drop = false; send_to_port(skb->ifindex); @@ -146,9 +146,9 @@ if (/* hdr->p4calc.isValid() */ case MAINCONTROLIMPL_CALCULATE_ACT_MAINCONTROLIMPL_OPERATION_XOR: { hdr->p4calc.res = (hdr->p4calc.operand_a ^ hdr->p4calc.operand_b); - tmp_5 = hdr->ethernet.dstAddr; - hdr->ethernet.dstAddr = hdr->ethernet.srcAddr; - hdr->ethernet.srcAddr = tmp_5; + storePrimitive64((u8 *)&tmp_5, 48, (getPrimitive64((u8 *)hdr->ethernet.dstAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48))); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)tmp_5, 48))); /* send_to_port(skb->ifindex) */ compiler_meta__->drop = false; send_to_port(skb->ifindex); @@ -230,7 +230,7 @@ if (/* hdr->p4calc.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -245,7 +245,7 @@ if (/* hdr->p4calc.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/calculator_parser.c b/testdata/p4tc_samples_outputs/calculator_parser.c index 0e4dad41d05..76782af8281 100644 --- a/testdata/p4tc_samples_outputs/calculator_parser.c +++ b/testdata/p4tc_samples_outputs/calculator_parser.c @@ -140,10 +140,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h hdr_start = hdr_start_save; ebpf_packetOffsetInBits = ebpf_packetOffsetInBits_save; } - u32 select_0; - select_0 = (((((u32)(((u16)tmp_0.p << 8) | ((u16)tmp_2.four & 0xff)) << 8) & ((1ULL << 24) - 1)) | (((u32)tmp_4.ver & 0xff) & ((1ULL << 24) - 1))) & ((1ULL << 24) - 1)); - if (select_0 == 0x503401)goto parse_p4calc; - if ((select_0 & 0x0) == (0x0 & 0x0))goto accept; + u8 select_0[3]; + storePrimitive32((u8 *)&select_0, 24, ((((((u32)(((u16)tmp_0.p << 8) | ((u16)tmp_2.four & 0xff)) << 8) & ((1ULL << 24) - 1)) | (((u32)tmp_4.ver & 0xff) & ((1ULL << 24) - 1))) & ((1ULL << 24) - 1)))); + if (getPrimitive32((u8 *)select_0, 24) == 0x503401)goto parse_p4calc; + if ((getPrimitive32((u8 *)select_0, 24) & 0x0) == (0x0 & 0x0))goto accept; else goto reject; } parse_p4calc: { @@ -188,10 +188,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/calculator_parser.h b/testdata/p4tc_samples_outputs/calculator_parser.h index 1645fd45158..1eb14dfcc56 100644 --- a/testdata/p4tc_samples_outputs/calculator_parser.h +++ b/testdata/p4tc_samples_outputs/calculator_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -56,3 +56,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/checksum_control_blocks.c b/testdata/p4tc_samples_outputs/checksum_control_blocks.c index 75453be3060..5a1a1d7654e 100644 --- a/testdata/p4tc_samples_outputs/checksum_control_blocks.c +++ b/testdata/p4tc_samples_outputs/checksum_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -89,8 +89,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/checksum_parser.h b/testdata/p4tc_samples_outputs/checksum_parser.h index 300b4de819a..a794e50d420 100644 --- a/testdata/p4tc_samples_outputs/checksum_parser.h +++ b/testdata/p4tc_samples_outputs/checksum_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -79,3 +79,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/const_entries_range_mask_parser.h b/testdata/p4tc_samples_outputs/const_entries_range_mask_parser.h index 2f2e0fa6123..bc634499931 100644 --- a/testdata/p4tc_samples_outputs/const_entries_range_mask_parser.h +++ b/testdata/p4tc_samples_outputs/const_entries_range_mask_parser.h @@ -47,3 +47,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/default_action_example_01_control_blocks.c b/testdata/p4tc_samples_outputs/default_action_example_01_control_blocks.c index fa15e70ed4e..f3fcf035216 100644 --- a/testdata/p4tc_samples_outputs/default_action_example_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/default_action_example_01_control_blocks.c @@ -220,7 +220,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -235,7 +235,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/default_action_example_01_parser.c b/testdata/p4tc_samples_outputs/default_action_example_01_parser.c index cd9f245e208..a8054b23906 100644 --- a/testdata/p4tc_samples_outputs/default_action_example_01_parser.c +++ b/testdata/p4tc_samples_outputs/default_action_example_01_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/default_action_example_01_parser.h b/testdata/p4tc_samples_outputs/default_action_example_01_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/default_action_example_01_parser.h +++ b/testdata/p4tc_samples_outputs/default_action_example_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/default_action_example_control_blocks.c b/testdata/p4tc_samples_outputs/default_action_example_control_blocks.c index df79cd64e45..fc8759edb95 100644 --- a/testdata/p4tc_samples_outputs/default_action_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/default_action_example_control_blocks.c @@ -240,7 +240,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -255,7 +255,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/default_action_example_parser.c b/testdata/p4tc_samples_outputs/default_action_example_parser.c index 108a626cb77..660a2f9fc3d 100644 --- a/testdata/p4tc_samples_outputs/default_action_example_parser.c +++ b/testdata/p4tc_samples_outputs/default_action_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/default_action_example_parser.h b/testdata/p4tc_samples_outputs/default_action_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/default_action_example_parser.h +++ b/testdata/p4tc_samples_outputs/default_action_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/default_action_with_param_01_control_blocks.c b/testdata/p4tc_samples_outputs/default_action_with_param_01_control_blocks.c index 2897a539871..680af68a7a3 100644 --- a/testdata/p4tc_samples_outputs/default_action_with_param_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/default_action_with_param_01_control_blocks.c @@ -240,7 +240,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -255,7 +255,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.c b/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.c index 5423435e2eb..80018a64ecc 100644 --- a/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.c +++ b/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.h b/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.h +++ b/testdata/p4tc_samples_outputs/default_action_with_param_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/default_action_with_param_control_blocks.c b/testdata/p4tc_samples_outputs/default_action_with_param_control_blocks.c index 949201b7b8a..6372bb8271b 100644 --- a/testdata/p4tc_samples_outputs/default_action_with_param_control_blocks.c +++ b/testdata/p4tc_samples_outputs/default_action_with_param_control_blocks.c @@ -240,7 +240,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -255,7 +255,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/default_action_with_param_parser.c b/testdata/p4tc_samples_outputs/default_action_with_param_parser.c index 7c7b9b1bf32..09f075d5197 100644 --- a/testdata/p4tc_samples_outputs/default_action_with_param_parser.c +++ b/testdata/p4tc_samples_outputs/default_action_with_param_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/default_action_with_param_parser.h b/testdata/p4tc_samples_outputs/default_action_with_param_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/default_action_with_param_parser.h +++ b/testdata/p4tc_samples_outputs/default_action_with_param_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/default_hit_const_example_control_blocks.c b/testdata/p4tc_samples_outputs/default_hit_const_example_control_blocks.c index b4f11a1fe74..bf6fe3c95d8 100644 --- a/testdata/p4tc_samples_outputs/default_hit_const_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/default_hit_const_example_control_blocks.c @@ -165,7 +165,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->eth.dstAddr = htonll(hdr->eth.dstAddr << 16); + storePrimitive64((u8 *)&hdr->eth.dstAddr, 48, (htonll(getPrimitive64(hdr->eth.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->eth.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->eth.dstAddr))[1]; @@ -180,7 +180,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->eth.srcAddr = htonll(hdr->eth.srcAddr << 16); + storePrimitive64((u8 *)&hdr->eth.srcAddr, 48, (htonll(getPrimitive64(hdr->eth.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->eth.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->eth.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/default_hit_const_example_parser.c b/testdata/p4tc_samples_outputs/default_hit_const_example_parser.c index 0547d47bdb5..36b8f84f04d 100644 --- a/testdata/p4tc_samples_outputs/default_hit_const_example_parser.c +++ b/testdata/p4tc_samples_outputs/default_hit_const_example_parser.c @@ -133,10 +133,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->eth.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->eth.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->eth.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->eth.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->eth.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/default_hit_const_example_parser.h b/testdata/p4tc_samples_outputs/default_hit_const_example_parser.h index 352d0815e9b..ecb84bce9f5 100644 --- a/testdata/p4tc_samples_outputs/default_hit_const_example_parser.h +++ b/testdata/p4tc_samples_outputs/default_hit_const_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -75,3 +75,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/digest_01_control_blocks.c b/testdata/p4tc_samples_outputs/digest_01_control_blocks.c index ba4a9e80e48..762cede3ac7 100644 --- a/testdata/p4tc_samples_outputs/digest_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/digest_01_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port; - u64 srcMac; - u64 dstMac; + u8 srcMac[6]; + u8 dstMac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -92,8 +92,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.srcMac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dstMac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.srcMac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dstMac, 48))); meta->ingress_port = skb->ifindex; meta->send_digest = true; /* send_to_port(value->u.ingress_send_nh.port) */ diff --git a/testdata/p4tc_samples_outputs/digest_01_parser.h b/testdata/p4tc_samples_outputs/digest_01_parser.h index 6cb04fcdf9a..4f4c4222de8 100644 --- a/testdata/p4tc_samples_outputs/digest_01_parser.h +++ b/testdata/p4tc_samples_outputs/digest_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -41,7 +41,7 @@ struct my_ingress_metadata_t { u8 send_digest; /* bool */ }; struct mac_learn_digest_t { - u64 srcAddr; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u32 ingress_port; /* PortId_t */ }; @@ -67,3 +67,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/digest_control_blocks.c b/testdata/p4tc_samples_outputs/digest_control_blocks.c index 3047bfc234d..d37ceb515fe 100644 --- a/testdata/p4tc_samples_outputs/digest_control_blocks.c +++ b/testdata/p4tc_samples_outputs/digest_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port; - u64 srcMac; - u64 dstMac; + u8 srcMac[6]; + u8 dstMac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -92,8 +92,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.srcMac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dstMac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.srcMac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dstMac, 48))); meta->ingress_port = skb->ifindex; meta->send_digest = true; /* send_to_port(value->u.ingress_send_nh.port) */ @@ -125,7 +125,7 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head __builtin_memset((void *) &mac_learn_digest_0, 0, sizeof(struct mac_learn_digest_t )); { if (meta->send_digest) { - mac_learn_digest_0.srcAddr = hdr->ethernet.srcAddr; + storePrimitive64((u8 *)&mac_learn_digest_0.srcAddr, 48, (getPrimitive64((u8 *)hdr->ethernet.srcAddr, 48))); mac_learn_digest_0.ingress_port = meta->ingress_port; /* digest_inst_0.pack(mac_learn_digest_0) */ diff --git a/testdata/p4tc_samples_outputs/digest_parser.h b/testdata/p4tc_samples_outputs/digest_parser.h index 6cb04fcdf9a..4f4c4222de8 100644 --- a/testdata/p4tc_samples_outputs/digest_parser.h +++ b/testdata/p4tc_samples_outputs/digest_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -41,7 +41,7 @@ struct my_ingress_metadata_t { u8 send_digest; /* bool */ }; struct mac_learn_digest_t { - u64 srcAddr; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u32 ingress_port; /* PortId_t */ }; @@ -67,3 +67,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/digest_parser_meta_control_blocks.c b/testdata/p4tc_samples_outputs/digest_parser_meta_control_blocks.c index 0a07f6218b8..c37f664a7ec 100644 --- a/testdata/p4tc_samples_outputs/digest_parser_meta_control_blocks.c +++ b/testdata/p4tc_samples_outputs/digest_parser_meta_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port; - u64 srcMac; - u64 dstMac; + u8 srcMac[6]; + u8 dstMac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -91,8 +91,8 @@ if (/* hdr->ipv4.isValid() */ switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.srcMac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dstMac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.srcMac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dstMac, 48))); /* send_to_port(value->u.ingress_send_nh.port) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port); diff --git a/testdata/p4tc_samples_outputs/digest_parser_meta_parser.h b/testdata/p4tc_samples_outputs/digest_parser_meta_parser.h index 3f9ca35c9c7..ab5238d4b06 100644 --- a/testdata/p4tc_samples_outputs/digest_parser_meta_parser.h +++ b/testdata/p4tc_samples_outputs/digest_parser_meta_parser.h @@ -15,8 +15,8 @@ struct my_ingress_metadata_t { u32 ingress_port; /* PortId_t */ }; struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -66,3 +66,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/direct_counter_example_control_blocks.c b/testdata/p4tc_samples_outputs/direct_counter_example_control_blocks.c index c218f0a5d7d..54d689ae432 100644 --- a/testdata/p4tc_samples_outputs/direct_counter_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/direct_counter_example_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -92,8 +92,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/direct_counter_example_parser.h b/testdata/p4tc_samples_outputs/direct_counter_example_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/direct_counter_example_parser.h +++ b/testdata/p4tc_samples_outputs/direct_counter_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/direct_meter_color_parser.h b/testdata/p4tc_samples_outputs/direct_meter_color_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/direct_meter_color_parser.h +++ b/testdata/p4tc_samples_outputs/direct_meter_color_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/direct_meter_parser.h b/testdata/p4tc_samples_outputs/direct_meter_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/direct_meter_parser.h +++ b/testdata/p4tc_samples_outputs/direct_meter_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/drop_packet_example_control_blocks.c b/testdata/p4tc_samples_outputs/drop_packet_example_control_blocks.c index 891f46879a7..87edcfa770e 100644 --- a/testdata/p4tc_samples_outputs/drop_packet_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/drop_packet_example_control_blocks.c @@ -163,7 +163,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -178,7 +178,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/drop_packet_example_parser.c b/testdata/p4tc_samples_outputs/drop_packet_example_parser.c index dc11f5f8b34..90e4adfe596 100644 --- a/testdata/p4tc_samples_outputs/drop_packet_example_parser.c +++ b/testdata/p4tc_samples_outputs/drop_packet_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/drop_packet_example_parser.h b/testdata/p4tc_samples_outputs/drop_packet_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/drop_packet_example_parser.h +++ b/testdata/p4tc_samples_outputs/drop_packet_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/global_action_example_01_control_blocks.c b/testdata/p4tc_samples_outputs/global_action_example_01_control_blocks.c index 553c4224a1e..4fb010a9384 100644 --- a/testdata/p4tc_samples_outputs/global_action_example_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/global_action_example_01_control_blocks.c @@ -28,7 +28,7 @@ struct __attribute__((__packed__)) ingress_nh_table2_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; + u8 dmac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -52,8 +52,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } _send_nh; struct { } ingress_drop; @@ -116,8 +116,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head /* send_to_port(value->u._send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u._send_nh.port_id); - hdr->ethernet.srcAddr = value->u._send_nh.smac; - hdr->ethernet.dstAddr = value->u._send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u._send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u._send_nh.dmac, 48))); } break; case INGRESS_NH_TABLE_ACT_INGRESS_DROP: @@ -163,7 +163,7 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE2_ACT_INGRESS_SEND_NH: { - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/global_action_example_01_parser.h b/testdata/p4tc_samples_outputs/global_action_example_01_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/global_action_example_01_parser.h +++ b/testdata/p4tc_samples_outputs/global_action_example_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/global_action_example_02_control_blocks.c b/testdata/p4tc_samples_outputs/global_action_example_02_control_blocks.c index 85ec5ae5be2..6ef1f440b31 100644 --- a/testdata/p4tc_samples_outputs/global_action_example_02_control_blocks.c +++ b/testdata/p4tc_samples_outputs/global_action_example_02_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table2_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -53,8 +53,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } _drop; @@ -114,8 +114,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); @@ -164,8 +164,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE2_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/global_action_example_02_parser.h b/testdata/p4tc_samples_outputs/global_action_example_02_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/global_action_example_02_parser.h +++ b/testdata/p4tc_samples_outputs/global_action_example_02_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/hash1_parser.h b/testdata/p4tc_samples_outputs/hash1_parser.h index 94841425a8e..4c36bc703c6 100644 --- a/testdata/p4tc_samples_outputs/hash1_parser.h +++ b/testdata/p4tc_samples_outputs/hash1_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -78,3 +78,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/hash_parser.h b/testdata/p4tc_samples_outputs/hash_parser.h index 893d1608192..303a941b0a1 100644 --- a/testdata/p4tc_samples_outputs/hash_parser.h +++ b/testdata/p4tc_samples_outputs/hash_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -78,3 +78,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/indirect_counter_01_example_control_blocks.c b/testdata/p4tc_samples_outputs/indirect_counter_01_example_control_blocks.c index 0670ed1c4a2..7bf30f612ce 100644 --- a/testdata/p4tc_samples_outputs/indirect_counter_01_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/indirect_counter_01_example_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -95,8 +95,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/indirect_counter_01_example_parser.h b/testdata/p4tc_samples_outputs/indirect_counter_01_example_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/indirect_counter_01_example_parser.h +++ b/testdata/p4tc_samples_outputs/indirect_counter_01_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/internetchecksum_01_control_blocks.c b/testdata/p4tc_samples_outputs/internetchecksum_01_control_blocks.c index 0c5bc31c61e..ce544cf7afc 100644 --- a/testdata/p4tc_samples_outputs/internetchecksum_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/internetchecksum_01_control_blocks.c @@ -33,7 +33,7 @@ struct __attribute__((__packed__)) ingress_fwd_table_value { u32 port; } ingress_set_ipip_csum; struct __attribute__((__packed__)) { - u64 dmac; + u8 dmac[6]; u32 port; } ingress_set_nh; struct { @@ -106,7 +106,7 @@ if (/* hdr->outer.isValid() */ break; case INGRESS_FWD_TABLE_ACT_INGRESS_SET_NH: { - hdr->ethernet.dstAddr = value->u.ingress_set_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_set_nh.dmac, 48))); /* send_to_port(value->u.ingress_set_nh.port) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_set_nh.port); diff --git a/testdata/p4tc_samples_outputs/internetchecksum_01_parser.h b/testdata/p4tc_samples_outputs/internetchecksum_01_parser.h index aa9b19c0bf4..d8b957fb643 100644 --- a/testdata/p4tc_samples_outputs/internetchecksum_01_parser.h +++ b/testdata/p4tc_samples_outputs/internetchecksum_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -85,3 +85,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/ipip_control_blocks.c b/testdata/p4tc_samples_outputs/ipip_control_blocks.c index bbbb11d6701..cbb547ec566 100644 --- a/testdata/p4tc_samples_outputs/ipip_control_blocks.c +++ b/testdata/p4tc_samples_outputs/ipip_control_blocks.c @@ -33,7 +33,7 @@ struct __attribute__((__packed__)) Main_fwd_table_value { u32 port; } Main_set_ipip; struct __attribute__((__packed__)) { - u64 dmac; + u8 dmac[6]; u32 port; } Main_set_nh; struct { @@ -106,7 +106,7 @@ if (/* hdr->outer.isValid() */ break; case MAIN_FWD_TABLE_ACT_MAIN_SET_NH: { - hdr->ethernet.dstAddr = value->u.Main_set_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.Main_set_nh.dmac, 48))); /* send_to_port(value->u.Main_set_nh.port) */ compiler_meta__->drop = false; send_to_port(value->u.Main_set_nh.port); diff --git a/testdata/p4tc_samples_outputs/ipip_parser.h b/testdata/p4tc_samples_outputs/ipip_parser.h index 48b48ac4944..41d75d36aad 100644 --- a/testdata/p4tc_samples_outputs/ipip_parser.h +++ b/testdata/p4tc_samples_outputs/ipip_parser.h @@ -17,8 +17,8 @@ struct metadata_t { u8 push; /* bool */ }; struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -65,3 +65,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/is_host_port_control_blocks.c b/testdata/p4tc_samples_outputs/is_host_port_control_blocks.c index b98c36a886e..a33c2987f1d 100644 --- a/testdata/p4tc_samples_outputs/is_host_port_control_blocks.c +++ b/testdata/p4tc_samples_outputs/is_host_port_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -100,8 +100,8 @@ if (/* hdr->ipv4.isValid() */ switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/is_host_port_parser.h b/testdata/p4tc_samples_outputs/is_host_port_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/is_host_port_parser.h +++ b/testdata/p4tc_samples_outputs/is_host_port_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/is_net_port_control_blocks.c b/testdata/p4tc_samples_outputs/is_net_port_control_blocks.c index 91f2bfb8f3c..b9da7a9e34f 100644 --- a/testdata/p4tc_samples_outputs/is_net_port_control_blocks.c +++ b/testdata/p4tc_samples_outputs/is_net_port_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -100,8 +100,8 @@ if (/* hdr->ipv4.isValid() */ switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/is_net_port_parser.h b/testdata/p4tc_samples_outputs/is_net_port_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/is_net_port_parser.h +++ b/testdata/p4tc_samples_outputs/is_net_port_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/matchtype_control_blocks.c b/testdata/p4tc_samples_outputs/matchtype_control_blocks.c index d5319ec9a1c..715563ece9a 100644 --- a/testdata/p4tc_samples_outputs/matchtype_control_blocks.c +++ b/testdata/p4tc_samples_outputs/matchtype_control_blocks.c @@ -388,7 +388,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -403,7 +403,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/matchtype_parser.c b/testdata/p4tc_samples_outputs/matchtype_parser.c index 6bca9c633c7..66ec6a2885c 100644 --- a/testdata/p4tc_samples_outputs/matchtype_parser.c +++ b/testdata/p4tc_samples_outputs/matchtype_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/matchtype_parser.h b/testdata/p4tc_samples_outputs/matchtype_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/matchtype_parser.h +++ b/testdata/p4tc_samples_outputs/matchtype_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/meter_color_parser.h b/testdata/p4tc_samples_outputs/meter_color_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/meter_color_parser.h +++ b/testdata/p4tc_samples_outputs/meter_color_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/meter_parser.h b/testdata/p4tc_samples_outputs/meter_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/meter_parser.h +++ b/testdata/p4tc_samples_outputs/meter_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/mix_matchtype_example_control_blocks.c b/testdata/p4tc_samples_outputs/mix_matchtype_example_control_blocks.c index fd2088e2766..07e0ef34707 100644 --- a/testdata/p4tc_samples_outputs/mix_matchtype_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/mix_matchtype_example_control_blocks.c @@ -394,7 +394,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -409,7 +409,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.c b/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.c index 0e2e8079452..805bbec82af 100644 --- a/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.c +++ b/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.h b/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.h +++ b/testdata/p4tc_samples_outputs/mix_matchtype_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/multiple_tables_example_01_control_blocks.c b/testdata/p4tc_samples_outputs/multiple_tables_example_01_control_blocks.c index 39e0fc5f4b3..262487b9df4 100644 --- a/testdata/p4tc_samples_outputs/multiple_tables_example_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/multiple_tables_example_01_control_blocks.c @@ -645,7 +645,7 @@ if (hdr->ipv4.protocol != 4 || (hdr->tcp.srcPort <= 3)) { return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -660,7 +660,7 @@ if (hdr->ipv4.protocol != 4 || (hdr->tcp.srcPort <= 3)) { write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.c b/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.c index d765938d0c5..a27bdf43ff7 100644 --- a/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.c +++ b/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.c @@ -133,10 +133,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.h b/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.h index 0bb6fe0e1ed..c5b9e1b9e7f 100644 --- a/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.h +++ b/testdata/p4tc_samples_outputs/multiple_tables_example_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -75,3 +75,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/multiple_tables_example_02_control_blocks.c b/testdata/p4tc_samples_outputs/multiple_tables_example_02_control_blocks.c index ac6fee0a896..57e392f95b6 100644 --- a/testdata/p4tc_samples_outputs/multiple_tables_example_02_control_blocks.c +++ b/testdata/p4tc_samples_outputs/multiple_tables_example_02_control_blocks.c @@ -645,7 +645,7 @@ if (hdr->ipv4.protocol == 6 || ((hdr->ipv4.version > 1) && (hdr->ipv4.ihl <= 2)) return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -660,7 +660,7 @@ if (hdr->ipv4.protocol == 6 || ((hdr->ipv4.version > 1) && (hdr->ipv4.ihl <= 2)) write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.c b/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.c index 866917aa245..c070f2d40e9 100644 --- a/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.c +++ b/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.c @@ -133,10 +133,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.h b/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.h index 0bb6fe0e1ed..c5b9e1b9e7f 100644 --- a/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.h +++ b/testdata/p4tc_samples_outputs/multiple_tables_example_02_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -75,3 +75,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/name_annotation_example_control_blocks.c b/testdata/p4tc_samples_outputs/name_annotation_example_control_blocks.c index 812f90fcde9..a9915adc074 100644 --- a/testdata/p4tc_samples_outputs/name_annotation_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/name_annotation_example_control_blocks.c @@ -238,7 +238,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -253,7 +253,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/name_annotation_example_parser.c b/testdata/p4tc_samples_outputs/name_annotation_example_parser.c index 0733cba3b7d..8f74f215f03 100644 --- a/testdata/p4tc_samples_outputs/name_annotation_example_parser.c +++ b/testdata/p4tc_samples_outputs/name_annotation_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/name_annotation_example_parser.h b/testdata/p4tc_samples_outputs/name_annotation_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/name_annotation_example_parser.h +++ b/testdata/p4tc_samples_outputs/name_annotation_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/no_table_example_control_blocks.c b/testdata/p4tc_samples_outputs/no_table_example_control_blocks.c index 0a660aa2c30..5625dfdce94 100644 --- a/testdata/p4tc_samples_outputs/no_table_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/no_table_example_control_blocks.c @@ -96,7 +96,7 @@ if ((u32)skb->ifindex == 4) { return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -111,7 +111,7 @@ if ((u32)skb->ifindex == 4) { write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/no_table_example_parser.c b/testdata/p4tc_samples_outputs/no_table_example_parser.c index 19bdd105c56..e498f1f9acc 100644 --- a/testdata/p4tc_samples_outputs/no_table_example_parser.c +++ b/testdata/p4tc_samples_outputs/no_table_example_parser.c @@ -108,10 +108,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/no_table_example_parser.h b/testdata/p4tc_samples_outputs/no_table_example_parser.h index 75006eeeca5..1fcda936eb2 100644 --- a/testdata/p4tc_samples_outputs/no_table_example_parser.h +++ b/testdata/p4tc_samples_outputs/no_table_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -69,3 +69,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/noaction_example_01_control_blocks.c b/testdata/p4tc_samples_outputs/noaction_example_01_control_blocks.c index b4f75605375..81ad1368071 100644 --- a/testdata/p4tc_samples_outputs/noaction_example_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/noaction_example_01_control_blocks.c @@ -238,7 +238,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -253,7 +253,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/noaction_example_01_parser.c b/testdata/p4tc_samples_outputs/noaction_example_01_parser.c index 4e0c2a54e70..e14af58ed1a 100644 --- a/testdata/p4tc_samples_outputs/noaction_example_01_parser.c +++ b/testdata/p4tc_samples_outputs/noaction_example_01_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/noaction_example_01_parser.h b/testdata/p4tc_samples_outputs/noaction_example_01_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/noaction_example_01_parser.h +++ b/testdata/p4tc_samples_outputs/noaction_example_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/noaction_example_02_control_blocks.c b/testdata/p4tc_samples_outputs/noaction_example_02_control_blocks.c index d2b3b7e6bed..042da7e4211 100644 --- a/testdata/p4tc_samples_outputs/noaction_example_02_control_blocks.c +++ b/testdata/p4tc_samples_outputs/noaction_example_02_control_blocks.c @@ -214,7 +214,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -229,7 +229,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/noaction_example_02_parser.c b/testdata/p4tc_samples_outputs/noaction_example_02_parser.c index 03e9a9fcdac..622a49069b6 100644 --- a/testdata/p4tc_samples_outputs/noaction_example_02_parser.c +++ b/testdata/p4tc_samples_outputs/noaction_example_02_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/noaction_example_02_parser.h b/testdata/p4tc_samples_outputs/noaction_example_02_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/noaction_example_02_parser.h +++ b/testdata/p4tc_samples_outputs/noaction_example_02_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/nummask_annotation_example_control_blocks.c b/testdata/p4tc_samples_outputs/nummask_annotation_example_control_blocks.c index 7083be009d6..5b646a5350a 100644 --- a/testdata/p4tc_samples_outputs/nummask_annotation_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/nummask_annotation_example_control_blocks.c @@ -161,7 +161,7 @@ if (((u32)skb->ifindex == 2 && /* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->eth.dstAddr = htonll(hdr->eth.dstAddr << 16); + storePrimitive64((u8 *)&hdr->eth.dstAddr, 48, (htonll(getPrimitive64(hdr->eth.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->eth.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->eth.dstAddr))[1]; @@ -176,7 +176,7 @@ if (((u32)skb->ifindex == 2 && /* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->eth.srcAddr = htonll(hdr->eth.srcAddr << 16); + storePrimitive64((u8 *)&hdr->eth.srcAddr, 48, (htonll(getPrimitive64(hdr->eth.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->eth.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->eth.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.c b/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.c index 6f59cc58fd1..278b02b26d9 100644 --- a/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.c +++ b/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.c @@ -133,10 +133,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->eth.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->eth.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->eth.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->eth.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->eth.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.h b/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.h index 352d0815e9b..ecb84bce9f5 100644 --- a/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.h +++ b/testdata/p4tc_samples_outputs/nummask_annotation_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -75,3 +75,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/send_to_port_example_control_blocks.c b/testdata/p4tc_samples_outputs/send_to_port_example_control_blocks.c index d99ba816271..11af1ae8387 100644 --- a/testdata/p4tc_samples_outputs/send_to_port_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/send_to_port_example_control_blocks.c @@ -193,7 +193,7 @@ ext_val = *ext_val_ptr; return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -208,7 +208,7 @@ ext_val = *ext_val_ptr; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/send_to_port_example_parser.c b/testdata/p4tc_samples_outputs/send_to_port_example_parser.c index a21b8aeabf5..9d688e6c7b2 100644 --- a/testdata/p4tc_samples_outputs/send_to_port_example_parser.c +++ b/testdata/p4tc_samples_outputs/send_to_port_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/send_to_port_example_parser.h b/testdata/p4tc_samples_outputs/send_to_port_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/send_to_port_example_parser.h +++ b/testdata/p4tc_samples_outputs/send_to_port_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/set_entry_timer_example_control_blocks.c b/testdata/p4tc_samples_outputs/set_entry_timer_example_control_blocks.c index df6f3416b98..cfb74c6b6f0 100644 --- a/testdata/p4tc_samples_outputs/set_entry_timer_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/set_entry_timer_example_control_blocks.c @@ -248,7 +248,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -263,7 +263,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.c b/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.c index 72e7fa57bdc..d5d24d429cf 100644 --- a/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.c +++ b/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.h b/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.h +++ b/testdata/p4tc_samples_outputs/set_entry_timer_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/simple_exact_example_control_blocks.c b/testdata/p4tc_samples_outputs/simple_exact_example_control_blocks.c index f4cc947d344..1ae9f7283dc 100644 --- a/testdata/p4tc_samples_outputs/simple_exact_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/simple_exact_example_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -89,8 +89,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/simple_exact_example_parser.h b/testdata/p4tc_samples_outputs/simple_exact_example_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/simple_exact_example_parser.h +++ b/testdata/p4tc_samples_outputs/simple_exact_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/simple_extern_example_control_blocks.c b/testdata/p4tc_samples_outputs/simple_extern_example_control_blocks.c index c46b97de1d7..d042bd9bdbf 100644 --- a/testdata/p4tc_samples_outputs/simple_extern_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/simple_extern_example_control_blocks.c @@ -32,8 +32,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } ingress_ext_reg; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -140,8 +140,8 @@ ext_val = *ext_val_ptr; break; case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/simple_extern_example_parser.h b/testdata/p4tc_samples_outputs/simple_extern_example_parser.h index 06164768dec..05f6638e748 100644 --- a/testdata/p4tc_samples_outputs/simple_extern_example_parser.h +++ b/testdata/p4tc_samples_outputs/simple_extern_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -67,3 +67,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/simple_lpm_example_control_blocks.c b/testdata/p4tc_samples_outputs/simple_lpm_example_control_blocks.c index 01f93dca0de..4d25dca449e 100644 --- a/testdata/p4tc_samples_outputs/simple_lpm_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/simple_lpm_example_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -89,8 +89,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/simple_lpm_example_parser.h b/testdata/p4tc_samples_outputs/simple_lpm_example_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/simple_lpm_example_parser.h +++ b/testdata/p4tc_samples_outputs/simple_lpm_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/simple_ternary_example_control_blocks.c b/testdata/p4tc_samples_outputs/simple_ternary_example_control_blocks.c index ce7dfbd4f69..ba386d937db 100644 --- a/testdata/p4tc_samples_outputs/simple_ternary_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/simple_ternary_example_control_blocks.c @@ -33,8 +33,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port_id; - u64 dmac; - u64 smac; + u8 dmac[6]; + u8 smac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -95,8 +95,8 @@ static __always_inline int process(struct __sk_buff *skb, struct my_ingress_head switch (value->action) { case INGRESS_NH_TABLE_ACT_INGRESS_SEND_NH: { - hdr->ethernet.srcAddr = value->u.ingress_send_nh.smac; - hdr->ethernet.dstAddr = value->u.ingress_send_nh.dmac; + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.smac, 48))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (getPrimitive64((u8 *)value->u.ingress_send_nh.dmac, 48))); /* send_to_port(value->u.ingress_send_nh.port_id) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port_id); diff --git a/testdata/p4tc_samples_outputs/simple_ternary_example_parser.h b/testdata/p4tc_samples_outputs/simple_ternary_example_parser.h index a4fbecb2270..521faaed0fc 100644 --- a/testdata/p4tc_samples_outputs/simple_ternary_example_parser.h +++ b/testdata/p4tc_samples_outputs/simple_ternary_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/size_param_example_control_blocks.c b/testdata/p4tc_samples_outputs/size_param_example_control_blocks.c index 22580bb3f9e..e485865a543 100644 --- a/testdata/p4tc_samples_outputs/size_param_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/size_param_example_control_blocks.c @@ -238,7 +238,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -253,7 +253,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/size_param_example_parser.c b/testdata/p4tc_samples_outputs/size_param_example_parser.c index 697c1199d51..77edc9c209e 100644 --- a/testdata/p4tc_samples_outputs/size_param_example_parser.c +++ b/testdata/p4tc_samples_outputs/size_param_example_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/size_param_example_parser.h b/testdata/p4tc_samples_outputs/size_param_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/size_param_example_parser.h +++ b/testdata/p4tc_samples_outputs/size_param_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/skb_meta_control_blocks.c b/testdata/p4tc_samples_outputs/skb_meta_control_blocks.c index 78a78259329..8e74ac5bc9c 100644 --- a/testdata/p4tc_samples_outputs/skb_meta_control_blocks.c +++ b/testdata/p4tc_samples_outputs/skb_meta_control_blocks.c @@ -28,8 +28,8 @@ struct __attribute__((__packed__)) ingress_nh_table_value { } _NoAction; struct __attribute__((__packed__)) { u32 port; - u64 srcMac; - u64 dstMac; + u8 srcMac[6]; + u8 dstMac[6]; } ingress_send_nh; struct { } ingress_drop; @@ -105,8 +105,8 @@ if (/* hdr->ipv4.isValid() */ /* skb_set_meta() */ bpf_p4tc_skb_meta_set(skb,&sa->set,sizeof(sa->set)); tmp_5 = bpf_p4tc_skb_get_tstamp(skb, &sa->get); - hdr->ethernet.srcAddr = (((u64)tmp_5 ^ bpf_cpu_to_be64(value->u.ingress_send_nh.srcMac)) & ((1ULL << 48) - 1)); - hdr->ethernet.dstAddr = ntohll(value->u.ingress_send_nh.dstMac << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (((getPrimitive64((u8 *)value->u.ingress_send_nh.srcMac, 48) ^ bpf_cpu_to_be64(getPrimitive64((u8 *)value->u.ingress_send_nh.srcMac, 48))) & ((1ULL << 48) - 1)))); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (ntohll(getPrimitive64((u8 *)value->u.ingress_send_nh.dstMac, 48) << 16))); /* send_to_port(value->u.ingress_send_nh.port) */ compiler_meta__->drop = false; send_to_port(value->u.ingress_send_nh.port); @@ -181,7 +181,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -196,7 +196,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/skb_meta_parser.c b/testdata/p4tc_samples_outputs/skb_meta_parser.c index feae2db4f8a..e05033545b9 100644 --- a/testdata/p4tc_samples_outputs/skb_meta_parser.c +++ b/testdata/p4tc_samples_outputs/skb_meta_parser.c @@ -89,10 +89,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct my_ingress_h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/skb_meta_parser.h b/testdata/p4tc_samples_outputs/skb_meta_parser.h index 8869b895e06..dbd5b875434 100644 --- a/testdata/p4tc_samples_outputs/skb_meta_parser.h +++ b/testdata/p4tc_samples_outputs/skb_meta_parser.h @@ -16,8 +16,8 @@ struct my_ingress_metadata_t { struct empty_metadata_t { }; struct ethernet_t { - u64 dstAddr; /* bit<48> */ - u64 srcAddr; /* bit<48> */ + u8 dstAddr[6]; /* bit<48> */ + u8 srcAddr[6]; /* bit<48> */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_01_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_01_control_blocks.c index e5917830cc5..7a7e141c132 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_01_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_01_control_blocks.c @@ -258,7 +258,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -273,7 +273,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.c index 16fbf526c82..e68001096eb 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.h index 8f8fb971a53..fe896e14b7e 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_01_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_02_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_02_control_blocks.c index f9f030850ff..adb83ce1d71 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_02_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_02_control_blocks.c @@ -240,7 +240,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -255,7 +255,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.c index 704339bc654..aced249a714 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_02_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_03_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_03_control_blocks.c index 37517dbd541..b026e33c5b5 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_03_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_03_control_blocks.c @@ -240,7 +240,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -255,7 +255,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.c index dfa2d245d8f..53de2347e4a 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_03_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_04_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_04_control_blocks.c index 5839f4c8d78..39d225d9337 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_04_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_04_control_blocks.c @@ -252,7 +252,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -267,7 +267,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.c index 10ba238ba22..fd37c7dd418 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.h index 06f3e6e418c..01a13bd8e3f 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_04_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -62,3 +62,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_05_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_05_control_blocks.c index 061b01e593f..00a0a37efec 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_05_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_05_control_blocks.c @@ -249,7 +249,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -264,7 +264,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.c index dcb5018eb10..cc7b28c23e3 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.h index 74b1b65a628..0202b29fd86 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_05_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -62,3 +62,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_06_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_06_control_blocks.c index 0254808af0e..e8824b7117f 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_06_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_06_control_blocks.c @@ -258,7 +258,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -273,7 +273,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.c index ea7398e3bf4..dfd41dfda9b 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.h index 8f8fb971a53..fe896e14b7e 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_06_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_07_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_07_control_blocks.c index 0539af21ebc..8d2db9971f2 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_07_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_07_control_blocks.c @@ -267,7 +267,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -282,7 +282,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.c index aafa7c3dedc..19a9afa213c 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.h index 8f8fb971a53..fe896e14b7e 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_07_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_08_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_08_control_blocks.c index 9bfbdd7ff76..5aa82dee7db 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_08_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_08_control_blocks.c @@ -267,7 +267,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -282,7 +282,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.c index dcb5a694879..dd5a983893e 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.h index 8f8fb971a53..fe896e14b7e 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_08_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -63,3 +63,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_09_control_blocks.c b/testdata/p4tc_samples_outputs/tc_may_override_example_09_control_blocks.c index 4bd6a9602e1..24b51048947 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_09_control_blocks.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_09_control_blocks.c @@ -258,7 +258,7 @@ if (/* hdr->ipv4.isValid() */ return TC_ACT_SHOT; } - hdr->ethernet.dstAddr = htonll(hdr->ethernet.dstAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (htonll(getPrimitive64(hdr->ethernet.dstAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.dstAddr))[1]; @@ -273,7 +273,7 @@ if (/* hdr->ipv4.isValid() */ write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 5, (ebpf_byte)); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = htonll(hdr->ethernet.srcAddr << 16); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (htonll(getPrimitive64(hdr->ethernet.srcAddr, 48) << 16))); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[0]; write_byte(pkt, BYTES(ebpf_packetOffsetInBits) + 0, (ebpf_byte)); ebpf_byte = ((char*)(&hdr->ethernet.srcAddr))[1]; diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.c b/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.c index 6ee7f4bae87..b9f0b7e9ab4 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.c +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.c @@ -85,10 +85,10 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h goto reject; } - hdr->ethernet.dstAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.dstAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; - hdr->ethernet.srcAddr = (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48)); + storePrimitive64((u8 *)&hdr->ethernet.srcAddr, 48, (u64)((load_dword(pkt, BYTES(ebpf_packetOffsetInBits)) >> 16) & EBPF_MASK(u64, 48))); ebpf_packetOffsetInBits += 48; hdr->ethernet.etherType = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.h b/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.h index 74b1b65a628..0202b29fd86 100644 --- a/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.h +++ b/testdata/p4tc_samples_outputs/tc_may_override_example_09_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -62,3 +62,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/tc_type_annotation_example_parser.h b/testdata/p4tc_samples_outputs/tc_type_annotation_example_parser.h index 4206db4108f..96da5da55a0 100644 --- a/testdata/p4tc_samples_outputs/tc_type_annotation_example_parser.h +++ b/testdata/p4tc_samples_outputs/tc_type_annotation_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -61,3 +61,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} + diff --git a/testdata/p4tc_samples_outputs/test_ipv6_example_control_blocks.c b/testdata/p4tc_samples_outputs/test_ipv6_example_control_blocks.c index 3657aa7c4d4..7865e326465 100644 --- a/testdata/p4tc_samples_outputs/test_ipv6_example_control_blocks.c +++ b/testdata/p4tc_samples_outputs/test_ipv6_example_control_blocks.c @@ -66,7 +66,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, struct MainControlImpl_tbl_default_key key; __builtin_memset(&key, 0, sizeof(key)); key.keysz = 128; - __builtin_memcpy(&(key.field0[0]), &(hdr->ipv6.srcAddr[0]), 16); + __builtin_memcpy(&(key.field0), &(hdr->ipv6.srcAddr), 16); struct p4tc_table_entry_act_bpf *act_bpf; /* value */ struct MainControlImpl_tbl_default_value *value = NULL; @@ -200,7 +200,7 @@ static __always_inline int process(struct __sk_buff *skb, struct headers_t *hdr, write_partial(pkt + BYTES(ebpf_packetOffsetInBits) + 0 + 1, 4, 4, (ebpf_byte)); ebpf_packetOffsetInBits += 8; - hdr->ipv6.flowLabel = htonl(hdr->ipv6.flowLabel << 12); + storePrimitive32((u8 *)&hdr->ipv6.flowLabel, 20, (htonl(getPrimitive32(hdr->ipv6.flowLabel, 20) << 12))); ebpf_byte = ((char*)(&hdr->ipv6.flowLabel))[0]; write_partial(pkt + BYTES(ebpf_packetOffsetInBits) + 0, 4, 0, (ebpf_byte >> 4)); write_partial(pkt + BYTES(ebpf_packetOffsetInBits) + 0 + 1, 4, 4, (ebpf_byte)); diff --git a/testdata/p4tc_samples_outputs/test_ipv6_example_parser.c b/testdata/p4tc_samples_outputs/test_ipv6_example_parser.c index 86f16611576..dac5b145a1a 100644 --- a/testdata/p4tc_samples_outputs/test_ipv6_example_parser.c +++ b/testdata/p4tc_samples_outputs/test_ipv6_example_parser.c @@ -41,7 +41,7 @@ static __always_inline int run_parser(struct __sk_buff *skb, struct headers_t *h hdr->ipv6.trafficClass = (u8)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)) >> 4) & EBPF_MASK(u8, 8)); ebpf_packetOffsetInBits += 8; - hdr->ipv6.flowLabel = (u32)((load_word(pkt, BYTES(ebpf_packetOffsetInBits)) >> 8) & EBPF_MASK(u32, 20)); + storePrimitive32((u8 *)&hdr->ipv6.flowLabel, 20, (u32)((load_word(pkt, BYTES(ebpf_packetOffsetInBits)) >> 8) & EBPF_MASK(u32, 20))); ebpf_packetOffsetInBits += 20; hdr->ipv6.payloadLength = (u16)((load_half(pkt, BYTES(ebpf_packetOffsetInBits)))); diff --git a/testdata/p4tc_samples_outputs/test_ipv6_example_parser.h b/testdata/p4tc_samples_outputs/test_ipv6_example_parser.h index 8d20a6e6c77..86aaef3b82e 100644 --- a/testdata/p4tc_samples_outputs/test_ipv6_example_parser.h +++ b/testdata/p4tc_samples_outputs/test_ipv6_example_parser.h @@ -12,8 +12,8 @@ struct ethernet_t { - u64 dstAddr; /* EthernetAddress */ - u64 srcAddr; /* EthernetAddress */ + u8 dstAddr[6]; /* EthernetAddress */ + u8 srcAddr[6]; /* EthernetAddress */ u16 etherType; /* bit<16> */ u8 ebpf_valid; }; @@ -35,7 +35,7 @@ struct ipv4_t { struct ipv6_t { u8 version; /* bit<4> */ u8 trafficClass; /* bit<8> */ - u32 flowLabel; /* bit<20> */ + u8 flowLabel[3]; /* bit<20> */ u16 payloadLength; /* bit<16> */ u8 nextHeader; /* bit<8> */ u8 hopLimit; /* bit<8> */ @@ -72,3 +72,48 @@ REGISTER_TABLE(hdr_md_cpumap, BPF_MAP_TYPE_PERCPU_ARRAY, u32, struct hdr_md, 2) BPF_ANNOTATE_KV_PAIR(hdr_md_cpumap, u32, struct hdr_md) REGISTER_END() +static inline u32 getPrimitive32(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + return ((((u32)a[2]) <<16) | (((u32)a[1]) << 8) | a[0]); +} +static inline u64 getPrimitive64(u8 *a, int size) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + if(size <= 40) { + return ((((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + if(size <= 48) { + return ((((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } else { + return ((((u64)a[6]) << 48) | (((u64)a[5]) << 40) | (((u64)a[4]) << 32) | (((u64)a[3]) << 24) | (((u64)a[2]) << 16) | (((u64)a[1]) << 8) | a[0]); + } + } +} +static inline void storePrimitive32(u8 *a, int size, u32 value) { + if(size <=16 || size >=24) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); +} +static inline void storePrimitive64(u8 *a, int size, u64 value) { + if(size <=32 || size >=56) { + bpf_printk("Invalid size."); + }; + a[0] = (u8)(value); + a[1] = (u8)(value >> 8); + a[2] = (u8)(value >> 16); + a[3] = (u8)(value >> 24); + a[4] = (u8)(value >> 32); + if (size > 40) { + a[5] = (u8)(value >> 40); + } + if (size > 48) { + a[6] = (u8)(value >> 48); + } +} +