2020-04-26 21:58:26 +00:00
|
|
|
package state
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-05-22 15:11:00 +00:00
|
|
|
"encoding/binary"
|
2020-04-26 21:58:26 +00:00
|
|
|
|
2020-05-22 15:11:00 +00:00
|
|
|
"github.com/VictoriaMetrics/fastcache"
|
2023-01-13 18:12:18 +00:00
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
2021-07-29 11:53:13 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2023-01-13 18:12:18 +00:00
|
|
|
|
2021-05-20 18:25:53 +00:00
|
|
|
"github.com/ledgerwatch/erigon/common"
|
|
|
|
"github.com/ledgerwatch/erigon/common/dbutils"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types/accounts"
|
2020-04-26 21:58:26 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Implements StateReader by wrapping database only, without trie
|
|
|
|
type DbStateReader struct {
|
2021-07-28 02:47:38 +00:00
|
|
|
db kv.Getter
|
2020-07-10 06:03:18 +00:00
|
|
|
accountCache *fastcache.Cache
|
|
|
|
storageCache *fastcache.Cache
|
|
|
|
codeCache *fastcache.Cache
|
|
|
|
codeSizeCache *fastcache.Cache
|
2020-04-26 21:58:26 +00:00
|
|
|
}
|
|
|
|
|
2021-07-28 02:47:38 +00:00
|
|
|
func NewDbStateReader(db kv.Getter) *DbStateReader {
|
2020-04-26 21:58:26 +00:00
|
|
|
return &DbStateReader{
|
2020-07-10 06:03:18 +00:00
|
|
|
db: db,
|
2020-04-26 21:58:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-22 15:11:00 +00:00
|
|
|
func (dbr *DbStateReader) SetAccountCache(accountCache *fastcache.Cache) {
|
2020-05-21 12:27:52 +00:00
|
|
|
dbr.accountCache = accountCache
|
|
|
|
}
|
|
|
|
|
2020-05-22 15:11:00 +00:00
|
|
|
func (dbr *DbStateReader) SetStorageCache(storageCache *fastcache.Cache) {
|
2020-05-21 12:27:52 +00:00
|
|
|
dbr.storageCache = storageCache
|
|
|
|
}
|
|
|
|
|
2020-05-22 15:11:00 +00:00
|
|
|
func (dbr *DbStateReader) SetCodeCache(codeCache *fastcache.Cache) {
|
2020-05-21 12:27:52 +00:00
|
|
|
dbr.codeCache = codeCache
|
|
|
|
}
|
|
|
|
|
2020-05-22 15:11:00 +00:00
|
|
|
func (dbr *DbStateReader) SetCodeSizeCache(codeSizeCache *fastcache.Cache) {
|
2020-05-21 12:27:52 +00:00
|
|
|
dbr.codeSizeCache = codeSizeCache
|
|
|
|
}
|
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
func (dbr *DbStateReader) ReadAccountData(address libcommon.Address) (*accounts.Account, error) {
|
2020-05-22 15:11:00 +00:00
|
|
|
var enc []byte
|
|
|
|
var ok bool
|
2020-05-21 12:27:52 +00:00
|
|
|
if dbr.accountCache != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
enc, ok = dbr.accountCache.HasGet(nil, address[:])
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
var err error
|
|
|
|
if addrHash, err1 := common.HashData(address[:]); err1 == nil {
|
2021-07-28 02:47:38 +00:00
|
|
|
enc, err = dbr.db.GetOne(kv.HashedAccounts, addrHash[:])
|
2020-05-22 15:11:00 +00:00
|
|
|
} else {
|
|
|
|
return nil, err1
|
|
|
|
}
|
2021-04-05 13:04:58 +00:00
|
|
|
if err != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
return nil, err
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
|
|
|
}
|
2020-05-22 15:11:00 +00:00
|
|
|
if !ok && dbr.accountCache != nil {
|
|
|
|
dbr.accountCache.Set(address[:], enc)
|
2020-04-26 21:58:26 +00:00
|
|
|
}
|
2020-05-22 15:11:00 +00:00
|
|
|
if enc == nil {
|
2020-04-26 21:58:26 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
2020-05-22 15:11:00 +00:00
|
|
|
acc := &accounts.Account{}
|
|
|
|
if err := acc.DecodeForStorage(enc); err != nil {
|
|
|
|
return nil, err
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
2020-05-22 15:11:00 +00:00
|
|
|
return acc, nil
|
2020-04-26 21:58:26 +00:00
|
|
|
}
|
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
func (dbr *DbStateReader) ReadAccountStorage(address libcommon.Address, incarnation uint64, key *libcommon.Hash) ([]byte, error) {
|
2020-04-26 21:58:26 +00:00
|
|
|
addrHash, err := common.HashData(address[:])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
seckey, err1 := common.HashData(key[:])
|
|
|
|
if err1 != nil {
|
|
|
|
return nil, err1
|
|
|
|
}
|
2020-05-22 15:11:00 +00:00
|
|
|
compositeKey := dbutils.GenerateCompositeStorageKey(addrHash, incarnation, seckey)
|
|
|
|
if dbr.storageCache != nil {
|
|
|
|
if enc, ok := dbr.storageCache.HasGet(nil, compositeKey); ok {
|
|
|
|
return enc, nil
|
|
|
|
}
|
|
|
|
}
|
2021-07-28 02:47:38 +00:00
|
|
|
enc, err2 := dbr.db.GetOne(kv.HashedStorage, compositeKey)
|
2021-07-24 07:14:11 +00:00
|
|
|
if err2 != nil {
|
2020-04-26 21:58:26 +00:00
|
|
|
return nil, err2
|
|
|
|
}
|
2020-05-21 12:27:52 +00:00
|
|
|
if dbr.storageCache != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
dbr.storageCache.Set(compositeKey, enc)
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
2020-04-26 21:58:26 +00:00
|
|
|
return enc, nil
|
|
|
|
}
|
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
func (dbr *DbStateReader) ReadAccountCode(address libcommon.Address, incarnation uint64, codeHash libcommon.Hash) ([]byte, error) {
|
2020-04-26 21:58:26 +00:00
|
|
|
if bytes.Equal(codeHash[:], emptyCodeHash) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2020-05-21 12:27:52 +00:00
|
|
|
if dbr.codeCache != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
if code, ok := dbr.codeCache.HasGet(nil, address[:]); ok {
|
|
|
|
return code, nil
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
|
|
|
}
|
2021-07-28 03:43:51 +00:00
|
|
|
code, err := dbr.db.GetOne(kv.Code, codeHash[:])
|
2021-07-24 07:14:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-21 20:55:39 +00:00
|
|
|
if dbr.codeCache != nil && len(code) <= 1024 {
|
2020-05-22 15:11:00 +00:00
|
|
|
dbr.codeCache.Set(address[:], code)
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
2020-05-21 20:55:39 +00:00
|
|
|
if dbr.codeSizeCache != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
var b [4]byte
|
|
|
|
binary.BigEndian.PutUint32(b[:], uint32(len(code)))
|
|
|
|
dbr.codeSizeCache.Set(address[:], b[:])
|
2020-05-21 20:55:39 +00:00
|
|
|
}
|
2020-04-26 21:58:26 +00:00
|
|
|
return code, err
|
|
|
|
}
|
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
func (dbr *DbStateReader) ReadAccountCodeSize(address libcommon.Address, incarnation uint64, codeHash libcommon.Hash) (codeSize int, err error) {
|
2020-04-26 21:58:26 +00:00
|
|
|
if bytes.Equal(codeHash[:], emptyCodeHash) {
|
|
|
|
return 0, nil
|
|
|
|
}
|
2020-05-21 12:27:52 +00:00
|
|
|
if dbr.codeSizeCache != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
if b, ok := dbr.codeSizeCache.HasGet(nil, address[:]); ok {
|
|
|
|
return int(binary.BigEndian.Uint32(b)), nil
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
|
|
|
}
|
2020-04-26 21:58:26 +00:00
|
|
|
var code []byte
|
2021-07-28 03:43:51 +00:00
|
|
|
code, err = dbr.db.GetOne(kv.Code, codeHash[:])
|
2020-04-26 21:58:26 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-05-21 12:27:52 +00:00
|
|
|
if dbr.codeSizeCache != nil {
|
2020-05-22 15:11:00 +00:00
|
|
|
var b [4]byte
|
|
|
|
binary.BigEndian.PutUint32(b[:], uint32(len(code)))
|
|
|
|
dbr.codeSizeCache.Set(address[:], b[:])
|
2020-05-21 12:27:52 +00:00
|
|
|
}
|
2020-04-26 21:58:26 +00:00
|
|
|
return len(code), nil
|
|
|
|
}
|
2020-05-02 18:00:42 +00:00
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
func (dbr *DbStateReader) ReadAccountIncarnation(address libcommon.Address) (uint64, error) {
|
2021-07-28 02:47:38 +00:00
|
|
|
b, err := dbr.db.GetOne(kv.IncarnationMap, address[:])
|
2021-07-24 07:14:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
if b == nil {
|
2020-05-02 18:00:42 +00:00
|
|
|
return 0, err
|
|
|
|
}
|
2021-07-24 07:14:11 +00:00
|
|
|
return binary.BigEndian.Uint64(b), nil
|
2020-05-02 18:00:42 +00:00
|
|
|
}
|