From 6c346a984571af04e8c9d68154b178582a117502 Mon Sep 17 00:00:00 2001 From: Karl Bartel Date: Tue, 13 Feb 2024 15:01:08 +0100 Subject: [PATCH] Use ReadOnlyStateDB in CeloBackend This prevents modification of the StateDB, which is necessary now that we don't copy the StateDB anymore. --- core/celo_backend.go | 3 +- core/celo_read_only_statedb.go | 144 +++++++++++++++++++++++++++++++++ 2 files changed, 146 insertions(+), 1 deletion(-) create mode 100644 core/celo_read_only_statedb.go diff --git a/core/celo_backend.go b/core/celo_backend.go index 370b814d65..2ed050a0f5 100644 --- a/core/celo_backend.go +++ b/core/celo_backend.go @@ -43,7 +43,8 @@ func (b *CeloBackend) CallContract(ctx context.Context, call ethereum.CallMsg, b txCtx := vm.TxContext{} vmConfig := vm.Config{} - evm := vm.NewEVM(blockCtx, txCtx, b.state, b.chainConfig, vmConfig) + readOnlyStateDB := ReadOnlyStateDB{StateDB: b.state} + evm := vm.NewEVM(blockCtx, txCtx, &readOnlyStateDB, b.chainConfig, vmConfig) ret, _, err := evm.StaticCall(vm.AccountRef(evm.Origin), *call.To, call.Data, call.Gas) return ret, err diff --git a/core/celo_read_only_statedb.go b/core/celo_read_only_statedb.go new file mode 100644 index 0000000000..99fe3832ec --- /dev/null +++ b/core/celo_read_only_statedb.go @@ -0,0 +1,144 @@ +package core + +import ( + "math/big" + + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/core/types" + "github.com/ethereum/go-ethereum/core/vm" + "github.com/ethereum/go-ethereum/params" +) + +// ReadOnlyStateDB wraps a StateDB to prevent modifications. Using it without copying it is safe. +// +// Only the following methods will be passed through to the original StateDB: +// +// GetState(common.Address, common.Hash) common.Hash +// GetCodeHash(common.Address) common.Hash +// GetCode(common.Address) []byte +// +// Gas calculations based on ReadOnlyStateDB will be wrong because the accessed storage slots and addresses are not tracked. +type ReadOnlyStateDB struct { + vm.StateDB +} + +func (r *ReadOnlyStateDB) CreateAccount(common.Address) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SubBalance(common.Address, *big.Int) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) AddBalance(_ common.Address, amount *big.Int) { + if amount.Cmp(common.Big0) == 0 { + // Adding zero is safe, so we can return here + return + } + panic("not implemented") +} + +func (r *ReadOnlyStateDB) GetBalance(common.Address) *big.Int { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) GetNonce(common.Address) uint64 { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SetNonce(common.Address, uint64) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SetCode(common.Address, []byte) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) GetCodeSize(common.Address) int { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) AddRefund(uint64) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SubRefund(uint64) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) GetRefund() uint64 { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) GetCommittedState(common.Address, common.Hash) common.Hash { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SetState(common.Address, common.Hash, common.Hash) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) GetTransientState(addr common.Address, key common.Hash) common.Hash { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SetTransientState(addr common.Address, key, value common.Hash) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) SelfDestruct(common.Address) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) HasSelfDestructed(common.Address) bool { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) Selfdestruct6780(common.Address) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) Exist(common.Address) bool { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) Empty(common.Address) bool { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) AddressInAccessList(addr common.Address) bool { + // We don't track access lists + return false +} + +func (r *ReadOnlyStateDB) SlotInAccessList(addr common.Address, slot common.Hash) (addressOk bool, slotOk bool) { + // We don't track access lists + return false, false +} + +func (r *ReadOnlyStateDB) AddAddressToAccessList(addr common.Address) { +} + +func (r *ReadOnlyStateDB) AddSlotToAccessList(addr common.Address, slot common.Hash) { +} + +func (r *ReadOnlyStateDB) Prepare(rules params.Rules, sender, coinbase common.Address, dest *common.Address, precompiles []common.Address, txAccesses types.AccessList) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) RevertToSnapshot(int) { + // No changes can be done, so reverting is a noop. +} + +func (r *ReadOnlyStateDB) Snapshot() int { + // We use id 0 for the single state this immutable StateDB can have. + return 0 +} + +func (r *ReadOnlyStateDB) AddLog(*types.Log) { + panic("not implemented") +} + +func (r *ReadOnlyStateDB) AddPreimage(common.Hash, []byte) { + panic("not implemented") +}