From bc55574a1628a73f5378670248e995b8cd97f029 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Thu, 7 Dec 2023 16:36:09 +0000 Subject: [PATCH 01/28] use StateAtBlock and reference states when recreating --- arbitrum/apibackend.go | 52 ++++++++++++++++-------- arbitrum/recreatestate.go | 21 ---------- eth/api_backend.go | 27 ++++++------ eth/catalyst/api_test.go | 3 +- eth/state_accessor.go | 14 +++++-- graphql/graphql.go | 18 ++++---- internal/ethapi/api.go | 30 +++++++++----- internal/ethapi/api_test.go | 6 +-- internal/ethapi/backend.go | 8 +++- internal/ethapi/transaction_args_test.go | 8 ++-- les/api_backend.go | 19 +++++---- 11 files changed, 116 insertions(+), 90 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 4419efce6074..baa7238fee58 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -229,10 +229,11 @@ func (a *APIBackend) FeeHistory( // use the most recent average compute rate for all blocks // note: while we could query this value for each block, it'd be prohibitively expensive - state, _, err := a.StateAndHeaderByNumber(ctx, newestBlock) + state, _, release, err := a.StateAndHeaderByNumber(ctx, newestBlock) if err != nil { return common.Big0, nil, nil, nil, err } + defer release() speedLimit, err := core.GetArbOSSpeedLimitPerSecond(state) if err != nil { return common.Big0, nil, nil, nil, err @@ -433,40 +434,59 @@ func (a *APIBackend) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc. return nil, errors.New("invalid arguments; neither block nor hash specified") } -func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types.Header, err error) (*state.StateDB, *types.Header, error) { +func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types.Header, err error) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { if err != nil { - return nil, header, err + return nil, header, nil, err } if header == nil { - return nil, nil, errors.New("header not found") + return nil, nil, nil, errors.New("header not found") } if !a.BlockChain().Config().IsArbitrumNitro(header.Number) { - return nil, header, types.ErrUseFallback + return nil, header, nil, types.ErrUseFallback } bc := a.BlockChain() stateFor := func(header *types.Header) (*state.StateDB, error) { - return bc.StateAt(header.Root) + if header.Root != (common.Hash{}) { + // Try referencing the root, if it isn't in dirties cache then Reference will have no effect + bc.StateCache().TrieDB().Reference(header.Root, common.Hash{}) + } + state, err := bc.StateAt(header.Root) + return state, err } - state, lastHeader, err := FindLastAvailableState(ctx, bc, stateFor, header, nil, a.b.config.MaxRecreateStateDepth) + lastState, lastHeader, err := FindLastAvailableState(ctx, bc, stateFor, header, nil, a.b.config.MaxRecreateStateDepth) if err != nil { - return nil, nil, err + return nil, nil, nil, err + } + release := func() { + if lastHeader.Root != (common.Hash{}) { + bc.StateCache().TrieDB().Dereference(lastHeader.Root) + } } if lastHeader == header { - return state, header, nil + return lastState, header, release, nil } - state, err = AdvanceStateUpToBlock(ctx, bc, state, header, lastHeader, nil) - if err != nil { - return nil, nil, err + defer release() + targetBlock := bc.GetBlockByNumber(header.Number.Uint64()) + if targetBlock == nil { + return nil, nil, nil, errors.New("target block not found") + } + lastBlock := bc.GetBlockByNumber(lastHeader.Number.Uint64()) + if lastBlock == nil { + return nil, nil, nil, errors.New("last block not found") } - return state, header, err + reexec := uint64(0) + checkLive := false + preferDisk := true + state, release, err := eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, targetBlock, reexec, lastState, lastBlock, checkLive, preferDisk) + return state, header, release, err } -func (a *APIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { +func (a *APIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { header, err := a.HeaderByNumber(ctx, number) return a.stateAndHeaderFromHeader(ctx, header, err) } -func (a *APIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { +func (a *APIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { header, err := a.HeaderByNumberOrHash(ctx, blockNrOrHash) return a.stateAndHeaderFromHeader(ctx, header, err) } @@ -476,7 +496,7 @@ func (a *APIBackend) StateAtBlock(ctx context.Context, block *types.Block, reexe return nil, nil, types.ErrUseFallback } // DEV: This assumes that `StateAtBlock` only accesses the blockchain and chainDb fields - return eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, block, reexec, base, checkLive, preferDisk) + return eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, block, reexec, base, nil, checkLive, preferDisk) } func (a *APIBackend) StateAtTransaction(ctx context.Context, block *types.Block, txIndex int, reexec uint64) (*core.Message, vm.BlockContext, *state.StateDB, tracers.StateReleaseFunc, error) { diff --git a/arbitrum/recreatestate.go b/arbitrum/recreatestate.go index 1b2dbb1a1816..e44262cdca83 100644 --- a/arbitrum/recreatestate.go +++ b/arbitrum/recreatestate.go @@ -80,24 +80,3 @@ func AdvanceStateByBlock(ctx context.Context, bc *core.BlockChain, state *state. } return state, block, nil } - -func AdvanceStateUpToBlock(ctx context.Context, bc *core.BlockChain, state *state.StateDB, targetHeader *types.Header, lastAvailableHeader *types.Header, logFunc StateBuildingLogFunction) (*state.StateDB, error) { - returnedBlockNumber := targetHeader.Number.Uint64() - blockToRecreate := lastAvailableHeader.Number.Uint64() + 1 - prevHash := lastAvailableHeader.Hash() - for ctx.Err() == nil { - state, block, err := AdvanceStateByBlock(ctx, bc, state, targetHeader, blockToRecreate, prevHash, logFunc) - if err != nil { - return nil, err - } - prevHash = block.Hash() - if blockToRecreate >= returnedBlockNumber { - if block.Hash() != targetHeader.Hash() { - return nil, fmt.Errorf("blockHash doesn't match when recreating number: %d expected: %v got: %v", blockToRecreate, targetHeader.Hash(), block.Hash()) - } - return state, nil - } - blockToRecreate++ - } - return nil, ctx.Err() -} diff --git a/eth/api_backend.go b/eth/api_backend.go index 77be44defc08..558ded2e9842 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -37,6 +37,7 @@ import ( "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/ethdb" "github.com/ethereum/go-ethereum/event" + "github.com/ethereum/go-ethereum/internal/ethapi" "github.com/ethereum/go-ethereum/miner" "github.com/ethereum/go-ethereum/params" "github.com/ethereum/go-ethereum/rpc" @@ -198,46 +199,46 @@ func (b *EthAPIBackend) PendingBlockAndReceipts() (*types.Block, types.Receipts) return b.eth.miner.PendingBlockAndReceipts() } -func (b *EthAPIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { +func (b *EthAPIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { // Pending state is only known by the miner if number == rpc.PendingBlockNumber { block, state := b.eth.miner.Pending() if block == nil || state == nil { - return nil, nil, errors.New("pending state is not available") + return nil, nil, nil, errors.New("pending state is not available") } - return state, block.Header(), nil + return state, block.Header(), ethapi.NoOpStateRelease, nil } // Otherwise resolve the block number and return its state header, err := b.HeaderByNumber(ctx, number) if err != nil { - return nil, nil, err + return nil, nil, nil, err } if header == nil { - return nil, nil, errors.New("header not found") + return nil, nil, nil, errors.New("header not found") } stateDb, err := b.eth.BlockChain().StateAt(header.Root) - return stateDb, header, err + return stateDb, header, ethapi.NoOpStateRelease, err } -func (b *EthAPIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { +func (b *EthAPIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { if blockNr, ok := blockNrOrHash.Number(); ok { return b.StateAndHeaderByNumber(ctx, blockNr) } if hash, ok := blockNrOrHash.Hash(); ok { header, err := b.HeaderByHash(ctx, hash) if err != nil { - return nil, nil, err + return nil, nil, nil, err } if header == nil { - return nil, nil, errors.New("header for hash not found") + return nil, nil, nil, errors.New("header for hash not found") } if blockNrOrHash.RequireCanonical && b.eth.blockchain.GetCanonicalHash(header.Number.Uint64()) != hash { - return nil, nil, errors.New("hash is not currently canonical") + return nil, nil, nil, errors.New("hash is not currently canonical") } stateDb, err := b.eth.BlockChain().StateAt(header.Root) - return stateDb, header, err + return stateDb, header, ethapi.NoOpStateRelease, err } - return nil, nil, errors.New("invalid arguments; neither block nor hash specified") + return nil, nil, nil, errors.New("invalid arguments; neither block nor hash specified") } func (b *EthAPIBackend) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) { @@ -423,7 +424,7 @@ func (b *EthAPIBackend) StartMining() error { } func (b *EthAPIBackend) StateAtBlock(ctx context.Context, block *types.Block, reexec uint64, base *state.StateDB, readOnly bool, preferDisk bool) (*state.StateDB, tracers.StateReleaseFunc, error) { - return b.eth.StateAtBlock(ctx, block, reexec, base, readOnly, preferDisk) + return b.eth.StateAtBlock(ctx, block, reexec, base, nil, readOnly, preferDisk) } func (b *EthAPIBackend) StateAtTransaction(ctx context.Context, block *types.Block, txIndex int, reexec uint64) (*core.Message, vm.BlockContext, *state.StateDB, tracers.StateReleaseFunc, error) { diff --git a/eth/catalyst/api_test.go b/eth/catalyst/api_test.go index 05ad3def48ae..cd846a499e70 100644 --- a/eth/catalyst/api_test.go +++ b/eth/catalyst/api_test.go @@ -1143,10 +1143,11 @@ func TestWithdrawals(t *testing.T) { } // 11: verify withdrawals were processed. - db, _, err := ethservice.APIBackend.StateAndHeaderByNumber(context.Background(), rpc.BlockNumber(execData.ExecutionPayload.Number)) + db, _, release, err := ethservice.APIBackend.StateAndHeaderByNumber(context.Background(), rpc.BlockNumber(execData.ExecutionPayload.Number)) if err != nil { t.Fatalf("unable to load db: %v", err) } + defer release() for i, w := range blockParams.Withdrawals { // w.Amount is in gwei, balance in wei if db.GetBalance(w.Address).Uint64() != w.Amount*params.GWei { diff --git a/eth/state_accessor.go b/eth/state_accessor.go index ff18d27054e8..cf780de7f192 100644 --- a/eth/state_accessor.go +++ b/eth/state_accessor.go @@ -53,7 +53,7 @@ import ( // - preferDisk: this arg can be used by the caller to signal that even though the 'base' is // provided, it would be preferable to start from a fresh state, if we have it // on disk. -func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexec uint64, base *state.StateDB, readOnly bool, preferDisk bool) (statedb *state.StateDB, release tracers.StateReleaseFunc, err error) { +func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexec uint64, base *state.StateDB, baseBlock *types.Block, readOnly bool, preferDisk bool) (statedb *state.StateDB, release tracers.StateReleaseFunc, err error) { var ( current *types.Block database state.Database @@ -66,8 +66,10 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe // The state is available in live database, create a reference // on top to prevent garbage collection and return a release // function to deref it. + + // Try referencing the root, if it isn't in dirties cache then Reference will have no effect + statedb.Database().TrieDB().Reference(block.Root(), common.Hash{}) if statedb, err = eth.blockchain.StateAt(block.Root()); err == nil { - statedb.Database().TrieDB().Reference(block.Root(), common.Hash{}) return statedb, func() { statedb.Database().TrieDB().Dereference(block.Root()) }, nil @@ -95,7 +97,11 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe } // The optional base statedb is given, mark the start point as parent block statedb, database, report = base, base.Database(), false - current = eth.blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1) + if baseBlock == nil { + current = eth.blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1) + } else { + current = baseBlock + } } else { // Otherwise, try to reexec blocks until we find a state or reach our limit current = block @@ -214,7 +220,7 @@ func (eth *Ethereum) stateAtTransaction(ctx context.Context, block *types.Block, } // Lookup the statedb of parent block from the live database, // otherwise regenerate it on the flight. - statedb, release, err := eth.StateAtBlock(ctx, parent, reexec, nil, true, false) + statedb, release, err := eth.StateAtBlock(ctx, parent, reexec, nil, nil, true, false) if err != nil { return nil, vm.BlockContext{}, nil, nil, err } diff --git a/graphql/graphql.go b/graphql/graphql.go index 5eea340f66cc..ac8b79ea5a40 100644 --- a/graphql/graphql.go +++ b/graphql/graphql.go @@ -86,9 +86,9 @@ type Account struct { } // getState fetches the StateDB object for an account. -func (a *Account) getState(ctx context.Context) (*state.StateDB, error) { - state, _, err := a.r.backend.StateAndHeaderByNumberOrHash(ctx, a.blockNrOrHash) - return state, err +func (a *Account) getState(ctx context.Context) (*state.StateDB, ethapi.StateReleaseFunc, error) { + state, _, release, err := a.r.backend.StateAndHeaderByNumberOrHash(ctx, a.blockNrOrHash) + return state, release, err } func (a *Account) Address(ctx context.Context) (common.Address, error) { @@ -96,10 +96,11 @@ func (a *Account) Address(ctx context.Context) (common.Address, error) { } func (a *Account) Balance(ctx context.Context) (hexutil.Big, error) { - state, err := a.getState(ctx) + state, release, err := a.getState(ctx) if err != nil { return hexutil.Big{}, err } + defer release() balance := state.GetBalance(a.address) if balance == nil { return hexutil.Big{}, fmt.Errorf("failed to load balance %x", a.address) @@ -116,26 +117,29 @@ func (a *Account) TransactionCount(ctx context.Context) (hexutil.Uint64, error) } return hexutil.Uint64(nonce), nil } - state, err := a.getState(ctx) + state, release, err := a.getState(ctx) if err != nil { return 0, err } + defer release() return hexutil.Uint64(state.GetNonce(a.address)), nil } func (a *Account) Code(ctx context.Context) (hexutil.Bytes, error) { - state, err := a.getState(ctx) + state, release, err := a.getState(ctx) if err != nil { return hexutil.Bytes{}, err } + defer release() return state.GetCode(a.address), nil } func (a *Account) Storage(ctx context.Context, args struct{ Slot common.Hash }) (common.Hash, error) { - state, err := a.getState(ctx) + state, release, err := a.getState(ctx) if err != nil { return common.Hash{}, err } + defer release() return state.GetState(a.address, args.Slot), nil } diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index 1fd916d903c2..c268e0858670 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -620,7 +620,7 @@ func (s *BlockChainAPI) BlockNumber() hexutil.Uint64 { // given block number. The rpc.LatestBlockNumber and rpc.PendingBlockNumber meta // block numbers are also allowed. func (s *BlockChainAPI) GetBalance(ctx context.Context, address common.Address, blockNrOrHash rpc.BlockNumberOrHash) (*hexutil.Big, error) { - state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Big @@ -629,6 +629,7 @@ func (s *BlockChainAPI) GetBalance(ctx context.Context, address common.Address, } return nil, err } + defer release() return (*hexutil.Big)(state.GetBalance(address)), state.Error() } @@ -681,10 +682,11 @@ func (s *BlockChainAPI) GetProof(ctx context.Context, address common.Address, st } } - state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return nil, err } + defer release() if storageTrie, err = state.StorageTrie(address); err != nil { return nil, err } @@ -867,7 +869,7 @@ func (s *BlockChainAPI) GetUncleCountByBlockHash(ctx context.Context, blockHash // GetCode returns the code stored at the given address in the state for the given block number. func (s *BlockChainAPI) GetCode(ctx context.Context, address common.Address, blockNrOrHash rpc.BlockNumberOrHash) (hexutil.Bytes, error) { - state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Bytes @@ -876,6 +878,7 @@ func (s *BlockChainAPI) GetCode(ctx context.Context, address common.Address, blo } return nil, err } + defer release() code := state.GetCode(address) return code, state.Error() } @@ -888,7 +891,7 @@ func (s *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Address if err != nil { return nil, fmt.Errorf("unable to decode storage key: %s", err) } - state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Bytes @@ -897,6 +900,7 @@ func (s *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Address } return nil, err } + defer release() res := state.GetState(address, key) return res[:], state.Error() } @@ -1125,10 +1129,11 @@ func doCall(ctx context.Context, b Backend, args TransactionArgs, state *state.S func DoCall(ctx context.Context, b Backend, args TransactionArgs, blockNrOrHash rpc.BlockNumberOrHash, overrides *StateOverride, blockOverrides *BlockOverrides, timeout time.Duration, globalGasCap uint64, runMode core.MessageRunMode) (*core.ExecutionResult, error) { defer func(start time.Time) { log.Debug("Executing EVM call finished", "runtime", time.Since(start)) }(time.Now()) - state, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return nil, err } + defer release() return doCall(ctx, b, args, state, header, overrides, blockOverrides, timeout, globalGasCap, runMode) } @@ -1232,10 +1237,11 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr } // Recap the highest gas limit with account's available balance. if feeCap.BitLen() != 0 { - state, _, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if err != nil { return 0, err } + defer release() err = overrides.Apply(state) if err != nil { return 0, err @@ -1265,10 +1271,11 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr // Arbitrum: raise the gas cap to ignore L1 costs so that it's compute-only vanillaGasCap := gasCap { - state, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return 0, err } + defer release() gasCap, err = args.L2OnlyGasCap(gasCap, header, state, core.MessageGasEstimationMode) if err != nil { return 0, err @@ -1295,10 +1302,11 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr } return result.Failed(), result, nil } - state, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return 0, err } + defer release() err = overrides.Apply(state) if err != nil { return 0, err @@ -1695,10 +1703,11 @@ func (s *BlockChainAPI) CreateAccessList(ctx context.Context, args TransactionAr // If the transaction itself fails, an vmErr is returned. func AccessList(ctx context.Context, b Backend, blockNrOrHash rpc.BlockNumberOrHash, args TransactionArgs) (acl types.AccessList, gasUsed uint64, vmErr error, err error) { // Retrieve the execution context - db, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + db, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if db == nil || err != nil { return nil, 0, nil, err } + defer release() // If the gas amount is not set, default to RPC gas cap. if args.Gas == nil { tmp := hexutil.Uint64(b.RPCGasCap()) @@ -1829,7 +1838,7 @@ func (s *TransactionAPI) GetTransactionCount(ctx context.Context, address common return (*hexutil.Uint64)(&nonce), nil } // Resolve block number and use its state to ask for the nonce - state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Uint64 @@ -1838,6 +1847,7 @@ func (s *TransactionAPI) GetTransactionCount(ctx context.Context, address common } return nil, err } + defer release() nonce := state.GetNonce(address) return (*hexutil.Uint64)(&nonce), state.Error() } diff --git a/internal/ethapi/api_test.go b/internal/ethapi/api_test.go index dd810a9a9671..cf87c553d0f7 100644 --- a/internal/ethapi/api_test.go +++ b/internal/ethapi/api_test.go @@ -437,7 +437,7 @@ func (b testBackend) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc. func (b testBackend) GetBody(ctx context.Context, hash common.Hash, number rpc.BlockNumber) (*types.Body, error) { return b.chain.GetBlock(hash, uint64(number.Int64())).Body(), nil } -func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { +func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, StateReleaseFunc, error) { if number == rpc.PendingBlockNumber { panic("pending state not implemented") } @@ -449,9 +449,9 @@ func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.Bloc return nil, nil, errors.New("header not found") } stateDb, err := b.chain.StateAt(header.Root) - return stateDb, header, err + return stateDb, header, NoOpStateRelease, err } -func (b testBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { +func (b testBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, StateReleaseFunc, error) { if blockNr, ok := blockNrOrHash.Number(); ok { return b.StateAndHeaderByNumber(ctx, blockNr) } diff --git a/internal/ethapi/backend.go b/internal/ethapi/backend.go index 6ebd22ddafd5..7017b8c010ac 100644 --- a/internal/ethapi/backend.go +++ b/internal/ethapi/backend.go @@ -37,6 +37,10 @@ import ( "github.com/ethereum/go-ethereum/rpc" ) +type StateReleaseFunc func() + +var NoOpStateRelease StateReleaseFunc = func() {} + // Backend interface provides the common API services (that are provided by // both full and light clients) with access to necessary functions. type Backend interface { @@ -66,8 +70,8 @@ type Backend interface { BlockByNumber(ctx context.Context, number rpc.BlockNumber) (*types.Block, error) BlockByHash(ctx context.Context, hash common.Hash) (*types.Block, error) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*types.Block, error) - StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) - StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) + StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, StateReleaseFunc, error) + StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, StateReleaseFunc, error) PendingBlockAndReceipts() (*types.Block, types.Receipts) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) GetTd(ctx context.Context, hash common.Hash) *big.Int diff --git a/internal/ethapi/transaction_args_test.go b/internal/ethapi/transaction_args_test.go index ba5dd3977f80..bc75c3dc1226 100644 --- a/internal/ethapi/transaction_args_test.go +++ b/internal/ethapi/transaction_args_test.go @@ -291,11 +291,11 @@ func (b *backendMock) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc func (b *backendMock) GetBody(ctx context.Context, hash common.Hash, number rpc.BlockNumber) (*types.Body, error) { return nil, nil } -func (b *backendMock) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { - return nil, nil, nil +func (b *backendMock) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, StateReleaseFunc, error) { + return nil, nil, NoOpStateRelease, nil } -func (b *backendMock) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { - return nil, nil, nil +func (b *backendMock) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, StateReleaseFunc, error) { + return nil, nil, NoOpStateRelease, nil } func (b *backendMock) PendingBlockAndReceipts() (*types.Block, types.Receipts) { return nil, nil } func (b *backendMock) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) { diff --git a/les/api_backend.go b/les/api_backend.go index 311db0b82824..323c97a9c3ee 100644 --- a/les/api_backend.go +++ b/les/api_backend.go @@ -36,6 +36,7 @@ import ( "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/ethdb" "github.com/ethereum/go-ethereum/event" + "github.com/ethereum/go-ethereum/internal/ethapi" "github.com/ethereum/go-ethereum/light" "github.com/ethereum/go-ethereum/params" "github.com/ethereum/go-ethereum/rpc" @@ -137,32 +138,32 @@ func (b *LesApiBackend) PendingBlockAndReceipts() (*types.Block, types.Receipts) return nil, nil } -func (b *LesApiBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { +func (b *LesApiBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { header, err := b.HeaderByNumber(ctx, number) if err != nil { - return nil, nil, err + return nil, nil, nil, err } if header == nil { - return nil, nil, errors.New("header not found") + return nil, nil, nil, errors.New("header not found") } - return light.NewState(ctx, header, b.eth.odr), header, nil + return light.NewState(ctx, header, b.eth.odr), header, ethapi.NoOpStateRelease, nil } -func (b *LesApiBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { +func (b *LesApiBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { if blockNr, ok := blockNrOrHash.Number(); ok { return b.StateAndHeaderByNumber(ctx, blockNr) } if hash, ok := blockNrOrHash.Hash(); ok { header := b.eth.blockchain.GetHeaderByHash(hash) if header == nil { - return nil, nil, errors.New("header for hash not found") + return nil, nil, nil, errors.New("header for hash not found") } if blockNrOrHash.RequireCanonical && b.eth.blockchain.GetCanonicalHash(header.Number.Uint64()) != hash { - return nil, nil, errors.New("hash is not currently canonical") + return nil, nil, nil, errors.New("hash is not currently canonical") } - return light.NewState(ctx, header, b.eth.odr), header, nil + return light.NewState(ctx, header, b.eth.odr), header, ethapi.NoOpStateRelease, nil } - return nil, nil, errors.New("invalid arguments; neither block nor hash specified") + return nil, nil, nil, errors.New("invalid arguments; neither block nor hash specified") } func (b *LesApiBackend) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) { From 0227c54150444a4fe111d9fcab08f696ff98b816 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Thu, 7 Dec 2023 16:46:58 +0000 Subject: [PATCH 02/28] fix ethapi test --- internal/ethapi/api_test.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/internal/ethapi/api_test.go b/internal/ethapi/api_test.go index cf87c553d0f7..dad135b79e9f 100644 --- a/internal/ethapi/api_test.go +++ b/internal/ethapi/api_test.go @@ -443,10 +443,10 @@ func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.Bloc } header, err := b.HeaderByNumber(ctx, number) if err != nil { - return nil, nil, err + return nil, nil, nil, err } if header == nil { - return nil, nil, errors.New("header not found") + return nil, nil, nil, errors.New("header not found") } stateDb, err := b.chain.StateAt(header.Root) return stateDb, header, NoOpStateRelease, err From 8d5951aa69681b7a35c7a1c5ce1f1d6f74ff60d1 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Mon, 11 Dec 2023 18:46:54 +0000 Subject: [PATCH 03/28] add baseBlock comment, fix referencing befor StateAt --- eth/state_accessor.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/eth/state_accessor.go b/eth/state_accessor.go index cf780de7f192..a440a8a8df68 100644 --- a/eth/state_accessor.go +++ b/eth/state_accessor.go @@ -47,6 +47,7 @@ import ( // - reexec: The maximum number of blocks to reprocess trying to obtain the desired state // - base: If the caller is tracing multiple blocks, the caller can provide the parent // state continuously from the callsite. +// - baseBlock: Arbitrum specific: caller can provide the block from which reprocessing should start. Previous argument (base) is assumed to be the state at the block. If base is not provided, baseBlock is ignored. // - readOnly: If true, then the live 'blockchain' state database is used. No mutation should // be made from caller, e.g. perform Commit or other 'save-to-disk' changes. // Otherwise, the trash generated by caller may be persisted permanently. @@ -68,7 +69,7 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe // function to deref it. // Try referencing the root, if it isn't in dirties cache then Reference will have no effect - statedb.Database().TrieDB().Reference(block.Root(), common.Hash{}) + eth.blockchain.StateCache().TrieDB().Reference(block.Root(), common.Hash{}) if statedb, err = eth.blockchain.StateAt(block.Root()); err == nil { return statedb, func() { statedb.Database().TrieDB().Dereference(block.Root()) From b158011a68df2450cbb1e9aa617abbe95ca0c879 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Mon, 18 Dec 2023 16:01:37 +0000 Subject: [PATCH 04/28] use finalizer instead of returning state release function --- arbitrum/apibackend.go | 62 +++++++++++++++--------- eth/api_backend.go | 25 +++++----- eth/catalyst/api_test.go | 3 +- graphql/graphql.go | 18 +++---- internal/ethapi/api.go | 30 ++++-------- internal/ethapi/api_test.go | 6 +-- internal/ethapi/backend.go | 8 +-- internal/ethapi/transaction_args_test.go | 8 +-- les/api_backend.go | 15 +++--- 9 files changed, 86 insertions(+), 89 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index baa7238fee58..b7b9f5731b81 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -5,6 +5,7 @@ import ( "errors" "fmt" "math/big" + "runtime" "strconv" "strings" "time" @@ -229,11 +230,10 @@ func (a *APIBackend) FeeHistory( // use the most recent average compute rate for all blocks // note: while we could query this value for each block, it'd be prohibitively expensive - state, _, release, err := a.StateAndHeaderByNumber(ctx, newestBlock) + state, _, err := a.StateAndHeaderByNumber(ctx, newestBlock) if err != nil { return common.Big0, nil, nil, nil, err } - defer release() speedLimit, err := core.GetArbOSSpeedLimitPerSecond(state) if err != nil { return common.Big0, nil, nil, nil, err @@ -434,15 +434,15 @@ func (a *APIBackend) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc. return nil, errors.New("invalid arguments; neither block nor hash specified") } -func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types.Header, err error) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types.Header, err error) (*state.StateDB, *types.Header, error) { if err != nil { - return nil, header, nil, err + return nil, header, err } if header == nil { - return nil, nil, nil, errors.New("header not found") + return nil, nil, errors.New("header not found") } if !a.BlockChain().Config().IsArbitrumNitro(header.Number) { - return nil, header, nil, types.ErrUseFallback + return nil, header, types.ErrUseFallback } bc := a.BlockChain() stateFor := func(header *types.Header) (*state.StateDB, error) { @@ -450,43 +450,61 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types // Try referencing the root, if it isn't in dirties cache then Reference will have no effect bc.StateCache().TrieDB().Reference(header.Root, common.Hash{}) } - state, err := bc.StateAt(header.Root) - return state, err + statedb, err := state.New(header.Root, bc.StateCache(), bc.Snapshots()) + if err != nil { + return nil, err + } + if header.Root != (common.Hash{}) { + // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream + headerRoot := header.Root + runtime.SetFinalizer(statedb, func(_ *state.StateDB) { + bc.StateCache().TrieDB().Dereference(headerRoot) + }) + } + return statedb, err } lastState, lastHeader, err := FindLastAvailableState(ctx, bc, stateFor, header, nil, a.b.config.MaxRecreateStateDepth) if err != nil { - return nil, nil, nil, err - } - release := func() { - if lastHeader.Root != (common.Hash{}) { - bc.StateCache().TrieDB().Dereference(lastHeader.Root) - } + return nil, nil, err } if lastHeader == header { - return lastState, header, release, nil + return lastState, header, nil + } + if lastHeader.Root != (common.Hash{}) { + defer bc.StateCache().TrieDB().Dereference(lastHeader.Root) } - defer release() targetBlock := bc.GetBlockByNumber(header.Number.Uint64()) if targetBlock == nil { - return nil, nil, nil, errors.New("target block not found") + return nil, nil, errors.New("target block not found") } lastBlock := bc.GetBlockByNumber(lastHeader.Number.Uint64()) if lastBlock == nil { - return nil, nil, nil, errors.New("last block not found") + return nil, nil, errors.New("last block not found") } reexec := uint64(0) checkLive := false preferDisk := true - state, release, err := eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, targetBlock, reexec, lastState, lastBlock, checkLive, preferDisk) - return state, header, release, err + statedb, release, err := eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, targetBlock, reexec, lastState, lastBlock, checkLive, preferDisk) + if err != nil { + return nil, nil, err + } + // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream + // to set a finalizer we need to allocated the obj in current block + statedb, err = state.New(header.Root, statedb.Database(), nil) + if header.Root != (common.Hash{}) { + runtime.SetFinalizer(statedb, func(_ *state.StateDB) { + release() + }) + } + return statedb, header, err } -func (a *APIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (a *APIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { header, err := a.HeaderByNumber(ctx, number) return a.stateAndHeaderFromHeader(ctx, header, err) } -func (a *APIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (a *APIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { header, err := a.HeaderByNumberOrHash(ctx, blockNrOrHash) return a.stateAndHeaderFromHeader(ctx, header, err) } diff --git a/eth/api_backend.go b/eth/api_backend.go index 558ded2e9842..7656d63c5855 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -37,7 +37,6 @@ import ( "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/ethdb" "github.com/ethereum/go-ethereum/event" - "github.com/ethereum/go-ethereum/internal/ethapi" "github.com/ethereum/go-ethereum/miner" "github.com/ethereum/go-ethereum/params" "github.com/ethereum/go-ethereum/rpc" @@ -199,46 +198,46 @@ func (b *EthAPIBackend) PendingBlockAndReceipts() (*types.Block, types.Receipts) return b.eth.miner.PendingBlockAndReceipts() } -func (b *EthAPIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (b *EthAPIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { // Pending state is only known by the miner if number == rpc.PendingBlockNumber { block, state := b.eth.miner.Pending() if block == nil || state == nil { - return nil, nil, nil, errors.New("pending state is not available") + return nil, nil, errors.New("pending state is not available") } - return state, block.Header(), ethapi.NoOpStateRelease, nil + return state, block.Header(), nil } // Otherwise resolve the block number and return its state header, err := b.HeaderByNumber(ctx, number) if err != nil { - return nil, nil, nil, err + return nil, nil, err } if header == nil { - return nil, nil, nil, errors.New("header not found") + return nil, nil, errors.New("header not found") } stateDb, err := b.eth.BlockChain().StateAt(header.Root) - return stateDb, header, ethapi.NoOpStateRelease, err + return stateDb, header, err } -func (b *EthAPIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (b *EthAPIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { if blockNr, ok := blockNrOrHash.Number(); ok { return b.StateAndHeaderByNumber(ctx, blockNr) } if hash, ok := blockNrOrHash.Hash(); ok { header, err := b.HeaderByHash(ctx, hash) if err != nil { - return nil, nil, nil, err + return nil, nil, err } if header == nil { - return nil, nil, nil, errors.New("header for hash not found") + return nil, nil, errors.New("header for hash not found") } if blockNrOrHash.RequireCanonical && b.eth.blockchain.GetCanonicalHash(header.Number.Uint64()) != hash { - return nil, nil, nil, errors.New("hash is not currently canonical") + return nil, nil, errors.New("hash is not currently canonical") } stateDb, err := b.eth.BlockChain().StateAt(header.Root) - return stateDb, header, ethapi.NoOpStateRelease, err + return stateDb, header, err } - return nil, nil, nil, errors.New("invalid arguments; neither block nor hash specified") + return nil, nil, errors.New("invalid arguments; neither block nor hash specified") } func (b *EthAPIBackend) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) { diff --git a/eth/catalyst/api_test.go b/eth/catalyst/api_test.go index cd846a499e70..05ad3def48ae 100644 --- a/eth/catalyst/api_test.go +++ b/eth/catalyst/api_test.go @@ -1143,11 +1143,10 @@ func TestWithdrawals(t *testing.T) { } // 11: verify withdrawals were processed. - db, _, release, err := ethservice.APIBackend.StateAndHeaderByNumber(context.Background(), rpc.BlockNumber(execData.ExecutionPayload.Number)) + db, _, err := ethservice.APIBackend.StateAndHeaderByNumber(context.Background(), rpc.BlockNumber(execData.ExecutionPayload.Number)) if err != nil { t.Fatalf("unable to load db: %v", err) } - defer release() for i, w := range blockParams.Withdrawals { // w.Amount is in gwei, balance in wei if db.GetBalance(w.Address).Uint64() != w.Amount*params.GWei { diff --git a/graphql/graphql.go b/graphql/graphql.go index ac8b79ea5a40..5eea340f66cc 100644 --- a/graphql/graphql.go +++ b/graphql/graphql.go @@ -86,9 +86,9 @@ type Account struct { } // getState fetches the StateDB object for an account. -func (a *Account) getState(ctx context.Context) (*state.StateDB, ethapi.StateReleaseFunc, error) { - state, _, release, err := a.r.backend.StateAndHeaderByNumberOrHash(ctx, a.blockNrOrHash) - return state, release, err +func (a *Account) getState(ctx context.Context) (*state.StateDB, error) { + state, _, err := a.r.backend.StateAndHeaderByNumberOrHash(ctx, a.blockNrOrHash) + return state, err } func (a *Account) Address(ctx context.Context) (common.Address, error) { @@ -96,11 +96,10 @@ func (a *Account) Address(ctx context.Context) (common.Address, error) { } func (a *Account) Balance(ctx context.Context) (hexutil.Big, error) { - state, release, err := a.getState(ctx) + state, err := a.getState(ctx) if err != nil { return hexutil.Big{}, err } - defer release() balance := state.GetBalance(a.address) if balance == nil { return hexutil.Big{}, fmt.Errorf("failed to load balance %x", a.address) @@ -117,29 +116,26 @@ func (a *Account) TransactionCount(ctx context.Context) (hexutil.Uint64, error) } return hexutil.Uint64(nonce), nil } - state, release, err := a.getState(ctx) + state, err := a.getState(ctx) if err != nil { return 0, err } - defer release() return hexutil.Uint64(state.GetNonce(a.address)), nil } func (a *Account) Code(ctx context.Context) (hexutil.Bytes, error) { - state, release, err := a.getState(ctx) + state, err := a.getState(ctx) if err != nil { return hexutil.Bytes{}, err } - defer release() return state.GetCode(a.address), nil } func (a *Account) Storage(ctx context.Context, args struct{ Slot common.Hash }) (common.Hash, error) { - state, release, err := a.getState(ctx) + state, err := a.getState(ctx) if err != nil { return common.Hash{}, err } - defer release() return state.GetState(a.address, args.Slot), nil } diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index c268e0858670..1fd916d903c2 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -620,7 +620,7 @@ func (s *BlockChainAPI) BlockNumber() hexutil.Uint64 { // given block number. The rpc.LatestBlockNumber and rpc.PendingBlockNumber meta // block numbers are also allowed. func (s *BlockChainAPI) GetBalance(ctx context.Context, address common.Address, blockNrOrHash rpc.BlockNumberOrHash) (*hexutil.Big, error) { - state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Big @@ -629,7 +629,6 @@ func (s *BlockChainAPI) GetBalance(ctx context.Context, address common.Address, } return nil, err } - defer release() return (*hexutil.Big)(state.GetBalance(address)), state.Error() } @@ -682,11 +681,10 @@ func (s *BlockChainAPI) GetProof(ctx context.Context, address common.Address, st } } - state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return nil, err } - defer release() if storageTrie, err = state.StorageTrie(address); err != nil { return nil, err } @@ -869,7 +867,7 @@ func (s *BlockChainAPI) GetUncleCountByBlockHash(ctx context.Context, blockHash // GetCode returns the code stored at the given address in the state for the given block number. func (s *BlockChainAPI) GetCode(ctx context.Context, address common.Address, blockNrOrHash rpc.BlockNumberOrHash) (hexutil.Bytes, error) { - state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Bytes @@ -878,7 +876,6 @@ func (s *BlockChainAPI) GetCode(ctx context.Context, address common.Address, blo } return nil, err } - defer release() code := state.GetCode(address) return code, state.Error() } @@ -891,7 +888,7 @@ func (s *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Address if err != nil { return nil, fmt.Errorf("unable to decode storage key: %s", err) } - state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Bytes @@ -900,7 +897,6 @@ func (s *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Address } return nil, err } - defer release() res := state.GetState(address, key) return res[:], state.Error() } @@ -1129,11 +1125,10 @@ func doCall(ctx context.Context, b Backend, args TransactionArgs, state *state.S func DoCall(ctx context.Context, b Backend, args TransactionArgs, blockNrOrHash rpc.BlockNumberOrHash, overrides *StateOverride, blockOverrides *BlockOverrides, timeout time.Duration, globalGasCap uint64, runMode core.MessageRunMode) (*core.ExecutionResult, error) { defer func(start time.Time) { log.Debug("Executing EVM call finished", "runtime", time.Since(start)) }(time.Now()) - state, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return nil, err } - defer release() return doCall(ctx, b, args, state, header, overrides, blockOverrides, timeout, globalGasCap, runMode) } @@ -1237,11 +1232,10 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr } // Recap the highest gas limit with account's available balance. if feeCap.BitLen() != 0 { - state, _, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if err != nil { return 0, err } - defer release() err = overrides.Apply(state) if err != nil { return 0, err @@ -1271,11 +1265,10 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr // Arbitrum: raise the gas cap to ignore L1 costs so that it's compute-only vanillaGasCap := gasCap { - state, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return 0, err } - defer release() gasCap, err = args.L2OnlyGasCap(gasCap, header, state, core.MessageGasEstimationMode) if err != nil { return 0, err @@ -1302,11 +1295,10 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr } return result.Failed(), result, nil } - state, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { return 0, err } - defer release() err = overrides.Apply(state) if err != nil { return 0, err @@ -1703,11 +1695,10 @@ func (s *BlockChainAPI) CreateAccessList(ctx context.Context, args TransactionAr // If the transaction itself fails, an vmErr is returned. func AccessList(ctx context.Context, b Backend, blockNrOrHash rpc.BlockNumberOrHash, args TransactionArgs) (acl types.AccessList, gasUsed uint64, vmErr error, err error) { // Retrieve the execution context - db, header, release, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + db, header, err := b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if db == nil || err != nil { return nil, 0, nil, err } - defer release() // If the gas amount is not set, default to RPC gas cap. if args.Gas == nil { tmp := hexutil.Uint64(b.RPCGasCap()) @@ -1838,7 +1829,7 @@ func (s *TransactionAPI) GetTransactionCount(ctx context.Context, address common return (*hexutil.Uint64)(&nonce), nil } // Resolve block number and use its state to ask for the nonce - state, _, release, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) + state, _, err := s.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash) if state == nil || err != nil { if client := fallbackClientFor(s.b, err); client != nil { var res hexutil.Uint64 @@ -1847,7 +1838,6 @@ func (s *TransactionAPI) GetTransactionCount(ctx context.Context, address common } return nil, err } - defer release() nonce := state.GetNonce(address) return (*hexutil.Uint64)(&nonce), state.Error() } diff --git a/internal/ethapi/api_test.go b/internal/ethapi/api_test.go index dad135b79e9f..c92b438b0336 100644 --- a/internal/ethapi/api_test.go +++ b/internal/ethapi/api_test.go @@ -437,7 +437,7 @@ func (b testBackend) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc. func (b testBackend) GetBody(ctx context.Context, hash common.Hash, number rpc.BlockNumber) (*types.Body, error) { return b.chain.GetBlock(hash, uint64(number.Int64())).Body(), nil } -func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, StateReleaseFunc, error) { +func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { if number == rpc.PendingBlockNumber { panic("pending state not implemented") } @@ -449,9 +449,9 @@ func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.Bloc return nil, nil, nil, errors.New("header not found") } stateDb, err := b.chain.StateAt(header.Root) - return stateDb, header, NoOpStateRelease, err + return stateDb, header, err } -func (b testBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, StateReleaseFunc, error) { +func (b testBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { if blockNr, ok := blockNrOrHash.Number(); ok { return b.StateAndHeaderByNumber(ctx, blockNr) } diff --git a/internal/ethapi/backend.go b/internal/ethapi/backend.go index 7017b8c010ac..6ebd22ddafd5 100644 --- a/internal/ethapi/backend.go +++ b/internal/ethapi/backend.go @@ -37,10 +37,6 @@ import ( "github.com/ethereum/go-ethereum/rpc" ) -type StateReleaseFunc func() - -var NoOpStateRelease StateReleaseFunc = func() {} - // Backend interface provides the common API services (that are provided by // both full and light clients) with access to necessary functions. type Backend interface { @@ -70,8 +66,8 @@ type Backend interface { BlockByNumber(ctx context.Context, number rpc.BlockNumber) (*types.Block, error) BlockByHash(ctx context.Context, hash common.Hash) (*types.Block, error) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*types.Block, error) - StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, StateReleaseFunc, error) - StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, StateReleaseFunc, error) + StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) + StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) PendingBlockAndReceipts() (*types.Block, types.Receipts) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) GetTd(ctx context.Context, hash common.Hash) *big.Int diff --git a/internal/ethapi/transaction_args_test.go b/internal/ethapi/transaction_args_test.go index bc75c3dc1226..ba5dd3977f80 100644 --- a/internal/ethapi/transaction_args_test.go +++ b/internal/ethapi/transaction_args_test.go @@ -291,11 +291,11 @@ func (b *backendMock) BlockByNumberOrHash(ctx context.Context, blockNrOrHash rpc func (b *backendMock) GetBody(ctx context.Context, hash common.Hash, number rpc.BlockNumber) (*types.Body, error) { return nil, nil } -func (b *backendMock) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, StateReleaseFunc, error) { - return nil, nil, NoOpStateRelease, nil +func (b *backendMock) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { + return nil, nil, nil } -func (b *backendMock) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, StateReleaseFunc, error) { - return nil, nil, NoOpStateRelease, nil +func (b *backendMock) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { + return nil, nil, nil } func (b *backendMock) PendingBlockAndReceipts() (*types.Block, types.Receipts) { return nil, nil } func (b *backendMock) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) { diff --git a/les/api_backend.go b/les/api_backend.go index 323c97a9c3ee..1ac58f9380ed 100644 --- a/les/api_backend.go +++ b/les/api_backend.go @@ -36,7 +36,6 @@ import ( "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/ethdb" "github.com/ethereum/go-ethereum/event" - "github.com/ethereum/go-ethereum/internal/ethapi" "github.com/ethereum/go-ethereum/light" "github.com/ethereum/go-ethereum/params" "github.com/ethereum/go-ethereum/rpc" @@ -138,7 +137,7 @@ func (b *LesApiBackend) PendingBlockAndReceipts() (*types.Block, types.Receipts) return nil, nil } -func (b *LesApiBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (b *LesApiBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { header, err := b.HeaderByNumber(ctx, number) if err != nil { return nil, nil, nil, err @@ -146,24 +145,24 @@ func (b *LesApiBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.B if header == nil { return nil, nil, nil, errors.New("header not found") } - return light.NewState(ctx, header, b.eth.odr), header, ethapi.NoOpStateRelease, nil + return light.NewState(ctx, header, b.eth.odr), header, nil } -func (b *LesApiBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, ethapi.StateReleaseFunc, error) { +func (b *LesApiBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { if blockNr, ok := blockNrOrHash.Number(); ok { return b.StateAndHeaderByNumber(ctx, blockNr) } if hash, ok := blockNrOrHash.Hash(); ok { header := b.eth.blockchain.GetHeaderByHash(hash) if header == nil { - return nil, nil, nil, errors.New("header for hash not found") + return nil, nil, errors.New("header for hash not found") } if blockNrOrHash.RequireCanonical && b.eth.blockchain.GetCanonicalHash(header.Number.Uint64()) != hash { - return nil, nil, nil, errors.New("hash is not currently canonical") + return nil, nil, errors.New("hash is not currently canonical") } - return light.NewState(ctx, header, b.eth.odr), header, ethapi.NoOpStateRelease, nil + return light.NewState(ctx, header, b.eth.odr), header, nil } - return nil, nil, nil, errors.New("invalid arguments; neither block nor hash specified") + return nil, nil, errors.New("invalid arguments; neither block nor hash specified") } func (b *LesApiBackend) GetReceipts(ctx context.Context, hash common.Hash) (types.Receipts, error) { From 8186f88540091efe44b7afcdbad0f35a2ef3c1a0 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Tue, 19 Dec 2023 13:47:31 +0000 Subject: [PATCH 05/28] clean up extra return values --- internal/ethapi/api_test.go | 4 ++-- les/api_backend.go | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/internal/ethapi/api_test.go b/internal/ethapi/api_test.go index c92b438b0336..dd810a9a9671 100644 --- a/internal/ethapi/api_test.go +++ b/internal/ethapi/api_test.go @@ -443,10 +443,10 @@ func (b testBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.Bloc } header, err := b.HeaderByNumber(ctx, number) if err != nil { - return nil, nil, nil, err + return nil, nil, err } if header == nil { - return nil, nil, nil, errors.New("header not found") + return nil, nil, errors.New("header not found") } stateDb, err := b.chain.StateAt(header.Root) return stateDb, header, err diff --git a/les/api_backend.go b/les/api_backend.go index 1ac58f9380ed..311db0b82824 100644 --- a/les/api_backend.go +++ b/les/api_backend.go @@ -140,10 +140,10 @@ func (b *LesApiBackend) PendingBlockAndReceipts() (*types.Block, types.Receipts) func (b *LesApiBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) { header, err := b.HeaderByNumber(ctx, number) if err != nil { - return nil, nil, nil, err + return nil, nil, err } if header == nil { - return nil, nil, nil, errors.New("header not found") + return nil, nil, errors.New("header not found") } return light.NewState(ctx, header, b.eth.odr), header, nil } From 5645cf052e8ea8c3491c3a20fcbf57282495ac46 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 12 Jan 2024 20:53:34 +0000 Subject: [PATCH 06/28] set statedb finalizer only in stateAndHeaderFromHeader --- arbitrum/apibackend.go | 23 ++++++++++------------- arbitrum/recordingdb.go | 7 ++++++- arbitrum/recreatestate.go | 29 ++++++++++++++++++++--------- 3 files changed, 36 insertions(+), 23 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index b7b9f5731b81..b7bbcab1bde3 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -445,34 +445,33 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types return nil, header, types.ErrUseFallback } bc := a.BlockChain() - stateFor := func(header *types.Header) (*state.StateDB, error) { + stateFor := func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) { if header.Root != (common.Hash{}) { // Try referencing the root, if it isn't in dirties cache then Reference will have no effect bc.StateCache().TrieDB().Reference(header.Root, common.Hash{}) } statedb, err := state.New(header.Root, bc.StateCache(), bc.Snapshots()) if err != nil { - return nil, err + return nil, noopStateRelease, err } if header.Root != (common.Hash{}) { - // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream headerRoot := header.Root - runtime.SetFinalizer(statedb, func(_ *state.StateDB) { - bc.StateCache().TrieDB().Dereference(headerRoot) - }) + return statedb, func() { bc.StateCache().TrieDB().Dereference(headerRoot) }, nil } - return statedb, err + return statedb, noopStateRelease, nil } - lastState, lastHeader, err := FindLastAvailableState(ctx, bc, stateFor, header, nil, a.b.config.MaxRecreateStateDepth) + lastState, lastHeader, lastStateRelease, err := FindLastAvailableState(ctx, bc, stateFor, header, nil, a.b.config.MaxRecreateStateDepth) if err != nil { return nil, nil, err } if lastHeader == header { + // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream + runtime.SetFinalizer(lastState, func(_ *state.StateDB) { + lastStateRelease() + }) return lastState, header, nil } - if lastHeader.Root != (common.Hash{}) { - defer bc.StateCache().TrieDB().Dereference(lastHeader.Root) - } + defer lastStateRelease() targetBlock := bc.GetBlockByNumber(header.Number.Uint64()) if targetBlock == nil { return nil, nil, errors.New("target block not found") @@ -489,8 +488,6 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types return nil, nil, err } // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream - // to set a finalizer we need to allocated the obj in current block - statedb, err = state.New(header.Root, statedb.Database(), nil) if header.Root != (common.Hash{}) { runtime.SetFinalizer(statedb, func(_ *state.StateDB) { release() diff --git a/arbitrum/recordingdb.go b/arbitrum/recordingdb.go index a5c08b99a8bc..5534ee10e1ae 100644 --- a/arbitrum/recordingdb.go +++ b/arbitrum/recordingdb.go @@ -302,7 +302,12 @@ func (r *RecordingDatabase) PreimagesFromRecording(chainContextIf core.ChainCont } func (r *RecordingDatabase) GetOrRecreateState(ctx context.Context, header *types.Header, logFunc StateBuildingLogFunction) (*state.StateDB, error) { - state, currentHeader, err := FindLastAvailableState(ctx, r.bc, r.StateFor, header, logFunc, -1) + stateFor := func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) { + state, err := r.StateFor(header) + // we don't use the release functor pattern here yet + return state, noopStateRelease, err + } + state, currentHeader, _, err := FindLastAvailableState(ctx, r.bc, stateFor, header, logFunc, -1) if err != nil { return nil, err } diff --git a/arbitrum/recreatestate.go b/arbitrum/recreatestate.go index e44262cdca83..9966074211fe 100644 --- a/arbitrum/recreatestate.go +++ b/arbitrum/recreatestate.go @@ -9,6 +9,7 @@ import ( "github.com/ethereum/go-ethereum/core/state" "github.com/ethereum/go-ethereum/core/types" "github.com/ethereum/go-ethereum/core/vm" + "github.com/ethereum/go-ethereum/eth/tracers" "github.com/pkg/errors" ) @@ -16,51 +17,61 @@ var ( ErrDepthLimitExceeded = errors.New("state recreation l2 gas depth limit exceeded") ) +type StateReleaseFunc tracers.StateReleaseFunc + +var noopStateRelease StateReleaseFunc = func() {} + type StateBuildingLogFunction func(targetHeader, header *types.Header, hasState bool) -type StateForHeaderFunction func(header *types.Header) (*state.StateDB, error) +type StateForHeaderFunction func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) // finds last available state and header checking it first for targetHeader then looking backwards // if maxDepthInL2Gas is positive, it constitutes a limit for cumulative l2 gas used of the traversed blocks // else if maxDepthInL2Gas is -1, the traversal depth is not limited // otherwise only targetHeader state is checked and no search is performed -func FindLastAvailableState(ctx context.Context, bc *core.BlockChain, stateFor StateForHeaderFunction, targetHeader *types.Header, logFunc StateBuildingLogFunction, maxDepthInL2Gas int64) (*state.StateDB, *types.Header, error) { +func FindLastAvailableState(ctx context.Context, bc *core.BlockChain, stateFor StateForHeaderFunction, targetHeader *types.Header, logFunc StateBuildingLogFunction, maxDepthInL2Gas int64) (*state.StateDB, *types.Header, StateReleaseFunc, error) { genesis := bc.Config().ArbitrumChainParams.GenesisBlockNum currentHeader := targetHeader var state *state.StateDB var err error var l2GasUsed uint64 + release := noopStateRelease for ctx.Err() == nil { lastHeader := currentHeader - state, err = stateFor(currentHeader) + state, release, err = stateFor(currentHeader) if err == nil { break } if maxDepthInL2Gas > 0 { receipts := bc.GetReceiptsByHash(currentHeader.Hash()) if receipts == nil { - return nil, lastHeader, fmt.Errorf("failed to get receipts for hash %v", currentHeader.Hash()) + release() + return nil, lastHeader, nil, fmt.Errorf("failed to get receipts for hash %v", currentHeader.Hash()) } for _, receipt := range receipts { l2GasUsed += receipt.GasUsed - receipt.GasUsedForL1 } if l2GasUsed > uint64(maxDepthInL2Gas) { - return nil, lastHeader, ErrDepthLimitExceeded + release() + return nil, lastHeader, nil, ErrDepthLimitExceeded } } else if maxDepthInL2Gas != InfiniteMaxRecreateStateDepth { - return nil, lastHeader, err + release() + return nil, lastHeader, nil, err } if logFunc != nil { logFunc(targetHeader, currentHeader, false) } if currentHeader.Number.Uint64() <= genesis { - return nil, lastHeader, errors.Wrap(err, fmt.Sprintf("moved beyond genesis looking for state %d, genesis %d", targetHeader.Number.Uint64(), genesis)) + release() + return nil, lastHeader, nil, errors.Wrap(err, fmt.Sprintf("moved beyond genesis looking for state %d, genesis %d", targetHeader.Number.Uint64(), genesis)) } currentHeader = bc.GetHeader(currentHeader.ParentHash, currentHeader.Number.Uint64()-1) if currentHeader == nil { - return nil, lastHeader, fmt.Errorf("chain doesn't contain parent of block %d hash %v", lastHeader.Number, lastHeader.Hash()) + release() + return nil, lastHeader, nil, fmt.Errorf("chain doesn't contain parent of block %d hash %v", lastHeader.Number, lastHeader.Hash()) } } - return state, currentHeader, ctx.Err() + return state, currentHeader, release, ctx.Err() } func AdvanceStateByBlock(ctx context.Context, bc *core.BlockChain, state *state.StateDB, targetHeader *types.Header, blockToRecreate uint64, prevBlockHash common.Hash, logFunc StateBuildingLogFunction) (*state.StateDB, *types.Block, error) { From 86298e8bf81183b33b2bb52027a4f9eb8cffef88 Mon Sep 17 00:00:00 2001 From: Ganesh Vanahalli Date: Mon, 22 Jan 2024 14:35:31 -0600 Subject: [PATCH 07/28] Add components that geth backend has in arbitrum's backend --- arbitrum/apibackend.go | 4 ++-- arbitrum/backend.go | 32 ++++++++++++++++++-------------- 2 files changed, 20 insertions(+), 16 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 4419efce6074..0f60353c92ba 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -145,7 +145,7 @@ func (a *APIBackend) GetAPIs(filterSystem *filters.FilterSystem) []rpc.API { } func (a *APIBackend) BlockChain() *core.BlockChain { - return a.b.arb.BlockChain() + return a.b.BlockChain() } func (a *APIBackend) GetArbitrumNode() interface{} { @@ -607,7 +607,7 @@ func (a *APIBackend) ChainConfig() *params.ChainConfig { } func (a *APIBackend) Engine() consensus.Engine { - return a.BlockChain().Engine() + return a.b.Engine() } func (b *APIBackend) PendingBlockAndReceipts() (*types.Block, types.Receipts) { diff --git a/arbitrum/backend.go b/arbitrum/backend.go index b1bb9c89de03..e3f23ef71a95 100644 --- a/arbitrum/backend.go +++ b/arbitrum/backend.go @@ -3,7 +3,9 @@ package arbitrum import ( "context" + "github.com/ethereum/go-ethereum/accounts" "github.com/ethereum/go-ethereum/arbitrum_types" + "github.com/ethereum/go-ethereum/consensus" "github.com/ethereum/go-ethereum/core" "github.com/ethereum/go-ethereum/core/bloombits" "github.com/ethereum/go-ethereum/core/types" @@ -12,6 +14,7 @@ import ( "github.com/ethereum/go-ethereum/event" "github.com/ethereum/go-ethereum/internal/shutdowncheck" "github.com/ethereum/go-ethereum/node" + "github.com/ethereum/go-ethereum/rpc" ) type Backend struct { @@ -32,6 +35,8 @@ type Backend struct { chanTxs chan *types.Transaction chanClose chan struct{} //close coroutine chanNewBlock chan struct{} //create new L2 block unless empty + + filterSystem *filters.FilterSystem } func NewBackend(stack *node.Node, config *Config, chainDb ethdb.Database, publisher ArbInterface, filterConfig filters.Config) (*Backend, *filters.FilterSystem, error) { @@ -64,15 +69,22 @@ func NewBackend(stack *node.Node, config *Config, chainDb ethdb.Database, publis if err != nil { return nil, nil, err } + backend.filterSystem = filterSystem return backend, filterSystem, nil } -func (b *Backend) APIBackend() *APIBackend { - return b.apiBackend -} - -func (b *Backend) ChainDb() ethdb.Database { - return b.chainDb +func (b *Backend) AccountManager() *accounts.Manager { return b.stack.AccountManager() } +func (b *Backend) APIBackend() *APIBackend { return b.apiBackend } +func (b *Backend) APIs() []rpc.API { return b.apiBackend.GetAPIs(b.filterSystem) } +func (b *Backend) ArbInterface() ArbInterface { return b.arb } +func (b *Backend) BlockChain() *core.BlockChain { return b.arb.BlockChain() } +func (b *Backend) BloomIndexer() *core.ChainIndexer { return b.bloomIndexer } +func (b *Backend) ChainDb() ethdb.Database { return b.chainDb } +func (b *Backend) Engine() consensus.Engine { return b.arb.BlockChain().Engine() } +func (b *Backend) Stack() *node.Node { return b.stack } + +func (b *Backend) ResetWithGenesisBlock(gb *types.Block) { + b.arb.BlockChain().ResetWithGenesisBlock(gb) } func (b *Backend) EnqueueL2Message(ctx context.Context, tx *types.Transaction, options *arbitrum_types.ConditionalOptions) error { @@ -83,14 +95,6 @@ func (b *Backend) SubscribeNewTxsEvent(ch chan<- core.NewTxsEvent) event.Subscri return b.scope.Track(b.txFeed.Subscribe(ch)) } -func (b *Backend) Stack() *node.Node { - return b.stack -} - -func (b *Backend) ArbInterface() ArbInterface { - return b.arb -} - // TODO: this is used when registering backend as lifecycle in stack func (b *Backend) Start() error { b.startBloomHandlers(b.config.BloomBitsBlocks) From ce1438bda1ab97ed3c228ac275fccd47cf5eb2f9 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Mon, 5 Feb 2024 14:41:02 +0000 Subject: [PATCH 08/28] save some states from triegc when stoping sparse archive node (same as for full node) --- core/blockchain.go | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/core/blockchain.go b/core/blockchain.go index 9a09b42c1d76..3ec18b57ef53 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -1074,7 +1074,8 @@ func (bc *BlockChain) Stop() { // - HEAD: So we don't need to reprocess any blocks in the general case // - HEAD-1: So we don't do large reorgs if our HEAD becomes an uncle // - HEAD-127: So we have a hard limit on the number of blocks reexecuted - if !bc.cacheConfig.TrieDirtyDisabled { + // It applies for both full node and sparse archive node + if !bc.cacheConfig.TrieDirtyDisabled || bc.cacheConfig.MaxNumberOfBlocksToSkipStateSaving > 0 || bc.cacheConfig.MaxAmountOfGasToSkipStateSaving > 0 { triedb := bc.triedb for _, offset := range []uint64{0, 1, bc.cacheConfig.TriesInMemory - 1, math.MaxUint64} { @@ -1499,7 +1500,7 @@ func (bc *BlockChain) writeBlockWithState(block *types.Block, receipts []*types. return nil } // If we're running an archive node, flush - // If MaxNumberOfBlocksToSkipStateSaving or MaxAmountOfGasToSkipStateSaving is not zero, then flushing of some blocks will be skipped: + // Sparse archive: if MaxNumberOfBlocksToSkipStateSaving or MaxAmountOfGasToSkipStateSaving is not zero, then flushing of some blocks will be skipped: // * at most MaxNumberOfBlocksToSkipStateSaving block state commits will be skipped // * sum of gas used in skipped blocks will be at most MaxAmountOfGasToSkipStateSaving archiveNode := bc.cacheConfig.TrieDirtyDisabled @@ -1529,7 +1530,7 @@ func (bc *BlockChain) writeBlockWithState(block *types.Block, receipts []*types. // we are skipping saving the trie to diskdb, so we need to keep the trie in memory and garbage collect it later } - // Full node or archive node that's not keeping all states, do proper garbage collection + // Full node or sparse archive node that's not keeping all states, do proper garbage collection bc.triedb.Reference(root, common.Hash{}) // metadata reference to keep trie alive bc.triegc.Push(trieGcEntry{root, block.Header().Time}, -int64(block.NumberU64())) From 5177f70f4b6fa4b63bb446d239f91827a0fde231 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Tue, 6 Feb 2024 15:20:03 +0000 Subject: [PATCH 09/28] add debug logs with state finalizers counters --- arbitrum/apibackend.go | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 5b9ec8becdc9..2801b4402c0c 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -8,6 +8,7 @@ import ( "runtime" "strconv" "strings" + "sync/atomic" "time" "github.com/ethereum/go-ethereum" @@ -38,6 +39,10 @@ type APIBackend struct { fallbackClient types.FallbackClient sync SyncProgressBackend + + // TODO remove + liveStateFinalizers atomic.Int64 + recreatedStateFinalizers atomic.Int64 } type timeoutFallbackClient struct { @@ -466,7 +471,11 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } if lastHeader == header { // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream + a.liveStateFinalizers.Add(1) + log.Debug("Live state finalizer set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) runtime.SetFinalizer(lastState, func(_ *state.StateDB) { + a.liveStateFinalizers.Add(-1) + log.Debug("Live state finalizer called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) lastStateRelease() }) return lastState, header, nil @@ -489,7 +498,12 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream if header.Root != (common.Hash{}) { + a.recreatedStateFinalizers.Add(1) + log.Debug("Recreated state finalizer set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) runtime.SetFinalizer(statedb, func(_ *state.StateDB) { + // TODO remove + a.recreatedStateFinalizers.Add(-1) + log.Debug("Recreated state finalizer called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) release() }) } From 97a51030390fd8443557d4faae4e74032aaea33b Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Thu, 8 Feb 2024 20:08:01 +0000 Subject: [PATCH 10/28] fix baseBlock usage in StateAtBlock --- eth/state_accessor.go | 22 ++++++++++++++-------- 1 file changed, 14 insertions(+), 8 deletions(-) diff --git a/eth/state_accessor.go b/eth/state_accessor.go index 940dd6c10312..812cd8b367c7 100644 --- a/eth/state_accessor.go +++ b/eth/state_accessor.go @@ -63,7 +63,18 @@ func (eth *Ethereum) hashState(ctx context.Context, block *types.Block, reexec u // The state is both for reading and writing, or it's unavailable in disk, // try to construct/recover the state over an ephemeral trie.Database for // isolating the live one. - if base != nil { + if baseBlock != nil { + // Create an ephemeral trie.Database for isolating the live one + triedb = trie.NewDatabase(eth.chainDb, trie.HashDefaults) + database = state.NewDatabaseWithNodeDB(eth.chainDb, triedb) + // there is no need of referencing baseBlock state as it's read from disk + statedb, err = state.New(baseBlock.Root(), database, nil) + if err == nil { + return statedb, noopReleaser, nil + } + report = false + current = baseBlock + } else if base != nil { if preferDisk { // Create an ephemeral trie.Database for isolating the live one. Otherwise // the internal junks created by tracing will be persisted into the disk. @@ -77,11 +88,7 @@ func (eth *Ethereum) hashState(ctx context.Context, block *types.Block, reexec u } // The optional base statedb is given, mark the start point as parent block statedb, database, triedb, report = base, base.Database(), base.Database().TrieDB(), false - if baseBlock == nil { - current = eth.blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1) - } else { - current = baseBlock - } + current = eth.blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1) } else { // Otherwise, try to reexec blocks until we find a state or reach our limit current = block @@ -209,8 +216,7 @@ func (eth *Ethereum) pathState(block *types.Block) (*state.StateDB, func(), erro // - base: If the caller is tracing multiple blocks, the caller can provide the parent // state continuously from the callsite. // - baseBlock: Arbitrum specific: caller can provide the block from which reprocessing should -// start. Previous argument (base) is assumed to be the state at the block. If base is not -// provided, baseBlock is ignored. +// start, if baseBlock is provided then base parameter is ignored // - readOnly: If true, then the live 'blockchain' state database is used. No mutation should // be made from caller, e.g. perform Commit or other 'save-to-disk' changes. // Otherwise, the trash generated by caller may be persisted permanently. From 65e4c8a20256e01e9cd6c4dc347540db18c215c3 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 9 Feb 2024 16:43:36 +0000 Subject: [PATCH 11/28] add state release method to StateDB --- arbitrum/apibackend.go | 17 ++++++++------- arbitrum/recreatestate.go | 5 ----- core/state/statedb.go | 46 +++++++++++++++++++++++++++++++++++++++ eth/state_accessor.go | 4 ++-- graphql/graphql.go | 3 +++ internal/ethapi/api.go | 9 ++++++++ 6 files changed, 69 insertions(+), 15 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 2801b4402c0c..6f3ec55a789f 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -5,7 +5,6 @@ import ( "errors" "fmt" "math/big" - "runtime" "strconv" "strings" "sync/atomic" @@ -240,6 +239,7 @@ func (a *APIBackend) FeeHistory( return common.Big0, nil, nil, nil, err } speedLimit, err := core.GetArbOSSpeedLimitPerSecond(state) + state.Release() if err != nil { return common.Big0, nil, nil, nil, err } @@ -472,12 +472,13 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types if lastHeader == header { // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream a.liveStateFinalizers.Add(1) - log.Debug("Live state finalizer set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) - runtime.SetFinalizer(lastState, func(_ *state.StateDB) { + lastState.SetRelease(func() { a.liveStateFinalizers.Add(-1) - log.Debug("Live state finalizer called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + // TODO remove logs and counters + log.Debug("Live state relase called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) lastStateRelease() }) + log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) return lastState, header, nil } defer lastStateRelease() @@ -499,13 +500,13 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream if header.Root != (common.Hash{}) { a.recreatedStateFinalizers.Add(1) - log.Debug("Recreated state finalizer set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) - runtime.SetFinalizer(statedb, func(_ *state.StateDB) { - // TODO remove + statedb.SetRelease(func() { + // TODO remove logs and counters a.recreatedStateFinalizers.Add(-1) - log.Debug("Recreated state finalizer called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + log.Debug("Recreated state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) release() }) + log.Debug("Recreated state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) } return statedb, header, err } diff --git a/arbitrum/recreatestate.go b/arbitrum/recreatestate.go index 9966074211fe..7d10c9c9e8ca 100644 --- a/arbitrum/recreatestate.go +++ b/arbitrum/recreatestate.go @@ -44,30 +44,25 @@ func FindLastAvailableState(ctx context.Context, bc *core.BlockChain, stateFor S if maxDepthInL2Gas > 0 { receipts := bc.GetReceiptsByHash(currentHeader.Hash()) if receipts == nil { - release() return nil, lastHeader, nil, fmt.Errorf("failed to get receipts for hash %v", currentHeader.Hash()) } for _, receipt := range receipts { l2GasUsed += receipt.GasUsed - receipt.GasUsedForL1 } if l2GasUsed > uint64(maxDepthInL2Gas) { - release() return nil, lastHeader, nil, ErrDepthLimitExceeded } } else if maxDepthInL2Gas != InfiniteMaxRecreateStateDepth { - release() return nil, lastHeader, nil, err } if logFunc != nil { logFunc(targetHeader, currentHeader, false) } if currentHeader.Number.Uint64() <= genesis { - release() return nil, lastHeader, nil, errors.Wrap(err, fmt.Sprintf("moved beyond genesis looking for state %d, genesis %d", targetHeader.Number.Uint64(), genesis)) } currentHeader = bc.GetHeader(currentHeader.ParentHash, currentHeader.Number.Uint64()-1) if currentHeader == nil { - release() return nil, lastHeader, nil, fmt.Errorf("chain doesn't contain parent of block %d hash %v", lastHeader.Number, lastHeader.Hash()) } } diff --git a/core/state/statedb.go b/core/state/statedb.go index d2e1227a69f0..8cc0ae568576 100644 --- a/core/state/statedb.go +++ b/core/state/statedb.go @@ -21,7 +21,9 @@ import ( "bytes" "fmt" "math/big" + "runtime" "sort" + "sync/atomic" "time" "github.com/ethereum/go-ethereum/common" @@ -145,9 +147,36 @@ type StateDB struct { // Testing hooks onCommit func(states *triestate.Set) // Hook invoked when commit is performed + // arbitrum: reference counted cleanup hook + release func() + releaseRefCount *atomic.Int64 + released bool + deterministic bool } +func (s *StateDB) SetRelease(release func()) { + if s.release == nil { + s.release = release + s.releaseRefCount = new(atomic.Int64) + s.releaseRefCount.Add(1) + } else { + // TODO + panic("StateDB.SetRelease called more then once") + } +} + +func (s *StateDB) Release() { + if s.release != nil && !s.released { + if ref := s.releaseRefCount.Add(-1); ref == 0 { + s.release() + } else { + log.Warn("statedb not release, refCount != 0", ref) + } + s.released = true + } +} + // New creates a new state from a given trie. func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error) { tr, err := db.OpenTrie(root) @@ -175,7 +204,14 @@ func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error) accessList: newAccessList(), transientStorage: newTransientStorage(), hasher: crypto.NewKeccakState(), + + release: nil, + releaseRefCount: nil, + released: false, } + runtime.SetFinalizer(sdb, func(s *StateDB) { + s.Release() + }) if sdb.snaps != nil { sdb.snap = sdb.snaps.Snapshot(root) } @@ -752,7 +788,17 @@ func (s *StateDB) Copy() *StateDB { // miner to operate trie-backed only. snaps: s.snaps, snap: s.snap, + + release: s.release, + releaseRefCount: s.releaseRefCount, + released: false, + } + if s.releaseRefCount != nil { + s.releaseRefCount.Add(1) } + runtime.SetFinalizer(state, func(s *StateDB) { + s.Release() + }) // Copy the dirty states, logs, and preimages for addr := range s.journal.dirties { // As documented [here](https://github.com/ethereum/go-ethereum/pull/16485#issuecomment-380438527), diff --git a/eth/state_accessor.go b/eth/state_accessor.go index 812cd8b367c7..9a0a49f411c6 100644 --- a/eth/state_accessor.go +++ b/eth/state_accessor.go @@ -69,8 +69,8 @@ func (eth *Ethereum) hashState(ctx context.Context, block *types.Block, reexec u database = state.NewDatabaseWithNodeDB(eth.chainDb, triedb) // there is no need of referencing baseBlock state as it's read from disk statedb, err = state.New(baseBlock.Root(), database, nil) - if err == nil { - return statedb, noopReleaser, nil + if err != nil { + return nil, nil, fmt.Errorf("state for base block missing: %w", err) } report = false current = baseBlock diff --git a/graphql/graphql.go b/graphql/graphql.go index ae4e5314d480..86ef4091109d 100644 --- a/graphql/graphql.go +++ b/graphql/graphql.go @@ -88,6 +88,9 @@ type Account struct { // getState fetches the StateDB object for an account. func (a *Account) getState(ctx context.Context) (*state.StateDB, error) { state, _, err := a.r.backend.StateAndHeaderByNumberOrHash(ctx, a.blockNrOrHash) + if state != nil && err == nil { + defer state.Release() + } return state, err } diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index 87a18f131bd9..2eb718f2e4a5 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -630,6 +630,7 @@ func (s *BlockChainAPI) GetBalance(ctx context.Context, address common.Address, } return nil, err } + defer state.Release() return (*hexutil.Big)(state.GetBalance(address)), state.Error() } @@ -686,6 +687,7 @@ func (s *BlockChainAPI) GetProof(ctx context.Context, address common.Address, st if state == nil || err != nil { return nil, err } + defer state.Release() if storageRoot := state.GetStorageRoot(address); storageRoot != types.EmptyRootHash && storageRoot != (common.Hash{}) { id := trie.StorageTrieID(header.Root, crypto.Keccak256Hash(address.Bytes()), storageRoot) tr, err := trie.NewStateTrie(id, state.Database().TrieDB()) @@ -884,6 +886,7 @@ func (s *BlockChainAPI) GetCode(ctx context.Context, address common.Address, blo } return nil, err } + defer state.Release() code := state.GetCode(address) return code, state.Error() } @@ -905,6 +908,7 @@ func (s *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Address } return nil, err } + defer state.Release() res := state.GetState(address, key) return res[:], state.Error() } @@ -1186,6 +1190,7 @@ func DoCall(ctx context.Context, b Backend, args TransactionArgs, blockNrOrHash if state == nil || err != nil { return nil, err } + defer state.Release() header = updateHeaderForPendingBlocks(blockNrOrHash, header) return doCall(ctx, b, args, state, header, overrides, blockOverrides, timeout, globalGasCap, runMode) @@ -1315,6 +1320,7 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr if state == nil || err != nil { return 0, err } + defer state.Release() if err := overrides.Apply(state); err != nil { return 0, err } @@ -1351,6 +1357,7 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr if state == nil || err != nil { return 0, err } + defer state.Release() gasCap, err = args.L2OnlyGasCap(gasCap, header, state, core.MessageGasEstimationMode) if err != nil { return 0, err @@ -1806,6 +1813,7 @@ func AccessList(ctx context.Context, b Backend, blockNrOrHash rpc.BlockNumberOrH if db == nil || err != nil { return nil, 0, nil, err } + defer db.Release() // If the gas amount is not set, default to RPC gas cap. if args.Gas == nil { tmp := hexutil.Uint64(b.RPCGasCap()) @@ -1945,6 +1953,7 @@ func (s *TransactionAPI) GetTransactionCount(ctx context.Context, address common } return nil, err } + defer state.Release() nonce := state.GetNonce(address) return (*hexutil.Uint64)(&nonce), state.Error() } From d1db8eb8f16ccc13eb56add844ed0404b6292e6a Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 9 Feb 2024 16:45:39 +0000 Subject: [PATCH 12/28] add todo comment --- core/state/statedb.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/core/state/statedb.go b/core/state/statedb.go index 8cc0ae568576..361ce0707e9c 100644 --- a/core/state/statedb.go +++ b/core/state/statedb.go @@ -171,7 +171,8 @@ func (s *StateDB) Release() { if ref := s.releaseRefCount.Add(-1); ref == 0 { s.release() } else { - log.Warn("statedb not release, refCount != 0", ref) + //TODO remove + log.Warn("statedb not released, refCount != 0", ref) } s.released = true } From f1fff92aa346aa8458b2de4d555fa1fbb5e1e877 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 16 Feb 2024 14:47:15 +0000 Subject: [PATCH 13/28] fix isolation of live state database --- arbitrum/apibackend.go | 57 +++++++++++++++++++++++++++++------------- core/state/statedb.go | 2 +- eth/state_accessor.go | 11 +------- 3 files changed, 42 insertions(+), 28 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 6f3ec55a789f..46cabb6bf407 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -15,6 +15,7 @@ import ( "github.com/ethereum/go-ethereum/eth" "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/log" + "github.com/ethereum/go-ethereum/trie" "github.com/ethereum/go-ethereum/accounts" "github.com/ethereum/go-ethereum/common" @@ -23,6 +24,7 @@ import ( "github.com/ethereum/go-ethereum/core/bloombits" "github.com/ethereum/go-ethereum/core/rawdb" "github.com/ethereum/go-ethereum/core/state" + "github.com/ethereum/go-ethereum/core/state/snapshot" "github.com/ethereum/go-ethereum/core/types" "github.com/ethereum/go-ethereum/core/vm" "github.com/ethereum/go-ethereum/eth/filters" @@ -450,32 +452,53 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types return nil, header, types.ErrUseFallback } bc := a.BlockChain() - stateFor := func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) { - if header.Root != (common.Hash{}) { - // Try referencing the root, if it isn't in dirties cache then Reference will have no effect - bc.StateCache().TrieDB().Reference(header.Root, common.Hash{}) - } - statedb, err := state.New(header.Root, bc.StateCache(), bc.Snapshots()) - if err != nil { - return nil, noopStateRelease, err - } - if header.Root != (common.Hash{}) { - headerRoot := header.Root - return statedb, func() { bc.StateCache().TrieDB().Dereference(headerRoot) }, nil + stateFor := func(db state.Database, snapshots *snapshot.Tree) func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) { + return func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) { + if header.Root != (common.Hash{}) { + // Try referencing the root, if it isn't in dirties cache then Reference will have no effect + db.TrieDB().Reference(header.Root, common.Hash{}) + } + statedb, err := state.New(header.Root, db, snapshots) + if err != nil { + return nil, nil, err + } + if header.Root != (common.Hash{}) { + headerRoot := header.Root + return statedb, func() { db.TrieDB().Dereference(headerRoot) }, nil + } + return statedb, noopStateRelease, nil } - return statedb, noopStateRelease, nil } - lastState, lastHeader, lastStateRelease, err := FindLastAvailableState(ctx, bc, stateFor, header, nil, a.b.config.MaxRecreateStateDepth) + liveState, liveStateRelease, err := stateFor(bc.StateCache(), bc.Snapshots())(header) + if err == nil { + a.liveStateFinalizers.Add(1) + liveState.SetRelease(func() { + a.liveStateFinalizers.Add(-1) + // TODO remove logs and counters + log.Debug("Live state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + liveStateRelease() + }) + log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + return liveState, header, nil + } + // else err != nil => we don't need to call liveStateRelease + + // Create an ephemeral trie.Database for isolating the live one + // note: triedb cleans cache is disabled in trie.HashDefaults + // note: only states committed to diskdb can be found as we're creating new triedb + // note: snapshots are not used here + ephemeral := state.NewDatabaseWithConfig(a.ChainDb(), trie.HashDefaults) + lastState, lastHeader, lastStateRelease, err := FindLastAvailableState(ctx, bc, stateFor(ephemeral, nil), header, nil, a.b.config.MaxRecreateStateDepth) if err != nil { return nil, nil, err } + // make sure that we haven't found the state in diskdb if lastHeader == header { - // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream a.liveStateFinalizers.Add(1) lastState.SetRelease(func() { a.liveStateFinalizers.Add(-1) // TODO remove logs and counters - log.Debug("Live state relase called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + log.Debug("Live state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) lastStateRelease() }) log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) @@ -492,7 +515,7 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } reexec := uint64(0) checkLive := false - preferDisk := true + preferDisk := false // preferDisk is ignored in this case statedb, release, err := eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, targetBlock, reexec, lastState, lastBlock, checkLive, preferDisk) if err != nil { return nil, nil, err diff --git a/core/state/statedb.go b/core/state/statedb.go index 361ce0707e9c..da65123e6db1 100644 --- a/core/state/statedb.go +++ b/core/state/statedb.go @@ -172,7 +172,7 @@ func (s *StateDB) Release() { s.release() } else { //TODO remove - log.Warn("statedb not released, refCount != 0", ref) + log.Warn("statedb not released, refCount != 0", "refcnt", ref) } s.released = true } diff --git a/eth/state_accessor.go b/eth/state_accessor.go index 9a0a49f411c6..b96a585ffdc3 100644 --- a/eth/state_accessor.go +++ b/eth/state_accessor.go @@ -64,16 +64,7 @@ func (eth *Ethereum) hashState(ctx context.Context, block *types.Block, reexec u // try to construct/recover the state over an ephemeral trie.Database for // isolating the live one. if baseBlock != nil { - // Create an ephemeral trie.Database for isolating the live one - triedb = trie.NewDatabase(eth.chainDb, trie.HashDefaults) - database = state.NewDatabaseWithNodeDB(eth.chainDb, triedb) - // there is no need of referencing baseBlock state as it's read from disk - statedb, err = state.New(baseBlock.Root(), database, nil) - if err != nil { - return nil, nil, fmt.Errorf("state for base block missing: %w", err) - } - report = false - current = baseBlock + current, statedb, database, triedb, report = baseBlock, base, base.Database(), base.Database().TrieDB(), false } else if base != nil { if preferDisk { // Create an ephemeral trie.Database for isolating the live one. Otherwise From e10de759ea54fb8e759d004cde5d21b037af32cb Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 16 Feb 2024 15:34:52 +0000 Subject: [PATCH 14/28] arbitrum/apibackend: add live and ephemeral states metrics --- arbitrum/apibackend.go | 37 ++++++++++++++++++++++--------------- 1 file changed, 22 insertions(+), 15 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 46cabb6bf407..0a329da2cf70 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -15,6 +15,7 @@ import ( "github.com/ethereum/go-ethereum/eth" "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/log" + "github.com/ethereum/go-ethereum/metrics" "github.com/ethereum/go-ethereum/trie" "github.com/ethereum/go-ethereum/accounts" @@ -35,6 +36,13 @@ import ( "github.com/ethereum/go-ethereum/rpc" ) +var ( + referencedLiveStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live/referenced", nil) + releasedLiveStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live/released", nil) + referencedEphemermalStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/ephemeral/referenced", nil) + releasedEphemermalStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/ephemeral/released", nil) +) + type APIBackend struct { b *Backend @@ -471,12 +479,12 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } liveState, liveStateRelease, err := stateFor(bc.StateCache(), bc.Snapshots())(header) if err == nil { - a.liveStateFinalizers.Add(1) + referencedLiveStatesCounter.Inc(1) liveState.SetRelease(func() { - a.liveStateFinalizers.Add(-1) // TODO remove logs and counters log.Debug("Live state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) liveStateRelease() + releasedLiveStatesCounter.Inc(1) }) log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) return liveState, header, nil @@ -494,12 +502,12 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } // make sure that we haven't found the state in diskdb if lastHeader == header { - a.liveStateFinalizers.Add(1) + referencedEphemermalStatesCounter.Inc(1) lastState.SetRelease(func() { - a.liveStateFinalizers.Add(-1) // TODO remove logs and counters log.Debug("Live state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) lastStateRelease() + releasedEphemermalStatesCounter.Inc(1) }) log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) return lastState, header, nil @@ -518,19 +526,18 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types preferDisk := false // preferDisk is ignored in this case statedb, release, err := eth.NewArbEthereum(a.b.arb.BlockChain(), a.ChainDb()).StateAtBlock(ctx, targetBlock, reexec, lastState, lastBlock, checkLive, preferDisk) if err != nil { - return nil, nil, err + return nil, nil, fmt.Errorf("failed to recreate state: %w", err) } // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream - if header.Root != (common.Hash{}) { - a.recreatedStateFinalizers.Add(1) - statedb.SetRelease(func() { - // TODO remove logs and counters - a.recreatedStateFinalizers.Add(-1) - log.Debug("Recreated state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) - release() - }) - log.Debug("Recreated state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) - } + referencedEphemermalStatesCounter.Inc(1) + statedb.SetRelease(func() { + // TODO remove logs and counters + a.recreatedStateFinalizers.Add(-1) + log.Debug("Recreated state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + release() + releasedEphemermalStatesCounter.Inc(1) + }) + log.Debug("Recreated state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) return statedb, header, err } From 94db0717f01d1473fdf421e0d3c255d6e330340d Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 16 Feb 2024 16:03:45 +0000 Subject: [PATCH 15/28] backport upstream hashdb locking changes (#28542) --- trie/triedb/hashdb/database.go | 69 +++++++++------------------------- 1 file changed, 18 insertions(+), 51 deletions(-) diff --git a/trie/triedb/hashdb/database.go b/trie/triedb/hashdb/database.go index be83c32e65c2..6ebd630a23f5 100644 --- a/trie/triedb/hashdb/database.go +++ b/trie/triedb/hashdb/database.go @@ -82,11 +82,6 @@ var Defaults = &Config{ // Database is an intermediate write layer between the trie data structures and // the disk database. The aim is to accumulate trie writes in-memory and only // periodically flush a couple tries to disk, garbage collecting the remainder. -// -// Note, the trie Database is **not** thread safe in its mutations, but it **is** -// thread safe in providing individual, independent node access. The rationale -// behind this split design is to provide read access to RPC handlers and sync -// servers even while the trie is executing expensive garbage collection. type Database struct { diskdb ethdb.Database // Persistent storage for matured trie nodes resolver ChildResolver // The handler to resolve children of nodes @@ -113,7 +108,7 @@ type Database struct { // cachedNode is all the information we know about a single cached trie node // in the memory database write layer. type cachedNode struct { - node []byte // Encoded node blob + node []byte // Encoded node blob, immutable parents uint32 // Number of live nodes referencing this one external map[common.Hash]struct{} // The set of external children flushPrev common.Hash // Previous node in the flush-list @@ -152,9 +147,9 @@ func New(diskdb ethdb.Database, config *Config, resolver ChildResolver) *Databas } } -// insert inserts a simplified trie node into the memory database. -// All nodes inserted by this function will be reference tracked -// and in theory should only used for **trie nodes** insertion. +// insert inserts a trie node into the memory database. All nodes inserted by +// this function will be reference tracked. This function assumes the lock is +// already held. func (db *Database) insert(hash common.Hash, node []byte) { // If the node's already cached, skip if _, ok := db.dirties[hash]; ok { @@ -183,7 +178,7 @@ func (db *Database) insert(hash common.Hash, node []byte) { db.dirtiesSize += common.StorageSize(common.HashLength + len(node)) } -// Node retrieves an encoded cached trie node from memory. If it cannot be found +// node retrieves an encoded cached trie node from memory. If it cannot be found // cached, the method queries the persistent database for the content. func (db *Database) Node(hash common.Hash) ([]byte, error) { // It doesn't make sense to retrieve the metaroot @@ -198,11 +193,14 @@ func (db *Database) Node(hash common.Hash) ([]byte, error) { return enc, nil } } - // Retrieve the node from the dirty cache if available + // Retrieve the node from the dirty cache if available. db.lock.RLock() dirty := db.dirties[hash] db.lock.RUnlock() + // Return the cached node if it's found in the dirty set. + // The dirty.node field is immutable and safe to read it + // even without lock guard. if dirty != nil { memcacheDirtyHitMeter.Mark(1) memcacheDirtyReadMeter.Mark(int64(len(dirty.node))) @@ -223,20 +221,6 @@ func (db *Database) Node(hash common.Hash) ([]byte, error) { return nil, errors.New("not found") } -// Nodes retrieves the hashes of all the nodes cached within the memory database. -// This method is extremely expensive and should only be used to validate internal -// states in test code. -func (db *Database) Nodes() []common.Hash { - db.lock.RLock() - defer db.lock.RUnlock() - - var hashes = make([]common.Hash, 0, len(db.dirties)) - for hash := range db.dirties { - hashes = append(hashes, hash) - } - return hashes -} - // Reference adds a new reference from a parent node to a child node. // This function is used to add reference between internal trie node // and external node(e.g. storage trie root), all internal trie nodes @@ -344,33 +328,28 @@ func (db *Database) dereference(hash common.Hash) { // Cap iteratively flushes old but still referenced trie nodes until the total // memory usage goes below the given threshold. -// -// Note, this method is a non-synchronized mutator. It is unsafe to call this -// concurrently with other mutators. func (db *Database) Cap(limit common.StorageSize) error { + db.lock.Lock() + defer db.lock.Unlock() + // Create a database batch to flush persistent data out. It is important that // outside code doesn't see an inconsistent state (referenced data removed from // memory cache during commit but not yet in persistent storage). This is ensured // by only uncaching existing data when the database write finalizes. - start := time.Now() batch := db.diskdb.NewBatch() - db.lock.RLock() - nodes, storage := len(db.dirties), db.dirtiesSize + nodes, storage, start := len(db.dirties), db.dirtiesSize, time.Now() // db.dirtiesSize only contains the useful data in the cache, but when reporting // the total memory consumption, the maintenance metadata is also needed to be // counted. size := db.dirtiesSize + common.StorageSize(len(db.dirties)*cachedNodeSize) size += db.childrenSize - db.lock.RUnlock() // Keep committing nodes from the flush-list until we're below allowance oldest := db.oldest for size > limit && oldest != (common.Hash{}) { // Fetch the oldest referenced node and push into the batch - db.lock.RLock() node := db.dirties[oldest] - db.lock.RUnlock() rawdb.WriteLegacyTrieNode(batch, oldest, node.node) // If we exceeded the ideal batch size, commit and reset @@ -396,9 +375,6 @@ func (db *Database) Cap(limit common.StorageSize) error { return err } // Write successful, clear out the flushed data - db.lock.Lock() - defer db.lock.Unlock() - for db.oldest != oldest { node := db.dirties[db.oldest] delete(db.dirties, db.oldest) @@ -429,14 +405,13 @@ func (db *Database) Cap(limit common.StorageSize) error { // Commit iterates over all the children of a particular node, writes them out // to disk, forcefully tearing down all references in both directions. As a side // effect, all pre-images accumulated up to this point are also written. -// -// Note, this method is a non-synchronized mutator. It is unsafe to call this -// concurrently with other mutators. func (db *Database) Commit(node common.Hash, report bool) error { if node == (common.Hash{}) { // There's no data to commit in this node return nil } + db.lock.Lock() + defer db.lock.Unlock() // Create a database batch to flush persistent data out. It is important that // outside code doesn't see an inconsistent state (referenced data removed from @@ -446,9 +421,7 @@ func (db *Database) Commit(node common.Hash, report bool) error { batch := db.diskdb.NewBatch() // Move the trie itself into the batch, flushing if enough data is accumulated - db.lock.RLock() nodes, storage := len(db.dirties), db.dirtiesSize - db.lock.RUnlock() uncacher := &cleaner{db} if err := db.commit(node, batch, uncacher); err != nil { @@ -461,8 +434,6 @@ func (db *Database) Commit(node common.Hash, report bool) error { return err } // Uncache any leftovers in the last batch - db.lock.Lock() - defer db.lock.Unlock() if err := batch.Replay(uncacher); err != nil { return err } @@ -490,9 +461,7 @@ func (db *Database) Commit(node common.Hash, report bool) error { // commit is the private locked version of Commit. func (db *Database) commit(hash common.Hash, batch ethdb.Batch, uncacher *cleaner) error { // If the node does not exist, it's a previously committed node - db.lock.RLock() node, ok := db.dirties[hash] - db.lock.RUnlock() if !ok { return nil } @@ -513,13 +482,11 @@ func (db *Database) commit(hash common.Hash, batch ethdb.Batch, uncacher *cleane if err := batch.Write(); err != nil { return err } - db.lock.Lock() err := batch.Replay(uncacher) - batch.Reset() - db.lock.Unlock() if err != nil { return err } + batch.Reset() } return nil } @@ -680,8 +647,8 @@ type reader struct { db *Database } -// Node retrieves the trie node with the given node hash. -// No error will be returned if the node is not found. +// Node retrieves the trie node with the given node hash. No error will be +// returned if the node is not found. func (reader *reader) Node(owner common.Hash, path []byte, hash common.Hash) ([]byte, error) { blob, _ := reader.db.Node(hash) return blob, nil From cf61e26fb7c6b81eb8a1782543ecec489b498faa Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Wed, 21 Feb 2024 12:48:18 +0000 Subject: [PATCH 16/28] update state recreation metrics --- arbitrum/apibackend.go | 28 ++++++++-------------------- eth/state_accessor.go | 11 ++++++++++- 2 files changed, 18 insertions(+), 21 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 0a329da2cf70..8d74000e4f3f 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -37,10 +37,8 @@ import ( ) var ( - referencedLiveStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live/referenced", nil) - releasedLiveStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live/released", nil) - referencedEphemermalStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/ephemeral/referenced", nil) - releasedEphemermalStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/ephemeral/released", nil) + liveStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live", nil) + recreatedStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/recreated", nil) ) type APIBackend struct { @@ -49,8 +47,6 @@ type APIBackend struct { fallbackClient types.FallbackClient sync SyncProgressBackend - // TODO remove - liveStateFinalizers atomic.Int64 recreatedStateFinalizers atomic.Int64 } @@ -479,14 +475,10 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } liveState, liveStateRelease, err := stateFor(bc.StateCache(), bc.Snapshots())(header) if err == nil { - referencedLiveStatesCounter.Inc(1) + liveStatesCounter.Inc(1) liveState.SetRelease(func() { - // TODO remove logs and counters - log.Debug("Live state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) liveStateRelease() - releasedLiveStatesCounter.Inc(1) }) - log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) return liveState, header, nil } // else err != nil => we don't need to call liveStateRelease @@ -502,14 +494,10 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } // make sure that we haven't found the state in diskdb if lastHeader == header { - referencedEphemermalStatesCounter.Inc(1) + liveStatesCounter.Inc(1) lastState.SetRelease(func() { - // TODO remove logs and counters - log.Debug("Live state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) lastStateRelease() - releasedEphemermalStatesCounter.Inc(1) }) - log.Debug("Live state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) return lastState, header, nil } defer lastStateRelease() @@ -529,15 +517,15 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types return nil, nil, fmt.Errorf("failed to recreate state: %w", err) } // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream - referencedEphemermalStatesCounter.Inc(1) + recreatedStatesCounter.Inc(1) + a.recreatedStateFinalizers.Add(1) statedb.SetRelease(func() { // TODO remove logs and counters a.recreatedStateFinalizers.Add(-1) - log.Debug("Recreated state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + log.Warn("Recreated state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load()) release() - releasedEphemermalStatesCounter.Inc(1) }) - log.Debug("Recreated state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load(), "liveStateFinalizers", a.liveStateFinalizers.Load()) + log.Warn("Recreated state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load()) return statedb, header, err } diff --git a/eth/state_accessor.go b/eth/state_accessor.go index b96a585ffdc3..439d88c2d8c4 100644 --- a/eth/state_accessor.go +++ b/eth/state_accessor.go @@ -30,9 +30,15 @@ import ( "github.com/ethereum/go-ethereum/core/vm" "github.com/ethereum/go-ethereum/eth/tracers" "github.com/ethereum/go-ethereum/log" + "github.com/ethereum/go-ethereum/metrics" "github.com/ethereum/go-ethereum/trie" ) +var ( + recreatedStatesCounter = metrics.NewRegisteredCounter("eth/stateaccessor/recreated/states", nil) + recreatedBytesMeter = metrics.NewRegisteredMeter("eth/stateaccessor/recreated/bytes", nil) +) + // noopReleaser is returned in case there is no operation expected // for releasing state. var noopReleaser = tracers.StateReleaseFunc(func() {}) @@ -171,10 +177,13 @@ func (eth *Ethereum) hashState(ctx context.Context, block *types.Block, reexec u } parent = root } + _, nodes, imgs := triedb.Size() // all memory is contained within the nodes return in hashdb if report { - _, nodes, imgs := triedb.Size() // all memory is contained within the nodes return in hashdb log.Info("Historical state regenerated", "block", current.NumberU64(), "elapsed", time.Since(start), "nodes", nodes, "preimages", imgs) } + recreatedStatesCounter.Inc(1) + recreatedBytesMeter.Mark(int64(nodes)) + return statedb, func() { triedb.Dereference(block.Root()) }, nil } From 27e28ede0056e5cb13b6fd97313781e8ef1b853f Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Thu, 22 Feb 2024 00:31:16 +0000 Subject: [PATCH 17/28] simplify StateDB.Release, don't set finalizer as it keeps StateDB live too long --- core/state/statedb.go | 36 ++++++------------------------------ 1 file changed, 6 insertions(+), 30 deletions(-) diff --git a/core/state/statedb.go b/core/state/statedb.go index da65123e6db1..d771609ba3f6 100644 --- a/core/state/statedb.go +++ b/core/state/statedb.go @@ -21,9 +21,7 @@ import ( "bytes" "fmt" "math/big" - "runtime" "sort" - "sync/atomic" "time" "github.com/ethereum/go-ethereum/common" @@ -147,10 +145,9 @@ type StateDB struct { // Testing hooks onCommit func(states *triestate.Set) // Hook invoked when commit is performed - // arbitrum: reference counted cleanup hook - release func() - releaseRefCount *atomic.Int64 - released bool + // arbitrum: cleanup hook + release func() + released bool deterministic bool } @@ -158,8 +155,6 @@ type StateDB struct { func (s *StateDB) SetRelease(release func()) { if s.release == nil { s.release = release - s.releaseRefCount = new(atomic.Int64) - s.releaseRefCount.Add(1) } else { // TODO panic("StateDB.SetRelease called more then once") @@ -168,12 +163,7 @@ func (s *StateDB) SetRelease(release func()) { func (s *StateDB) Release() { if s.release != nil && !s.released { - if ref := s.releaseRefCount.Add(-1); ref == 0 { - s.release() - } else { - //TODO remove - log.Warn("statedb not released, refCount != 0", "refcnt", ref) - } + s.release() s.released = true } } @@ -206,13 +196,9 @@ func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error) transientStorage: newTransientStorage(), hasher: crypto.NewKeccakState(), - release: nil, - releaseRefCount: nil, - released: false, + release: nil, + released: false, } - runtime.SetFinalizer(sdb, func(s *StateDB) { - s.Release() - }) if sdb.snaps != nil { sdb.snap = sdb.snaps.Snapshot(root) } @@ -789,17 +775,7 @@ func (s *StateDB) Copy() *StateDB { // miner to operate trie-backed only. snaps: s.snaps, snap: s.snap, - - release: s.release, - releaseRefCount: s.releaseRefCount, - released: false, - } - if s.releaseRefCount != nil { - s.releaseRefCount.Add(1) } - runtime.SetFinalizer(state, func(s *StateDB) { - s.Release() - }) // Copy the dirty states, logs, and preimages for addr := range s.journal.dirties { // As documented [here](https://github.com/ethereum/go-ethereum/pull/16485#issuecomment-380438527), From 3e1f80f784e2f4e2109b80a4e214cb029086dc26 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 23 Feb 2024 10:50:13 +0000 Subject: [PATCH 18/28] bring back AdvanceStateUpToBlock --- arbitrum/apibackend.go | 2 +- arbitrum/recordingdb.go | 2 +- arbitrum/recreatestate.go | 25 +++++++++++++++++++++++-- 3 files changed, 25 insertions(+), 4 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 8d74000e4f3f..4dd7bbec6cb6 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -470,7 +470,7 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types headerRoot := header.Root return statedb, func() { db.TrieDB().Dereference(headerRoot) }, nil } - return statedb, noopStateRelease, nil + return statedb, NoopStateRelease, nil } } liveState, liveStateRelease, err := stateFor(bc.StateCache(), bc.Snapshots())(header) diff --git a/arbitrum/recordingdb.go b/arbitrum/recordingdb.go index 7d89b14c6d68..30cafe5a7765 100644 --- a/arbitrum/recordingdb.go +++ b/arbitrum/recordingdb.go @@ -312,7 +312,7 @@ func (r *RecordingDatabase) GetOrRecreateState(ctx context.Context, header *type stateFor := func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) { state, err := r.StateFor(header) // we don't use the release functor pattern here yet - return state, noopStateRelease, err + return state, NoopStateRelease, err } state, currentHeader, _, err := FindLastAvailableState(ctx, r.bc, stateFor, header, logFunc, -1) if err != nil { diff --git a/arbitrum/recreatestate.go b/arbitrum/recreatestate.go index 7d10c9c9e8ca..bd16c9d3a9a5 100644 --- a/arbitrum/recreatestate.go +++ b/arbitrum/recreatestate.go @@ -19,7 +19,7 @@ var ( type StateReleaseFunc tracers.StateReleaseFunc -var noopStateRelease StateReleaseFunc = func() {} +var NoopStateRelease StateReleaseFunc = func() {} type StateBuildingLogFunction func(targetHeader, header *types.Header, hasState bool) type StateForHeaderFunction func(header *types.Header) (*state.StateDB, StateReleaseFunc, error) @@ -34,7 +34,7 @@ func FindLastAvailableState(ctx context.Context, bc *core.BlockChain, stateFor S var state *state.StateDB var err error var l2GasUsed uint64 - release := noopStateRelease + release := NoopStateRelease for ctx.Err() == nil { lastHeader := currentHeader state, release, err = stateFor(currentHeader) @@ -86,3 +86,24 @@ func AdvanceStateByBlock(ctx context.Context, bc *core.BlockChain, state *state. } return state, block, nil } + +func AdvanceStateUpToBlock(ctx context.Context, bc *core.BlockChain, state *state.StateDB, targetHeader *types.Header, lastAvailableHeader *types.Header, logFunc StateBuildingLogFunction) (*state.StateDB, error) { + returnedBlockNumber := targetHeader.Number.Uint64() + blockToRecreate := lastAvailableHeader.Number.Uint64() + 1 + prevHash := lastAvailableHeader.Hash() + for ctx.Err() == nil { + state, block, err := AdvanceStateByBlock(ctx, bc, state, targetHeader, blockToRecreate, prevHash, logFunc) + if err != nil { + return nil, err + } + prevHash = block.Hash() + if blockToRecreate >= returnedBlockNumber { + if block.Hash() != targetHeader.Hash() { + return nil, fmt.Errorf("blockHash doesn't match when recreating number: %d expected: %v got: %v", blockToRecreate, targetHeader.Hash(), block.Hash()) + } + return state, nil + } + blockToRecreate++ + } + return nil, ctx.Err() +} From af4fb22f472ef9242b69a8bf1a17ac3ff0ebb661 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 23 Feb 2024 11:11:05 +0000 Subject: [PATCH 19/28] cleanup debug log --- arbitrum/apibackend.go | 8 -------- 1 file changed, 8 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 4dd7bbec6cb6..ac5cce70da49 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -7,7 +7,6 @@ import ( "math/big" "strconv" "strings" - "sync/atomic" "time" "github.com/ethereum/go-ethereum" @@ -46,8 +45,6 @@ type APIBackend struct { fallbackClient types.FallbackClient sync SyncProgressBackend - - recreatedStateFinalizers atomic.Int64 } type timeoutFallbackClient struct { @@ -518,14 +515,9 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream recreatedStatesCounter.Inc(1) - a.recreatedStateFinalizers.Add(1) statedb.SetRelease(func() { - // TODO remove logs and counters - a.recreatedStateFinalizers.Add(-1) - log.Warn("Recreated state release called", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load()) release() }) - log.Warn("Recreated state release set", "recreatedStateFinalizers", a.recreatedStateFinalizers.Load()) return statedb, header, err } From a2c45fb6efef261bf73e87eac39f51ebb55be388 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Fri, 23 Feb 2024 11:15:03 +0000 Subject: [PATCH 20/28] don't panic if StateDB.SetRelease is called more then once --- core/state/statedb.go | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/core/state/statedb.go b/core/state/statedb.go index d771609ba3f6..4763f3c83198 100644 --- a/core/state/statedb.go +++ b/core/state/statedb.go @@ -156,8 +156,7 @@ func (s *StateDB) SetRelease(release func()) { if s.release == nil { s.release = release } else { - // TODO - panic("StateDB.SetRelease called more then once") + log.Error("StateDB.SetRelease called more then once, may cause memory leak") } } From cb9f8f577c2cf0db3789783dfd7c95bc99de12a0 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Thu, 29 Feb 2024 16:34:28 +0000 Subject: [PATCH 21/28] expose hashdb.Database.node method --- trie/triedb/hashdb/database.go | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/trie/triedb/hashdb/database.go b/trie/triedb/hashdb/database.go index 6ebd630a23f5..78df83de2d5e 100644 --- a/trie/triedb/hashdb/database.go +++ b/trie/triedb/hashdb/database.go @@ -180,7 +180,7 @@ func (db *Database) insert(hash common.Hash, node []byte) { // node retrieves an encoded cached trie node from memory. If it cannot be found // cached, the method queries the persistent database for the content. -func (db *Database) Node(hash common.Hash) ([]byte, error) { +func (db *Database) node(hash common.Hash) ([]byte, error) { // It doesn't make sense to retrieve the metaroot if hash == (common.Hash{}) { return nil, errors.New("not found") @@ -221,6 +221,11 @@ func (db *Database) Node(hash common.Hash) ([]byte, error) { return nil, errors.New("not found") } +// arbitrum: exposing hashdb.Database.Node for triedb.Database.Node currently used by arbitrum.RecordingKV.Get +func (db *Database) Node(hash common.Hash) ([]byte, error) { + return db.node(hash) +} + // Reference adds a new reference from a parent node to a child node. // This function is used to add reference between internal trie node // and external node(e.g. storage trie root), all internal trie nodes @@ -555,7 +560,7 @@ func (db *Database) Initialized(genesisRoot common.Hash) bool { func (db *Database) Update(root common.Hash, parent common.Hash, block uint64, nodes *trienode.MergedNodeSet, states *triestate.Set) error { // Ensure the parent state is present and signal a warning if not. if parent != types.EmptyRootHash { - if blob, _ := db.Node(parent); len(blob) == 0 { + if blob, _ := db.node(parent); len(blob) == 0 { log.Error("parent state is not present") } } @@ -636,7 +641,7 @@ func (db *Database) Scheme() string { // Reader retrieves a node reader belonging to the given state root. // An error will be returned if the requested state is not available. func (db *Database) Reader(root common.Hash) (*reader, error) { - if _, err := db.Node(root); err != nil { + if _, err := db.node(root); err != nil { return nil, fmt.Errorf("state %#x is not available, %v", root, err) } return &reader{db: db}, nil @@ -650,6 +655,6 @@ type reader struct { // Node retrieves the trie node with the given node hash. No error will be // returned if the node is not found. func (reader *reader) Node(owner common.Hash, path []byte, hash common.Hash) ([]byte, error) { - blob, _ := reader.db.Node(hash) + blob, _ := reader.db.node(hash) return blob, nil } From b432ef83821041f713842ac2b93f39c0be504959 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Wed, 6 Mar 2024 12:54:24 +0000 Subject: [PATCH 22/28] bring back working finalizers --- arbitrum/apibackend.go | 22 +++++++++------ core/state/statedb.go | 51 +++++++++++----------------------- core/state/statedb_arbitrum.go | 12 +++++++- graphql/graphql.go | 3 -- internal/ethapi/api.go | 9 ------ 5 files changed, 40 insertions(+), 57 deletions(-) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index ac5cce70da49..6000679582ef 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -36,8 +36,10 @@ import ( ) var ( - liveStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live", nil) - recreatedStatesCounter = metrics.NewRegisteredCounter("arb/apibackend/states/recreated", nil) + liveStatesReferencedCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live/referenced", nil) + liveStatesDereferencedCounter = metrics.NewRegisteredCounter("arb/apibackend/states/live/dereferenced", nil) + recreatedStatesReferencedCounter = metrics.NewRegisteredCounter("arb/apibackend/states/recreated/referenced", nil) + recreatedStatesDereferencedCounter = metrics.NewRegisteredCounter("arb/apibackend/states/recreated/dereferenced", nil) ) type APIBackend struct { @@ -242,7 +244,6 @@ func (a *APIBackend) FeeHistory( return common.Big0, nil, nil, nil, err } speedLimit, err := core.GetArbOSSpeedLimitPerSecond(state) - state.Release() if err != nil { return common.Big0, nil, nil, nil, err } @@ -472,9 +473,10 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } liveState, liveStateRelease, err := stateFor(bc.StateCache(), bc.Snapshots())(header) if err == nil { - liveStatesCounter.Inc(1) - liveState.SetRelease(func() { + liveStatesReferencedCounter.Inc(1) + liveState.SetArbFinalizer(func(*state.ArbitrumExtraData) { liveStateRelease() + liveStatesDereferencedCounter.Inc(1) }) return liveState, header, nil } @@ -491,9 +493,10 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types } // make sure that we haven't found the state in diskdb if lastHeader == header { - liveStatesCounter.Inc(1) - lastState.SetRelease(func() { + liveStatesReferencedCounter.Inc(1) + lastState.SetArbFinalizer(func(*state.ArbitrumExtraData) { lastStateRelease() + liveStatesDereferencedCounter.Inc(1) }) return lastState, header, nil } @@ -514,9 +517,10 @@ func (a *APIBackend) stateAndHeaderFromHeader(ctx context.Context, header *types return nil, nil, fmt.Errorf("failed to recreate state: %w", err) } // we are setting finalizer instead of returning a StateReleaseFunc to avoid changing ethapi.Backend interface to minimize diff to upstream - recreatedStatesCounter.Inc(1) - statedb.SetRelease(func() { + recreatedStatesReferencedCounter.Inc(1) + statedb.SetArbFinalizer(func(*state.ArbitrumExtraData) { release() + recreatedStatesDereferencedCounter.Inc(1) }) return statedb, header, err } diff --git a/core/state/statedb.go b/core/state/statedb.go index 4763f3c83198..3c65a61b057e 100644 --- a/core/state/statedb.go +++ b/core/state/statedb.go @@ -62,8 +62,7 @@ type revision struct { // must be created with new root and updated database for accessing post- // commit states. type StateDB struct { - // Arbitrum: track the total balance change across all accounts - unexpectedBalanceDelta *big.Int + arbExtraData *ArbitrumExtraData // must be a pointer - can't be a part of StateDB allocation, otherwise its finalizer might not get called db Database prefetcher *triePrefetcher @@ -145,28 +144,9 @@ type StateDB struct { // Testing hooks onCommit func(states *triestate.Set) // Hook invoked when commit is performed - // arbitrum: cleanup hook - release func() - released bool - deterministic bool } -func (s *StateDB) SetRelease(release func()) { - if s.release == nil { - s.release = release - } else { - log.Error("StateDB.SetRelease called more then once, may cause memory leak") - } -} - -func (s *StateDB) Release() { - if s.release != nil && !s.released { - s.release() - s.released = true - } -} - // New creates a new state from a given trie. func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error) { tr, err := db.OpenTrie(root) @@ -174,7 +154,9 @@ func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error) return nil, err } sdb := &StateDB{ - unexpectedBalanceDelta: new(big.Int), + arbExtraData: &ArbitrumExtraData{ + unexpectedBalanceDelta: new(big.Int), + }, db: db, trie: tr, @@ -194,9 +176,6 @@ func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error) accessList: newAccessList(), transientStorage: newTransientStorage(), hasher: crypto.NewKeccakState(), - - release: nil, - released: false, } if sdb.snaps != nil { sdb.snap = sdb.snaps.Snapshot(root) @@ -417,7 +396,7 @@ func (s *StateDB) HasSelfDestructed(addr common.Address) bool { func (s *StateDB) AddBalance(addr common.Address, amount *big.Int) { stateObject := s.GetOrNewStateObject(addr) if stateObject != nil { - s.unexpectedBalanceDelta.Add(s.unexpectedBalanceDelta, amount) + s.arbExtraData.unexpectedBalanceDelta.Add(s.arbExtraData.unexpectedBalanceDelta, amount) stateObject.AddBalance(amount) } } @@ -426,7 +405,7 @@ func (s *StateDB) AddBalance(addr common.Address, amount *big.Int) { func (s *StateDB) SubBalance(addr common.Address, amount *big.Int) { stateObject := s.GetOrNewStateObject(addr) if stateObject != nil { - s.unexpectedBalanceDelta.Sub(s.unexpectedBalanceDelta, amount) + s.arbExtraData.unexpectedBalanceDelta.Sub(s.arbExtraData.unexpectedBalanceDelta, amount) stateObject.SubBalance(amount) } } @@ -438,8 +417,8 @@ func (s *StateDB) SetBalance(addr common.Address, amount *big.Int) { amount = big.NewInt(0) } prevBalance := stateObject.Balance() - s.unexpectedBalanceDelta.Add(s.unexpectedBalanceDelta, amount) - s.unexpectedBalanceDelta.Sub(s.unexpectedBalanceDelta, prevBalance) + s.arbExtraData.unexpectedBalanceDelta.Add(s.arbExtraData.unexpectedBalanceDelta, amount) + s.arbExtraData.unexpectedBalanceDelta.Sub(s.arbExtraData.unexpectedBalanceDelta, prevBalance) stateObject.SetBalance(amount) } } @@ -448,7 +427,7 @@ func (s *StateDB) ExpectBalanceBurn(amount *big.Int) { if amount.Sign() < 0 { panic(fmt.Sprintf("ExpectBalanceBurn called with negative amount %v", amount)) } - s.unexpectedBalanceDelta.Add(s.unexpectedBalanceDelta, amount) + s.arbExtraData.unexpectedBalanceDelta.Add(s.arbExtraData.unexpectedBalanceDelta, amount) } func (s *StateDB) SetNonce(addr common.Address, nonce uint64) { @@ -510,7 +489,7 @@ func (s *StateDB) SelfDestruct(addr common.Address) { }) stateObject.markSelfdestructed() - s.unexpectedBalanceDelta.Sub(s.unexpectedBalanceDelta, stateObject.data.Balance) + s.arbExtraData.unexpectedBalanceDelta.Sub(s.arbExtraData.unexpectedBalanceDelta, stateObject.data.Balance) stateObject.data.Balance = new(big.Int) } @@ -748,7 +727,9 @@ func (s *StateDB) CreateAccount(addr common.Address) { func (s *StateDB) Copy() *StateDB { // Copy all the basic fields, initialize the memory ones state := &StateDB{ - unexpectedBalanceDelta: new(big.Int).Set(s.unexpectedBalanceDelta), + arbExtraData: &ArbitrumExtraData{ + unexpectedBalanceDelta: new(big.Int).Set(s.arbExtraData.unexpectedBalanceDelta), + }, db: s.db, trie: s.db.CopyTrie(s.trie), @@ -853,7 +834,7 @@ func (s *StateDB) Copy() *StateDB { func (s *StateDB) Snapshot() int { id := s.nextRevisionId s.nextRevisionId++ - s.validRevisions = append(s.validRevisions, revision{id, s.journal.length(), new(big.Int).Set(s.unexpectedBalanceDelta)}) + s.validRevisions = append(s.validRevisions, revision{id, s.journal.length(), new(big.Int).Set(s.arbExtraData.unexpectedBalanceDelta)}) return id } @@ -868,7 +849,7 @@ func (s *StateDB) RevertToSnapshot(revid int) { } revision := s.validRevisions[idx] snapshot := revision.journalIndex - s.unexpectedBalanceDelta = new(big.Int).Set(revision.unexpectedBalanceDelta) + s.arbExtraData.unexpectedBalanceDelta = new(big.Int).Set(revision.unexpectedBalanceDelta) // Replay the journal to undo changes and remove invalidated snapshots s.journal.revert(s, snapshot) @@ -1344,7 +1325,7 @@ func (s *StateDB) Commit(block uint64, deleteEmptyObjects bool) (common.Hash, er s.snap = nil } - s.unexpectedBalanceDelta.Set(new(big.Int)) + s.arbExtraData.unexpectedBalanceDelta.Set(new(big.Int)) if root == (common.Hash{}) { root = types.EmptyRootHash diff --git a/core/state/statedb_arbitrum.go b/core/state/statedb_arbitrum.go index a77e4dd3ae2d..ce4b19b7ae31 100644 --- a/core/state/statedb_arbitrum.go +++ b/core/state/statedb_arbitrum.go @@ -19,6 +19,7 @@ package state import ( "math/big" + "runtime" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/core/types" @@ -26,13 +27,22 @@ import ( "github.com/ethereum/go-ethereum/trie" ) +type ArbitrumExtraData struct { + // track the total balance change across all accounts + unexpectedBalanceDelta *big.Int +} + +func (s *StateDB) SetArbFinalizer(f func(*ArbitrumExtraData)) { + runtime.SetFinalizer(s.arbExtraData, f) +} + func (s *StateDB) GetCurrentTxLogs() []*types.Log { return s.logs[s.thash] } // GetUnexpectedBalanceDelta returns the total unexpected change in balances since the last commit to the database. func (s *StateDB) GetUnexpectedBalanceDelta() *big.Int { - return new(big.Int).Set(s.unexpectedBalanceDelta) + return new(big.Int).Set(s.arbExtraData.unexpectedBalanceDelta) } func (s *StateDB) GetSelfDestructs() []common.Address { diff --git a/graphql/graphql.go b/graphql/graphql.go index 86ef4091109d..ae4e5314d480 100644 --- a/graphql/graphql.go +++ b/graphql/graphql.go @@ -88,9 +88,6 @@ type Account struct { // getState fetches the StateDB object for an account. func (a *Account) getState(ctx context.Context) (*state.StateDB, error) { state, _, err := a.r.backend.StateAndHeaderByNumberOrHash(ctx, a.blockNrOrHash) - if state != nil && err == nil { - defer state.Release() - } return state, err } diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index af085483c235..2d3fe10d1172 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -630,7 +630,6 @@ func (s *BlockChainAPI) GetBalance(ctx context.Context, address common.Address, } return nil, err } - defer state.Release() return (*hexutil.Big)(state.GetBalance(address)), state.Error() } @@ -687,7 +686,6 @@ func (s *BlockChainAPI) GetProof(ctx context.Context, address common.Address, st if state == nil || err != nil { return nil, err } - defer state.Release() if storageRoot := state.GetStorageRoot(address); storageRoot != types.EmptyRootHash && storageRoot != (common.Hash{}) { id := trie.StorageTrieID(header.Root, crypto.Keccak256Hash(address.Bytes()), storageRoot) tr, err := trie.NewStateTrie(id, state.Database().TrieDB()) @@ -886,7 +884,6 @@ func (s *BlockChainAPI) GetCode(ctx context.Context, address common.Address, blo } return nil, err } - defer state.Release() code := state.GetCode(address) return code, state.Error() } @@ -908,7 +905,6 @@ func (s *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Address } return nil, err } - defer state.Release() res := state.GetState(address, key) return res[:], state.Error() } @@ -1200,7 +1196,6 @@ func DoCall(ctx context.Context, b Backend, args TransactionArgs, blockNrOrHash if state == nil || err != nil { return nil, err } - defer state.Release() header = updateHeaderForPendingBlocks(blockNrOrHash, header) return doCall(ctx, b, args, state, header, overrides, blockOverrides, timeout, globalGasCap, runMode) @@ -1330,7 +1325,6 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr if state == nil || err != nil { return 0, err } - defer state.Release() if err := overrides.Apply(state); err != nil { return 0, err } @@ -1367,7 +1361,6 @@ func DoEstimateGas(ctx context.Context, b Backend, args TransactionArgs, blockNr if state == nil || err != nil { return 0, err } - defer state.Release() gasCap, err = args.L2OnlyGasCap(gasCap, header, state, core.MessageGasEstimationMode) if err != nil { return 0, err @@ -1823,7 +1816,6 @@ func AccessList(ctx context.Context, b Backend, blockNrOrHash rpc.BlockNumberOrH if db == nil || err != nil { return nil, 0, nil, err } - defer db.Release() // If the gas amount is not set, default to RPC gas cap. if args.Gas == nil { tmp := hexutil.Uint64(b.RPCGasCap()) @@ -1963,7 +1955,6 @@ func (s *TransactionAPI) GetTransactionCount(ctx context.Context, address common } return nil, err } - defer state.Release() nonce := state.GetNonce(address) return (*hexutil.Uint64)(&nonce), state.Error() } From 4f76df8b3ca8f25056409d661c669f6b04458494 Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Wed, 6 Mar 2024 17:49:48 +0000 Subject: [PATCH 23/28] eth/tracers: add canonical hash check in API.blockByHash --- eth/tracers/api.go | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/eth/tracers/api.go b/eth/tracers/api.go index cfbd683d894e..afeed9f7d228 100644 --- a/eth/tracers/api.go +++ b/eth/tracers/api.go @@ -128,6 +128,15 @@ func (api *API) blockByHash(ctx context.Context, hash common.Hash) (*types.Block if block == nil { return nil, fmt.Errorf("block %s not found", hash.Hex()) } + + number := block.NumberU64() + canonical := rawdb.ReadCanonicalHash(api.backend.ChainDb(), number) + if canonical == (common.Hash{}) { + return nil, fmt.Errorf("canonical hash for block %s not found", hash.Hex()) + } + if hash.Cmp(canonical) != 0 { + return nil, fmt.Errorf("canonical hash doesn't match requested hash, hash %s, canonical %s, block #%d", hash.Hex(), canonical.Hex(), number) + } return block, nil } From dae9eb1802e4ed3aa1afe752d76cc9f634122dcd Mon Sep 17 00:00:00 2001 From: Nodar Ambroladze Date: Tue, 12 Mar 2024 00:32:32 +0100 Subject: [PATCH 24/28] Postfix duplicate solidity function bindings with 4 byte signature --- accounts/abi/bind/bind.go | 22 +++++++++++++++++++++- accounts/abi/bind/bind_test.go | 4 ++-- 2 files changed, 23 insertions(+), 3 deletions(-) diff --git a/accounts/abi/bind/bind.go b/accounts/abi/bind/bind.go index 8a54a0e6ef04..bbc1c7becb4a 100644 --- a/accounts/abi/bind/bind.go +++ b/accounts/abi/bind/bind.go @@ -77,6 +77,21 @@ func isKeyWord(arg string) bool { return true } +func duplicates(methods map[string]abi.Method) map[string]bool { + var ( + identifiers = make(map[string]bool) + dups = make(map[string]bool) + ) + for _, method := range methods { + identifiers, dups := identifiers, dups + if identifiers[method.RawName] { + dups[method.RawName] = true + } + identifiers[method.RawName] = true + } + return dups +} + // Bind generates a Go wrapper around a contract ABI. This wrapper isn't meant // to be used as is in client code, but rather as an intermediate struct which // enforces compile time type safety and naming convention opposed to having to @@ -121,6 +136,7 @@ func Bind(types []string, abis []string, bytecodes []string, fsigs []map[string] callIdentifiers = make(map[string]bool) transactIdentifiers = make(map[string]bool) eventIdentifiers = make(map[string]bool) + dups = duplicates(evmABI.Methods) ) for _, input := range evmABI.Constructor.Inputs { @@ -132,12 +148,16 @@ func Bind(types []string, abis []string, bytecodes []string, fsigs []map[string] for _, original := range evmABI.Methods { // Normalize the method for capital cases and non-anonymous inputs/outputs normalized := original - normalizedName := methodNormalizer[lang](alias(aliases, original.Name)) // Ensure there is no duplicated identifier var identifiers = callIdentifiers if !original.IsConstant() { identifiers = transactIdentifiers } + name := original.RawName + if dups[original.RawName] { + name = fmt.Sprintf("%s%x", original.RawName, original.ID) + } + normalizedName := methodNormalizer[lang](alias(aliases, name)) // Name shouldn't start with a digit. It will make the generated code invalid. if len(normalizedName) > 0 && unicode.IsDigit(rune(normalizedName[0])) { normalizedName = fmt.Sprintf("M%s", normalizedName) diff --git a/accounts/abi/bind/bind_test.go b/accounts/abi/bind/bind_test.go index 1069f3d396d4..3b08bdf8900e 100644 --- a/accounts/abi/bind/bind_test.go +++ b/accounts/abi/bind/bind_test.go @@ -1486,7 +1486,7 @@ var bindTests = []struct { } } }() - contract.Foo(auth, big.NewInt(1), big.NewInt(2)) + contract.Foo04bc52f8(auth, big.NewInt(1), big.NewInt(2)) sim.Commit() select { case n := <-resCh: @@ -1497,7 +1497,7 @@ var bindTests = []struct { t.Fatalf("Wait bar0 event timeout") } - contract.Foo0(auth, big.NewInt(1)) + contract.Foo2fbebd38(auth, big.NewInt(1)) sim.Commit() select { case n := <-resCh: From 57fcba9a2212c137eb17dfcd05308e3922a352ea Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Tue, 12 Mar 2024 21:57:10 +0000 Subject: [PATCH 25/28] add check for recent block in StateAndHeaderByNumberOrHash --- arbitrum/apibackend.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 08cb0a8d000d..581ebf5e8ae5 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -532,6 +532,11 @@ func (a *APIBackend) StateAndHeaderByNumber(ctx context.Context, number rpc.Bloc func (a *APIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) { header, err := a.HeaderByNumberOrHash(ctx, blockNrOrHash) + hash, ishash := blockNrOrHash.Hash() + bc := a.BlockChain() + if ishash && header.Number.Cmp(bc.CurrentBlock().Number) > 0 && bc.GetCanonicalHash(header.Number.Uint64()) != hash { + return nil, nil, errors.New("requested block ahead of current block and the hash is not currently canonical") + } return a.stateAndHeaderFromHeader(ctx, header, err) } From 088149d73d7b39c844050e63f8a9c988ed8bdb2d Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Tue, 12 Mar 2024 22:03:16 +0000 Subject: [PATCH 26/28] add comment --- arbitrum/apibackend.go | 1 + 1 file changed, 1 insertion(+) diff --git a/arbitrum/apibackend.go b/arbitrum/apibackend.go index 581ebf5e8ae5..77af9239f37d 100644 --- a/arbitrum/apibackend.go +++ b/arbitrum/apibackend.go @@ -534,6 +534,7 @@ func (a *APIBackend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOr header, err := a.HeaderByNumberOrHash(ctx, blockNrOrHash) hash, ishash := blockNrOrHash.Hash() bc := a.BlockChain() + // check if we are not trying to get recent state that is not yet triedb referenced or committed in Blockchain.writeBlockWithState if ishash && header.Number.Cmp(bc.CurrentBlock().Number) > 0 && bc.GetCanonicalHash(header.Number.Uint64()) != hash { return nil, nil, errors.New("requested block ahead of current block and the hash is not currently canonical") } From 01b68594f694364d095f9f3d50251f8479df287f Mon Sep 17 00:00:00 2001 From: Maciej Kulawik Date: Tue, 12 Mar 2024 22:29:41 +0000 Subject: [PATCH 27/28] simpify recent block check --- eth/tracers/api.go | 11 +++-------- 1 file changed, 3 insertions(+), 8 deletions(-) diff --git a/eth/tracers/api.go b/eth/tracers/api.go index afeed9f7d228..e8ba938d59ca 100644 --- a/eth/tracers/api.go +++ b/eth/tracers/api.go @@ -128,14 +128,9 @@ func (api *API) blockByHash(ctx context.Context, hash common.Hash) (*types.Block if block == nil { return nil, fmt.Errorf("block %s not found", hash.Hex()) } - - number := block.NumberU64() - canonical := rawdb.ReadCanonicalHash(api.backend.ChainDb(), number) - if canonical == (common.Hash{}) { - return nil, fmt.Errorf("canonical hash for block %s not found", hash.Hex()) - } - if hash.Cmp(canonical) != 0 { - return nil, fmt.Errorf("canonical hash doesn't match requested hash, hash %s, canonical %s, block #%d", hash.Hex(), canonical.Hex(), number) + canonical := rawdb.ReadCanonicalHash(api.backend.ChainDb(), block.NumberU64()) + if hash != canonical { + return nil, fmt.Errorf("hash %s is not currently canonical", hash.Hex()) } return block, nil } From 8f491a030ac7a5bac886a050037d7a893ccff865 Mon Sep 17 00:00:00 2001 From: Ganesh Vanahalli Date: Fri, 15 Mar 2024 15:45:51 -0500 Subject: [PATCH 28/28] arbBlockHash for pending blocks should return latest block hash [NIT-2121] --- internal/ethapi/api.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index 2d3fe10d1172..744e43624bfc 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -1178,13 +1178,14 @@ func doCall(ctx context.Context, b Backend, args TransactionArgs, state *state.S func updateHeaderForPendingBlocks(blockNrOrHash rpc.BlockNumberOrHash, header *types.Header) *types.Header { if blockNrOrHash.BlockNumber != nil && *blockNrOrHash.BlockNumber == rpc.PendingBlockNumber { - headerCopy := *header + headerCopy := types.CopyHeader(header) now := uint64(time.Now().Unix()) if now > headerCopy.Time { headerCopy.Time = now } headerCopy.Number = new(big.Int).Add(headerCopy.Number, common.Big1) - return &headerCopy + headerCopy.ParentHash = header.Hash() + return headerCopy } return header }