Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

all: live chain-aware tracing #27629

Closed
wants to merge 113 commits into from
Closed
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
113 commits
Select commit Hold shift + click to select a range
642a374
Initial support for extended tracer
s1na Jun 15, 2023
e25065b
capture keccak preimage
s1na Jun 15, 2023
f67ac09
add hooks to state object & genesis
s1na Jun 20, 2023
84d6432
add log and newAccount hooks
s1na Jun 22, 2023
ee791b2
add gas consumption hook
s1na Jun 23, 2023
0d25fbc
pass logger to BC through vmConfig
s1na Jun 26, 2023
8db1078
fix genesis logging
s1na Jun 26, 2023
788851b
disable tracing for miner processed txes
s1na Jun 26, 2023
057bbda
add comments
s1na Jun 26, 2023
a7de17e
only emit event for new accounts
s1na Jun 26, 2023
3787816
Upgrade TxStart and TxEnd hooks
s1na Jun 28, 2023
3975c7c
minor
s1na Jun 28, 2023
619ac29
provide StateLogger for short-lived tracers
s1na Jun 28, 2023
54171ee
pass env in TxStart
s1na Jun 28, 2023
3cabef4
capture block end errors
s1na Jun 30, 2023
f1bd415
rm extra CaptureTxEnd
s1na Jul 10, 2023
d799c68
nicer printer output
s1na Jul 10, 2023
add825e
fix BlockEnd in case of err
s1na Jul 10, 2023
0aeec7f
mv td to OnBlockStart
s1na Jul 12, 2023
b692a68
add final & safe headers to BlockStart
s1na Jul 12, 2023
2a59d24
add genesis alloc to hook
s1na Jul 12, 2023
4370527
fix prefetcher double emit
s1na Jul 24, 2023
5466fa3
Capture balance reason (#12)
maoueh Jul 26, 2023
2faf3db
emit err on tx validation failure
s1na Jul 26, 2023
b2b32e6
fix merge conflict
s1na Jul 27, 2023
09ee197
fix merge conflict
s1na Jul 27, 2023
f9f377d
capture call validation errors
s1na Jul 27, 2023
b7ff573
minor refactor
s1na Jul 28, 2023
00a19d3
Fixed tests compilation (#13)
maoueh Aug 2, 2023
216a4b0
Added writing of `genesis.Alloc` on bootstrap (#15)
maoueh Aug 30, 2023
659043a
pass tracer name via cli
s1na Aug 30, 2023
9c999c3
Use noopTracer as base for loggers
s1na Aug 31, 2023
0be6e22
add comment to statedb logger
s1na Aug 31, 2023
abd8807
use defer for OnBlockEnd
s1na Aug 31, 2023
f7ca31e
fix import cycle in blockchain test
s1na Sep 4, 2023
651c438
fix runtime tests
s1na Sep 4, 2023
74c1f30
Fixed `BlockchainLogger` tracer not being correctly set up up to `Blo…
maoueh Sep 12, 2023
ff3c15f
Full OnGasConsumed loop and added `GasChangeReason` (#16)
maoueh Sep 12, 2023
6ee4fb8
resolve merge conflicts
s1na Sep 13, 2023
1914bc6
error code for VM failures (#18)
maoueh Oct 27, 2023
429dcc9
fix merge conflict
s1na Nov 2, 2023
e65f14a
update state processor
s1na Nov 2, 2023
e4399a6
Use applyTransaction in tracer instead of applyMessage
s1na Nov 3, 2023
14d603b
add comment re newAccount precompile
s1na Nov 3, 2023
023ade6
instrument when inserting known block
s1na Nov 3, 2023
ee58cc7
statedb: precompile check before onNewAccount
s1na Nov 7, 2023
dc1175e
rm unnecessary line
s1na Nov 7, 2023
da8b362
minor
s1na Nov 7, 2023
0d268b7
fix TxStart for traceCall
s1na Nov 8, 2023
cc6b68e
Fix prestate create issue, add test
s1na Nov 11, 2023
6b8d216
fix setBalance reasons
s1na Nov 13, 2023
b80aa1e
fix double-allocation
s1na Nov 24, 2023
1d13645
move balance reason to metadata file
s1na Nov 24, 2023
856c079
move traced block processing to own func
s1na Nov 24, 2023
9c00b12
fix merge conflict
s1na Nov 24, 2023
379b8d3
fix atomic followupInterrupt
s1na Nov 24, 2023
56600e9
fix VMError.Unwrap
s1na Nov 24, 2023
8a1f67c
forgot prestate testcase
s1na Nov 24, 2023
5bc3f62
move makeTest to own file
s1na Nov 24, 2023
cd8c4c5
force load live tracer set
s1na Nov 24, 2023
8610f8f
add hooks for beacon block root processing
s1na Nov 24, 2023
00ee953
rm comment
s1na Nov 28, 2023
72410c3
Fixed `BlockchainLogger.OnBlockEnd` called too much times (#19)
maoueh Nov 30, 2023
c341c79
Merge remote-tracking branch 's1na/extended-tracer' into extended-tracer
s1na Dec 4, 2023
5190cfc
rm extra galloc capturing
s1na Dec 5, 2023
9e5c96f
add chainConfig to block events
s1na Dec 5, 2023
f862531
rm chainConfig from OnGenesisBlock
s1na Dec 5, 2023
c5e407b
resolve merge conflicts
s1na Dec 5, 2023
9360cab
assign values for balance reasons
s1na Dec 7, 2023
0967291
Track burnt and selfdestruct withdraw bal changes
s1na Dec 7, 2023
ddd1b0a
fix gas purchase
s1na Dec 11, 2023
0484068
fix lint issues
s1na Dec 11, 2023
5e82cbe
nil check
s1na Dec 11, 2023
0d254e4
fix test
s1na Dec 11, 2023
a52c80d
Fix test case
s1na Dec 12, 2023
89d79d6
fix t8n tracer nil check
s1na Dec 12, 2023
54f4bc7
fix balance check
s1na Dec 13, 2023
64e35e9
revert buyGas refactor
s1na Dec 13, 2023
8b3180d
revert gas refund
s1na Dec 13, 2023
2d14bb5
rename and document balance change reasons
s1na Dec 13, 2023
68bd6c6
Removed the `OnGenesisBlock` in `genesis.Commit` (#20)
maoueh Jan 11, 2024
f241a76
fix comment'
s1na Jan 12, 2024
580f025
has call reverted to snapshot
s1na Jan 12, 2024
aa0db68
resolve merge conflict
s1na Jan 12, 2024
f21efbc
fix traceWriter
s1na Jan 12, 2024
c331930
add test case for frontier create out of storage
s1na Jan 15, 2024
626fee4
fix withdraw balance change name
s1na Jan 15, 2024
d07264a
no balance change reason on 0 balance selfdestruct
s1na Jan 16, 2024
4c1d675
resolve merge conflict
s1na Jan 29, 2024
9a6b9a2
re-add simulated backend removed accidentally
s1na Jan 30, 2024
ae522d6
rm empty line
s1na Jan 30, 2024
56862cf
emit onNewAccount when reseting account
s1na Jan 30, 2024
68342ef
improve processBlock
s1na Jan 31, 2024
3078dfe
minor fix
s1na Jan 31, 2024
6c44a59
fix self destruct burn condition
s1na Feb 5, 2024
188cd41
remove precompile check for newAccount
s1na Feb 5, 2024
f3c0a89
add OnBlockchainInit for chainConfig
s1na Feb 6, 2024
95b5029
replace printer with noop live tracer
s1na Feb 6, 2024
2cc0954
rename
s1na Feb 6, 2024
57cd0c3
rm OnNewAccount
s1na Feb 8, 2024
056f229
indicate known block
s1na Feb 8, 2024
27f662d
fix merge conflict
s1na Feb 8, 2024
706c869
fix remaining conflict
s1na Feb 9, 2024
03206be
fix blockchain -> logger circular import
s1na Feb 9, 2024
08cb623
live tracer err -> warn
s1na Feb 9, 2024
cf6a315
fix mdLogger
s1na Feb 9, 2024
4cd7cb3
resolve merge conflict
s1na Feb 14, 2024
674a38e
add config for tracers
s1na Feb 14, 2024
0db7a19
minor fix
s1na Feb 14, 2024
35291e6
resolve merge conflict
s1na Feb 15, 2024
3ba6b92
remove onBeaconBlockRoot events
s1na Feb 15, 2024
5f0a543
refactor onBlockStart params, new skip method
s1na Feb 15, 2024
490efc5
resolve merge conflict
s1na Feb 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions cmd/evm/internal/t8ntool/execution.go
Original file line number Diff line number Diff line change
Expand Up @@ -183,6 +183,7 @@ func (pre *Prestate) Apply(vmConfig vm.Config, chainConfig *params.ChainConfig,
)
evm := vm.NewEVM(vmContext, txContext, statedb, chainConfig, vmConfig)

tracer.CaptureTxStart(tx)
s1na marked this conversation as resolved.
Show resolved Hide resolved
// (ret []byte, usedGas uint64, failed bool, err error)
msgResult, err := core.ApplyMessage(evm, msg, gaspool)
if err != nil {
Expand Down Expand Up @@ -231,6 +232,7 @@ func (pre *Prestate) Apply(vmConfig vm.Config, chainConfig *params.ChainConfig,
//receipt.BlockNumber
receipt.TransactionIndex = uint(txIndex)
receipts = append(receipts, receipt)
tracer.CaptureTxEnd(receipt)
}

txIndex++
Expand Down
1 change: 1 addition & 0 deletions cmd/geth/chaincmd.go
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,7 @@ if one is set. Otherwise it prints the genesis from the datadir.`,
utils.MetricsInfluxDBBucketFlag,
utils.MetricsInfluxDBOrganizationFlag,
utils.TxLookupLimitFlag,
utils.VMTraceFlag,
}, utils.DatabasePathFlags),
Description: `
The import command imports blocks from an RLP-encoded form. The form can be one file
Expand Down
1 change: 1 addition & 0 deletions cmd/geth/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,7 @@ var (
utils.DeveloperPeriodFlag,
utils.DeveloperGasLimitFlag,
utils.VMEnableDebugFlag,
utils.VMTraceFlag,
utils.NetworkIdFlag,
utils.EthStatsURLFlag,
utils.NoCompactionFlag,
Expand Down
13 changes: 12 additions & 1 deletion cmd/utils/flags.go
Original file line number Diff line number Diff line change
Expand Up @@ -529,6 +529,11 @@ var (
Usage: "Record information useful for VM and contract debugging",
Category: flags.VMCategory,
}
VMTraceFlag = &cli.BoolFlag{
Name: "vmtrace",
Usage: "Record internal VM operations (costly)",
Category: flags.VMCategory,
}

// API options.
RPCGlobalGasCapFlag = &cli.Uint64Flag{
Expand Down Expand Up @@ -1736,6 +1741,9 @@ func SetEthConfig(ctx *cli.Context, stack *node.Node, cfg *ethconfig.Config) {
// TODO(fjl): force-enable this in --dev mode
cfg.EnablePreimageRecording = ctx.Bool(VMEnableDebugFlag.Name)
}
if ctx.IsSet(VMTraceFlag.Name) {
cfg.LiveTrace = ctx.Bool(VMTraceFlag.Name)
}

if ctx.IsSet(RPCGlobalGasCapFlag.Name) {
cfg.RPCGasCap = ctx.Uint64(RPCGlobalGasCapFlag.Name)
Expand Down Expand Up @@ -2158,12 +2166,15 @@ func MakeChain(ctx *cli.Context, stack *node.Node, readonly bool) (*core.BlockCh
cache.TrieDirtyLimit = ctx.Int(CacheFlag.Name) * ctx.Int(CacheGCFlag.Name) / 100
}
vmcfg := vm.Config{EnablePreimageRecording: ctx.Bool(VMEnableDebugFlag.Name)}

if ctx.IsSet(VMTraceFlag.Name) {
vmcfg.Tracer = tracers.NewPrinter()
s1na marked this conversation as resolved.
Show resolved Hide resolved
}
// Disable transaction indexing/unindexing by default.
chain, err := core.NewBlockChain(chainDb, cache, gspec, nil, engine, vmcfg, nil, nil)
if err != nil {
Fatalf("Can't create BlockChain: %v", err)
}

return chain, chainDb
}

Expand Down
28 changes: 27 additions & 1 deletion core/blockchain.go
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,17 @@ var defaultCacheConfig = &CacheConfig{
SnapshotWait: true,
}

// BlockchainLogger is used to collect traces during chain processing.
// Please make a copy of the referenced types if you intend to retain them.
type BlockchainLogger interface {
vm.EVMLogger
state.StateLogger
OnBlockStart(*types.Block)
OnBlockEnd(td *big.Int, err error)
s1na marked this conversation as resolved.
Show resolved Hide resolved
OnBlockValidationError(block *types.Block, err error)
OnGenesisBlock(*types.Block)
}

// BlockChain represents the canonical chain given a database with a genesis
// block. The Blockchain manages chain imports, reverts, chain reorganisations.
//
Expand Down Expand Up @@ -226,6 +237,7 @@ type BlockChain struct {
processor Processor // Block transaction processor interface
forker *ForkChoice
vmConfig vm.Config
logger BlockchainLogger
}

// NewBlockChain returns a fully initialised block chain using information
Expand All @@ -241,10 +253,19 @@ func NewBlockChain(db ethdb.Database, cacheConfig *CacheConfig, genesis *Genesis
Journal: cacheConfig.TrieCleanJournal,
Preimages: cacheConfig.Preimages,
})
var logger BlockchainLogger
if vmConfig.Tracer != nil {
l, ok := vmConfig.Tracer.(BlockchainLogger)
if !ok {
return nil, fmt.Errorf("only extended tracers are supported for live mode")
s1na marked this conversation as resolved.
Show resolved Hide resolved
}
logger = l
}
// Setup the genesis block, commit the provided genesis specification
// to database if the genesis block is not present yet, or load the
// stored one from database.
chainConfig, genesisHash, genesisErr := SetupGenesisBlockWithOverride(db, triedb, genesis, overrides)
// TODO: pass in blockchainLogger here to catch genesis block and allocs
chainConfig, genesisHash, genesisErr := SetupGenesisBlockWithOverride(db, triedb, genesis, overrides, logger)
if _, ok := genesisErr.(*params.ConfigCompatError); genesisErr != nil && !ok {
return nil, genesisErr
}
Expand Down Expand Up @@ -272,6 +293,7 @@ func NewBlockChain(db ethdb.Database, cacheConfig *CacheConfig, genesis *Genesis
futureBlocks: lru.NewCache[common.Hash, *types.Block](maxFutureBlocks),
engine: engine,
vmConfig: vmConfig,
logger: logger,
}
bc.flushInterval.Store(int64(cacheConfig.TrieTimeLimit))
bc.forker = NewForkChoice(bc, shouldPreserve)
Expand Down Expand Up @@ -1726,6 +1748,7 @@ func (bc *BlockChain) insertChain(chain types.Blocks, setHead bool) (int, error)
if err != nil {
return it.index, err
}
statedb.SetLogger(bc.logger)
s1na marked this conversation as resolved.
Show resolved Hide resolved

// Enable prefetching to pull in trie node paths while processing transactions
statedb.StartPrefetcher("chain")
s1na marked this conversation as resolved.
Show resolved Hide resolved
Expand Down Expand Up @@ -1763,6 +1786,9 @@ func (bc *BlockChain) insertChain(chain types.Blocks, setHead bool) (int, error)
if err := bc.validator.ValidateState(block, statedb, receipts, usedGas); err != nil {
bc.reportBlock(block, receipts, err)
followupInterrupt.Store(true)
if bc.logger != nil {
bc.logger.OnBlockValidationError(block, err)
}
s1na marked this conversation as resolved.
Show resolved Hide resolved
return it.index, err
}
vtime := time.Since(vstart)
Expand Down
2 changes: 1 addition & 1 deletion core/chain_makers.go
Original file line number Diff line number Diff line change
Expand Up @@ -350,7 +350,7 @@ func GenerateChain(config *params.ChainConfig, parent *types.Block, engine conse
// then generate chain on top.
func GenerateChainWithGenesis(genesis *Genesis, engine consensus.Engine, n int, gen func(int, *BlockGen)) (ethdb.Database, []*types.Block, []types.Receipts) {
db := rawdb.NewMemoryDatabase()
_, err := genesis.Commit(db, trie.NewDatabase(db))
_, err := genesis.Commit(db, trie.NewDatabase(db), nil)
if err != nil {
panic(err)
}
Expand Down
22 changes: 13 additions & 9 deletions core/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -138,8 +138,9 @@ func (ga *GenesisAlloc) deriveHash() (common.Hash, error) {
// flush is very similar with deriveHash, but the main difference is
// all the generated states will be persisted into the given database.
// Also, the genesis state specification will be flushed as well.
func (ga *GenesisAlloc) flush(db ethdb.Database, triedb *trie.Database, blockhash common.Hash) error {
func (ga *GenesisAlloc) flush(db ethdb.Database, triedb *trie.Database, blockhash common.Hash, bcLogger BlockchainLogger) error {
s1na marked this conversation as resolved.
Show resolved Hide resolved
statedb, err := state.New(types.EmptyRootHash, state.NewDatabaseWithNodeDB(db, triedb), nil)
statedb.SetLogger(bcLogger)
if err != nil {
return err
}
Expand Down Expand Up @@ -200,7 +201,7 @@ func CommitGenesisState(db ethdb.Database, triedb *trie.Database, blockhash comm
return errors.New("not found")
}
}
return alloc.flush(db, triedb, blockhash)
return alloc.flush(db, triedb, blockhash, nil)
}

// GenesisAccount is an account in the state of the genesis block.
Expand Down Expand Up @@ -282,10 +283,10 @@ type ChainOverrides struct {
//
// The returned chain configuration is never nil.
func SetupGenesisBlock(db ethdb.Database, triedb *trie.Database, genesis *Genesis) (*params.ChainConfig, common.Hash, error) {
return SetupGenesisBlockWithOverride(db, triedb, genesis, nil)
return SetupGenesisBlockWithOverride(db, triedb, genesis, nil, nil)
}

func SetupGenesisBlockWithOverride(db ethdb.Database, triedb *trie.Database, genesis *Genesis, overrides *ChainOverrides) (*params.ChainConfig, common.Hash, error) {
func SetupGenesisBlockWithOverride(db ethdb.Database, triedb *trie.Database, genesis *Genesis, overrides *ChainOverrides, bcLogger BlockchainLogger) (*params.ChainConfig, common.Hash, error) {
if genesis != nil && genesis.Config == nil {
return params.AllEthashProtocolChanges, common.Hash{}, errGenesisNoConfig
}
Expand All @@ -305,7 +306,7 @@ func SetupGenesisBlockWithOverride(db ethdb.Database, triedb *trie.Database, gen
} else {
log.Info("Writing custom genesis block")
}
block, err := genesis.Commit(db, triedb)
block, err := genesis.Commit(db, triedb, bcLogger)
if err != nil {
return genesis.Config, common.Hash{}, err
}
Expand All @@ -324,7 +325,7 @@ func SetupGenesisBlockWithOverride(db ethdb.Database, triedb *trie.Database, gen
if hash != stored {
return genesis.Config, hash, &GenesisMismatchError{stored, hash}
}
block, err := genesis.Commit(db, triedb)
block, err := genesis.Commit(db, triedb, bcLogger)
if err != nil {
return genesis.Config, hash, err
}
Expand Down Expand Up @@ -468,7 +469,7 @@ func (g *Genesis) ToBlock() *types.Block {

// Commit writes the block and state of a genesis specification to the database.
// The block is committed as the canonical head block.
func (g *Genesis) Commit(db ethdb.Database, triedb *trie.Database) (*types.Block, error) {
func (g *Genesis) Commit(db ethdb.Database, triedb *trie.Database, bcLogger BlockchainLogger) (*types.Block, error) {
block := g.ToBlock()
if block.Number().Sign() != 0 {
return nil, errors.New("can't commit genesis block with number > 0")
Expand All @@ -483,10 +484,13 @@ func (g *Genesis) Commit(db ethdb.Database, triedb *trie.Database) (*types.Block
if config.Clique != nil && len(block.Extra()) < 32+crypto.SignatureLength {
return nil, errors.New("can't start clique chain without signers")
}
if bcLogger != nil {
bcLogger.OnGenesisBlock(block)
s1na marked this conversation as resolved.
Show resolved Hide resolved
}
// All the checks has passed, flush the states derived from the genesis
// specification as well as the specification itself into the provided
// database.
if err := g.Alloc.flush(db, triedb, block.Hash()); err != nil {
if err := g.Alloc.flush(db, triedb, block.Hash(), bcLogger); err != nil {
s1na marked this conversation as resolved.
Show resolved Hide resolved
return nil, err
}
rawdb.WriteTd(db, block.Hash(), block.NumberU64(), block.Difficulty())
Expand All @@ -505,7 +509,7 @@ func (g *Genesis) Commit(db ethdb.Database, triedb *trie.Database) (*types.Block
// Note the state changes will be committed in hash-based scheme, use Commit
// if path-scheme is preferred.
func (g *Genesis) MustCommit(db ethdb.Database) *types.Block {
block, err := g.Commit(db, trie.NewDatabase(db))
block, err := g.Commit(db, trie.NewDatabase(db), nil)
if err != nil {
panic(err)
}
Expand Down
12 changes: 12 additions & 0 deletions core/state/state_object.go
Original file line number Diff line number Diff line change
Expand Up @@ -236,6 +236,9 @@ func (s *stateObject) SetState(db Database, key, value common.Hash) {
key: key,
prevalue: prev,
})
if s.db.logger != nil {
s.db.logger.OnStorageChange(s.address, key, prev, value)
}
s.setState(key, value)
}

Expand Down Expand Up @@ -399,6 +402,9 @@ func (s *stateObject) SetBalance(amount *big.Int) {
account: &s.address,
prev: new(big.Int).Set(s.data.Balance),
})
if s.db.logger != nil {
s.db.logger.OnBalanceChange(s.address, s.Balance(), amount)
}
s.setBalance(amount)
}

Expand Down Expand Up @@ -470,6 +476,9 @@ func (s *stateObject) SetCode(codeHash common.Hash, code []byte) {
prevhash: s.CodeHash(),
prevcode: prevcode,
})
if s.db.logger != nil {
s.db.logger.OnCodeChange(s.address, common.BytesToHash(s.CodeHash()), prevcode, codeHash, code)
}
s.setCode(codeHash, code)
}

Expand All @@ -484,6 +493,9 @@ func (s *stateObject) SetNonce(nonce uint64) {
account: &s.address,
prev: s.data.Nonce,
})
if s.db.logger != nil {
s.db.logger.OnNonceChange(s.address, s.data.Nonce, nonce)
}
s.setNonce(nonce)
}

Expand Down
27 changes: 27 additions & 0 deletions core/state/statedb.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,19 @@ func (n *proofList) Delete(key []byte) error {
panic("not supported")
}

// StateLogger is used to collect state update traces from EVM transaction
// execution.
// Note that reference types are actual VM data structures; make copies
// if you need to retain them beyond the current call.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please also document that these all are invoked post-execution, and the values reported via these methods are the finalized values (as opposed to runtime values which might revert)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please also document that these all are invoked post-execution

Good tip

these methods are the finalized values (as opposed to runtime values which might revert)

Is that the case? I thought values in statedb could still revert.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right, my bad

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

SO, how does a "tracer" (generically, I know there differen types), track

  1. A transfer in some subscope, and the subscope reverts five ops later
  2. A transfer in some subscope, and the grand-parent scope reverts, 100 ops later
  3. A transfer in some subscope, but the transaction goes OOG and reverts later on
  4. A transfer in some subscope, but the block A is not yet finalized. The same tx is also executed on a sibling block B, and the transfer is different there. The same tx is also executed on sibling block C, and there the transfer never happens.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm unsure about some of your terminology. I'll assume subscope means a new CALL/STATIC/DELEGATE/etc. execution context and ops means opcode executed.

A transfer in some subscope, and the subscope reverts five ops later

- Trx (also root call)
-   CALL [pure transfer] -> Fails

Then CaptureExit will have the err to determine the call failed, reverted/failed can be checked using error.Is(err, evm.ExecutionReverted).

A transfer in some subscope, and the grand-parent scope reverts, 100 ops later
A transfer in some subscope, but the transaction goes OOG and reverts later on

- Trx (also root call)
-   CALL (contract's call)
-       CALL (another one)
-           CALL [pure transfer]
-           <-- Failure Point -->

In those cases, when the transaction terminates, in CaptureTxEnd, we re-create the call tree structure of the trx and if a parent node is reverted/failed, we flag all the calls in its substree with a special value to tell that this call did not make it to the chain.

A transfer in some subscope, but the block A is not yet finalized. The same tx is also executed on a sibling block B, and the transfer is different there. The same tx is also executed on sibling block C, and there the transfer never happens.

I think in those it depends on the use case(s) of how the tracer is going to be used. Some tracers could wait until the block is actually finalized (there is implementation challenges for this but it's feasible. In our case, we deal with fork in a later step. So we receive all blocks from the tracer, then we deal with forks after the tracer by keeping the parent/child relationship in a "tree".

type StateLogger interface {
OnBalanceChange(addr common.Address, prev, new *big.Int)
OnNonceChange(addr common.Address, prev, new uint64)
OnCodeChange(addr common.Address, prevCodeHash common.Hash, prevCode []byte, codeHash common.Hash, code []byte)
OnStorageChange(addr common.Address, slot common.Hash, prev, new common.Hash)
OnLog(log *types.Log)
OnNewAccount(addr common.Address)
s1na marked this conversation as resolved.
Show resolved Hide resolved
}

// StateDB structs within the ethereum protocol are used to store anything
// within the merkle trie. StateDBs take care of caching and storing
// nested states. It's the general query interface to retrieve:
Expand All @@ -63,6 +76,7 @@ type StateDB struct {
prefetcher *triePrefetcher
trie Trie
hasher crypto.KeccakState
logger StateLogger

// originalRoot is the pre-state root, before any changes were made.
// It will be updated when the Commit is called.
Expand Down Expand Up @@ -161,6 +175,11 @@ func New(root common.Hash, db Database, snaps *snapshot.Tree) (*StateDB, error)
return sdb, nil
}

// SetLogger sets the logger for account update hooks.
func (s *StateDB) SetLogger(l StateLogger) {
s.logger = l
}

// StartPrefetcher initializes a new trie prefetcher to pull in nodes from the
// state trie concurrently while the state is mutated so that when we reach the
// commit phase, most of the needed data is already hot.
Expand Down Expand Up @@ -201,6 +220,9 @@ func (s *StateDB) AddLog(log *types.Log) {
log.TxHash = s.thash
log.TxIndex = uint(s.txIndex)
log.Index = s.logSize
if s.logger != nil {
s.logger.OnLog(log)
}
s.logs[s.thash] = append(s.logs[s.thash], log)
s.logSize++
}
Expand Down Expand Up @@ -626,6 +648,10 @@ func (s *StateDB) createObject(addr common.Address) (newobj, prev *stateObject)
newobj = newObject(s, addr, types.StateAccount{})
if prev == nil {
s.journal.append(createObjectChange{account: &addr})
// TODO: add isPrecompile check
s1na marked this conversation as resolved.
Show resolved Hide resolved
if s.logger != nil {
s.logger.OnNewAccount(addr)
}
} else {
// The original account should be marked as destructed and all cached
// account and storage data should be cleared as well. Note, it must
Expand Down Expand Up @@ -656,6 +682,7 @@ func (s *StateDB) createObject(addr common.Address) (newobj, prev *stateObject)
prevStorage: storage,
})
}

s.setStateObject(newobj)
if prev != nil && !prev.deleted {
return newobj, prev
Expand Down
Loading