mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-06-22 14:44:30 +00:00
## Why this should be merged In lieu of modifications to `core.IntrinsicGas()`, required for support of Warp in SAE. If we were to introduce variadic `options.Option`s to `IntrinsicGas()`, it's impossible to guarantee that they would always be passed. The `types.Transaction` itself isn't actually passed to `IntrinsicGas()` either, so we can't rely on it to carry payloads/hooks. ## How this works `vm.Hooks` are extended to include `PreprocessingGasCharge(tx common.Hash) (uint64, error)`, which returns the amount of gas to be charged between `core.IntrinsicGas` and `vm.EVM` execution charges. The two entry points to execution, `vm.EVM.Call()` and `vm.EVM.Create()` are modified to first spend said charge before running upstream logic. The new hook is defined on a separate interface definition, embedded in `vm.Hooks`, to allow types to implement just that method and enforce it with the `var _ vm.Preprocessor = (*impl)(nil)` pattern. ## How this was tested Integration tests via both `core.ApplyTransaction()` and `core.ApplyMessage()`. The former is more comprehensive, while the latter allows for inspection of interim error values. --------- Signed-off-by: Arran Schlosberg <519948+ARR4N@users.noreply.github.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Austin Larson <78000745+alarso16@users.noreply.github.com>
109 lines
4 KiB
Go
109 lines
4 KiB
Go
// Copyright 2025 the libevm authors.
|
|
//
|
|
// The libevm additions to go-ethereum are free software: you can redistribute
|
|
// them and/or modify them under the terms of the GNU Lesser General Public License
|
|
// as published by the Free Software Foundation, either version 3 of the License,
|
|
// or (at your option) any later version.
|
|
//
|
|
// The libevm additions are distributed in the hope that they will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser
|
|
// General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see
|
|
// <http://www.gnu.org/licenses/>.
|
|
|
|
package vm
|
|
|
|
import (
|
|
"github.com/holiman/uint256"
|
|
|
|
"github.com/ava-labs/libevm/common"
|
|
"github.com/ava-labs/libevm/libevm"
|
|
"github.com/ava-labs/libevm/log"
|
|
)
|
|
|
|
// canCreateContract is a convenience wrapper for calling the
|
|
// [params.RulesHooks.CanCreateContract] hook.
|
|
func (evm *EVM) canCreateContract(caller ContractRef, contractToCreate common.Address, gas uint64) (remainingGas uint64, _ error) {
|
|
addrs := &libevm.AddressContext{
|
|
Origin: evm.Origin,
|
|
EVMSemantic: libevm.CallerAndSelf{
|
|
Caller: caller.Address(),
|
|
Self: contractToCreate,
|
|
},
|
|
// The "raw" caller isn't guaranteed to be known if the caller is a
|
|
// delegate so the `Raw` field is documented as always being nil.
|
|
}
|
|
gas, err := evm.chainRules.Hooks().CanCreateContract(addrs, gas, evm.StateDB)
|
|
|
|
// NOTE that this block only performs logging and that all paths propagate
|
|
// `(gas, err)` unmodified.
|
|
if err != nil {
|
|
log.Debug(
|
|
"Contract creation blocked by libevm hook",
|
|
"origin", addrs.Origin,
|
|
"caller", addrs.EVMSemantic.Caller,
|
|
"contract", addrs.EVMSemantic.Self,
|
|
"hooks", log.TypeOf(evm.chainRules.Hooks()),
|
|
"reason", err,
|
|
)
|
|
}
|
|
|
|
return gas, err
|
|
}
|
|
|
|
// Call executes the contract associated with the addr with the given input as
|
|
// parameters. It also handles any necessary value transfer required and takes
|
|
// the necessary steps to create accounts and reverses the state in case of an
|
|
// execution error or failed value transfer.
|
|
func (evm *EVM) Call(caller ContractRef, addr common.Address, input []byte, gas uint64, value *uint256.Int) (ret []byte, leftOverGas uint64, err error) {
|
|
gas, err = evm.spendPreprocessingGas(gas)
|
|
if err != nil {
|
|
return nil, gas, err
|
|
}
|
|
return evm.call(caller, addr, input, gas, value)
|
|
}
|
|
|
|
// create wraps the original geth method of the same name, now named
|
|
// [EVM.createCommon], first spending preprocessing gas.
|
|
func (evm *EVM) create(caller ContractRef, codeAndHash *codeAndHash, gas uint64, value *uint256.Int, address common.Address, typ OpCode) ([]byte, common.Address, uint64, error) {
|
|
gas, err := evm.spendPreprocessingGas(gas)
|
|
if err != nil {
|
|
return nil, common.Address{}, gas, err
|
|
}
|
|
return evm.createCommon(caller, codeAndHash, gas, value, address, typ)
|
|
}
|
|
|
|
func (evm *EVM) spendPreprocessingGas(gas uint64) (uint64, error) {
|
|
if internalCall := evm.depth > 0; internalCall || !libevmHooks.Registered() {
|
|
return gas, nil
|
|
}
|
|
c, err := libevmHooks.Get().PreprocessingGasCharge(evm.StateDB.TxHash())
|
|
if err != nil {
|
|
return gas, err
|
|
}
|
|
if c > gas {
|
|
return 0, ErrOutOfGas
|
|
}
|
|
return gas - c, nil
|
|
}
|
|
|
|
// InvalidateExecution sets the error that will be returned by
|
|
// [EVM.ExecutionInvalidated] for the length of the current transaction; i.e.
|
|
// until [EVM.Reset] is called. This is honoured by state-transition logic to
|
|
// render the execution itself void (as against reverted).
|
|
//
|
|
// This method MUST NOT be exposed in a manner that allows contracts to set
|
|
// the error; it MAY be exposed to precompiles.
|
|
func (evm *EVM) InvalidateExecution(err error) {
|
|
evm.executionInvalidated = err
|
|
}
|
|
|
|
// ExecutionInvalidated returns the last value passed to
|
|
// [EVM.InvalidateExecution] or nil if no such call has occurred or if
|
|
// [EVM.Reset] has been called.
|
|
func (evm *EVM) ExecutionInvalidated() error {
|
|
return evm.executionInvalidated
|
|
}
|