Skip to content

Commit

Permalink
Pay attention to side effects in some optimizations; fixes p4lang#2287
Browse files Browse the repository at this point in the history
  • Loading branch information
Mihai Budiu committed Apr 6, 2020
1 parent e73006f commit c67fe1e
Show file tree
Hide file tree
Showing 10 changed files with 720 additions and 5 deletions.
18 changes: 13 additions & 5 deletions frontends/p4/strengthReduction.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -321,11 +321,18 @@ const IR::Node* DoStrengthReduction::postorder(IR::Slice* expr) {
while (auto cat = expr->e0->to<IR::Concat>()) {
unsigned rwidth = cat->right->type->width_bits();
if (expr->getL() >= rwidth) {
expr->e0 = cat->left;
expr->e1 = new IR::Constant(expr->getH() - rwidth);
expr->e2 = new IR::Constant(expr->getL() - rwidth);
if (!hasSideEffects(cat->right)) {
expr->e0 = cat->left;
expr->e1 = new IR::Constant(expr->getH() - rwidth);
expr->e2 = new IR::Constant(expr->getL() - rwidth);
} else {
break;
}
} else if (expr->getH() < rwidth) {
expr->e0 = cat->right;
if (!hasSideEffects(cat->left))
expr->e0 = cat->right;
else
break;
} else {
return new IR::Concat(expr->type,
new IR::Slice(cat->left, expr->getH() - rwidth, 0),
Expand All @@ -342,7 +349,8 @@ const IR::Node* DoStrengthReduction::postorder(IR::Slice* expr) {
}

auto slice_width = expr->getH() - expr->getL() + 1;
if (slice_width == (unsigned)expr->e0->type->width_bits())
if (slice_width == (unsigned)expr->e0->type->width_bits() &&
!hasSideEffects(expr->e1))
return expr->e0;

return expr;
Expand Down
95 changes: 95 additions & 0 deletions testdata/p4_16_samples/issue2287-bmv2.p4
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
#include <core.p4>
#include <v1model.p4>

header ethernet_t {
bit<48> dst_addr;
bit<48> src_addr;
bit<16> eth_type;
}

header H {
bit<8> a;
bit<8> b;
bit<8> c;
bit<8> d;
bit<8> e;
bit<8> f;
bit<8> g;
bit<8> h;
bit<8> i;
bit<8> j;
bit<8> k;
bit<8> l;
bit<8> m;
}
header B {
bit<8> a;
bit<8> b;
bit<8> c;
bit<8> d;
}



struct Headers {
ethernet_t eth_hdr;
H h;
B b;
}

struct Meta {
}

bit<8> function_with_side_effect(inout bit<8> val) {
val = 1;
return 8w2;
}

bool bool_with_side_effect(inout bit<8> val) {
val = 1;
return true;
}

parser p(packet_in pkt, out Headers hdr, inout Meta m, inout standard_metadata_t sm) {
state start {
pkt.extract(hdr.eth_hdr);
pkt.extract(hdr.h);
pkt.extract(hdr.b);
transition accept;
}
}

control ingress(inout Headers h, inout Meta m, inout standard_metadata_t sm) {
apply {
bit<8> dummy_var;
bool dummy_bool;
dummy_var = 8w0 & function_with_side_effect(h.h.a);
dummy_var = 8w0 * function_with_side_effect(h.h.b);
dummy_var = 8w0 / function_with_side_effect(h.h.c);
dummy_var = 8w0 >> function_with_side_effect(h.h.d);
dummy_var = 8w0 << function_with_side_effect(h.h.e);
dummy_var = 8w0 % function_with_side_effect(h.h.f);
dummy_var = 8w0 ^ function_with_side_effect(h.h.g);
dummy_var = 8w0 |-| function_with_side_effect(h.h.h);
dummy_var = 8w255 |+| function_with_side_effect(h.h.i);
dummy_var = 8w255 + function_with_side_effect(h.h.j);
dummy_var = 8w255 | function_with_side_effect(h.h.k);
dummy_var = 8w0 - function_with_side_effect(h.h.l);
dummy_var = (16w1 ++ function_with_side_effect(h.h.m))[15:8];

dummy_bool = true || bool_with_side_effect(h.b.a);
dummy_bool = false && bool_with_side_effect(h.b.b);
dummy_bool = function_with_side_effect(h.b.c) != function_with_side_effect(h.b.c);
dummy_bool = function_with_side_effect(h.b.d) == function_with_side_effect(h.b.d);
}
}

control vrfy(inout Headers h, inout Meta m) { apply {} }

control update(inout Headers h, inout Meta m) { apply {} }

control egress(inout Headers h, inout Meta m, inout standard_metadata_t sm) { apply {} }

control deparser(packet_out b, in Headers h) { apply {b.emit(h);} }

V1Switch(p(), vrfy(), ingress(), egress(), update(), deparser()) main;
2 changes: 2 additions & 0 deletions testdata/p4_16_samples/issue2287-bmv2.stf
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
packet 0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
expect 0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 01 01 01 01 01 01 01 01 01 01 01 01 01 00 00 01 01
105 changes: 105 additions & 0 deletions testdata/p4_16_samples_outputs/issue2287-bmv2-first.p4
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
#include <core.p4>
#include <v1model.p4>

header ethernet_t {
bit<48> dst_addr;
bit<48> src_addr;
bit<16> eth_type;
}

header H {
bit<8> a;
bit<8> b;
bit<8> c;
bit<8> d;
bit<8> e;
bit<8> f;
bit<8> g;
bit<8> h;
bit<8> i;
bit<8> j;
bit<8> k;
bit<8> l;
bit<8> m;
}

header B {
bit<8> a;
bit<8> b;
bit<8> c;
bit<8> d;
}

struct Headers {
ethernet_t eth_hdr;
H h;
B b;
}

struct Meta {
}

bit<8> function_with_side_effect(inout bit<8> val) {
val = 8w1;
return 8w2;
}
bool bool_with_side_effect(inout bit<8> val) {
val = 8w1;
return true;
}
parser p(packet_in pkt, out Headers hdr, inout Meta m, inout standard_metadata_t sm) {
state start {
pkt.extract<ethernet_t>(hdr.eth_hdr);
pkt.extract<H>(hdr.h);
pkt.extract<B>(hdr.b);
transition accept;
}
}

control ingress(inout Headers h, inout Meta m, inout standard_metadata_t sm) {
apply {
bit<8> dummy_var;
bool dummy_bool;
dummy_var = 8w0 & function_with_side_effect(h.h.a);
dummy_var = 8w0 * function_with_side_effect(h.h.b);
dummy_var = 8w0 / function_with_side_effect(h.h.c);
dummy_var = 8w0 >> function_with_side_effect(h.h.d);
dummy_var = 8w0 << function_with_side_effect(h.h.e);
dummy_var = 8w0 % function_with_side_effect(h.h.f);
dummy_var = function_with_side_effect(h.h.g);
dummy_var = 8w0 |-| function_with_side_effect(h.h.h);
dummy_var = 8w255 |+| function_with_side_effect(h.h.i);
dummy_var = 8w255 + function_with_side_effect(h.h.j);
dummy_var = 8w255 | function_with_side_effect(h.h.k);
dummy_var = -function_with_side_effect(h.h.l);
dummy_var = (16w1 ++ function_with_side_effect(h.h.m))[15:8];
dummy_bool = true;
dummy_bool = false;
dummy_bool = function_with_side_effect(h.b.c) != function_with_side_effect(h.b.c);
dummy_bool = function_with_side_effect(h.b.d) == function_with_side_effect(h.b.d);
}
}

control vrfy(inout Headers h, inout Meta m) {
apply {
}
}

control update(inout Headers h, inout Meta m) {
apply {
}
}

control egress(inout Headers h, inout Meta m, inout standard_metadata_t sm) {
apply {
}
}

control deparser(packet_out b, in Headers h) {
apply {
b.emit<Headers>(h);
}
}

V1Switch<Headers, Meta>(p(), vrfy(), ingress(), egress(), update(), deparser()) main;

Loading

0 comments on commit c67fe1e

Please sign in to comment.