-
Notifications
You must be signed in to change notification settings - Fork 5.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Change the constant optimizer to make use of PUSH0
#14117
Conversation
b62a72e
to
2c1d785
Compare
libevmasm/ConstantOptimiser.cpp
Outdated
AssemblyItems static copyRoutine{ | ||
// back up memory | ||
// mload(0) | ||
Instruction::PUSH0, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could use u256(0)
in these places, but the explicit instruction felt better for readability and ensuring the actualCopyRoutine[3]
position is fixed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We probably should use u256(0)
in these places. The constructor of AssemblyItem
should technically really assert against the instruction being any push or forward to the Push
constructor.
Currently, the convention is "never use push as instruction in AssemblyItem
", which was kind of hard to violate since all pushes had push data - but now we have AssemblyItem(u256(0)) != AssemblyItem(Instruction::PUSH0)
- and should only ever use the former for consistency and for the other optimization steps to work properly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fair enough, changed it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
After I changed to u256(0)
a few test cases fail because codecopy does not seem to be selected. That suggest the cost is more than expected, i.e. u256(0)
generates larger code than push0
. Need to investigate.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@ekpyron I won't have time to debug this for the next 2 weeks. Feel free to take it over.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Alright - even though it'd be nicer to have it, we don't absolutely need to change this in the immediate release with Shanghai support anyways, so we'll just postpone it for now (we aim to release tomorrow or Wednesday).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If I understood this right, codecopy cost is more than expected and thus is not being selected because when computing the byte size, PUSH0
is not taken in account.
At AssemblyItem::bytesRequired
, the return value for any PUSHN
is at minimum 2
but for PUSH0
should be always 1
:
solidity/libevmasm/AssemblyItem.cpp
Lines 126 to 127 in 374a6fd
case Push: | |
return 1 + max<size_t>(1, numberEncodingSize(data())); |
libevmasm/ConstantOptimiser.cpp
Outdated
if (m_params.evmVersion.hasPush0()) | ||
{ | ||
// This costs ~29 gas. | ||
AssemblyItems static copyRoutine{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
When I had a bug in this version some tests failed, so the CodeCopyMethod
gets triggered, but someone should double check this before merging.
25ac63f
to
6c756d2
Compare
bacab83
to
ca1394e
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The fix does look like the right direction to go here. AssemblyItem:::bytesRequired
should generally be fixed to account for PUSH0
and we should make sure it gets a correct (or reasonable) EVM version passed in at all times.
libevmasm/ConstantOptimiser.cpp
Outdated
if (m_params.evmVersion.hasPush0()) | ||
actualCopyRoutine[3] = _assembly.newData(data); | ||
else | ||
actualCopyRoutine[4] = _assembly.newData(data); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It looks like it would be relatively easy for someone to change copyRoutine()
and either forget the number or use a wrong one. I'd add an assert that the item we're replacing is really PushData
just to be safe.
Actually, it looks like it would be even better to pass the number via an argument (with the u256(1) << 16
placeholder being the default) and avoid this brittle construct altogether. I wonder if there's any good reason it was done this way.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I added a parameter to copyRoutine
. Since AssemblyItem
is forward declared in ConstantOptimiser.h
, we need to use a pointer.
443f226
to
43ddf2d
Compare
3d2fb1f
to
d6792cf
Compare
9df26c1
to
0831f1b
Compare
0x00 | ||
dup1 | ||
revert |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So one thing is that this is no longer de-duplicated with tag_2
below for some reason (may be that the 0x00
is a PUSH0
in one case and a PUSH1 0x00
in the other?). Still a bit strange how the changes in this PR are causing this.
But another thing is: I'd have assumed that the libevmasm/CommonSubexpressionEliminator should actually be clever enough to transform 0x00 dup1 revert
to 0x00 0x00 revert
, if it gets the correct push0 pricing.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So, here's what happens in this case. The inliner
sees that the cost of inlining is less than the pushtag 2
, which was not the case before push0
. So the code of tag_2
gets inlined. After that, the peephole
optimizer finds that it can replace the code with revert dup1 push0
, because its method DoublePush
is not prepared to handle push0
. Another problem is that the peephole
optimizer was able to remove the pushtag 2 , jump
next to tag 2
, but it cannot do it now, because of the inlining.
Finally, the CSE is not able to transform dup1 push0
to push0 push0
. I guess it needs a rule for that in SimplificationRules
.
Not sure about what to do regarding the inliner
situation.
I will try to make the CSE and peephole
optimizer consider push0
, although for the latter that depends on getting the evm version in a static function of the optimizer (DoublePush::applySimple
).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, right, that explains things - I hadn't considered that inlining would kick in since assuming a tag size of two bytes PUSH0 PUSH0 REVERT
actually now is already the same size as PUSH<tag> JUMP
! I mean, letting the inlining happen is just fine then...
The peephole optimizer should definitely consider push0
- and it also looks like some deduplication still doesn't happen correctly (but in cases in which it also didn't happen before), so we don't necessarily need to fix it here directly.
c2d8b49
to
7178bce
Compare
0de5d5e
to
8cdc70f
Compare
I investigated the last 6 problematic cases of the hardhat tests and all cases are related to the introduction of the pragma solidity ^0.8.0;
contract C {
modifier m2(bool b) {
require(b);
_;
}
function test(bool b) m1(b) m2(b) public {
revert();
}
modifier m1(bool b) {
_;
}
} The stack trace expectation shows it expected the revert error raised from the {
"transactions": [
{
"file": "c.sol",
"contract": "C"
},
{
"to": 0,
"params": [false],
"function": "test",
"stackTrace": [
{
"type": "CALLSTACK_ENTRY",
"sourceReference": {
"contract": "C",
"file": "c.sol",
"function": "test",
"line": 10
}
},
{
"type": "REVERT_ERROR",
"sourceReference": {
"contract": "C",
"file": "c.sol",
"function": "m2",
"line": 6
}
}
]
}
]
} The code generated by the compiler before this PR contains the part related to that:
But the code generated with the introduction of this PR is optimized further and removes the asm related to the
The final result contains only the code corresponding to the
The |
1e39984
to
c5e2877
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Consider this comment a soft-approval. The logic looks sound now, all testing issues seem to be resolved. Missing is only changelog entries (for the constant optimiser taking into account push0 and for the peephole optimizer optimizing identical code snippets that terminate if they occur one after the other), and potentially a bit of commit cleanup.
7924c39
to
d7fb2a3
Compare
Gas cost benchmarks
|
project | bytecode_size | deployment_gas | method_gas |
---|---|---|---|
brink | +0% |
||
colony | +0.04% ❌ |
||
elementfi | +0% |
||
ens | +0% |
||
euler | |||
gnosis | |||
gp2 | -0.02% ✅ |
||
pool-together | 0% |
||
uniswap | 0% |
||
yield_liquidator | +0% |
+0% |
0% |
zeppelin |
ir-optimize-evm+yul
project | bytecode_size | deployment_gas | method_gas |
---|---|---|---|
brink | 0% |
||
colony | +0.02% ❌ |
||
elementfi | -0% |
||
ens | -0% |
0% |
-0% |
euler | -0% |
||
gnosis | |||
gp2 | -0.03% ✅ |
||
pool-together | +0% |
||
uniswap | -0% |
||
yield_liquidator | +0.27% ❌ |
+0.3% ❌ |
-0.01% ✅ |
zeppelin | -0.01% ✅ |
-0.01% ✅ |
+0.13% ❌ |
ir-optimize-evm-only
project | bytecode_size | deployment_gas | method_gas |
---|---|---|---|
brink | +0.01% ❌ |
||
colony | +0.01% ❌ |
||
elementfi | -0.01% ✅ |
||
ens | -0% |
+0.01% ❌ |
0% |
euler | |||
gnosis | |||
gp2 | +0.01% ❌ |
||
pool-together | +0.01% ❌ |
||
uniswap | -0.01% ✅ |
||
yield_liquidator | +0% |
+0% |
0% |
zeppelin | -0.01% ✅ |
legacy-no-optimize
project | bytecode_size | deployment_gas | method_gas |
---|---|---|---|
brink | +0.03% ❌ |
||
colony | +0.04% ❌ |
||
elementfi | +0.1% ❌ |
||
ens | +0.04% ❌ |
||
euler | +0.04% ❌ |
||
gnosis | +0.05% ❌ |
||
gp2 | -0% |
||
pool-together | +0.06% ❌ |
||
uniswap | +0.02% ❌ |
||
yield_liquidator | +0.04% ❌ |
+0.04% ❌ |
-0.01% ✅ |
zeppelin | +0.06% ❌ |
+3.36% ❌ |
+0.01% ❌ |
legacy-optimize-evm+yul
project | bytecode_size | deployment_gas | method_gas |
---|---|---|---|
brink | 0% |
||
colony | +0.02% ❌ |
||
elementfi | +0.01% ❌ |
||
ens | +0.02% ❌ |
+0% |
-0% |
euler | +0.02% ❌ |
||
gnosis | 0% |
||
gp2 | -0.06% ✅ |
||
pool-together | +0.01% ❌ |
||
uniswap | -0% |
||
yield_liquidator | 0% |
+0% |
-0% |
zeppelin | -0.02% ✅ |
-0% |
+0.07% ❌ |
legacy-optimize-evm-only
project | bytecode_size | deployment_gas | method_gas |
---|---|---|---|
brink | 0% |
||
colony | +0.01% ❌ |
||
elementfi | +0.01% ❌ |
||
ens | +0.02% ❌ |
-0% |
-0% |
euler | +0.04% ❌ |
||
gnosis | 0% |
||
gp2 | -0.05% ✅ |
||
pool-together | -0% |
||
uniswap | -0% |
||
yield_liquidator | 0% |
-0% |
-0% |
zeppelin | -0.01% ✅ |
-0% |
-0% |
!V
= version mismatch
!B
= no value in the "before" version
!A
= no value in the "after" version
!T
= one or both values were not numeric and could not be compared
-0
= very small negative value rounded to zero
+0
= very small positive value rounded to zero
This pull request is stale because it has been open for 14 days with no activity. |
d7fb2a3
to
0aa28ff
Compare
0aa28ff
to
7431bcb
Compare
7431bcb
to
a500a63
Compare
Part of #14073.
This change only affects large copies using codecopy (which are supposedly rare).
(It will be fun changing this code for EOF, i.e. use
DATACOPY
or replace it entirely with a singleDATALOADN
.)