erigon-pulse/core/state/plain_state_writer.go

150 lines
3.9 KiB
Go

package state
import (
"encoding/binary"
"github.com/holiman/uint256"
libcommon "github.com/ledgerwatch/erigon-lib/common"
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon/common/dbutils"
"github.com/ledgerwatch/erigon/core/types/accounts"
"github.com/ledgerwatch/erigon/turbo/shards"
)
var _ WriterWithChangeSets = (*PlainStateWriter)(nil)
type putDel interface {
kv.Putter
kv.Deleter
IncrementSequence(bucket string, amount uint64) (uint64, error)
}
type PlainStateWriter struct {
db putDel
csw *ChangeSetWriter
accumulator *shards.Accumulator
}
func NewPlainStateWriter(db putDel, changeSetsDB kv.RwTx, blockNumber uint64) *PlainStateWriter {
return &PlainStateWriter{
db: db,
csw: NewChangeSetWriterPlain(changeSetsDB, blockNumber),
}
}
func NewPlainStateWriterNoHistory(db putDel) *PlainStateWriter {
return &PlainStateWriter{
db: db,
}
}
func (w *PlainStateWriter) SetAccumulator(accumulator *shards.Accumulator) *PlainStateWriter {
w.accumulator = accumulator
return w
}
func (w *PlainStateWriter) UpdateAccountData(address libcommon.Address, original, account *accounts.Account) error {
//fmt.Printf("balance,%x,%d\n", address, &account.Balance)
if w.csw != nil {
if err := w.csw.UpdateAccountData(address, original, account); err != nil {
return err
}
}
value := make([]byte, account.EncodingLengthForStorage())
account.EncodeForStorage(value)
if w.accumulator != nil {
w.accumulator.ChangeAccount(address, account.Incarnation, value)
}
return w.db.Put(kv.PlainState, address[:], value)
}
func (w *PlainStateWriter) UpdateAccountCode(address libcommon.Address, incarnation uint64, codeHash libcommon.Hash, code []byte) error {
//fmt.Printf("code,%x,%x\n", address, code)
if w.csw != nil {
if err := w.csw.UpdateAccountCode(address, incarnation, codeHash, code); err != nil {
return err
}
}
if w.accumulator != nil {
w.accumulator.ChangeCode(address, incarnation, code)
}
if err := w.db.Put(kv.Code, codeHash[:], code); err != nil {
return err
}
return w.db.Put(kv.PlainContractCode, dbutils.PlainGenerateStoragePrefix(address[:], incarnation), codeHash[:])
}
func (w *PlainStateWriter) DeleteAccount(address libcommon.Address, original *accounts.Account) error {
//fmt.Printf("delete,%x\n", address)
if w.csw != nil {
if err := w.csw.DeleteAccount(address, original); err != nil {
return err
}
}
if w.accumulator != nil {
w.accumulator.DeleteAccount(address)
}
if err := w.db.Delete(kv.PlainState, address[:]); err != nil {
return err
}
if original.Incarnation > 0 {
var b [8]byte
binary.BigEndian.PutUint64(b[:], original.Incarnation)
if err := w.db.Put(kv.IncarnationMap, address[:], b[:]); err != nil {
return err
}
}
return nil
}
func (w *PlainStateWriter) WriteAccountStorage(address libcommon.Address, incarnation uint64, key *libcommon.Hash, original, value *uint256.Int) error {
//fmt.Printf("storage,%x,%x,%x\n", address, *key, value.Bytes())
if w.csw != nil {
if err := w.csw.WriteAccountStorage(address, incarnation, key, original, value); err != nil {
return err
}
}
if *original == *value {
return nil
}
compositeKey := dbutils.PlainGenerateCompositeStorageKey(address.Bytes(), incarnation, key.Bytes())
v := value.Bytes()
if w.accumulator != nil {
w.accumulator.ChangeStorage(address, incarnation, *key, v)
}
if len(v) == 0 {
return w.db.Delete(kv.PlainState, compositeKey)
}
return w.db.Put(kv.PlainState, compositeKey, v)
}
func (w *PlainStateWriter) CreateContract(address libcommon.Address) error {
if w.csw != nil {
if err := w.csw.CreateContract(address); err != nil {
return err
}
}
return nil
}
func (w *PlainStateWriter) WriteChangeSets() error {
if w.csw != nil {
return w.csw.WriteChangeSets()
}
return nil
}
func (w *PlainStateWriter) WriteHistory() error {
if w.csw != nil {
return w.csw.WriteHistory()
}
return nil
}
func (w *PlainStateWriter) ChangeSetWriter() *ChangeSetWriter {
return w.csw
}