mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-18 08:38:46 +00:00
92 lines
2.9 KiB
Go
92 lines
2.9 KiB
Go
package state
|
|
|
|
import (
|
|
"bytes"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types/accounts"
|
|
"github.com/ledgerwatch/erigon/turbo/shards"
|
|
)
|
|
|
|
// CachedReader is a wrapper for an instance of type StateReader
|
|
// This wrapper only makes calls to the underlying reader if the item is not in the cache
|
|
type CachedReader struct {
|
|
r StateReader
|
|
cache *shards.StateCache
|
|
}
|
|
|
|
// NewCachedReader wraps a given state reader into the cached reader
|
|
func NewCachedReader(r StateReader, cache *shards.StateCache) *CachedReader {
|
|
return &CachedReader{r: r, cache: cache}
|
|
}
|
|
|
|
// ReadAccountData is called when an account needs to be fetched from the state
|
|
func (cr *CachedReader) ReadAccountData(address common.Address) (*accounts.Account, error) {
|
|
addrBytes := address.Bytes()
|
|
if a, ok := cr.cache.GetAccount(addrBytes); ok {
|
|
return a, nil
|
|
}
|
|
a, err := cr.r.ReadAccountData(address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if a == nil {
|
|
cr.cache.SetAccountAbsent(addrBytes)
|
|
} else {
|
|
cr.cache.SetAccountRead(addrBytes, a)
|
|
}
|
|
return a, nil
|
|
}
|
|
|
|
// ReadAccountStorage is called when a storage item needs to be fetched from the state
|
|
func (cr *CachedReader) ReadAccountStorage(address common.Address, incarnation uint64, key *common.Hash) ([]byte, error) {
|
|
addrBytes := address.Bytes()
|
|
if s, ok := cr.cache.GetStorage(addrBytes, incarnation, key.Bytes()); ok {
|
|
return s, nil
|
|
}
|
|
v, err := cr.r.ReadAccountStorage(address, incarnation, key)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(v) == 0 {
|
|
cr.cache.SetStorageAbsent(addrBytes, incarnation, key.Bytes())
|
|
} else {
|
|
cr.cache.SetStorageRead(addrBytes, incarnation, key.Bytes(), v)
|
|
}
|
|
return v, nil
|
|
}
|
|
|
|
// ReadAccountCode is called when code of an account needs to be fetched from the state
|
|
// Usually, one of (address;incarnation) or codeHash is enough to uniquely identify the code
|
|
func (cr *CachedReader) ReadAccountCode(address common.Address, incarnation uint64, codeHash common.Hash) ([]byte, error) {
|
|
if bytes.Equal(codeHash[:], emptyCodeHash) {
|
|
return nil, nil
|
|
}
|
|
if c, ok := cr.cache.GetCode(address.Bytes(), incarnation); ok {
|
|
return c, nil
|
|
}
|
|
c, err := cr.r.ReadAccountCode(address, incarnation, codeHash)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if cr.cache != nil && len(c) <= 1024 {
|
|
cr.cache.SetCodeRead(address.Bytes(), incarnation, c)
|
|
}
|
|
return c, nil
|
|
}
|
|
|
|
func (cr *CachedReader) ReadAccountCodeSize(address common.Address, incarnation uint64, codeHash common.Hash) (int, error) {
|
|
c, err := cr.ReadAccountCode(address, incarnation, codeHash)
|
|
return len(c), err
|
|
}
|
|
|
|
// ReadAccountIncarnation is called when incarnation of the account is required (to create and recreate contract)
|
|
func (cr *CachedReader) ReadAccountIncarnation(address common.Address) (uint64, error) {
|
|
deleted := cr.cache.GetDeletedAccount(address.Bytes())
|
|
if deleted != nil {
|
|
return deleted.Incarnation, nil
|
|
}
|
|
return cr.r.ReadAccountIncarnation(address)
|
|
}
|