erigon-pulse/core/state/stateless.go
ledgerwatch 5ea590c18e
State cache switching writes to reads during commit (#1368)
* State cache init

* More code

* Fix lint

* More tests

* More tests

* More tests

* Fix test

* Transformations

* remove writeQueue, before fixing the tests

* Fix tests

* Add more tests, incarnation to the code items

* Fix lint

* Fix lint

* Remove shards prototype, add incarnation to the state reader code

* Clean up and replace cache in call_traces stage

* fix flaky test

* Save changes

* Readers to use addrHash, writes - addresses

* Fix lint

* Fix lint

* More accurate tracking of size

* Optimise for smaller write batches

* Attempt to integrate state cache into Execution stage

* cacheSize to default flags

* Print correct cache sizes and batch sizes

* cacheSize in the integration

* Fix tests

* Fix lint

* Remove print

* Fix exec stage

* Fix test

* Refresh sequence on write

* No double increment

* heap.Remove

* Try to fix alignment

* Refactoring, adding hashItems

* More changes

* Fix compile errors

* Fix lint

* Wrapping cached reader

* Wrap writer into cached writer

* Turn state cache off by default

* Fix plain state writer

* Fix for code/storage mixup

* Fix tests

* Fix clique test

* Better fix for the tests

* Add test and fix some more

* Fix compile error|

* More functions

* Fixes

* Fix for the tests

* sepatate DeletedFlag and AbsentFlag

* Minor fixes

* Test refactoring

* More changes

* Fix some tests

* More test fixes

* More test fixes

* Fix lint

* Move blockchain_test to be able to use stagedsync

* More fixes

* Fixes and cleanup

* Fix tests in turbo/stages

* Fix lint

* Fix lint

* Intemediate

* Fix tests

* Intemediate

* More fixes

* Compilation fixes

* More fixes

* Fix compile errors

* More test fixes

* More fixes

* More test fixes

* Fix compile error

* Fixes

* Fix

* Fix

* More fixes

* Fixes

* More fixes and cleanup

* Further fix

* Check gas used and bloom with header

Co-authored-by: Alexey Sharp <alexeysharp@Alexeys-iMac.local>
2020-12-08 09:44:29 +00:00

348 lines
11 KiB
Go

// Copyright 2019 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 (
"bytes"
"context"
"fmt"
"os"
"github.com/holiman/uint256"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/common/dbutils"
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
"github.com/ledgerwatch/turbo-geth/turbo/trie"
)
var (
_ StateReader = (*Stateless)(nil)
_ StateWriter = (*Stateless)(nil)
)
// Stateless is the inter-block cache for stateless client prototype, iteration 2
// It creates the initial state trie during the construction, and then updates it
// during the execution of block(s)
type Stateless struct {
t *trie.Trie // State trie
codeUpdates map[common.Hash][]byte // Lookup index from code hashes to corresponding bytecode
blockNr uint64 // Current block number
storageUpdates map[common.Hash]map[common.Hash][]byte
accountUpdates map[common.Hash]*accounts.Account
deleted map[common.Hash]struct{}
created map[common.Hash]struct{}
trace bool
}
// NewStateless creates a new instance of Stateless
// It deserialises the block witness and creates the state trie out of it, checking that the root of the constructed
// state trie matches the value of `stateRoot` parameter
func NewStateless(stateRoot common.Hash, blockWitness *trie.Witness, blockNr uint64, trace bool, isBinary bool) (*Stateless, error) {
t, err := trie.BuildTrieFromWitness(blockWitness, isBinary, trace)
if err != nil {
return nil, err
}
if !isBinary {
if t.Hash() != stateRoot {
filename := fmt.Sprintf("root_%d.txt", blockNr)
f, err := os.Create(filename)
if err == nil {
defer f.Close()
t.Print(f)
}
return nil, fmt.Errorf("state root mistmatch when creating Stateless2, got %x, expected %x", t.Hash(), stateRoot)
}
}
return &Stateless{
t: t,
codeUpdates: make(map[common.Hash][]byte),
storageUpdates: make(map[common.Hash]map[common.Hash][]byte),
accountUpdates: make(map[common.Hash]*accounts.Account),
deleted: make(map[common.Hash]struct{}),
created: make(map[common.Hash]struct{}),
blockNr: blockNr,
trace: trace,
}, nil
}
// SetBlockNr changes the block number associated with this
func (s *Stateless) SetBlockNr(blockNr uint64) {
s.blockNr = blockNr
}
// ReadAccountData is a part of the StateReader interface
// This implementation attempts to look up account data in the state trie, and fails if it is not found
func (s *Stateless) ReadAccountData(address common.Address) (*accounts.Account, error) {
addrHash, err := common.HashData(address[:])
if err != nil {
return nil, err
}
acc, ok := s.t.GetAccount(addrHash[:])
if ok {
return acc, nil
}
return nil, fmt.Errorf("could not find account with address %x", address)
}
// ReadAccountStorage is a part of the StateReader interface
// This implementation attempts to look up the storage in the state trie, and fails if it is not found
func (s *Stateless) ReadAccountStorage(address common.Address, incarnation uint64, key *common.Hash) ([]byte, error) {
seckey, err := common.HashData(key[:])
if err != nil {
return nil, err
}
addrHash, err := common.HashData(address[:])
if err != nil {
return nil, err
}
if enc, ok := s.t.Get(dbutils.GenerateCompositeTrieKey(addrHash, seckey)); ok {
return enc, nil
}
return nil, fmt.Errorf("could not find storage item %x in account with address %x", key, address)
}
// ReadAccountCode is a part of the StateReader interface
func (s *Stateless) ReadAccountCode(address common.Address, incarnation uint64, codeHash common.Hash) (code []byte, err error) {
if bytes.Equal(codeHash[:], emptyCodeHash) {
return nil, nil
}
if s.trace {
fmt.Printf("Getting code for %x\n", codeHash)
}
addrHash, err := common.HashData(address[:])
if err != nil {
return nil, err
}
if code, ok := s.codeUpdates[addrHash]; ok {
return code, nil
}
if code, ok := s.t.GetAccountCode(addrHash[:]); ok {
return code, nil
}
return nil, fmt.Errorf("could not find bytecode for acc: %x hash %x", address, codeHash)
}
// ReadAccountCodeSize is a part of the StateReader interface
// This implementation looks the code up in the codeMap, and returns its size
// It fails if the code is not found in the map
func (s *Stateless) ReadAccountCodeSize(address common.Address, incarnation uint64, codeHash common.Hash) (codeSize int, err error) {
if bytes.Equal(codeHash[:], emptyCodeHash) {
return 0, nil
}
addrHash, err := common.HashData(address[:])
if err != nil {
return 0, err
}
if code, ok := s.codeUpdates[addrHash]; ok {
return len(code), nil
}
if code, ok := s.t.GetAccountCode(addrHash[:]); ok {
return len(code), nil
}
if codeSize, ok := s.t.GetAccountCodeSize(addrHash[:]); ok {
return codeSize, nil
}
return 0, fmt.Errorf("could not find bytecode for hash %x", codeHash)
}
func (s *Stateless) ReadAccountIncarnation(address common.Address) (uint64, error) {
return 0, nil
}
// UpdateAccountData is a part of the StateWriter interface
// This implementation registers the account update in the `accountUpdates` map
func (s *Stateless) UpdateAccountData(_ context.Context, address common.Address, original, account *accounts.Account) error {
addrHash, err := common.HashData(address[:])
if err != nil {
return err
}
if s.trace {
fmt.Printf("UpdateAccountData for address %x, addrHash %x\n", address, addrHash)
}
s.accountUpdates[addrHash] = account
return nil
}
// DeleteAccount is a part of the StateWriter interface
// This implementation registers the deletion of the account in two internal maps
func (s *Stateless) DeleteAccount(_ context.Context, address common.Address, original *accounts.Account) error {
addrHash, err := common.HashData(address[:])
if err != nil {
return err
}
s.accountUpdates[addrHash] = nil
s.deleted[addrHash] = struct{}{}
if s.trace {
fmt.Printf("Stateless: DeleteAccount %x hash %x\n", address, addrHash)
}
return nil
}
// UpdateAccountCode is a part of the StateWriter interface
// This implementation adds the code to the codeMap to make it available for further accesses
func (s *Stateless) UpdateAccountCode(address common.Address, incarnation uint64, codeHash common.Hash, code []byte) error {
s.codeUpdates[codeHash] = code
if s.trace {
fmt.Printf("Stateless: UpdateAccountCode %x codeHash %x\n", address, codeHash)
}
return nil
}
// WriteAccountStorage is a part of the StateWriter interface
// This implementation registeres the change of the account's storage in the internal double map `storageUpdates`
func (s *Stateless) WriteAccountStorage(_ context.Context, address common.Address, incarnation uint64, key *common.Hash, original, value *uint256.Int) error {
addrHash, err := common.HashData(address[:])
if err != nil {
return err
}
v := value.Bytes()
m, ok := s.storageUpdates[addrHash]
if !ok {
m = make(map[common.Hash][]byte)
s.storageUpdates[addrHash] = m
}
seckey, err := common.HashData(key[:])
if err != nil {
return err
}
if len(v) > 0 {
m[seckey] = v
} else {
m[seckey] = nil
}
if s.trace {
fmt.Printf("Stateless: WriteAccountStorage %x key %x val %x\n", address, *key, *value)
}
return nil
}
// CreateContract is a part of StateWriter interface
// This implementation registers given address in the internal map `created`
func (s *Stateless) CreateContract(address common.Address) error {
addrHash, err := common.HashData(address[:])
if err != nil {
return err
}
if s.trace {
fmt.Printf("Stateless: CreateContract %x hash %x\n", address, addrHash)
}
s.created[addrHash] = struct{}{}
return nil
}
// CheckRoot finalises the execution of a block and computes the resulting state root
func (s *Stateless) CheckRoot(expected common.Hash) error {
// The following map is to prevent repeated clearouts of the storage
alreadyCreated := make(map[common.Hash]struct{})
// New contracts are being created at these addresses. Therefore, we need to clear the storage items
// that might be remaining in the trie and figure out the next incarnations
for addrHash := range s.created {
// Prevent repeated storage clearouts
if _, ok := alreadyCreated[addrHash]; ok {
continue
}
alreadyCreated[addrHash] = struct{}{}
if account, ok := s.accountUpdates[addrHash]; ok && account != nil {
account.Root = trie.EmptyRoot
}
// The only difference between Delete and DeleteSubtree is that Delete would delete accountNode too,
// wherewas DeleteSubtree will keep the accountNode, but will make the storage sub-trie empty
s.t.DeleteSubtree(addrHash[:])
}
for addrHash, account := range s.accountUpdates {
if account != nil {
s.t.UpdateAccount(addrHash[:], account)
} else {
s.t.Delete(addrHash[:])
}
}
for addrHash, m := range s.storageUpdates {
if _, ok := s.deleted[addrHash]; ok {
// Deleted contracts will be dealth with later, in the next loop
continue
}
for keyHash, v := range m {
cKey := dbutils.GenerateCompositeTrieKey(addrHash, keyHash)
if len(v) > 0 {
s.t.Update(cKey, v)
} else {
s.t.Delete(cKey)
}
}
if account, ok := s.accountUpdates[addrHash]; ok && account != nil {
ok, root := s.t.DeepHash(addrHash[:])
if ok {
account.Root = root
} else {
account.Root = trie.EmptyRoot
}
}
}
// For the contracts that got deleted
for addrHash := range s.deleted {
if _, ok := s.created[addrHash]; ok {
// In some rather artificial circumstances, an account can be recreated after having been self-destructed
// in the same block. It can only happen when contract is introduced in the genesis state with nonce 0
// rather than created by a transaction (in that case, its starting nonce is 1). The self-destructed
// contract actually gets removed from the state only at the end of the block, so if its nonce is not 0,
// it will prevent any re-creation within the same block. However, if the contract is introduced in
// the genesis state, its nonce is 0, and that means it can be self-destructed, and then re-created,
// all in the same block. In such cases, we must preserve storage modifications happening after the
// self-destruction
continue
}
if account, ok := s.accountUpdates[addrHash]; ok && account != nil {
account.Root = trie.EmptyRoot
}
s.t.DeleteSubtree(addrHash[:])
}
myRoot := s.t.Hash()
if myRoot != expected {
filename := fmt.Sprintf("root_%d.txt", s.blockNr)
f, err := os.Create(filename)
if err == nil {
defer f.Close()
s.t.Print(f)
}
return fmt.Errorf("final root: %x, expected: %x", myRoot, expected)
}
s.storageUpdates = make(map[common.Hash]map[common.Hash][]byte)
s.accountUpdates = make(map[common.Hash]*accounts.Account)
s.deleted = make(map[common.Hash]struct{})
s.created = make(map[common.Hash]struct{})
return nil
}
func (s *Stateless) GetTrie() *trie.Trie {
return s.t
}