2017-06-27 08:57:06 -05:00
|
|
|
// Copyright 2017 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 state
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2022-11-14 08:41:56 -06:00
|
|
|
"github.com/ethereum/go-ethereum/common/lru"
|
2020-08-21 07:10:40 -05:00
|
|
|
"github.com/ethereum/go-ethereum/core/rawdb"
|
2024-09-05 05:10:47 -05:00
|
|
|
"github.com/ethereum/go-ethereum/core/state/snapshot"
|
2021-09-28 03:48:07 -05:00
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
core/state, light, les: make signature of ContractCode hash-independent (#27209)
* core/state, light, les: make signature of ContractCode hash-independent
* push current state for feedback
* les: fix unit test
* core, les, light: fix les unittests
* core/state, trie, les, light: fix state iterator
* core, les: address comments
* les: fix lint
---------
Co-authored-by: Gary Rong <garyrong0905@gmail.com>
2023-06-28 04:11:02 -05:00
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
2017-06-27 08:57:06 -05:00
|
|
|
"github.com/ethereum/go-ethereum/ethdb"
|
|
|
|
"github.com/ethereum/go-ethereum/trie"
|
2023-05-09 02:11:04 -05:00
|
|
|
"github.com/ethereum/go-ethereum/trie/trienode"
|
2023-11-14 06:09:40 -06:00
|
|
|
"github.com/ethereum/go-ethereum/trie/utils"
|
2024-02-13 07:49:53 -06:00
|
|
|
"github.com/ethereum/go-ethereum/triedb"
|
2017-06-27 08:57:06 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// Number of codehash->size associations to keep.
|
|
|
|
codeSizeCacheSize = 100000
|
2020-08-21 07:10:40 -05:00
|
|
|
|
|
|
|
// Cache size granted for caching clean code.
|
|
|
|
codeCacheSize = 64 * 1024 * 1024
|
2023-11-14 06:09:40 -06:00
|
|
|
|
2024-05-10 13:13:11 -05:00
|
|
|
// Number of address->curve point associations to keep.
|
|
|
|
pointCacheSize = 4096
|
2017-06-27 08:57:06 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// Database wraps access to tries and contract code.
|
|
|
|
type Database interface {
|
2024-09-05 05:10:47 -05:00
|
|
|
// Reader returns a state reader associated with the specified state root.
|
|
|
|
Reader(root common.Hash) (Reader, error)
|
|
|
|
|
2017-06-27 08:57:06 -05:00
|
|
|
// OpenTrie opens the main account trie.
|
|
|
|
OpenTrie(root common.Hash) (Trie, error)
|
2018-02-05 10:40:32 -06:00
|
|
|
|
|
|
|
// OpenStorageTrie opens the storage trie of an account.
|
2023-11-14 06:09:40 -06:00
|
|
|
OpenStorageTrie(stateRoot common.Hash, address common.Address, root common.Hash, trie Trie) (Trie, error)
|
2018-02-05 10:40:32 -06:00
|
|
|
|
2024-05-10 13:13:11 -05:00
|
|
|
// PointCache returns the cache holding points used in verkle tree key computation
|
|
|
|
PointCache() *utils.PointCache
|
|
|
|
|
all: activate pbss as experimental feature (#26274)
* all: activate pbss
* core/rawdb: fix compilation error
* cma, core, eth, les, trie: address comments
* cmd, core, eth, trie: polish code
* core, cmd, eth: address comments
* cmd, core, eth, les, light, tests: address comment
* cmd/utils: shorten log message
* trie/triedb/pathdb: limit node buffer size to 1gb
* cmd/utils: fix opening non-existing db
* cmd/utils: rename flag name
* cmd, core: group chain history flags and fix tests
* core, eth, trie: fix memory leak in snapshot generation
* cmd, eth, internal: deprecate flags
* all: enable state tests for pathdb, fixes
* cmd, core: polish code
* trie/triedb/pathdb: limit the node buffer size to 256mb
---------
Co-authored-by: Martin Holst Swende <martin@swende.se>
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
2023-08-10 14:21:36 -05:00
|
|
|
// TrieDB returns the underlying trie database for managing trie nodes.
|
2024-02-13 07:49:53 -06:00
|
|
|
TrieDB() *triedb.Database
|
2024-09-05 05:10:47 -05:00
|
|
|
|
|
|
|
// Snapshot returns the underlying state snapshot.
|
|
|
|
Snapshot() *snapshot.Tree
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2019-03-14 08:25:12 -05:00
|
|
|
// Trie is a Ethereum Merkle Patricia trie.
|
2017-06-27 08:57:06 -05:00
|
|
|
type Trie interface {
|
2019-03-14 08:25:12 -05:00
|
|
|
// GetKey returns the sha3 preimage of a hashed key that was previously used
|
|
|
|
// to store a value.
|
|
|
|
//
|
2022-08-04 09:13:18 -05:00
|
|
|
// TODO(fjl): remove this when StateTrie is removed
|
2019-03-14 08:25:12 -05:00
|
|
|
GetKey([]byte) []byte
|
|
|
|
|
2023-03-27 03:48:46 -05:00
|
|
|
// GetAccount abstracts an account read from the trie. It retrieves the
|
2023-01-03 07:41:40 -06:00
|
|
|
// account blob from the trie with provided account address and decodes it
|
|
|
|
// with associated decoding algorithm. If the specified account is not in
|
|
|
|
// the trie, nil will be returned. If the trie is corrupted(e.g. some nodes
|
|
|
|
// are missing or the account blob is incorrect for decoding), an error will
|
|
|
|
// be returned.
|
2023-03-27 03:48:46 -05:00
|
|
|
GetAccount(address common.Address) (*types.StateAccount, error)
|
2021-09-28 03:48:07 -05:00
|
|
|
|
2023-11-14 06:09:40 -06:00
|
|
|
// GetStorage returns the value for key stored in the trie. The value bytes
|
|
|
|
// must not be modified by the caller. If a node was not found in the database,
|
|
|
|
// a trie.MissingNodeError is returned.
|
|
|
|
GetStorage(addr common.Address, key []byte) ([]byte, error)
|
2019-03-14 08:25:12 -05:00
|
|
|
|
2023-03-27 03:48:46 -05:00
|
|
|
// UpdateAccount abstracts an account write to the trie. It encodes the
|
2023-01-03 07:41:40 -06:00
|
|
|
// provided account object with associated algorithm and then updates it
|
|
|
|
// in the trie with provided address.
|
2024-08-30 07:13:02 -05:00
|
|
|
UpdateAccount(address common.Address, account *types.StateAccount, codeLen int) error
|
2022-08-04 09:13:18 -05:00
|
|
|
|
2023-11-14 06:09:40 -06:00
|
|
|
// UpdateStorage associates key with value in the trie. If value has length zero,
|
|
|
|
// any existing value is deleted from the trie. The value bytes must not be modified
|
|
|
|
// by the caller while they are stored in the trie. If a node was not found in the
|
|
|
|
// database, a trie.MissingNodeError is returned.
|
|
|
|
UpdateStorage(addr common.Address, key, value []byte) error
|
|
|
|
|
|
|
|
// DeleteAccount abstracts an account deletion from the trie.
|
|
|
|
DeleteAccount(address common.Address) error
|
2023-06-22 07:52:52 -05:00
|
|
|
|
2023-03-27 03:48:46 -05:00
|
|
|
// DeleteStorage removes any existing value for key from the trie. If a node
|
2023-03-23 05:52:22 -05:00
|
|
|
// was not found in the database, a trie.MissingNodeError is returned.
|
2023-03-27 03:48:46 -05:00
|
|
|
DeleteStorage(addr common.Address, key []byte) error
|
2019-03-14 08:25:12 -05:00
|
|
|
|
2023-11-14 06:09:40 -06:00
|
|
|
// UpdateContractCode abstracts code write to the trie. It is expected
|
|
|
|
// to be moved to the stateWriter interface when the latter is ready.
|
|
|
|
UpdateContractCode(address common.Address, codeHash common.Hash, code []byte) error
|
2022-08-17 06:14:49 -05:00
|
|
|
|
2019-03-14 08:25:12 -05:00
|
|
|
// Hash returns the root hash of the trie. It does not write to the database and
|
|
|
|
// can be used even if the trie doesn't have one.
|
2017-06-27 08:57:06 -05:00
|
|
|
Hash() common.Hash
|
2019-03-14 08:25:12 -05:00
|
|
|
|
2022-08-04 03:03:20 -05:00
|
|
|
// Commit collects all dirty nodes in the trie and replace them with the
|
|
|
|
// corresponding node hash. All collected nodes(including dirty leaves if
|
|
|
|
// collectLeaf is true) will be encapsulated into a nodeset for return.
|
|
|
|
// The returned nodeset can be nil if the trie is clean(nothing to commit).
|
|
|
|
// Once the trie is committed, it's not usable anymore. A new trie must
|
|
|
|
// be created with new root and updated trie database for following usage
|
2024-06-12 04:23:16 -05:00
|
|
|
Commit(collectLeaf bool) (common.Hash, *trienode.NodeSet)
|
2019-03-14 08:25:12 -05:00
|
|
|
|
2024-06-25 06:48:08 -05:00
|
|
|
// Witness returns a set containing all trie nodes that have been accessed.
|
|
|
|
// The returned map could be nil if the witness is empty.
|
|
|
|
Witness() map[string]struct{}
|
|
|
|
|
2019-03-14 08:25:12 -05:00
|
|
|
// NodeIterator returns an iterator that returns nodes of the trie. Iteration
|
cmd, core/state, eth, tests, trie: improve state reader (#27428)
The state availability is checked during the creation of a state reader.
- In hash-based database, if the specified root node does not exist on disk disk, then
the state reader won't be created and an error will be returned.
- In path-based database, if the specified state layer is not available, then the
state reader won't be created and an error will be returned.
This change also contains a stricter semantics regarding the `Commit` operation: once it has been performed, the trie is no longer usable, and certain operations will return an error.
2023-06-20 14:31:45 -05:00
|
|
|
// starts at the key after the given start key. And error will be returned
|
|
|
|
// if fails to create node iterator.
|
|
|
|
NodeIterator(startKey []byte) (trie.NodeIterator, error)
|
2019-03-14 08:25:12 -05:00
|
|
|
|
|
|
|
// Prove constructs a Merkle proof for key. The result contains all encoded nodes
|
|
|
|
// on the path to the value at key. The value itself is also included in the last
|
|
|
|
// node and can be retrieved by verifying the proof.
|
|
|
|
//
|
|
|
|
// If the trie does not contain a value for key, the returned proof contains all
|
|
|
|
// nodes of the longest existing prefix of the key (at least the root), ending
|
|
|
|
// with the node that proves the absence of the key.
|
2023-06-19 09:28:40 -05:00
|
|
|
Prove(key []byte, proofDb ethdb.KeyValueWriter) error
|
2024-05-10 13:13:11 -05:00
|
|
|
|
|
|
|
// IsVerkle returns true if the trie is verkle-tree based
|
|
|
|
IsVerkle() bool
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2024-09-05 05:10:47 -05:00
|
|
|
// CachingDB is an implementation of Database interface. It leverages both trie and
|
|
|
|
// state snapshot to provide functionalities for state access. It's meant to be a
|
|
|
|
// long-live object and has a few caches inside for sharing between blocks.
|
|
|
|
type CachingDB struct {
|
|
|
|
disk ethdb.KeyValueStore
|
|
|
|
triedb *triedb.Database
|
|
|
|
snap *snapshot.Tree
|
|
|
|
codeCache *lru.SizeConstrainedCache[common.Hash, []byte]
|
|
|
|
codeSizeCache *lru.Cache[common.Hash, int]
|
|
|
|
pointCache *utils.PointCache
|
2018-11-12 10:47:34 -06:00
|
|
|
}
|
|
|
|
|
2024-09-05 05:10:47 -05:00
|
|
|
// NewDatabase creates a state database with the provided data sources.
|
|
|
|
func NewDatabase(triedb *triedb.Database, snap *snapshot.Tree) *CachingDB {
|
|
|
|
return &CachingDB{
|
|
|
|
disk: triedb.Disk(),
|
|
|
|
triedb: triedb,
|
|
|
|
snap: snap,
|
2022-11-14 08:41:56 -06:00
|
|
|
codeCache: lru.NewSizeConstrainedCache[common.Hash, []byte](codeCacheSize),
|
2024-09-05 05:10:47 -05:00
|
|
|
codeSizeCache: lru.NewCache[common.Hash, int](codeSizeCacheSize),
|
2024-05-10 13:13:11 -05:00
|
|
|
pointCache: utils.NewPointCache(pointCacheSize),
|
2022-11-28 07:31:28 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-05 05:10:47 -05:00
|
|
|
// NewDatabaseForTesting is similar to NewDatabase, but it initializes the caching
|
|
|
|
// db by using an ephemeral memory db with default config for testing.
|
|
|
|
func NewDatabaseForTesting() *CachingDB {
|
|
|
|
return NewDatabase(triedb.NewDatabase(rawdb.NewMemoryDatabase(), nil), nil)
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2024-09-05 05:10:47 -05:00
|
|
|
// Reader returns a state reader associated with the specified state root.
|
|
|
|
func (db *CachingDB) Reader(stateRoot common.Hash) (Reader, error) {
|
core/state: introduce code reader interface (#30816)
This PR introduces a `ContractCodeReader` interface with functions defined:
type ContractCodeReader interface {
Code(addr common.Address, codeHash common.Hash) ([]byte, error)
CodeSize(addr common.Address, codeHash common.Hash) (int, error)
}
This interface can be implemented in various ways. Although the codebase
currently includes only one implementation, additional implementations
could be created for different purposes and scenarios, such as a code
reader designed for the Verkle tree approach or one that reads code from
the witness.
*Notably, this interface modifies the function’s semantics. If the
contract code is not found, no error will be returned. An error should
only be returned in the event of an unexpected issue, primarily for
future implementations.*
The original state.Reader interface is extended with ContractCodeReader
methods, it gives us more flexibility to manipulate the reader with additional
logic on top, e.g. Hooks.
type Reader interface {
ContractCodeReader
StateReader
}
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
2024-11-29 08:32:45 -06:00
|
|
|
var readers []StateReader
|
2024-09-05 05:10:47 -05:00
|
|
|
|
|
|
|
// Set up the state snapshot reader if available. This feature
|
|
|
|
// is optional and may be partially useful if it's not fully
|
|
|
|
// generated.
|
|
|
|
if db.snap != nil {
|
2024-12-09 20:10:49 -06:00
|
|
|
// If standalone state snapshot is available (hash scheme),
|
|
|
|
// then construct the legacy snap reader.
|
2024-11-08 18:08:06 -06:00
|
|
|
snap := db.snap.Snapshot(stateRoot)
|
|
|
|
if snap != nil {
|
core/state: introduce code reader interface (#30816)
This PR introduces a `ContractCodeReader` interface with functions defined:
type ContractCodeReader interface {
Code(addr common.Address, codeHash common.Hash) ([]byte, error)
CodeSize(addr common.Address, codeHash common.Hash) (int, error)
}
This interface can be implemented in various ways. Although the codebase
currently includes only one implementation, additional implementations
could be created for different purposes and scenarios, such as a code
reader designed for the Verkle tree approach or one that reads code from
the witness.
*Notably, this interface modifies the function’s semantics. If the
contract code is not found, no error will be returned. An error should
only be returned in the event of an unexpected issue, primarily for
future implementations.*
The original state.Reader interface is extended with ContractCodeReader
methods, it gives us more flexibility to manipulate the reader with additional
logic on top, e.g. Hooks.
type Reader interface {
ContractCodeReader
StateReader
}
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
2024-11-29 08:32:45 -06:00
|
|
|
readers = append(readers, newFlatReader(snap))
|
2024-09-05 05:10:47 -05:00
|
|
|
}
|
2024-12-09 20:10:49 -06:00
|
|
|
} else {
|
|
|
|
// If standalone state snapshot is not available, try to construct
|
|
|
|
// the state reader with database.
|
|
|
|
reader, err := db.triedb.StateReader(stateRoot)
|
|
|
|
if err == nil {
|
|
|
|
readers = append(readers, newFlatReader(reader)) // state reader is optional
|
|
|
|
}
|
2024-09-05 05:10:47 -05:00
|
|
|
}
|
|
|
|
// Set up the trie reader, which is expected to always be available
|
|
|
|
// as the gatekeeper unless the state is corrupted.
|
|
|
|
tr, err := newTrieReader(stateRoot, db.triedb, db.pointCache)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
readers = append(readers, tr)
|
|
|
|
|
core/state: introduce code reader interface (#30816)
This PR introduces a `ContractCodeReader` interface with functions defined:
type ContractCodeReader interface {
Code(addr common.Address, codeHash common.Hash) ([]byte, error)
CodeSize(addr common.Address, codeHash common.Hash) (int, error)
}
This interface can be implemented in various ways. Although the codebase
currently includes only one implementation, additional implementations
could be created for different purposes and scenarios, such as a code
reader designed for the Verkle tree approach or one that reads code from
the witness.
*Notably, this interface modifies the function’s semantics. If the
contract code is not found, no error will be returned. An error should
only be returned in the event of an unexpected issue, primarily for
future implementations.*
The original state.Reader interface is extended with ContractCodeReader
methods, it gives us more flexibility to manipulate the reader with additional
logic on top, e.g. Hooks.
type Reader interface {
ContractCodeReader
StateReader
}
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
2024-11-29 08:32:45 -06:00
|
|
|
combined, err := newMultiStateReader(readers...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return newReader(newCachingCodeReader(db.disk, db.codeCache, db.codeSizeCache), combined), nil
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2019-03-14 08:25:12 -05:00
|
|
|
// OpenTrie opens the main account trie at a specific root hash.
|
2024-09-05 05:10:47 -05:00
|
|
|
func (db *CachingDB) OpenTrie(root common.Hash) (Trie, error) {
|
2023-11-14 06:09:40 -06:00
|
|
|
if db.triedb.IsVerkle() {
|
2024-05-10 13:13:11 -05:00
|
|
|
return trie.NewVerkleTrie(root, db.triedb, db.pointCache)
|
2023-11-14 06:09:40 -06:00
|
|
|
}
|
2022-11-28 07:31:28 -06:00
|
|
|
tr, err := trie.NewStateTrie(trie.StateTrieID(root), db.triedb)
|
2020-02-05 06:12:09 -06:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return tr, nil
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2018-02-05 10:40:32 -06:00
|
|
|
// OpenStorageTrie opens the storage trie of an account.
|
2024-09-05 05:10:47 -05:00
|
|
|
func (db *CachingDB) OpenStorageTrie(stateRoot common.Hash, address common.Address, root common.Hash, self Trie) (Trie, error) {
|
2023-11-14 06:09:40 -06:00
|
|
|
// In the verkle case, there is only one tree. But the two-tree structure
|
|
|
|
// is hardcoded in the codebase. So we need to return the same trie in this
|
|
|
|
// case.
|
|
|
|
if db.triedb.IsVerkle() {
|
|
|
|
return self, nil
|
|
|
|
}
|
core/state, light, les: make signature of ContractCode hash-independent (#27209)
* core/state, light, les: make signature of ContractCode hash-independent
* push current state for feedback
* les: fix unit test
* core, les, light: fix les unittests
* core/state, trie, les, light: fix state iterator
* core, les: address comments
* les: fix lint
---------
Co-authored-by: Gary Rong <garyrong0905@gmail.com>
2023-06-28 04:11:02 -05:00
|
|
|
tr, err := trie.NewStateTrie(trie.StorageTrieID(stateRoot, crypto.Keccak256Hash(address.Bytes()), root), db.triedb)
|
2020-02-05 06:12:09 -06:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return tr, nil
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2020-08-21 07:10:40 -05:00
|
|
|
// ContractCodeWithPrefix retrieves a particular contract's code. If the
|
|
|
|
// code can't be found in the cache, then check the existence with **new**
|
|
|
|
// db scheme.
|
core/state: introduce code reader interface (#30816)
This PR introduces a `ContractCodeReader` interface with functions defined:
type ContractCodeReader interface {
Code(addr common.Address, codeHash common.Hash) ([]byte, error)
CodeSize(addr common.Address, codeHash common.Hash) (int, error)
}
This interface can be implemented in various ways. Although the codebase
currently includes only one implementation, additional implementations
could be created for different purposes and scenarios, such as a code
reader designed for the Verkle tree approach or one that reads code from
the witness.
*Notably, this interface modifies the function’s semantics. If the
contract code is not found, no error will be returned. An error should
only be returned in the event of an unexpected issue, primarily for
future implementations.*
The original state.Reader interface is extended with ContractCodeReader
methods, it gives us more flexibility to manipulate the reader with additional
logic on top, e.g. Hooks.
type Reader interface {
ContractCodeReader
StateReader
}
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
2024-11-29 08:32:45 -06:00
|
|
|
func (db *CachingDB) ContractCodeWithPrefix(address common.Address, codeHash common.Hash) []byte {
|
2022-11-14 08:41:56 -06:00
|
|
|
code, _ := db.codeCache.Get(codeHash)
|
|
|
|
if len(code) > 0 {
|
core/state: introduce code reader interface (#30816)
This PR introduces a `ContractCodeReader` interface with functions defined:
type ContractCodeReader interface {
Code(addr common.Address, codeHash common.Hash) ([]byte, error)
CodeSize(addr common.Address, codeHash common.Hash) (int, error)
}
This interface can be implemented in various ways. Although the codebase
currently includes only one implementation, additional implementations
could be created for different purposes and scenarios, such as a code
reader designed for the Verkle tree approach or one that reads code from
the witness.
*Notably, this interface modifies the function’s semantics. If the
contract code is not found, no error will be returned. An error should
only be returned in the event of an unexpected issue, primarily for
future implementations.*
The original state.Reader interface is extended with ContractCodeReader
methods, it gives us more flexibility to manipulate the reader with additional
logic on top, e.g. Hooks.
type Reader interface {
ContractCodeReader
StateReader
}
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
2024-11-29 08:32:45 -06:00
|
|
|
return code
|
2020-08-21 07:10:40 -05:00
|
|
|
}
|
2022-11-14 08:41:56 -06:00
|
|
|
code = rawdb.ReadCodeWithPrefix(db.disk, codeHash)
|
2020-08-21 07:10:40 -05:00
|
|
|
if len(code) > 0 {
|
2022-11-09 01:06:02 -06:00
|
|
|
db.codeCache.Add(codeHash, code)
|
2017-06-27 08:57:06 -05:00
|
|
|
db.codeSizeCache.Add(codeHash, len(code))
|
|
|
|
}
|
core/state: introduce code reader interface (#30816)
This PR introduces a `ContractCodeReader` interface with functions defined:
type ContractCodeReader interface {
Code(addr common.Address, codeHash common.Hash) ([]byte, error)
CodeSize(addr common.Address, codeHash common.Hash) (int, error)
}
This interface can be implemented in various ways. Although the codebase
currently includes only one implementation, additional implementations
could be created for different purposes and scenarios, such as a code
reader designed for the Verkle tree approach or one that reads code from
the witness.
*Notably, this interface modifies the function’s semantics. If the
contract code is not found, no error will be returned. An error should
only be returned in the event of an unexpected issue, primarily for
future implementations.*
The original state.Reader interface is extended with ContractCodeReader
methods, it gives us more flexibility to manipulate the reader with additional
logic on top, e.g. Hooks.
type Reader interface {
ContractCodeReader
StateReader
}
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
2024-11-29 08:32:45 -06:00
|
|
|
return code
|
2017-06-27 08:57:06 -05:00
|
|
|
}
|
|
|
|
|
2018-02-05 10:40:32 -06:00
|
|
|
// TrieDB retrieves any intermediate trie-node caching layer.
|
2024-09-05 05:10:47 -05:00
|
|
|
func (db *CachingDB) TrieDB() *triedb.Database {
|
2022-11-28 07:31:28 -06:00
|
|
|
return db.triedb
|
2018-02-05 10:40:32 -06:00
|
|
|
}
|
2024-05-10 13:13:11 -05:00
|
|
|
|
|
|
|
// PointCache returns the cache of evaluated curve points.
|
2024-09-05 05:10:47 -05:00
|
|
|
func (db *CachingDB) PointCache() *utils.PointCache {
|
2024-05-10 13:13:11 -05:00
|
|
|
return db.pointCache
|
|
|
|
}
|
2024-09-05 05:10:47 -05:00
|
|
|
|
|
|
|
// Snapshot returns the underlying state snapshot.
|
|
|
|
func (db *CachingDB) Snapshot() *snapshot.Tree {
|
|
|
|
return db.snap
|
|
|
|
}
|
|
|
|
|
|
|
|
// mustCopyTrie returns a deep-copied trie.
|
|
|
|
func mustCopyTrie(t Trie) Trie {
|
|
|
|
switch t := t.(type) {
|
|
|
|
case *trie.StateTrie:
|
|
|
|
return t.Copy()
|
|
|
|
case *trie.VerkleTrie:
|
|
|
|
return t.Copy()
|
|
|
|
default:
|
|
|
|
panic(fmt.Errorf("unknown trie type %T", t))
|
|
|
|
}
|
|
|
|
}
|