erigon-pulse/core/state/plain_state_writer.go
Alex Sharov b3f1915d09
ChangeSets dupsort (#1342)
* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* squash

* squash

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* history_early_stop

* history_early_stop

* vmConfig with ReadOnly false

* auto_increment

* auto_increment

* rebase master

Co-authored-by: Alexey Akhunov <akhounov@gmail.com>
2020-11-16 12:08:28 +00:00

205 lines
5.6 KiB
Go

package state
import (
"context"
"encoding/binary"
"github.com/VictoriaMetrics/fastcache"
"github.com/holiman/uint256"
"github.com/ledgerwatch/turbo-geth/common/changeset"
"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"
)
var _ WriterWithChangeSets = (*PlainStateWriter)(nil)
type PlainStateWriter struct {
db ethdb.Database
changeSetsDB ethdb.Database
csw *ChangeSetWriter
blockNumber uint64
accountCache *fastcache.Cache
storageCache *fastcache.Cache
codeCache *fastcache.Cache
codeSizeCache *fastcache.Cache
}
func NewPlainStateWriter(db ethdb.Database, changeSetsDB ethdb.Database, blockNumber uint64) *PlainStateWriter {
return &PlainStateWriter{
db: db,
changeSetsDB: changeSetsDB,
csw: NewChangeSetWriterPlain(blockNumber),
blockNumber: blockNumber,
}
}
func (w *PlainStateWriter) SetAccountCache(accountCache *fastcache.Cache) {
w.accountCache = accountCache
}
func (w *PlainStateWriter) SetStorageCache(storageCache *fastcache.Cache) {
w.storageCache = storageCache
}
func (w *PlainStateWriter) SetCodeCache(codeCache *fastcache.Cache) {
w.codeCache = codeCache
}
func (w *PlainStateWriter) SetCodeSizeCache(codeSizeCache *fastcache.Cache) {
w.codeSizeCache = codeSizeCache
}
func (w *PlainStateWriter) UpdateAccountData(ctx context.Context, address common.Address, original, account *accounts.Account) error {
if err := w.csw.UpdateAccountData(ctx, address, original, account); err != nil {
return err
}
value := make([]byte, account.EncodingLengthForStorage())
account.EncodeForStorage(value)
if w.accountCache != nil {
w.accountCache.Set(address[:], value)
}
return w.db.Put(dbutils.PlainStateBucket, address[:], value)
}
func (w *PlainStateWriter) UpdateAccountCode(address common.Address, incarnation uint64, codeHash common.Hash, code []byte) error {
if err := w.csw.UpdateAccountCode(address, incarnation, codeHash, code); err != nil {
return err
}
if w.codeCache != nil {
if len(code) <= 1024 {
w.codeCache.Set(address[:], code)
} else {
w.codeCache.Del(address[:])
}
}
if w.codeSizeCache != nil {
var b [4]byte
binary.BigEndian.PutUint32(b[:], uint32(len(code)))
w.codeSizeCache.Set(address[:], b[:])
}
if err := w.db.Put(dbutils.CodeBucket, codeHash[:], code); err != nil {
return err
}
return w.db.Put(dbutils.PlainContractCodeBucket, dbutils.PlainGenerateStoragePrefix(address[:], incarnation), codeHash[:])
}
func (w *PlainStateWriter) DeleteAccount(ctx context.Context, address common.Address, original *accounts.Account) error {
if err := w.csw.DeleteAccount(ctx, address, original); err != nil {
return err
}
if w.accountCache != nil {
w.accountCache.Set(address[:], nil)
}
if w.codeCache != nil {
w.codeCache.Set(address[:], nil)
}
if w.codeSizeCache != nil {
var b [4]byte
binary.BigEndian.PutUint32(b[:], 0)
w.codeSizeCache.Set(address[:], b[:])
}
if err := w.db.Delete(dbutils.PlainStateBucket, address[:], nil); err != nil {
return err
}
if original.Incarnation > 0 {
var b [8]byte
binary.BigEndian.PutUint64(b[:], original.Incarnation)
if err := w.db.Put(dbutils.IncarnationMapBucket, address[:], b[:]); err != nil {
return err
}
}
return nil
}
func (w *PlainStateWriter) WriteAccountStorage(ctx context.Context, address common.Address, incarnation uint64, key *common.Hash, original, value *uint256.Int) error {
if err := w.csw.WriteAccountStorage(ctx, address, incarnation, key, original, value); err != nil {
return err
}
if *original == *value {
return nil
}
compositeKey := dbutils.PlainGenerateCompositeStorageKey(address, incarnation, *key)
v := value.Bytes()
if w.storageCache != nil {
w.storageCache.Set(compositeKey, v)
}
if len(v) == 0 {
return w.db.Delete(dbutils.PlainStateBucket, compositeKey, nil)
}
return w.db.Put(dbutils.PlainStateBucket, compositeKey, v)
}
func (w *PlainStateWriter) CreateContract(address common.Address) error {
if err := w.csw.CreateContract(address); err != nil {
return err
}
if err := w.db.Delete(dbutils.IncarnationMapBucket, address[:], nil); err != nil {
return err
}
return nil
}
func (w *PlainStateWriter) WriteChangeSets() error {
db := w.db
if w.changeSetsDB != nil {
db = w.changeSetsDB
}
accountChanges, err := w.csw.GetAccountChanges()
if err != nil {
return err
}
if err = changeset.Mapper[dbutils.PlainAccountChangeSetBucket].Encode(w.blockNumber, accountChanges, func(k, v []byte) error {
return db.Append(dbutils.PlainAccountChangeSetBucket, k, v)
}); err != nil {
return err
}
storageChanges, err := w.csw.GetStorageChanges()
if err != nil {
return err
}
if storageChanges.Len() == 0 {
return nil
}
if err = changeset.Mapper[dbutils.PlainStorageChangeSetBucket].Encode(w.blockNumber, storageChanges, func(k, v []byte) error {
return db.Append(dbutils.PlainStorageChangeSetBucket, k, v)
}); err != nil {
return err
}
return nil
}
func (w *PlainStateWriter) WriteHistory() error {
db := w.db
if w.changeSetsDB != nil {
db = w.changeSetsDB
}
accountChanges, err := w.csw.GetAccountChanges()
if err != nil {
return err
}
err = writeIndex(w.blockNumber, accountChanges, dbutils.AccountsHistoryBucket, db)
if err != nil {
return err
}
storageChanges, err := w.csw.GetStorageChanges()
if err != nil {
return err
}
err = writeIndex(w.blockNumber, storageChanges, dbutils.StorageHistoryBucket, db)
if err != nil {
return err
}
return nil
}
func (w *PlainStateWriter) ChangeSetWriter() *ChangeSetWriter {
return w.csw
}