mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-06-15 19:31:37 +00:00
cmd/geth, core/state, tests: rework EIP7610 check
This commit is contained in:
parent
77e7e5ad1a
commit
cb298098aa
11 changed files with 297 additions and 19 deletions
|
|
@ -18,15 +18,18 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"slices"
|
"slices"
|
||||||
|
"sort"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/cmd/utils"
|
"github.com/ethereum/go-ethereum/cmd/utils"
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/core"
|
||||||
"github.com/ethereum/go-ethereum/core/rawdb"
|
"github.com/ethereum/go-ethereum/core/rawdb"
|
||||||
"github.com/ethereum/go-ethereum/core/state"
|
"github.com/ethereum/go-ethereum/core/state"
|
||||||
"github.com/ethereum/go-ethereum/core/state/pruner"
|
"github.com/ethereum/go-ethereum/core/state/pruner"
|
||||||
|
|
@ -159,6 +162,22 @@ block is used.
|
||||||
Description: `
|
Description: `
|
||||||
The export-preimages command exports hash preimages to a flat file, in exactly
|
The export-preimages command exports hash preimages to a flat file, in exactly
|
||||||
the expected order for the overlay tree migration.
|
the expected order for the overlay tree migration.
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "list-eip-7610-accounts",
|
||||||
|
Aliases: []string{"eip7610"},
|
||||||
|
Usage: "list EIP7610 eligible accounts",
|
||||||
|
Action: listEIP7610EligibleAccounts,
|
||||||
|
Flags: slices.Concat(utils.NetworkFlags, utils.DatabaseFlags),
|
||||||
|
Description: `
|
||||||
|
geth snapshot list-eip-7610-accounts
|
||||||
|
traverses the post–EIP-161 state and returns all accounts that are eligible
|
||||||
|
under EIP-7610: accounts with zero nonce, empty runtime code, and non-empty
|
||||||
|
storage. The traversal will be aborted immediately if the state is prior to
|
||||||
|
EIP-161.
|
||||||
|
|
||||||
|
The exported accounts are identified by their address.
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
@ -690,3 +709,92 @@ func checkAccount(ctx *cli.Context) error {
|
||||||
log.Info("Checked the snapshot journalled storage", "time", common.PrettyDuration(time.Since(start)))
|
log.Info("Checked the snapshot journalled storage", "time", common.PrettyDuration(time.Since(start)))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// listEIP7610EligibleAccounts traverses the post–EIP-161 state and returns all
|
||||||
|
// accounts that are eligible under EIP-7610: accounts with zero nonce, empty
|
||||||
|
// runtime code, and non-empty storage.
|
||||||
|
//
|
||||||
|
// Such accounts could only have been created before EIP-161, since after that
|
||||||
|
// all newly created contracts are initialized with a nonce of one.
|
||||||
|
//
|
||||||
|
// This helper should be generally applicable to all networks, including the
|
||||||
|
// Ethereum mainnet. For most networks where EIP-161 was enabled from genesis,
|
||||||
|
// the resulting set is expected to be empty. Otherwise, network operators are
|
||||||
|
// responsible for generating the eligible account set themselves.
|
||||||
|
//
|
||||||
|
// Notably, the exported accounts are identified by their address.
|
||||||
|
func listEIP7610EligibleAccounts(ctx *cli.Context) error {
|
||||||
|
stack, _ := makeConfigNode(ctx)
|
||||||
|
defer stack.Close()
|
||||||
|
|
||||||
|
chaindb := utils.MakeChainDatabase(ctx, stack, true)
|
||||||
|
defer chaindb.Close()
|
||||||
|
|
||||||
|
headBlock := rawdb.ReadHeadBlock(chaindb)
|
||||||
|
if headBlock == nil {
|
||||||
|
log.Error("Failed to load head block")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
config, _, err := core.LoadChainConfig(chaindb, utils.MakeGenesis(ctx))
|
||||||
|
if err != nil {
|
||||||
|
log.Error("Failed to load chain config", "err", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !config.IsEIP158(headBlock.Number()) {
|
||||||
|
log.Info("Local head is prior to EIP-161", "head", headBlock.Number(), "eip-161", *config.EIP158Block)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
triedb := utils.MakeTrieDatabase(ctx, stack, chaindb, false, true, false)
|
||||||
|
defer triedb.Close()
|
||||||
|
|
||||||
|
if triedb.Scheme() != rawdb.PathScheme {
|
||||||
|
log.Error("Hash scheme is not supported")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
iter, err := triedb.AccountIterator(headBlock.Root(), common.Hash{})
|
||||||
|
if err != nil {
|
||||||
|
log.Error("Failed to get account iterator", "err", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
var (
|
||||||
|
start = time.Now()
|
||||||
|
accounts []common.Address
|
||||||
|
)
|
||||||
|
for iter.Next() {
|
||||||
|
blob := iter.Account()
|
||||||
|
if blob == nil {
|
||||||
|
log.Error("Failed to get account blob")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
var account types.SlimAccount
|
||||||
|
if err := rlp.DecodeBytes(blob, &account); err != nil {
|
||||||
|
log.Error("Failed to decode", "err", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// EIP-7610 account eligibility:
|
||||||
|
// - account.nonce == 0
|
||||||
|
// - account.runtime_code == empty
|
||||||
|
// - account.storage != empty
|
||||||
|
if len(account.CodeHash) == 0 && account.Nonce == 0 && len(account.Root) != 0 {
|
||||||
|
preimage := rawdb.ReadPreimage(chaindb, iter.Hash())
|
||||||
|
if preimage == nil {
|
||||||
|
log.Error("Failed to read preimage", "hash", iter.Hash().Hex())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
accounts = append(accounts, common.BytesToAddress(preimage))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(accounts) == 0 {
|
||||||
|
log.Info("Traversed state", "eligible", len(accounts), "elapsed", common.PrettyDuration(time.Since(start)))
|
||||||
|
} else {
|
||||||
|
sort.Slice(accounts, func(i, j int) bool {
|
||||||
|
return accounts[i].Cmp(accounts[j]) < 0
|
||||||
|
})
|
||||||
|
buf := make([]byte, len(accounts)*common.AddressLength)
|
||||||
|
for i, h := range accounts {
|
||||||
|
copy(buf[i*common.AddressLength:], h[:])
|
||||||
|
}
|
||||||
|
log.Info("Traversed state", "eligible", len(accounts), "elapsed", common.PrettyDuration(time.Since(start)), "output", hex.EncodeToString(buf))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -341,6 +341,9 @@ func (s *StateDB) GetNonce(addr common.Address) uint64 {
|
||||||
|
|
||||||
// GetStorageRoot retrieves the storage root from the given address or empty
|
// GetStorageRoot retrieves the storage root from the given address or empty
|
||||||
// if object not found.
|
// if object not found.
|
||||||
|
//
|
||||||
|
// Note: the storage root returned corresponds to the trie since last Intermediate
|
||||||
|
// operation, some recent in-memory changes are excluded.
|
||||||
func (s *StateDB) GetStorageRoot(addr common.Address) common.Hash {
|
func (s *StateDB) GetStorageRoot(addr common.Address) common.Hash {
|
||||||
stateObject := s.getStateObject(addr)
|
stateObject := s.getStateObject(addr)
|
||||||
if stateObject != nil {
|
if stateObject != nil {
|
||||||
|
|
|
||||||
|
|
@ -98,10 +98,6 @@ func (s *hookedStateDB) GetState(addr common.Address, hash common.Hash) common.H
|
||||||
return s.inner.GetState(addr, hash)
|
return s.inner.GetState(addr, hash)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *hookedStateDB) GetStorageRoot(addr common.Address) common.Hash {
|
|
||||||
return s.inner.GetStorageRoot(addr)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *hookedStateDB) GetTransientState(addr common.Address, key common.Hash) common.Hash {
|
func (s *hookedStateDB) GetTransientState(addr common.Address, key common.Hash) common.Hash {
|
||||||
return s.inner.GetTransientState(addr, key)
|
return s.inner.GetTransientState(addr, key)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -247,16 +247,16 @@ func TestCopyWithDirtyJournal(t *testing.T) {
|
||||||
|
|
||||||
orig.Finalise(true)
|
orig.Finalise(true)
|
||||||
for i := byte(0); i < 255; i++ {
|
for i := byte(0); i < 255; i++ {
|
||||||
root := orig.GetStorageRoot(common.BytesToAddress([]byte{i}))
|
balance := orig.GetBalance(common.BytesToAddress([]byte{i}))
|
||||||
if root != (common.Hash{}) {
|
if !balance.IsZero() {
|
||||||
t.Errorf("Unexpected storage root %x", root)
|
t.Errorf("Unexpected balance %x", root)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
cpy.Finalise(true)
|
cpy.Finalise(true)
|
||||||
for i := byte(0); i < 255; i++ {
|
for i := byte(0); i < 255; i++ {
|
||||||
root := cpy.GetStorageRoot(common.BytesToAddress([]byte{i}))
|
balance := cpy.GetBalance(common.BytesToAddress([]byte{i}))
|
||||||
if root != (common.Hash{}) {
|
if !balance.IsZero() {
|
||||||
t.Errorf("Unexpected storage root %x", root)
|
t.Errorf("Unexpected balance %x", root)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if cpy.IntermediateRoot(true) != orig.IntermediateRoot(true) {
|
if cpy.IntermediateRoot(true) != orig.IntermediateRoot(true) {
|
||||||
|
|
@ -394,9 +394,7 @@ func newTestAction(addr common.Address, r *rand.Rand) testAction {
|
||||||
}
|
}
|
||||||
contractHash := s.GetCodeHash(addr)
|
contractHash := s.GetCodeHash(addr)
|
||||||
emptyCode := contractHash == (common.Hash{}) || contractHash == types.EmptyCodeHash
|
emptyCode := contractHash == (common.Hash{}) || contractHash == types.EmptyCodeHash
|
||||||
storageRoot := s.GetStorageRoot(addr)
|
if s.GetNonce(addr) == 0 && emptyCode {
|
||||||
emptyStorage := storageRoot == (common.Hash{}) || storageRoot == types.EmptyRootHash
|
|
||||||
if s.GetNonce(addr) == 0 && emptyCode && emptyStorage {
|
|
||||||
s.CreateContract(addr)
|
s.CreateContract(addr)
|
||||||
// We also set some code here, to prevent the
|
// We also set some code here, to prevent the
|
||||||
// CreateContract action from being performed twice in a row,
|
// CreateContract action from being performed twice in a row,
|
||||||
|
|
|
||||||
|
|
@ -86,18 +86,17 @@ func TestVerklePrefetcher(t *testing.T) {
|
||||||
root, _ := state.Commit(0, true, false)
|
root, _ := state.Commit(0, true, false)
|
||||||
|
|
||||||
state, _ = New(root, sdb)
|
state, _ = New(root, sdb)
|
||||||
sRoot := state.GetStorageRoot(addr)
|
|
||||||
fetcher := newTriePrefetcher(sdb, root, "", false)
|
fetcher := newTriePrefetcher(sdb, root, "", false)
|
||||||
|
|
||||||
// Read account
|
// Read account
|
||||||
fetcher.prefetch(common.Hash{}, root, common.Address{}, []common.Address{addr}, nil, false)
|
fetcher.prefetch(common.Hash{}, root, common.Address{}, []common.Address{addr}, nil, false)
|
||||||
|
|
||||||
// Read storage slot
|
// Read storage slot
|
||||||
fetcher.prefetch(crypto.Keccak256Hash(addr.Bytes()), sRoot, addr, nil, []common.Hash{skey}, false)
|
fetcher.prefetch(crypto.Keccak256Hash(addr.Bytes()), common.Hash{}, addr, nil, []common.Hash{skey}, false)
|
||||||
|
|
||||||
fetcher.terminate(false)
|
fetcher.terminate(false)
|
||||||
accountTrie := fetcher.trie(common.Hash{}, root)
|
accountTrie := fetcher.trie(common.Hash{}, root)
|
||||||
storageTrie := fetcher.trie(crypto.Keccak256Hash(addr.Bytes()), sRoot)
|
storageTrie := fetcher.trie(crypto.Keccak256Hash(addr.Bytes()), common.Hash{})
|
||||||
|
|
||||||
rootA := accountTrie.Hash()
|
rootA := accountTrie.Hash()
|
||||||
rootB := storageTrie.Hash()
|
rootB := storageTrie.Hash()
|
||||||
|
|
|
||||||
92
core/vm/eip7610.go
Normal file
92
core/vm/eip7610.go
Normal file
|
|
@ -0,0 +1,92 @@
|
||||||
|
// Copyright 2026 The go-ethereum Authors
|
||||||
|
// This file is part of the go-ethereum library.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||||
|
// it 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 go-ethereum library is distributed in the hope that it 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 (
|
||||||
|
"encoding/hex"
|
||||||
|
"fmt"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/params"
|
||||||
|
)
|
||||||
|
|
||||||
|
func decodeEIP7610AccountSet(str string) map[common.Address]struct{} {
|
||||||
|
if str == "" {
|
||||||
|
return make(map[common.Address]struct{})
|
||||||
|
}
|
||||||
|
b, err := hex.DecodeString(str)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
if len(b)%common.AddressLength != 0 {
|
||||||
|
panic(fmt.Sprintf("invalid length, %d", len(b)))
|
||||||
|
}
|
||||||
|
addresses := make(map[common.Address]struct{}, len(b)/common.AddressLength)
|
||||||
|
for i := 0; i < len(b)/common.AddressLength; i++ {
|
||||||
|
addresses[common.BytesToAddress(b[i*common.AddressLength:(i+1)*common.AddressLength])] = struct{}{}
|
||||||
|
}
|
||||||
|
return addresses
|
||||||
|
}
|
||||||
|
|
||||||
|
const mainnetEIP7610Data = "02820e4bee488c40f7455fdca53125565148708f14725085d004f1b10ee07234a4ab28c5ad2a7b9e19272418753b90d9a3e3efc8430b1612c55fcb3a2c081ed1949d7dd9447f9d96e509befe576d44613311c08066580cb906a7287b6786e504c2ebd09f361d7a60b43587c7f6bba4f9fd9642747f65210a40490c9c468622d5c89646d6f3097f8eaf80c4114d149eb99bdeefc1f858f8fd22289c6beae99f2c5071cb62aa170b7f66b26cae8004d90e6078bb1e50b1497068bae652df3562eb8ea7677ff84477fa5983c6ac846dcf85fbbc4303f43eb91c379f79ae59ec0410867828e3b8c23dd8a29d9796ef523b175cc182fabfb81a056b6080d4200bc5150673d06f6f156dbf8ed30e53f7c9df73144e69f65cbb7e947d6ae067de8d44ae1a08750e7d626d61a623c44a8398ff6c618e9515468c1c4b198d53666cbe8462a21b22389bfc1cd6bc7ba19a4fc96adc3d0fe074add92e0650457c5db0c4c08cbf7ca580175d33d2ae3703584494ade958ad27ec2d289b7a67c19e90b619f45637c39ca49a41ac64c11637a0a194455ed8253352f6044cfe55bcc0748c3fa37b7df81f98db7c577b93baeb56dab50af4d6f86f99a06b96a2de425ad4b8d2d9e0e12f65cbcd6d55f447b44083e62dc49c92fa799033644d2a9afd7e3babe5a80af468bcbc4a0bfdb06336e773382c5202e674db71f4a835ec1364809003de3925685f24cd360bdffefc4465f84b29a1f8794dc753f41bef1f4b025ed2fee7707fa4b8c0a923a0e40399db3e7ce26069c6"
|
||||||
|
const sepoliaEIP7610Data = ""
|
||||||
|
const holeskyEIP7610Data = ""
|
||||||
|
const hoodiEIP7610Data = ""
|
||||||
|
|
||||||
|
var (
|
||||||
|
mainnetEIP7610Accounts = decodeEIP7610AccountSet(mainnetEIP7610Data)
|
||||||
|
sepoliaEIP7610Accounts = decodeEIP7610AccountSet(sepoliaEIP7610Data)
|
||||||
|
holeskyEIP7610Accounts = decodeEIP7610AccountSet(hoodiEIP7610Data)
|
||||||
|
hoodiEIP7610Accounts = decodeEIP7610AccountSet(holeskyEIP7610Data)
|
||||||
|
)
|
||||||
|
|
||||||
|
// isEIP7610RejectedAccount reports whether the account identified by the
|
||||||
|
// address is eligible for contract deployment rejection due to having
|
||||||
|
// non-empty storage.
|
||||||
|
//
|
||||||
|
// Note that, historically, there has been no case where a contract deployment
|
||||||
|
// targets an already existing account in Ethereum. This situation would only
|
||||||
|
// occur in the event of an address collision, which is extremely unlikely.
|
||||||
|
//
|
||||||
|
// This check is skipped for blocks prior to EIP-158, serving as a safeguard
|
||||||
|
// against potential address collisions in the future.
|
||||||
|
func isEIP7610RejectedAccount(chainID *big.Int, addr common.Address, isEIP158 bool) bool {
|
||||||
|
// Short circuit for blocks prior to EIP-158.
|
||||||
|
if !isEIP158 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
var accountSet map[common.Address]struct{}
|
||||||
|
switch chainID {
|
||||||
|
case params.MainnetChainConfig.ChainID:
|
||||||
|
accountSet = mainnetEIP7610Accounts
|
||||||
|
case params.SepoliaChainConfig.ChainID:
|
||||||
|
accountSet = sepoliaEIP7610Accounts
|
||||||
|
case params.HoleskyChainConfig.ChainID:
|
||||||
|
accountSet = holeskyEIP7610Accounts
|
||||||
|
case params.HoodiChainConfig.ChainID:
|
||||||
|
accountSet = hoodiEIP7610Accounts
|
||||||
|
default:
|
||||||
|
// The network is unknown, so the account set must be provided by the
|
||||||
|
// network operators themselves. Notably, only a small number of
|
||||||
|
// networks enabled EIP-158 after genesis; for all others, this set
|
||||||
|
// will always be empty.
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
_, exist := accountSet[addr]
|
||||||
|
return exist
|
||||||
|
}
|
||||||
61
core/vm/eip7610_test.go
Normal file
61
core/vm/eip7610_test.go
Normal file
|
|
@ -0,0 +1,61 @@
|
||||||
|
// Copyright 2026 The go-ethereum Authors
|
||||||
|
// This file is part of the go-ethereum library.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||||
|
// it 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 go-ethereum library is distributed in the hope that it 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 (
|
||||||
|
"fmt"
|
||||||
|
"maps"
|
||||||
|
"slices"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Example_mainnetEIP7610Accounts() {
|
||||||
|
list := slices.SortedFunc(maps.Keys(mainnetEIP7610Accounts), common.Address.Cmp)
|
||||||
|
for _, addr := range list {
|
||||||
|
fmt.Println(addr.Hex())
|
||||||
|
}
|
||||||
|
// Output:
|
||||||
|
// 0x02820E4bEE488C40f7455fDCa53125565148708F
|
||||||
|
// 0x14725085d004f1b10Ee07234A4ab28c5Ad2a7b9E
|
||||||
|
// 0x19272418753B90D9a3E3Efc8430b1612c55fcB3A
|
||||||
|
// 0x2c081Ed1949D7Dd9447F9d96e509befE576D4461
|
||||||
|
// 0x3311c08066580cb906a7287b6786E504C2EBD09f
|
||||||
|
// 0x361d7a60b43587c7f6bbA4f9fD9642747F65210A
|
||||||
|
// 0x40490C9c468622d5c89646D6F3097F8Eaf80c411
|
||||||
|
// 0x4d149EB99BDEEFC1f858f8fd22289C6beAE99f2c
|
||||||
|
// 0x5071cb62aA170b7f66b26cae8004d90E6078Bb1E
|
||||||
|
// 0x50b1497068bAE652Df3562EB8Ea7677ff84477FA
|
||||||
|
// 0x5983C6aC846DcF85fbBC4303F43eb91C379F79ae
|
||||||
|
// 0x59EC0410867828E3b8c23Dd8A29d9796ef523b17
|
||||||
|
// 0x5cC182faBFb81A056B6080d4200BC5150673D06f
|
||||||
|
// 0x6f156dbf8Ed30e53F7C9Df73144E69f65cBB7E94
|
||||||
|
// 0x7D6ae067De8d44Ae1A08750e7D626D61A623C44A
|
||||||
|
// 0x8398fF6c618e9515468c1c4b198d53666CBe8462
|
||||||
|
// 0xA21B22389bfC1cd6Bc7BA19A4Fc96aDC3D0FE074
|
||||||
|
// 0xaDD92e0650457C5Db0c4c08cbf7cA580175d33d2
|
||||||
|
// 0xAE3703584494Ade958AD27EC2d289b7a67c19E90
|
||||||
|
// 0xb619f45637C39Ca49A41ac64c11637A0A194455E
|
||||||
|
// 0xD8253352f6044cFE55bcC0748C3FA37b7dF81F98
|
||||||
|
// 0xDB7C577B93Baeb56dAB50aF4D6f86F99A06B96a2
|
||||||
|
// 0xdE425ad4B8d2d9E0E12F65CBcD6D55F447B44083
|
||||||
|
// 0xe62dc49C92fA799033644d2A9aFD7e3BAbE5A80a
|
||||||
|
// 0xF468BcBC4a0BFDB06336E773382C5202E674db71
|
||||||
|
// 0xF4a835ec1364809003dE3925685F24cD360bdffe
|
||||||
|
// 0xFc4465F84B29a1F8794Dc753F41BeF1F4b025ED2
|
||||||
|
// 0xfeE7707fa4b8C0A923A0E40399Db3e7Ce26069C6
|
||||||
|
}
|
||||||
|
|
@ -531,10 +531,9 @@ func (evm *EVM) create(caller common.Address, code []byte, gas uint64, value *ui
|
||||||
// - the code is non-empty
|
// - the code is non-empty
|
||||||
// - the storage is non-empty
|
// - the storage is non-empty
|
||||||
contractHash := evm.StateDB.GetCodeHash(address)
|
contractHash := evm.StateDB.GetCodeHash(address)
|
||||||
storageRoot := evm.StateDB.GetStorageRoot(address)
|
|
||||||
if evm.StateDB.GetNonce(address) != 0 ||
|
if evm.StateDB.GetNonce(address) != 0 ||
|
||||||
(contractHash != (common.Hash{}) && contractHash != types.EmptyCodeHash) || // non-empty code
|
(contractHash != (common.Hash{}) && contractHash != types.EmptyCodeHash) || // non-empty code
|
||||||
(storageRoot != (common.Hash{}) && storageRoot != types.EmptyRootHash) { // non-empty storage
|
isEIP7610RejectedAccount(evm.ChainConfig().ChainID, address, evm.chainRules.IsEIP158) {
|
||||||
if evm.Config.Tracer != nil && evm.Config.Tracer.OnGasChange != nil {
|
if evm.Config.Tracer != nil && evm.Config.Tracer.OnGasChange != nil {
|
||||||
evm.Config.Tracer.OnGasChange(gas, 0, tracing.GasChangeCallFailedExecution)
|
evm.Config.Tracer.OnGasChange(gas, 0, tracing.GasChangeCallFailedExecution)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -52,7 +52,6 @@ type StateDB interface {
|
||||||
GetStateAndCommittedState(common.Address, common.Hash) (common.Hash, common.Hash)
|
GetStateAndCommittedState(common.Address, common.Hash) (common.Hash, common.Hash)
|
||||||
GetState(common.Address, common.Hash) common.Hash
|
GetState(common.Address, common.Hash) common.Hash
|
||||||
SetState(common.Address, common.Hash, common.Hash) common.Hash
|
SetState(common.Address, common.Hash, common.Hash) common.Hash
|
||||||
GetStorageRoot(addr common.Address) common.Hash
|
|
||||||
|
|
||||||
GetTransientState(addr common.Address, key common.Hash) common.Hash
|
GetTransientState(addr common.Address, key common.Hash) common.Hash
|
||||||
SetTransientState(addr common.Address, key, value common.Hash)
|
SetTransientState(addr common.Address, key, value common.Hash)
|
||||||
|
|
|
||||||
|
|
@ -66,6 +66,12 @@ func TestBlockchain(t *testing.T) {
|
||||||
// This directory contains no test.
|
// This directory contains no test.
|
||||||
bt.skipLoad(`.*\.meta/.*`)
|
bt.skipLoad(`.*\.meta/.*`)
|
||||||
|
|
||||||
|
// Broken tests
|
||||||
|
bt.skipLoad(`RevertInCreateInInit`)
|
||||||
|
bt.skipLoad(`InitCollisionParis`)
|
||||||
|
bt.skipLoad(`dynamicAccountOverwriteEmpty_Paris`)
|
||||||
|
bt.skipLoad(`create2collisionStorageParis`)
|
||||||
|
|
||||||
bt.walk(t, blockTestDir, func(t *testing.T, name string, test *BlockTest) {
|
bt.walk(t, blockTestDir, func(t *testing.T, name string, test *BlockTest) {
|
||||||
execBlockTest(t, bt, test)
|
execBlockTest(t, bt, test)
|
||||||
})
|
})
|
||||||
|
|
@ -85,6 +91,12 @@ func TestExecutionSpecBlocktests(t *testing.T) {
|
||||||
bt.skipLoad(".*prague/eip7251_consolidations/test_system_contract_deployment.json")
|
bt.skipLoad(".*prague/eip7251_consolidations/test_system_contract_deployment.json")
|
||||||
bt.skipLoad(".*prague/eip7002_el_triggerable_withdrawals/test_system_contract_deployment.json")
|
bt.skipLoad(".*prague/eip7002_el_triggerable_withdrawals/test_system_contract_deployment.json")
|
||||||
|
|
||||||
|
// Broken tests
|
||||||
|
bt.skipLoad(`RevertInCreateInInit`)
|
||||||
|
bt.skipLoad(`InitCollisionParis`)
|
||||||
|
bt.skipLoad(`dynamicAccountOverwriteEmpty_Paris`)
|
||||||
|
bt.skipLoad(`create2collisionStorageParis`)
|
||||||
|
|
||||||
bt.walk(t, executionSpecBlockchainTestDir, func(t *testing.T, name string, test *BlockTest) {
|
bt.walk(t, executionSpecBlockchainTestDir, func(t *testing.T, name string, test *BlockTest) {
|
||||||
execBlockTest(t, bt, test)
|
execBlockTest(t, bt, test)
|
||||||
})
|
})
|
||||||
|
|
|
||||||
|
|
@ -57,6 +57,11 @@ func initMatcher(st *testMatcher) {
|
||||||
// Broken tests:
|
// Broken tests:
|
||||||
// EOF is not part of cancun
|
// EOF is not part of cancun
|
||||||
st.skipLoad(`^stEOF/`)
|
st.skipLoad(`^stEOF/`)
|
||||||
|
|
||||||
|
st.skipLoad(`RevertInCreateInInit`)
|
||||||
|
st.skipLoad(`InitCollisionParis`)
|
||||||
|
st.skipLoad(`dynamicAccountOverwriteEmpty_Paris`)
|
||||||
|
st.skipLoad(`create2collisionStorageParis`)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestState(t *testing.T) {
|
func TestState(t *testing.T) {
|
||||||
|
|
@ -92,6 +97,12 @@ func TestExecutionSpecState(t *testing.T) {
|
||||||
}
|
}
|
||||||
st := new(testMatcher)
|
st := new(testMatcher)
|
||||||
|
|
||||||
|
// Broken tests
|
||||||
|
st.skipLoad(`RevertInCreateInInit`)
|
||||||
|
st.skipLoad(`InitCollisionParis`)
|
||||||
|
st.skipLoad(`dynamicAccountOverwriteEmpty_Paris`)
|
||||||
|
st.skipLoad(`create2collisionStorageParis`)
|
||||||
|
|
||||||
st.walk(t, executionSpecStateTestDir, func(t *testing.T, name string, test *StateTest) {
|
st.walk(t, executionSpecStateTestDir, func(t *testing.T, name string, test *StateTest) {
|
||||||
execStateTest(t, st, test)
|
execStateTest(t, st, test)
|
||||||
})
|
})
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue