Mark Holt 19bc328a07
Added db loggers to all db callers and fixed flag settings (#9099)
Mdbx now takes a logger - but this has not been pushed to all callers -
meaning it had an invalid logger

This fixes the log propagation.

It also fixed a start-up issue for http.enabled and txpool.disable
created by a previous merge
2023-12-31 17:10:08 +07:00

332 lines
10 KiB
Go

package exec3
import (
"context"
"math/big"
"sync"
"github.com/ledgerwatch/log/v3"
"golang.org/x/sync/errgroup"
"github.com/ledgerwatch/erigon-lib/chain"
libcommon "github.com/ledgerwatch/erigon-lib/common"
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon/cmd/state/exec22"
"github.com/ledgerwatch/erigon/consensus"
"github.com/ledgerwatch/erigon/core"
"github.com/ledgerwatch/erigon/core/rawdb"
"github.com/ledgerwatch/erigon/core/state"
"github.com/ledgerwatch/erigon/core/types"
"github.com/ledgerwatch/erigon/core/vm"
"github.com/ledgerwatch/erigon/core/vm/evmtypes"
"github.com/ledgerwatch/erigon/rlp"
"github.com/ledgerwatch/erigon/turbo/services"
)
type Worker struct {
lock sync.Locker
chainDb kv.RoDB
chainTx kv.Tx
background bool // if true - worker does manage RoTx (begin/rollback) in .ResetTx()
blockReader services.FullBlockReader
in *exec22.QueueWithRetry
rs *state.StateV3
stateWriter *state.StateWriterBufferedV3
stateReader *state.StateReaderV3
chainConfig *chain.Config
getHeader func(hash libcommon.Hash, number uint64) *types.Header
ctx context.Context
engine consensus.Engine
genesis *types.Genesis
resultCh *exec22.ResultsQueue
chain ChainReader
callTracer *CallTracer
taskGasPool *core.GasPool
evm *vm.EVM
ibs *state.IntraBlockState
}
func NewWorker(lock sync.Locker, ctx context.Context, background bool, chainDb kv.RoDB, rs *state.StateV3, in *exec22.QueueWithRetry, blockReader services.FullBlockReader, chainConfig *chain.Config, genesis *types.Genesis, results *exec22.ResultsQueue, engine consensus.Engine) *Worker {
w := &Worker{
lock: lock,
chainDb: chainDb,
in: in,
rs: rs,
background: background,
blockReader: blockReader,
stateWriter: state.NewStateWriterBufferedV3(rs),
stateReader: state.NewStateReaderV3(rs),
chainConfig: chainConfig,
ctx: ctx,
genesis: genesis,
resultCh: results,
engine: engine,
evm: vm.NewEVM(evmtypes.BlockContext{}, evmtypes.TxContext{}, nil, chainConfig, vm.Config{}),
callTracer: NewCallTracer(),
taskGasPool: new(core.GasPool),
}
w.getHeader = func(hash libcommon.Hash, number uint64) *types.Header {
h, err := blockReader.Header(ctx, w.chainTx, hash, number)
if err != nil {
panic(err)
}
return h
}
w.ibs = state.New(w.stateReader)
return w
}
func (rw *Worker) Tx() kv.Tx { return rw.chainTx }
func (rw *Worker) DiscardReadList() { rw.stateReader.DiscardReadList() }
func (rw *Worker) ResetTx(chainTx kv.Tx) {
if rw.background && rw.chainTx != nil {
rw.chainTx.Rollback()
rw.chainTx = nil
}
if chainTx != nil {
rw.chainTx = chainTx
rw.stateReader.SetTx(rw.chainTx)
rw.chain = ChainReader{config: rw.chainConfig, tx: rw.chainTx, blockReader: rw.blockReader}
}
}
func (rw *Worker) Run() error {
for txTask, ok := rw.in.Next(rw.ctx); ok; txTask, ok = rw.in.Next(rw.ctx) {
rw.RunTxTask(txTask)
if err := rw.resultCh.Add(rw.ctx, txTask); err != nil {
return err
}
}
return nil
}
func (rw *Worker) RunTxTask(txTask *exec22.TxTask) {
rw.lock.Lock()
defer rw.lock.Unlock()
rw.RunTxTaskNoLock(txTask)
}
func (rw *Worker) RunTxTaskNoLock(txTask *exec22.TxTask) {
if rw.background && rw.chainTx == nil {
var err error
if rw.chainTx, err = rw.chainDb.BeginRo(rw.ctx); err != nil {
panic(err)
}
rw.stateReader.SetTx(rw.chainTx)
rw.chain = ChainReader{config: rw.chainConfig, tx: rw.chainTx, blockReader: rw.blockReader}
}
txTask.Error = nil
rw.stateReader.SetTxNum(txTask.TxNum)
rw.stateWriter.SetTxNum(txTask.TxNum)
rw.stateReader.ResetReadSet()
rw.stateWriter.ResetWriteSet()
rw.ibs.Reset()
ibs := rw.ibs
//ibs.SetTrace(true)
rules := txTask.Rules
var err error
header := txTask.Header
var logger = log.New("worker-tx")
switch {
case txTask.TxIndex == -1:
if txTask.BlockNum == 0 {
// Genesis block
// fmt.Printf("txNum=%d, blockNum=%d, Genesis\n", txTask.TxNum, txTask.BlockNum)
_, ibs, err = core.GenesisToBlock(rw.genesis, "", logger)
if err != nil {
panic(err)
}
// For Genesis, rules should be empty, so that empty accounts can be included
rules = &chain.Rules{}
break
}
// Block initialisation
//fmt.Printf("txNum=%d, blockNum=%d, initialisation of the block\n", txTask.TxNum, txTask.BlockNum)
syscall := func(contract libcommon.Address, data []byte, ibs *state.IntraBlockState, header *types.Header, constCall bool) ([]byte, error) {
return core.SysCallContract(contract, data, rw.chainConfig, ibs, header, rw.engine, constCall /* constCall */)
}
rw.engine.Initialize(rw.chainConfig, rw.chain, header, ibs, syscall, logger)
txTask.Error = ibs.FinalizeTx(rules, noop)
case txTask.Final:
if txTask.BlockNum == 0 {
break
}
//fmt.Printf("txNum=%d, blockNum=%d, finalisation of the block\n", txTask.TxNum, txTask.BlockNum)
// End of block transaction in a block
syscall := func(contract libcommon.Address, data []byte) ([]byte, error) {
return core.SysCallContract(contract, data, rw.chainConfig, ibs, header, rw.engine, false /* constCall */)
}
if _, _, err := rw.engine.Finalize(rw.chainConfig, types.CopyHeader(header), ibs, txTask.Txs, txTask.Uncles, nil, txTask.Withdrawals, rw.chain, syscall, logger); err != nil {
//fmt.Printf("error=%v\n", err)
txTask.Error = err
} else {
//rw.callTracer.AddCoinbase(txTask.Coinbase, txTask.Uncles)
//txTask.TraceTos = rw.callTracer.Tos()
txTask.TraceTos = map[libcommon.Address]struct{}{}
txTask.TraceTos[txTask.Coinbase] = struct{}{}
for _, uncle := range txTask.Uncles {
txTask.TraceTos[uncle.Coinbase] = struct{}{}
}
}
default:
//fmt.Printf("txNum=%d, blockNum=%d, txIndex=%d\n", txTask.TxNum, txTask.BlockNum, txTask.TxIndex)
txHash := txTask.Tx.Hash()
rw.taskGasPool.Reset(txTask.Tx.GetGas())
rw.callTracer.Reset()
vmConfig := vm.Config{Debug: true, Tracer: rw.callTracer, SkipAnalysis: txTask.SkipAnalysis}
ibs.SetTxContext(txHash, txTask.BlockHash, txTask.TxIndex)
msg := txTask.TxAsMessage
blockContext := txTask.EvmBlockContext
if !rw.background {
getHashFn := core.GetHashFn(header, rw.getHeader)
blockContext = core.NewEVMBlockContext(header, getHashFn, rw.engine, nil /* author */)
}
rw.evm.ResetBetweenBlocks(blockContext, core.NewEVMTxContext(msg), ibs, vmConfig, rules)
// MA applytx
vmenv := rw.evm
applyRes, err := core.ApplyMessage(vmenv, msg, rw.taskGasPool, true /* refunds */, false /* gasBailout */)
if err != nil {
txTask.Error = err
} else {
txTask.UsedGas = applyRes.UsedGas
// Update the state with pending changes
txTask.Error = ibs.FinalizeTx(rules, noop)
txTask.Logs = ibs.GetLogs(txHash)
txTask.TraceFroms = rw.callTracer.Froms()
txTask.TraceTos = rw.callTracer.Tos()
}
}
// Prepare read set, write set and balanceIncrease set and send for serialisation
if txTask.Error == nil {
txTask.BalanceIncreaseSet = ibs.BalanceIncreaseSet()
//for addr, bal := range txTask.BalanceIncreaseSet {
// fmt.Printf("BalanceIncreaseSet [%x]=>[%d]\n", addr, &bal)
//}
if err = ibs.MakeWriteSet(rules, rw.stateWriter); err != nil {
panic(err)
}
txTask.ReadLists = rw.stateReader.ReadSet()
txTask.WriteLists = rw.stateWriter.WriteSet()
txTask.AccountPrevs, txTask.AccountDels, txTask.StoragePrevs, txTask.CodePrevs = rw.stateWriter.PrevAndDels()
}
}
type ChainReader struct {
config *chain.Config
tx kv.Tx
blockReader services.FullBlockReader
}
func NewChainReader(config *chain.Config, tx kv.Tx, blockReader services.FullBlockReader) ChainReader {
return ChainReader{config: config, tx: tx, blockReader: blockReader}
}
func (cr ChainReader) Config() *chain.Config { return cr.config }
func (cr ChainReader) CurrentHeader() *types.Header { panic("") }
func (cr ChainReader) GetHeader(hash libcommon.Hash, number uint64) *types.Header {
if cr.blockReader != nil {
h, _ := cr.blockReader.Header(context.Background(), cr.tx, hash, number)
return h
}
return rawdb.ReadHeader(cr.tx, hash, number)
}
func (cr ChainReader) GetHeaderByNumber(number uint64) *types.Header {
if cr.blockReader != nil {
h, _ := cr.blockReader.HeaderByNumber(context.Background(), cr.tx, number)
return h
}
return rawdb.ReadHeaderByNumber(cr.tx, number)
}
func (cr ChainReader) GetHeaderByHash(hash libcommon.Hash) *types.Header {
if cr.blockReader != nil {
number := rawdb.ReadHeaderNumber(cr.tx, hash)
if number == nil {
return nil
}
return cr.GetHeader(hash, *number)
}
h, _ := rawdb.ReadHeaderByHash(cr.tx, hash)
return h
}
func (cr ChainReader) GetTd(hash libcommon.Hash, number uint64) *big.Int {
td, err := rawdb.ReadTd(cr.tx, hash, number)
if err != nil {
log.Error("ReadTd failed", "err", err)
return nil
}
return td
}
func (cr ChainReader) FrozenBlocks() uint64 {
return cr.blockReader.FrozenBlocks()
}
func (cr ChainReader) GetBlock(hash libcommon.Hash, number uint64) *types.Block {
panic("")
}
func (cr ChainReader) HasBlock(hash libcommon.Hash, number uint64) bool {
panic("")
}
func (cr ChainReader) BorEventsByBlock(hash libcommon.Hash, number uint64) []rlp.RawValue {
panic("")
}
func (cr ChainReader) BorSpan(spanId uint64) []byte { panic("") }
func NewWorkersPool(lock sync.Locker, ctx context.Context, background bool, chainDb kv.RoDB, rs *state.StateV3, in *exec22.QueueWithRetry, blockReader services.FullBlockReader, chainConfig *chain.Config, genesis *types.Genesis, engine consensus.Engine, workerCount int) (reconWorkers []*Worker, applyWorker *Worker, rws *exec22.ResultsQueue, clear func(), wait func()) {
reconWorkers = make([]*Worker, workerCount)
resultChSize := workerCount * 8
rws = exec22.NewResultsQueue(resultChSize, workerCount) // workerCount * 4
{
// we all errors in background workers (except ctx.Cancel), because applyLoop will detect this error anyway.
// and in applyLoop all errors are critical
ctx, cancel := context.WithCancel(ctx)
g, ctx := errgroup.WithContext(ctx)
for i := 0; i < workerCount; i++ {
reconWorkers[i] = NewWorker(lock, ctx, background, chainDb, rs, in, blockReader, chainConfig, genesis, rws, engine)
}
if background {
for i := 0; i < workerCount; i++ {
i := i
g.Go(func() error {
return reconWorkers[i].Run()
})
}
wait = func() { g.Wait() }
}
var clearDone bool
clear = func() {
if clearDone {
return
}
clearDone = true
cancel()
g.Wait()
for _, w := range reconWorkers {
w.ResetTx(nil)
}
//applyWorker.ResetTx(nil)
}
}
applyWorker = NewWorker(lock, ctx, false, chainDb, rs, in, blockReader, chainConfig, genesis, rws, engine)
return reconWorkers, applyWorker, rws, clear, wait
}