erigon-pulse/core/state/database.go

1190 lines
35 KiB
Go
Raw Normal View History

// 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"
"encoding/binary"
"fmt"
"io"
"runtime"
"sort"
lru "github.com/hashicorp/golang-lru"
"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/ethdb"
"github.com/ledgerwatch/turbo-geth/log"
"github.com/ledgerwatch/turbo-geth/params"
"github.com/ledgerwatch/turbo-geth/trie"
)
// Trie cache generation limit after which to evict trie nodes from memory.
var MaxTrieCacheGen = uint32(1024 * 1024)
const IncarnationLength = 8
type StateReader interface {
ReadAccountData(address common.Address) (*accounts.Account, error)
ReadAccountStorage(address common.Address, incarnation uint64, key *common.Hash) ([]byte, error)
ReadAccountCode(codeHash common.Hash) ([]byte, error)
ReadAccountCodeSize(codeHash common.Hash) (int, error)
}
type StateWriter interface {
UpdateAccountData(ctx context.Context, address common.Address, original, account *accounts.Account) error
UpdateAccountCode(codeHash common.Hash, code []byte) error
DeleteAccount(ctx context.Context, address common.Address, original *accounts.Account) error
WriteAccountStorage(ctx context.Context, address common.Address, incarnation uint64, key, original, value *common.Hash) error
CreateContract(address common.Address) error
}
type NoopWriter struct {
}
func NewNoopWriter() *NoopWriter {
return &NoopWriter{}
}
func (nw *NoopWriter) UpdateAccountData(_ context.Context, address common.Address, original, account *accounts.Account) error {
return nil
}
func (nw *NoopWriter) DeleteAccount(_ context.Context, address common.Address, original *accounts.Account) error {
return nil
}
func (nw *NoopWriter) UpdateAccountCode(codeHash common.Hash, code []byte) error {
return nil
}
func (nw *NoopWriter) WriteAccountStorage(_ context.Context, address common.Address, incarnation uint64, key, original, value *common.Hash) error {
return nil
}
func (nw *NoopWriter) CreateContract(address common.Address) error {
return nil
}
// Structure holding updates, deletes, and reads registered within one change period
// A change period can be transaction within a block, or a block within group of blocks
type Buffer struct {
storageUpdates map[addressHashWithIncarnation]map[common.Hash][]byte
storageReads map[addressHashWithIncarnation]map[common.Hash]struct{}
accountUpdates map[common.Hash]*accounts.Account
accountReads map[common.Hash]struct{}
deleted map[common.Address]struct{}
deletedHashes map[common.Hash]struct{}
created map[common.Address]struct{}
}
func newAddressHashWithIncarnation(addrHash common.Hash, incarnation uint64) addressHashWithIncarnation {
var res addressHashWithIncarnation
copy(res[:common.HashLength], addrHash[:])
buf := make([]byte, IncarnationLength)
binary.BigEndian.PutUint64(buf, incarnation^^uint64(0))
copy(res[common.HashLength:], buf[:])
return res
}
type addressHashWithIncarnation [common.HashLength + IncarnationLength]byte
// returns address hash
func (a *addressHashWithIncarnation) Hash() common.Hash {
var addrHash common.Hash
copy(addrHash[:], a[:common.HashLength])
return addrHash
}
func (a *addressHashWithIncarnation) Incarnation() uint64 {
return ^uint64(0) ^ binary.BigEndian.Uint64(a[common.HashLength:common.HashLength+IncarnationLength])
}
// Prepares buffer for work or clears previous data
func (b *Buffer) initialise() {
b.storageUpdates = make(map[addressHashWithIncarnation]map[common.Hash][]byte)
b.storageReads = make(map[addressHashWithIncarnation]map[common.Hash]struct{})
b.accountUpdates = make(map[common.Hash]*accounts.Account)
b.accountReads = make(map[common.Hash]struct{})
b.deleted = make(map[common.Address]struct{})
b.created = make(map[common.Address]struct{})
}
// Replaces account pointer with pointers to the copies
func (b *Buffer) detachAccounts() {
for addrHash, account := range b.accountUpdates {
if account != nil {
var c accounts.Account
c.Copy(account)
b.accountUpdates[addrHash] = &c
}
}
}
// Merges the content of another buffer into this one
func (b *Buffer) merge(other *Buffer) {
for address, om := range other.storageUpdates {
m, ok := b.storageUpdates[address]
if !ok {
m = make(map[common.Hash][]byte)
b.storageUpdates[address] = m
}
for keyHash, v := range om {
m[keyHash] = v
}
}
for address, om := range other.storageReads {
m, ok := b.storageReads[address]
if !ok {
m = make(map[common.Hash]struct{})
b.storageReads[address] = m
}
for keyHash := range om {
m[keyHash] = struct{}{}
}
}
for addrHash, account := range other.accountUpdates {
b.accountUpdates[addrHash] = account
}
for addrHash := range other.accountReads {
b.accountReads[addrHash] = struct{}{}
}
for address := range other.deleted {
b.deleted[address] = struct{}{}
}
for address := range other.created {
b.created[address] = struct{}{}
}
}
// TrieDbState implements StateReader by wrapping a trie and a database, where trie acts as a cache for the database
type TrieDbState struct {
t *trie.Trie
db ethdb.Database
blockNr uint64
buffers []*Buffer
aggregateBuffer *Buffer // Merge of all buffers
currentBuffer *Buffer
codeCache *lru.Cache
codeSizeCache *lru.Cache
historical bool
noHistory bool
resolveReads bool
pg *trie.ProofGenerator
tp *trie.TriePruning
}
func NewTrieDbState(root common.Hash, db ethdb.Database, blockNr uint64) (*TrieDbState, error) {
csc, err := lru.New(100000)
if err != nil {
return nil, err
}
cc, err := lru.New(10000)
if err != nil {
return nil, err
}
t := trie.New(root)
tp := trie.NewTriePruning(blockNr)
tds := TrieDbState{
t: t,
db: db,
blockNr: blockNr,
codeCache: cc,
codeSizeCache: csc,
pg: trie.NewProofGenerator(),
tp: tp,
}
t.SetTouchFunc(func(hex []byte, del bool) {
tp.Touch(hex, del)
})
return &tds, nil
}
func (tds *TrieDbState) SetHistorical(h bool) {
tds.historical = h
}
func (tds *TrieDbState) SetResolveReads(rr bool) {
tds.resolveReads = rr
}
func (tds *TrieDbState) SetNoHistory(nh bool) {
tds.noHistory = nh
}
func (tds *TrieDbState) Copy() *TrieDbState {
tcopy := *tds.t
tp := trie.NewTriePruning(tds.blockNr)
cpy := TrieDbState{
t: &tcopy,
db: tds.db,
blockNr: tds.blockNr,
tp: tp,
}
return &cpy
}
func (tds *TrieDbState) Database() ethdb.Database {
return tds.db
}
func (tds *TrieDbState) Trie() *trie.Trie {
return tds.t
}
func (tds *TrieDbState) StartNewBuffer() {
if tds.currentBuffer != nil {
if tds.aggregateBuffer == nil {
tds.aggregateBuffer = &Buffer{}
tds.aggregateBuffer.initialise()
}
tds.aggregateBuffer.merge(tds.currentBuffer)
tds.currentBuffer.detachAccounts()
}
tds.currentBuffer = &Buffer{}
tds.currentBuffer.initialise()
tds.buffers = append(tds.buffers, tds.currentBuffer)
}
func (tds *TrieDbState) LastRoot() common.Hash {
return tds.t.Hash()
}
// DESCRIBED: docs/programmers_guide/guide.md#organising-ethereum-state-into-a-merkle-tree
func (tds *TrieDbState) ComputeTrieRoots() ([]common.Hash, error) {
roots, err := tds.computeTrieRoots(true)
tds.clearUpdates()
return roots, err
}
func (tds *TrieDbState) PrintTrie(w io.Writer) {
tds.t.Print(w)
fmt.Fprintln(w, "") //nolint
}
// Hashes are a slice of hashes.
type Hashes []common.Hash
func (hashes Hashes) Len() int {
return len(hashes)
}
func (hashes Hashes) Less(i, j int) bool {
return bytes.Compare(hashes[i][:], hashes[j][:]) == -1
}
func (hashes Hashes) Swap(i, j int) {
hashes[i], hashes[j] = hashes[j], hashes[i]
}
// Builds a map where for each address (of a smart contract) there is
// a sorted list of all key hashes that were touched within the
// period for which we are aggregating updates
func (tds *TrieDbState) buildStorageTouches() map[addressHashWithIncarnation]Hashes {
storageTouches := make(map[addressHashWithIncarnation]Hashes)
for addressHash, m := range tds.aggregateBuffer.storageUpdates {
var hashes Hashes
mRead := tds.aggregateBuffer.storageReads[addressHash]
i := 0
hashes = make(Hashes, len(m)+len(mRead))
for keyHash := range m {
hashes[i] = keyHash
i++
}
for keyHash := range mRead {
if _, ok := m[keyHash]; !ok {
hashes[i] = keyHash
i++
}
}
if i > 0 {
sort.Sort(hashes[:i])
storageTouches[addressHash] = hashes[:i]
}
}
for address, m := range tds.aggregateBuffer.storageReads {
if _, ok := tds.aggregateBuffer.storageUpdates[address]; ok {
continue
}
hashes := make(Hashes, len(m))
i := 0
for keyHash := range m {
hashes[i] = keyHash
i++
}
sort.Sort(hashes)
storageTouches[address] = hashes
}
return storageTouches
}
// Expands the storage tries (by loading data from the database) if it is required
// for accessing storage slots containing in the storageTouches map
func (tds *TrieDbState) resolveStorageTouches(storageTouches map[addressHashWithIncarnation]Hashes) error {
var resolver *trie.Resolver
for addressHash, hashes := range storageTouches {
var addrHash = addressHash.Hash()
for _, keyHash := range hashes {
//todo @need resolution for prefix
if need, req := tds.t.NeedResolution(addrHash[:], keyHash[:]); need {
if resolver == nil {
resolver = trie.NewResolver(0, false, tds.blockNr)
resolver.SetHistorical(tds.historical)
}
//fmt.Printf("Storage resolve request: %s\n", req.String())
resolver.AddRequest(req)
//fmt.Printf("Need resolution for %x %x, %s\n", addrHash, keyHash, req.String())
} else { //nolint
//fmt.Printf("Don't need resolution for %x %x\n", addrHash, keyHash)
}
}
}
if resolver != nil {
if err := resolver.ResolveWithDb(tds.db, tds.blockNr); err != nil {
return err
}
}
return nil
}
// Populate pending block proof so that it will be sufficient for accessing all storage slots in storageTouches
func (tds *TrieDbState) populateStorageBlockProof(storageTouches map[addressHashWithIncarnation]Hashes) error { //nolint
for addresHash, hashes := range storageTouches {
addrHash := addresHash.Hash()
if _, ok := tds.aggregateBuffer.deletedHashes[addrHash]; ok && len(tds.aggregateBuffer.storageReads[addresHash]) == 0 {
// We can only skip the proof of storage entirely if
// there were no reads before writes and account got deleted
continue
}
for _, keyHash := range hashes {
tds.pg.AddTouch(dbutils.GenerateCompositeTrieKey(addrHash, keyHash))
//tds.t.PopulateBlockProofData(addrHash[:], keyHash[:], tds.pg)
}
}
return nil
}
// Builds a sorted list of all address hashes that were touched within the
// period for which we are aggregating updates
func (tds *TrieDbState) buildAccountTouches() Hashes {
accountTouches := make(Hashes, len(tds.aggregateBuffer.accountUpdates)+len(tds.aggregateBuffer.accountReads))
i := 0
for addrHash := range tds.aggregateBuffer.accountUpdates {
accountTouches[i] = addrHash
i++
}
for addrHash := range tds.aggregateBuffer.accountReads {
if _, ok := tds.aggregateBuffer.accountUpdates[addrHash]; !ok {
accountTouches[i] = addrHash
i++
}
}
sort.Sort(accountTouches[:i])
return accountTouches[:i]
}
func (tds *TrieDbState) buildDeletedAccountTouches() error {
for i := range tds.buffers {
tds.buffers[i].deletedHashes = make(map[common.Hash]struct{}, len(tds.buffers[i].deleted))
for k := range tds.buffers[i].deleted {
h, err := tds.HashAddress(k, false)
if err != nil {
return err
}
tds.buffers[i].deletedHashes[h] = struct{}{}
}
}
return nil
}
// Expands the accounts trie (by loading data from the database) if it is required
// for accessing accounts whose addresses are contained in the accountTouches
func (tds *TrieDbState) resolveAccountTouches(accountTouches Hashes) error {
var resolver *trie.Resolver
for _, addrHash := range accountTouches {
if need, req := tds.t.NeedResolution(nil, addrHash[:]); need {
if resolver == nil {
resolver = trie.NewResolver(0, true, tds.blockNr)
resolver.SetHistorical(tds.historical)
}
resolver.AddRequest(req)
}
}
if resolver != nil {
if err := resolver.ResolveWithDb(tds.db, tds.blockNr); err != nil {
return err
}
resolver = nil
}
return nil
}
func (tds *TrieDbState) populateAccountBlockProof(accountTouches Hashes) {
for _, addrHash := range accountTouches {
if addrHash == (common.Hash{}) {
fmt.Printf("Empty hash\n")
}
a := addrHash
tds.pg.AddTouch(a[:])
//tds.t.PopulateBlockProofData(nil, addrHash[:], tds.pg)
}
}
func (tds *TrieDbState) ExtractTouches() [][]byte {
return tds.pg.ExtractTouches()
}
// forward is `true` if the function is used to progress the state forward (by adding blocks)
// forward is `false` if the function is used to rewind the state (for reorgs, for example)
func (tds *TrieDbState) computeTrieRoots(forward bool) ([]common.Hash, error) {
// Aggregating the current buffer, if any
if tds.currentBuffer != nil {
if tds.aggregateBuffer == nil {
tds.aggregateBuffer = &Buffer{}
tds.aggregateBuffer.initialise()
}
tds.aggregateBuffer.merge(tds.currentBuffer)
}
if tds.aggregateBuffer == nil {
return nil, nil
}
// Prepare (resolve) storage tries so that actual modifications can proceed without database access
storageTouches := tds.buildStorageTouches()
// Prepare (resolve) accounts trie so that actual modifications can proceed without database access
accountTouches := tds.buildAccountTouches()
if err := tds.resolveAccountTouches(accountTouches); err != nil {
return nil, err
}
if tds.resolveReads {
tds.populateAccountBlockProof(accountTouches)
}
err := tds.buildDeletedAccountTouches()
if err != nil {
return nil, err
}
if err := tds.resolveStorageTouches(storageTouches); err != nil {
return nil, err
}
if tds.resolveReads {
if err := tds.populateStorageBlockProof(storageTouches); err != nil {
return nil, err
}
}
accountUpdates := tds.aggregateBuffer.accountUpdates
// The following map is to prevent repeated clearouts of the storage
alreadyCreated := make(map[common.Address]struct{})
// Perform actual updates on the tries, and compute one trie root per buffer
// These roots can be used to populate receipt.PostState on pre-Byzantium
roots := make([]common.Hash, len(tds.buffers))
for i, b := range tds.buffers {
// 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 address := range b.created {
// Prevent repeated storage clearouts
if _, ok := alreadyCreated[address]; ok {
continue
}
alreadyCreated[address] = struct{}{}
addrHash, err := tds.HashAddress(address, false /*save*/)
if err != nil {
return nil, err
}
var incarnation uint64
incarnation, err = tds.nextIncarnation(address)
if err != nil {
return nil, err
}
if account, ok := b.accountUpdates[addrHash]; ok && account != nil {
account.SetIncarnation(incarnation)
account.Root = trie.EmptyRoot
}
if account, ok := accountUpdates[addrHash]; ok && account != nil {
account.SetIncarnation(incarnation)
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
tds.t.DeleteSubtree(addrHash[:], tds.blockNr)
}
for addrHash, account := range b.accountUpdates {
if account != nil {
tds.t.UpdateAccount(addrHash[:], account)
} else {
tds.t.Delete(addrHash[:], tds.blockNr)
}
}
for addressHash, m := range b.storageUpdates {
addrHash := addressHash.Hash()
if _, ok := b.deletedHashes[addressHash.Hash()]; ok {
// Deleted contracts will be dealth with later, in the next loop
continue
}
for keyHash, v := range m {
cKey := dbutils.GenerateCompositeTrieKey(addressHash.Hash(), keyHash)
if len(v) > 0 {
//fmt.Printf("Update storage trie addrHash %x, keyHash %x: %x\n", addrHash, keyHash, v)
if forward {
tds.t.Update(cKey, v, tds.blockNr)
} else {
// If rewinding, it might not be possible to execute storage item update.
// If we rewind from the state where a contract does not exist anymore (it was self-destructed)
// to the point where it existed (with storage), then rewinding to the point of existence
// will not bring back the full storage trie. Instead there will be one hashNode.
// So we probe for this situation first
if _, ok := tds.t.Get(cKey); ok {
tds.t.Update(cKey, v, tds.blockNr)
}
}
} else {
//fmt.Printf("Delete storage trie addrHash %x, keyHash %x\n", addrHash, keyHash)
if forward {
tds.t.Delete(cKey, tds.blockNr)
} else {
// If rewinding, it might not be possible to execute storage item update.
// If we rewind from the state where a contract does not exist anymore (it was self-destructed)
// to the point where it existed (with storage), then rewinding to the point of existence
// will not bring back the full storage trie. Instead there will be one hashNode.
// So we probe for this situation first
if _, ok := tds.t.Get(cKey); ok {
tds.t.Delete(cKey, tds.blockNr)
}
}
}
}
if forward {
if account, ok := b.accountUpdates[addrHash]; ok && account != nil {
ok, root := tds.t.DeepHash(addrHash[:])
if ok {
account.Root = root
} else {
//fmt.Printf("(b)Set empty root for addrHash %x\n", addrHash)
account.Root = trie.EmptyRoot
}
}
if account, ok := accountUpdates[addrHash]; ok && account != nil {
ok, root := tds.t.DeepHash(addrHash[:])
if ok {
account.Root = root
} else {
//fmt.Printf("Set empty root for addrHash %x\n", addrHash)
account.Root = trie.EmptyRoot
}
}
} else {
// Simply comparing the correctness of the storageRoot computations
if account, ok := b.accountUpdates[addrHash]; ok && account != nil {
ok, h := tds.t.DeepHash(addrHash[:])
if !ok {
h = trie.EmptyRoot
}
if account.Root != h {
return nil, fmt.Errorf("mismatched storage root for %x: expected %x, got %x", addressHash, account.Root, h)
}
}
if account, ok := accountUpdates[addrHash]; ok && account != nil {
ok, h := tds.t.DeepHash(addrHash[:])
if !ok {
h = trie.EmptyRoot
}
if account.Root != h {
return nil, fmt.Errorf("mismatched storage root for %x: expected %x, got %x", addressHash, account.Root, h)
}
}
}
}
// For the contracts that got deleted
for address := range b.deleted {
addrHash, err := tds.HashAddress(address, false /*save*/)
if err != nil {
return nil, err
}
if account, ok := b.accountUpdates[addrHash]; ok && account != nil {
account.Root = trie.EmptyRoot
}
if account, ok := accountUpdates[addrHash]; ok && account != nil {
account.Root = trie.EmptyRoot
}
tds.t.DeleteSubtree(addrHash[:], tds.blockNr)
}
roots[i] = tds.t.Hash()
}
return roots, nil
}
func (tds *TrieDbState) clearUpdates() {
tds.buffers = nil
tds.currentBuffer = nil
tds.aggregateBuffer = nil
}
func (tds *TrieDbState) Rebuild() error {
if err := tds.Trie().Rebuild(tds.db, tds.blockNr); err != nil {
return err
}
var m runtime.MemStats
runtime.ReadMemStats(&m)
log.Info("Memory after rebuild", "nodes", tds.tp.NodeCount(), "alloc", int(m.Alloc/1024), "sys", int(m.Sys/1024), "numGC", int(m.NumGC))
return nil
}
func (tds *TrieDbState) SetBlockNr(blockNr uint64) {
tds.blockNr = blockNr
tds.tp.SetBlockNr(blockNr)
}
func (tds *TrieDbState) UnwindTo(blockNr uint64) error {
tds.StartNewBuffer()
b := tds.currentBuffer
if err := tds.db.RewindData(tds.blockNr, blockNr, func(bucket, key, value []byte) error {
//fmt.Printf("bucket: %x, key: %x, value: %x\n", bucket, key, value)
if bytes.Equal(bucket, dbutils.AccountsHistoryBucket) {
var addrHash common.Hash
copy(addrHash[:], key)
if len(value) > 0 {
var acc accounts.Account
if err := acc.DecodeForStorage(value); err != nil {
return err
}
b.accountUpdates[addrHash] = &acc
if err := tds.db.Put(dbutils.AccountsBucket, addrHash[:], value); err != nil {
return err
}
} else {
b.accountUpdates[addrHash] = nil
if err := tds.db.Delete(dbutils.AccountsBucket, addrHash[:]); err != nil {
return err
}
}
} else if bytes.Equal(bucket, dbutils.StorageHistoryBucket) {
var addrHash common.Hash
copy(addrHash[:], key[:common.HashLength])
var keyHash common.Hash
copy(keyHash[:], key[common.HashLength+IncarnationLength:])
var addrHashWithVersion addressHashWithIncarnation
copy(addrHashWithVersion[:], key[:common.HashLength+IncarnationLength])
m, ok := b.storageUpdates[addrHashWithVersion]
if !ok {
m = make(map[common.Hash][]byte)
b.storageUpdates[addrHashWithVersion] = m
}
if len(value) > 0 {
m[keyHash] = AddExtraRLPLevel(value)
if err := tds.db.Put(dbutils.StorageBucket, key[:common.HashLength+IncarnationLength+common.HashLength], value); err != nil {
return err
}
} else {
m[keyHash] = nil
if err := tds.db.Delete(dbutils.StorageBucket, key[:common.HashLength+IncarnationLength+common.HashLength]); err != nil {
return err
}
}
}
return nil
}); err != nil {
return err
}
if _, err := tds.computeTrieRoots(false); err != nil {
return err
}
for i := tds.blockNr; i > blockNr; i-- {
if err := tds.db.DeleteTimestamp(i); err != nil {
return err
}
}
tds.clearUpdates()
tds.blockNr = blockNr
return nil
}
func (tds *TrieDbState) readAccountDataByHash(addrHash common.Hash) (*accounts.Account, error) {
acc, ok := tds.t.GetAccount(addrHash[:])
if ok {
return acc, nil
}
// Not present in the trie, try the database
var err error
var enc []byte
if tds.historical {
enc, err = tds.db.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, addrHash[:], tds.blockNr+1)
if err != nil {
enc = nil
}
} else {
enc, err = tds.db.Get(dbutils.AccountsBucket, addrHash[:])
if err != nil {
enc = nil
}
}
if len(enc) == 0 {
return nil, nil
}
var a accounts.Account
if err := a.DecodeForStorage(enc); err != nil {
return nil, err
}
return &a, nil
}
func (tds *TrieDbState) ReadAccountData(address common.Address) (*accounts.Account, error) {
addrHash, err := common.HashData(address[:])
if err != nil {
return nil, err
}
if tds.resolveReads {
if _, ok := tds.currentBuffer.accountUpdates[addrHash]; !ok {
tds.currentBuffer.accountReads[addrHash] = struct{}{}
}
}
return tds.readAccountDataByHash(addrHash)
}
func (tds *TrieDbState) savePreimage(save bool, hash, preimage []byte) error {
if !save {
return nil
}
// Following check is to minimise the overwriting the same value of preimage
// in the database, which would cause extra write churn
if p, _ := tds.db.Get(dbutils.PreimagePrefix, hash); p != nil {
return nil
}
return tds.db.Put(dbutils.PreimagePrefix, hash, preimage)
}
func (tds *TrieDbState) HashAddress(address common.Address, save bool) (common.Hash, error) {
addrHash, err := common.HashData(address[:])
if err != nil {
return common.Hash{}, err
}
return addrHash, tds.savePreimage(save, addrHash[:], address[:])
}
func (tds *TrieDbState) HashKey(key *common.Hash, save bool) (common.Hash, error) {
keyHash, err := common.HashData(key[:])
if err != nil {
return common.Hash{}, err
}
return keyHash, tds.savePreimage(save, keyHash[:], key[:])
}
func (tds *TrieDbState) GetKey(shaKey []byte) []byte {
key, _ := tds.db.Get(dbutils.PreimagePrefix, shaKey)
return key
}
func (tds *TrieDbState) ReadAccountStorage(address common.Address, incarnation uint64, key *common.Hash) ([]byte, error) {
if tds.currentBuffer != nil {
if _, ok := tds.currentBuffer.deleted[address]; ok {
return nil, nil
}
}
if tds.aggregateBuffer != nil {
if _, ok := tds.aggregateBuffer.deleted[address]; ok {
return nil, nil
}
}
seckey, err := tds.HashKey(key, false /*save*/)
if err != nil {
return nil, err
}
addrHash, err := tds.HashAddress(address, false /*save*/)
if err != nil {
return nil, err
}
if tds.resolveReads {
var addReadRecord = false
if mWrite, ok := tds.currentBuffer.storageUpdates[newAddressHashWithIncarnation(addrHash, incarnation)]; ok {
if _, ok1 := mWrite[seckey]; !ok1 {
addReadRecord = true
}
} else {
addReadRecord = true
}
if addReadRecord {
m, ok := tds.currentBuffer.storageReads[newAddressHashWithIncarnation(addrHash, incarnation)]
if !ok {
m = make(map[common.Hash]struct{})
tds.currentBuffer.storageReads[newAddressHashWithIncarnation(addrHash, incarnation)] = m
}
m[seckey] = struct{}{}
}
}
enc, ok := tds.t.Get(dbutils.GenerateCompositeTrieKey(addrHash, seckey))
if ok {
// Unwrap one RLP level
if len(enc) > 1 {
enc = enc[1:]
}
} else {
// Not present in the trie, try database
if tds.historical {
enc, err = tds.db.GetAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, dbutils.GenerateCompositeStorageKey(addrHash, incarnation, seckey), tds.blockNr)
if err != nil {
enc = nil
}
} else {
enc, err = tds.db.Get(dbutils.StorageBucket, dbutils.GenerateCompositeStorageKey(addrHash, incarnation, seckey))
if err != nil {
enc = nil
}
}
}
return enc, nil
}
func (tds *TrieDbState) ReadAccountCode(codeHash common.Hash) (code []byte, err error) {
if bytes.Equal(codeHash[:], emptyCodeHash) {
return nil, nil
}
if cached, ok := tds.codeCache.Get(codeHash); ok {
code, err = cached.([]byte), nil
} else {
code, err = tds.db.Get(dbutils.CodeBucket, codeHash[:])
if err == nil {
tds.codeSizeCache.Add(codeHash, len(code))
tds.codeCache.Add(codeHash, code)
}
}
if tds.resolveReads {
tds.pg.ReadCode(codeHash, code)
}
return code, err
}
func (tds *TrieDbState) ReadAccountCodeSize(codeHash common.Hash) (codeSize int, err error) {
var code []byte
if cached, ok := tds.codeSizeCache.Get(codeHash); ok {
codeSize, err = cached.(int), nil
if tds.resolveReads {
if cachedCode, ok := tds.codeCache.Get(codeHash); ok {
code, err = cachedCode.([]byte), nil
} else {
code, err = tds.ReadAccountCode(codeHash)
if err != nil {
return 0, err
}
}
}
} else {
code, err = tds.ReadAccountCode(codeHash)
if err != nil {
return 0, err
}
codeSize = len(code)
}
if tds.resolveReads {
tds.pg.ReadCode(codeHash, code)
}
return codeSize, nil
}
// nextIncarnation determines what should be the next incarnation of an account (i.e. how many time it has existed before at this address)
func (tds *TrieDbState) nextIncarnation(address common.Address) (uint64, error) {
addrHash, err := tds.HashAddress(address, false /*save*/)
if err != nil {
return 0, err
}
var found bool
var incarnationBytes [IncarnationLength]byte
if tds.historical {
// We reserve ethdb.MaxTimestampLength (8) at the end of the key to accomodate any possible timestamp
// (timestamp's encoding may have variable length)
startkey := make([]byte, common.HashLength+IncarnationLength+common.HashLength+ethdb.MaxTimestampLength)
var fixedbits uint = 8 * common.HashLength
copy(startkey, addrHash[:])
err = tds.db.WalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, startkey, fixedbits, tds.blockNr, func(k, _ []byte) (bool, error) {
copy(incarnationBytes[:], k[common.HashLength:])
found = true
return false, nil
})
if err != nil {
return 0, err
}
} else {
startkey := make([]byte, common.HashLength+IncarnationLength+common.HashLength)
var fixedbits uint = 8 * common.HashLength
copy(startkey, addrHash[:])
err = tds.db.Walk(dbutils.StorageBucket, startkey, fixedbits, func(k, v []byte) (bool, error) {
copy(incarnationBytes[:], k[common.HashLength:])
found = true
return false, nil
})
if err != nil {
return 0, err
}
}
if found {
return (^uint64(0) ^ binary.BigEndian.Uint64(incarnationBytes[:])) + 1, nil
}
return 0, nil
}
var prevMemStats runtime.MemStats
type TrieStateWriter struct {
tds *TrieDbState
}
func (tds *TrieDbState) PruneTries(print bool) {
if print {
prunableNodes := tds.t.CountPrunableNodes()
fmt.Printf("[Before] Actual prunable nodes: %d, accounted: %d\n", prunableNodes, tds.tp.NodeCount())
}
tds.tp.PruneTo(tds.t, int(MaxTrieCacheGen))
if print {
prunableNodes := tds.t.CountPrunableNodes()
fmt.Printf("[After] Actual prunable nodes: %d, accounted: %d\n", prunableNodes, tds.tp.NodeCount())
}
var m runtime.MemStats
runtime.ReadMemStats(&m)
log.Info("Memory", "nodes", tds.tp.NodeCount(), "alloc", int(m.Alloc/1024), "sys", int(m.Sys/1024), "numGC", int(m.NumGC))
if print {
fmt.Printf("Pruning done. Nodes: %d, alloc: %d, sys: %d, numGC: %d\n", tds.tp.NodeCount(), int(m.Alloc/1024), int(m.Sys/1024), int(m.NumGC))
}
}
type DbStateWriter struct {
tds *TrieDbState
}
func (tds *TrieDbState) TrieStateWriter() *TrieStateWriter {
return &TrieStateWriter{tds: tds}
}
func (tds *TrieDbState) DbStateWriter() *DbStateWriter {
return &DbStateWriter{tds: tds}
}
func accountsEqual(a1, a2 *accounts.Account) bool {
if a1.Nonce != a2.Nonce {
return false
}
if !a1.Initialised {
if a2.Initialised {
return false
}
} else if !a2.Initialised {
return false
} else if a1.Balance.Cmp(&a2.Balance) != 0 {
return false
}
if a1.Root != a2.Root {
return false
}
if a1.CodeHash == (common.Hash{}) {
if a2.CodeHash != (common.Hash{}) {
return false
}
} else if a2.CodeHash == (common.Hash{}) {
return false
} else if a1.CodeHash != a2.CodeHash {
return false
}
return true
}
func (tsw *TrieStateWriter) UpdateAccountData(_ context.Context, address common.Address, original, account *accounts.Account) error {
addrHash, err := tsw.tds.HashAddress(address, false /*save*/)
if err != nil {
return err
}
tsw.tds.currentBuffer.accountUpdates[addrHash] = account
addrHashWithInc := newAddressHashWithIncarnation(addrHash, account.GetIncarnation())
if _, ok := tsw.tds.currentBuffer.storageUpdates[addrHashWithInc]; !ok && account.GetIncarnation() > 0 {
tsw.tds.currentBuffer.storageUpdates[addrHashWithInc] = map[common.Hash][]byte{}
}
return nil
}
func (dsw *DbStateWriter) UpdateAccountData(ctx context.Context, address common.Address, original, account *accounts.Account) error {
dataLen := account.EncodingLengthForStorage()
data := make([]byte, dataLen)
account.EncodeForStorage(data)
addrHash, err := dsw.tds.HashAddress(address, true /*save*/)
if err != nil {
return err
}
if err = dsw.tds.db.Put(dbutils.AccountsBucket, addrHash[:], data); err != nil {
return err
}
_, noHistory := params.GetNoHistory(ctx)
if dsw.tds.noHistory || noHistory {
return nil
}
// Don't write historical record if the account did not change
if accountsEqual(original, account) {
return nil
}
var originalData []byte
if !original.Initialised {
originalData = []byte{}
} else {
originalDataLen := original.EncodingLengthForStorage()
originalData = make([]byte, originalDataLen)
original.EncodeForStorage(originalData)
}
return dsw.tds.db.PutS(dbutils.AccountsHistoryBucket, addrHash[:], originalData, dsw.tds.blockNr)
}
func (tsw *TrieStateWriter) DeleteAccount(_ context.Context, address common.Address, original *accounts.Account) error {
addrHash, err := tsw.tds.HashAddress(address, false /*save*/)
if err != err {
return err
}
tsw.tds.currentBuffer.accountUpdates[addrHash] = nil
tsw.tds.currentBuffer.deleted[address] = struct{}{}
return nil
}
func (dsw *DbStateWriter) DeleteAccount(ctx context.Context, address common.Address, original *accounts.Account) error {
addrHash, err := dsw.tds.HashAddress(address, true /*save*/)
if err != nil {
return err
}
if err := dsw.tds.db.Delete(dbutils.AccountsBucket, addrHash[:]); err != nil {
return err
}
_, noHistory := params.GetNoHistory(ctx)
if dsw.tds.noHistory || noHistory {
return nil
}
var originalData []byte
if !original.Initialised {
// Account has been created and deleted in the same block
originalData = []byte{}
} else {
originalDataLen := original.EncodingLengthForStorage()
originalData = make([]byte, originalDataLen)
original.EncodeForStorage(originalData)
}
return dsw.tds.db.PutS(dbutils.AccountsHistoryBucket, addrHash[:], originalData, dsw.tds.blockNr)
}
func (tsw *TrieStateWriter) UpdateAccountCode(codeHash common.Hash, code []byte) error {
if tsw.tds.resolveReads {
tsw.tds.pg.CreateCode(codeHash, code)
}
return nil
}
func (dsw *DbStateWriter) UpdateAccountCode(codeHash common.Hash, code []byte) error {
if dsw.tds.resolveReads {
dsw.tds.pg.CreateCode(codeHash, code)
}
return dsw.tds.db.Put(dbutils.CodeBucket, codeHash[:], code)
}
func (tsw *TrieStateWriter) WriteAccountStorage(_ context.Context, address common.Address, incarnation uint64, key, original, value *common.Hash) error {
addrHash, err := tsw.tds.HashAddress(address, false /*save*/)
if err != nil {
return err
}
v := bytes.TrimLeft(value[:], "\x00")
m, ok := tsw.tds.currentBuffer.storageUpdates[newAddressHashWithIncarnation(addrHash, incarnation)]
if !ok {
m = make(map[common.Hash][]byte)
tsw.tds.currentBuffer.storageUpdates[newAddressHashWithIncarnation(addrHash, incarnation)] = m
}
seckey, err := tsw.tds.HashKey(key, false /*save*/)
if err != nil {
return err
}
if len(v) > 0 {
// Write into 1 extra RLP level
m[seckey] = AddExtraRLPLevel(v)
} else {
m[seckey] = nil
}
//fmt.Printf("WriteAccountStorage %x %x: %x, buffer %d\n", addrHash, seckey, value, len(tsw.tds.buffers))
return nil
}
func (dsw *DbStateWriter) WriteAccountStorage(ctx context.Context, address common.Address, incarnation uint64, key, original, value *common.Hash) error {
if *original == *value {
return nil
}
seckey, err := dsw.tds.HashKey(key, true /*save*/)
if err != nil {
return err
}
v := bytes.TrimLeft(value[:], "\x00")
vv := make([]byte, len(v))
copy(vv, v)
addrHash, err := dsw.tds.HashAddress(address, false /*save*/)
if err != nil {
return err
}
compositeKey := dbutils.GenerateCompositeStorageKey(addrHash, incarnation, seckey)
if len(v) == 0 {
err = dsw.tds.db.Delete(dbutils.StorageBucket, compositeKey)
} else {
err = dsw.tds.db.Put(dbutils.StorageBucket, compositeKey, vv)
}
//fmt.Printf("WriteAccountStorage (db) %x %d %x: %x\n", address, incarnation, key, value)
if err != nil {
return err
}
_, noHistory := params.GetNoHistory(ctx)
if dsw.tds.noHistory || noHistory {
return nil
}
o := bytes.TrimLeft(original[:], "\x00")
oo := make([]byte, len(o))
copy(oo, o)
return dsw.tds.db.PutS(dbutils.StorageHistoryBucket, compositeKey, oo, dsw.tds.blockNr)
}
func (tds *TrieDbState) ExtractProofs(trace bool) trie.BlockProof {
return tds.pg.ExtractProofs(trace)
}
func (tsw *TrieStateWriter) CreateContract(address common.Address) error {
tsw.tds.currentBuffer.created[address] = struct{}{}
return nil
}
func (dsw *DbStateWriter) CreateContract(address common.Address) error {
return nil
}
func (tds *TrieDbState) TriePruningDebugDump() string {
return tds.tp.DebugDump()
}