|
8 | 8 | (["eq", 1, 2], [0]), |
9 | 9 | (["lt", 1, 2], [1]), |
10 | 10 | (["eq", "x", 0], ["iszero", "x"]), |
| 11 | + (["eq", ["sload", 0], 0], ["iszero", ["sload", 0]]), |
11 | 12 | # branch pruner |
12 | 13 | (["if", ["eq", 1, 2], "pass"], ["seq"]), |
13 | 14 | (["if", ["eq", 1, 1], 3, 4], [3]), |
|
22 | 23 | (["mstore", 0, ["eq", 1, 2]], ["mstore", 0, 0]), |
23 | 24 | # conditions |
24 | 25 | (["ge", "x", 0], [1]), # x >= 0 == True |
| 26 | + (["ge", ["sload", 0], 0], None), # no-op |
25 | 27 | (["iszero", ["gt", "x", 2 ** 256 - 1]], [1]), # x >= MAX_UINT256 == False |
26 | 28 | (["iszero", ["sgt", "x", 2 ** 255 - 1]], [1]), # signed x >= MAX_INT256 == False |
27 | 29 | (["le", "x", 0], ["iszero", "x"]), |
28 | 30 | (["le", 0, "x"], [1]), |
| 31 | + (["le", 0, ["sload", 0]], None), # no-op |
29 | 32 | (["lt", "x", 0], [0]), |
30 | 33 | (["lt", 0, "x"], ["iszero", ["iszero", "x"]]), |
31 | | - (["gt", 5, "x"], ["lt", "x", 5]), |
32 | | - (["ge", 5, "x"], ["le", "x", 5]), |
33 | | - (["lt", 5, "x"], ["gt", "x", 5]), |
34 | | - (["le", 5, "x"], ["ge", "x", 5]), |
35 | | - (["sgt", 5, "x"], ["slt", "x", 5]), |
36 | | - (["sge", 5, "x"], ["sle", "x", 5]), |
37 | | - (["slt", 5, "x"], ["sgt", "x", 5]), |
38 | | - (["sle", 5, "x"], ["sge", "x", 5]), |
| 34 | + (["gt", 5, "x"], None), |
| 35 | + (["ge", 5, "x"], None), |
| 36 | + (["lt", 5, "x"], None), |
| 37 | + (["le", 5, "x"], None), |
| 38 | + (["sgt", 5, "x"], None), |
| 39 | + (["sge", 5, "x"], None), |
| 40 | + (["slt", 5, "x"], None), |
| 41 | + (["sle", 5, "x"], None), |
39 | 42 | (["slt", "x", -(2 ** 255)], ["slt", "x", -(2 ** 255)]), # unimplemented |
40 | 43 | # tricky conditions |
41 | 44 | (["sgt", 2 ** 256 - 1, 0], [0]), # -1 > 0 |
|
55 | 58 | (["add", 0, "x"], ["x"]), |
56 | 59 | (["sub", "x", 0], ["x"]), |
57 | 60 | (["sub", "x", "x"], [0]), |
58 | | - (["sub", ["sload", 0], ["sload", 0]], ["sub", ["sload", 0], ["sload", 0]]), # no-op |
59 | | - (["sub", ["callvalue"], ["callvalue"]], ["sub", ["callvalue"], ["callvalue"]]), # no-op |
| 61 | + (["sub", ["sload", 0], ["sload", 0]], None), |
| 62 | + (["sub", ["callvalue"], ["callvalue"]], None), |
60 | 63 | (["mul", "x", 1], ["x"]), |
61 | 64 | (["div", "x", 1], ["x"]), |
62 | 65 | (["sdiv", "x", 1], ["x"]), |
63 | 66 | (["mod", "x", 1], [0]), |
| 67 | + (["mod", ["sload", 0], 1], None), |
64 | 68 | (["smod", "x", 1], [0]), |
65 | 69 | (["mul", "x", -1], ["sub", 0, "x"]), |
66 | 70 | (["sdiv", "x", -1], ["sub", 0, "x"]), |
67 | 71 | (["mul", "x", 0], [0]), |
| 72 | + (["mul", ["sload", 0], 0], None), |
68 | 73 | (["div", "x", 0], [0]), |
| 74 | + (["div", ["sload", 0], 0], None), |
69 | 75 | (["sdiv", "x", 0], [0]), |
| 76 | + (["sdiv", ["sload", 0], 0], None), |
70 | 77 | (["mod", "x", 0], [0]), |
| 78 | + (["mod", ["sload", 0], 0], None), |
71 | 79 | (["smod", "x", 0], [0]), |
72 | 80 | (["mul", "x", 32], ["shl", 5, "x"]), |
73 | 81 | (["div", "x", 64], ["shr", 6, "x"]), |
74 | 82 | (["mod", "x", 128], ["and", "x", 127]), |
75 | | - (["sdiv", "x", 64], ["sdiv", "x", 64]), # no-op |
76 | | - (["smod", "x", 64], ["smod", "x", 64]), # no-op |
| 83 | + (["sdiv", "x", 64], None), |
| 84 | + (["smod", "x", 64], None), |
77 | 85 | # bitwise ops |
78 | 86 | (["shr", 0, "x"], ["x"]), |
79 | 87 | (["sar", 0, "x"], ["x"]), |
80 | 88 | (["shl", 0, "x"], ["x"]), |
| 89 | + (["shr", 256, "x"], None), |
| 90 | + (["sar", 256, "x"], None), |
| 91 | + (["shl", 256, "x"], None), |
81 | 92 | (["and", 1, 2], [0]), |
82 | 93 | (["or", 1, 2], [3]), |
83 | 94 | (["xor", 1, 2], [3]), |
|
87 | 98 | (["or", "x", 0], ["x"]), |
88 | 99 | (["or", 0, "x"], ["x"]), |
89 | 100 | (["xor", "x", 0], ["x"]), |
90 | | - (["xor", "x", 1], ["xor", "x", 1]), # no-op |
91 | | - (["and", "x", 1], ["and", "x", 1]), # no-op |
92 | | - (["or", "x", 1], ["or", "x", 1]), # no-op |
| 101 | + (["xor", "x", 1], None), |
| 102 | + (["and", "x", 1], None), |
| 103 | + (["or", "x", 1], None), |
93 | 104 | (["xor", 0, "x"], ["x"]), |
94 | 105 | (["iszero", ["or", "x", 1]], [0]), |
95 | 106 | (["iszero", ["or", 2, "x"]], [0]), |
| 107 | + (["iszero", ["or", 1, ["sload", 0]]], None), |
96 | 108 | # nested optimizations |
97 | 109 | (["eq", 0, ["sub", 1, 1]], [1]), |
98 | 110 | (["eq", 0, ["add", 2 ** 255, 2 ** 255]], [1]), # test compile-time wrapping |
|
108 | 120 | def test_ir_optimizer(ir): |
109 | 121 | optimized = optimizer.optimize(IRnode.from_list(ir[0])) |
110 | 122 | optimized.repr_show_gas = True |
111 | | - hand_optimized = IRnode.from_list(ir[1]) |
112 | | - hand_optimized.repr_show_gas = True |
113 | | - assert optimized == hand_optimized |
| 123 | + if ir[1] is None: |
| 124 | + # no-op, assert optimizer does nothing |
| 125 | + expected = IRnode.from_list(ir[0]) |
| 126 | + else: |
| 127 | + expected = IRnode.from_list(ir[1]) |
| 128 | + expected.repr_show_gas = True |
| 129 | + assert optimized == expected |
114 | 130 |
|
115 | 131 |
|
116 | 132 | static_assertions_list = [ |
|
0 commit comments