erigon-pulse/eth/calltracer/calltracer.go
ledgerwatch 97c9a9108d
Native tracers - step 4 (#6363)
Remove `callType` argument from `CaptureStart`, and change `callType
vm.CallType` to `typ vm.OpCode` in `CaptureEnter` and move to the first
position (as it is in geth)

Co-authored-by: Alex Sharp <alexsharp@Alexs-MacBook-Pro-2.local>
2022-12-19 03:12:08 +00:00

116 lines
3.2 KiB
Go

package calltracer
import (
"encoding/binary"
"sort"
"time"
"github.com/holiman/uint256"
"github.com/ledgerwatch/erigon-lib/common/length"
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon/common"
"github.com/ledgerwatch/erigon/core/types"
"github.com/ledgerwatch/erigon/core/vm"
)
type CallTracer struct {
froms map[common.Address]struct{}
tos map[common.Address]bool // address -> isCreated
}
func NewCallTracer() *CallTracer {
return &CallTracer{
froms: make(map[common.Address]struct{}),
tos: make(map[common.Address]bool),
}
}
func (ct *CallTracer) CaptureTxStart(gasLimit uint64) {}
func (ct *CallTracer) CaptureTxEnd(restGas uint64) {}
func (ct *CallTracer) captureStartOrEnter(from, to common.Address, create bool, code []byte) {
ct.froms[from] = struct{}{}
created, ok := ct.tos[to]
if !ok {
ct.tos[to] = false
}
if !created && create {
if len(code) > 0 {
ct.tos[to] = true
}
}
}
func (ct *CallTracer) CaptureStart(env *vm.EVM, from common.Address, to common.Address, precompile bool, create bool, input []byte, gas uint64, value *uint256.Int, code []byte) {
ct.captureStartOrEnter(from, to, create, code)
}
func (ct *CallTracer) CaptureEnter(typ vm.OpCode, from common.Address, to common.Address, precompile bool, create bool, input []byte, gas uint64, value *uint256.Int, code []byte) {
ct.captureStartOrEnter(from, to, create, code)
}
func (ct *CallTracer) CaptureState(pc uint64, op vm.OpCode, gas, cost uint64, scope *vm.ScopeContext, rData []byte, depth int, err error) {
}
func (ct *CallTracer) CaptureFault(pc uint64, op vm.OpCode, gas, cost uint64, scope *vm.ScopeContext, depth int, err error) {
}
func (ct *CallTracer) CaptureEnd(output []byte, startGas, endGas uint64, t time.Duration, err error) {
}
func (ct *CallTracer) CaptureExit(output []byte, startGas, endGas uint64, t time.Duration, err error) {
}
func (ct *CallTracer) CaptureSelfDestruct(from common.Address, to common.Address, value *uint256.Int) {
ct.froms[from] = struct{}{}
ct.tos[to] = false
}
func (ct *CallTracer) CaptureAccountRead(account common.Address) error {
return nil
}
func (ct *CallTracer) CaptureAccountWrite(account common.Address) error {
return nil
}
func (ct *CallTracer) WriteToDb(tx kv.StatelessWriteTx, block *types.Block, vmConfig vm.Config) error {
ct.tos[block.Coinbase()] = false
for _, uncle := range block.Uncles() {
ct.tos[uncle.Coinbase] = false
}
list := make(common.Addresses, len(ct.froms)+len(ct.tos))
i := 0
for addr := range ct.froms {
copy(list[i][:], addr[:])
i++
}
for addr := range ct.tos {
copy(list[i][:], addr[:])
i++
}
sort.Sort(list)
// List may contain duplicates
var blockNumEnc [8]byte
binary.BigEndian.PutUint64(blockNumEnc[:], block.Number().Uint64())
var prev common.Address
for j, addr := range list {
if j > 0 && prev == addr {
continue
}
var v [length.Addr + 1]byte
copy(v[:], addr[:])
if _, ok := ct.froms[addr]; ok {
v[length.Addr] |= 1
}
if _, ok := ct.tos[addr]; ok {
v[length.Addr] |= 2
}
if j == 0 {
if err := tx.Append(kv.CallTraceSet, blockNumEnc[:], v[:]); err != nil {
return err
}
} else {
if err := tx.AppendDup(kv.CallTraceSet, blockNumEnc[:], v[:]); err != nil {
return err
}
}
copy(prev[:], addr[:])
}
return nil
}