-
Notifications
You must be signed in to change notification settings - Fork 871
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
Do not return gas cost along with ExceptionalHaltReason in OperationResult unless OOG #7919
base: main
Are you sure you want to change the base?
Do not return gas cost along with ExceptionalHaltReason in OperationResult unless OOG #7919
Conversation
2f5956f
to
073356f
Compare
Signed-off-by: Luis Pinto <[email protected]>
…t in halt cases Signed-off-by: Luis Pinto <[email protected]>
allow for gasCost reporting in the case of InsufficientGas Signed-off-by: Luis Pinto <[email protected]>
073356f
to
c62c68d
Compare
evm/src/main/java/org/hyperledger/besu/evm/operation/BalanceOperation.java
Show resolved
Hide resolved
evm/src/main/java/org/hyperledger/besu/evm/operation/ExtCodeHashOperation.java
Show resolved
Hide resolved
* @return halted OperationResult | ||
*/ | ||
public static OperationResult underFlow() { | ||
return new OperationResult(ExceptionalHaltReason.INSUFFICIENT_STACK_ITEMS); |
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.
nit : why we need to recreate a new one everytime ?
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.
There was some inconsistency between cached and non cached instances of OperationResult
with halt. I decided to not cache them because a halt is a terminating event for the current EVM execution context and caching them also takes memory. I think with not caching we will be better of as I don't think it will impact the EVM considerably.
if (result.getHaltReason() != null) { | ||
return result; | ||
} | ||
return new OperationResult(result.getGasCost(), 0); |
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.
why we need to create a new one in this case and not returning result ?
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 think it was like that before because we don't want to increment the ProgramCounter so it's fixed to 0
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 just for confirmation it's not a problem to not fix it in case of failure as before ?
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 I got you now! For halt cases you don't need PC because you essentially exit the EVM code execution so it doesn't matter what value you use.
This pr is stale because it has been open for 30 days with no activity. |
PR description
Rationale
While working on something unrelated I discovered that EVM halt events also require to pass in a gas cost. The meaning of this gas cost is not totally clear across all halt event types (insufficient gas, stack underflow/overflow, invalid jump destination, illegal state change, ...), so this inconsistency makes it harder for devs building new EVM features, see example on code from initial verkle gas schedule efforts. In the Ethereum Yellow paper it is mentioned that at the start of a transaction an upfront gas cost is taken captive and, then, if code is executed correctly the remaining unused gas is refunded to the account. In the event of an EVM error, halting the code execution, no gas is refunded and so the entire upfront gas made captive is consumed. Therefore, clients do not need any cost information upon halting.
Tracer cases
There is the case of tracers that have access to
OperationResult
when callingtracePostExecution
. IMO, the only case where returning a gas cost along with the halt condition might make sense is the insufficient gas case. This seems to show exactly in which amount of gas charging did the EVM halted on. It does not make any sense at all to report a gas cost when the reason for the halt was not related to gas. In fact, I would question if it's really useful to report the gas cost at all even in the insufficient gas case because this is not specific enough (was it the static gas or a dynamic part? which type of gas exactly?). Currently the default in clients is to compound all gas costs for an opcode execution in advance of any DB lookups or computation. But one could incrementally compute the cost before any computation and check for out of gas incrementally also. I believe this will happen with stateless clients. This is also used by tracing APIs (in particular trace_* calls) but the current spec or future spec with EOF do not mention any standards around EVM halting cases. Right now I'm told that Nethermind and Reth implement these but I haven't checked how they handle this case in particular. A quick scan on Reth's website didn't show many details on this edge case either.Changes
This PR refactors and streamlines the
OperationResult
, the outcome of executing an opcode, to split up between halting and successful results. It also enforces the creation of new haltOperationResults
through static factory methods for better encapsulation, that always create a new instance since, on a halt event, the execution context is not recoverable so the overhead is negligible.I'm proposing that we get rid of this gas cost reporting in
OperationResult
when anExceptionalHaltReason
is used, to make it all consistent. I changed some tests to cope with this modification so please review carefully. On the other hand, if someone feels strongly we could very well keep reporting gas cost upon insufficient gas.EDIT: Latest tip allows for gasCost reporting for insufficient gas only.
Thanks for sending a pull request! Have you done the following?
doc-change-required
label to this PR if updates are required.Locally, you can run these tests to catch failures early:
./gradlew build
./gradlew acceptanceTest
./gradlew integrationTest
./gradlew ethereum:referenceTests:referenceTests