Skip to content

Commit

Permalink
don't close triedb too early in eth.StateAtBlock
Browse files Browse the repository at this point in the history
  • Loading branch information
magicxyyz committed Nov 16, 2023
1 parent 6cacfc9 commit 6b6fd12
Showing 1 changed file with 20 additions and 5 deletions.
25 changes: 20 additions & 5 deletions eth/state_accessor.go
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,7 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe
}, nil
}
}
closeTrieDBInDefer := true
// 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.
Expand All @@ -85,10 +86,15 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe
// Create an ephemeral trie.Database for isolating the live one. Otherwise
// the internal junks created by tracing will be persisted into the disk.
database = state.NewDatabaseWithConfig(eth.chainDb, &trie.Config{Cache: 16})
defer database.TrieDB().Close()
defer func() {
if closeTrieDBInDefer {
database.TrieDB().Close()
}
}()
if statedb, err = state.New(block.Root(), database, nil); err == nil {
log.Info("Found disk backend for state trie", "root", block.Root(), "number", block.Number())
return statedb, noopReleaser, nil
closeTrieDBInDefer = false
return statedb, func() { database.TrieDB().Close() }, nil
}
}
// The optional base statedb is given, mark the start point as parent block
Expand All @@ -101,15 +107,20 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe
// Create an ephemeral trie.Database for isolating the live one. Otherwise
// the internal junks created by tracing will be persisted into the disk.
database = state.NewDatabaseWithConfig(eth.chainDb, &trie.Config{Cache: 16})
defer database.TrieDB().Close()
defer func() {
if closeTrieDBInDefer {
database.TrieDB().Close()
}
}()

// If we didn't check the live database, do check state over ephemeral database,
// otherwise we would rewind past a persisted block (specific corner case is
// chain tracing from the genesis).
if !readOnly {
statedb, err = state.New(current.Root(), database, nil)
if err == nil {
return statedb, noopReleaser, nil
closeTrieDBInDefer = false
return statedb, func() { database.TrieDB().Close() }, nil
}
}
// Database does not have the state for the given block, try to regenerate
Expand Down Expand Up @@ -187,7 +198,11 @@ func (eth *Ethereum) StateAtBlock(ctx context.Context, block *types.Block, reexe
nodes, imgs := database.TrieDB().Size()
log.Info("Historical state regenerated", "block", current.NumberU64(), "elapsed", time.Since(start), "nodes", nodes, "preimages", imgs)
}
return statedb, func() { database.TrieDB().Dereference(block.Root()) }, nil
closeTrieDBInDefer = false
return statedb, func() {
database.TrieDB().Dereference(block.Root())
database.TrieDB().Close()
}, nil
}

// stateAtTransaction returns the execution environment of a certain transaction.
Expand Down

0 comments on commit 6b6fd12

Please sign in to comment.