2023-07-14 14:38:17 +00:00
|
|
|
package eth1
|
|
|
|
|
|
|
|
import (
|
2023-07-15 16:31:15 +00:00
|
|
|
"context"
|
|
|
|
|
2023-07-27 12:37:49 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/chain"
|
2023-07-15 16:31:15 +00:00
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
2023-07-20 16:35:02 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces"
|
2023-07-14 14:38:17 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces/execution"
|
2023-07-20 16:35:02 +00:00
|
|
|
types2 "github.com/ledgerwatch/erigon-lib/gointerfaces/types"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
2023-07-23 17:10:24 +00:00
|
|
|
"golang.org/x/sync/semaphore"
|
2023-07-14 14:38:17 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2023-07-27 12:37:49 +00:00
|
|
|
"github.com/ledgerwatch/erigon/core"
|
2023-07-15 16:31:15 +00:00
|
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
2023-07-20 16:35:02 +00:00
|
|
|
"github.com/ledgerwatch/erigon/eth/stagedsync"
|
|
|
|
"github.com/ledgerwatch/erigon/eth/stagedsync/stages"
|
2023-07-27 12:37:49 +00:00
|
|
|
"github.com/ledgerwatch/erigon/turbo/builder"
|
2023-07-23 17:10:24 +00:00
|
|
|
"github.com/ledgerwatch/erigon/turbo/engineapi/engine_helpers"
|
|
|
|
"github.com/ledgerwatch/erigon/turbo/engineapi/engine_types"
|
2023-07-14 14:38:17 +00:00
|
|
|
"github.com/ledgerwatch/erigon/turbo/services"
|
|
|
|
)
|
|
|
|
|
|
|
|
// EthereumExecutionModule describes ethereum execution logic and indexing.
|
|
|
|
type EthereumExecutionModule struct {
|
|
|
|
// Snapshots + MDBX
|
|
|
|
blockReader services.FullBlockReader
|
|
|
|
|
|
|
|
// MDBX database
|
2023-07-20 16:35:02 +00:00
|
|
|
db kv.RwDB // main database
|
2023-07-23 17:10:24 +00:00
|
|
|
semaphore *semaphore.Weighted
|
2023-07-20 16:35:02 +00:00
|
|
|
executionPipeline *stagedsync.Sync
|
2023-07-23 17:10:24 +00:00
|
|
|
forkValidator *engine_helpers.ForkValidator
|
2023-07-20 16:35:02 +00:00
|
|
|
|
|
|
|
logger log.Logger
|
2023-07-27 12:37:49 +00:00
|
|
|
// Block building
|
|
|
|
nextPayloadId uint64
|
|
|
|
lastParameters *core.BlockBuilderParameters
|
|
|
|
builderFunc builder.BlockBuilderFunc
|
|
|
|
builders map[uint64]*builder.BlockBuilder
|
|
|
|
|
|
|
|
// configuration
|
|
|
|
config *chain.Config
|
2023-07-14 14:38:17 +00:00
|
|
|
|
|
|
|
execution.UnimplementedExecutionServer
|
|
|
|
}
|
|
|
|
|
2023-07-27 12:37:49 +00:00
|
|
|
func NewEthereumExecutionModule(blockReader services.FullBlockReader, db kv.RwDB, executionPipeline *stagedsync.Sync, forkValidator *engine_helpers.ForkValidator, config *chain.Config, builderFunc builder.BlockBuilderFunc, logger log.Logger) *EthereumExecutionModule {
|
2023-07-15 16:31:15 +00:00
|
|
|
return &EthereumExecutionModule{
|
2023-07-20 16:35:02 +00:00
|
|
|
blockReader: blockReader,
|
|
|
|
db: db,
|
|
|
|
executionPipeline: executionPipeline,
|
|
|
|
logger: logger,
|
2023-07-23 17:10:24 +00:00
|
|
|
forkValidator: forkValidator,
|
2023-07-27 12:37:49 +00:00
|
|
|
builders: make(map[uint64]*builder.BlockBuilder),
|
|
|
|
builderFunc: builderFunc,
|
|
|
|
config: config,
|
2023-07-23 17:10:24 +00:00
|
|
|
semaphore: semaphore.NewWeighted(1),
|
2023-07-15 16:31:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-20 16:35:02 +00:00
|
|
|
func (e *EthereumExecutionModule) getHeader(ctx context.Context, tx kv.Tx, blockHash libcommon.Hash, blockNumber uint64) (*types.Header, error) {
|
2023-07-15 16:31:15 +00:00
|
|
|
if e.blockReader == nil {
|
2023-07-20 16:35:02 +00:00
|
|
|
return rawdb.ReadHeader(tx, blockHash, blockNumber), nil
|
2023-07-15 16:31:15 +00:00
|
|
|
}
|
2023-07-20 16:35:02 +00:00
|
|
|
return e.blockReader.Header(ctx, tx, blockHash, blockNumber)
|
2023-07-15 16:31:15 +00:00
|
|
|
}
|
2023-07-14 14:38:17 +00:00
|
|
|
|
2023-07-20 16:35:02 +00:00
|
|
|
func (e *EthereumExecutionModule) getBody(ctx context.Context, tx kv.Tx, blockHash libcommon.Hash, blockNumber uint64) (*types.Body, error) {
|
2023-07-15 16:31:15 +00:00
|
|
|
if e.blockReader == nil {
|
2023-07-20 16:35:02 +00:00
|
|
|
body, _, _ := rawdb.ReadBody(tx, blockHash, blockNumber)
|
2023-07-15 16:31:15 +00:00
|
|
|
return body, nil
|
|
|
|
}
|
2023-07-20 16:35:02 +00:00
|
|
|
return e.blockReader.BodyWithTransactions(ctx, tx, blockHash, blockNumber)
|
2023-07-15 16:31:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (e *EthereumExecutionModule) canonicalHash(ctx context.Context, tx kv.Tx, blockNumber uint64) (libcommon.Hash, error) {
|
|
|
|
if e.blockReader == nil {
|
|
|
|
return rawdb.ReadCanonicalHash(tx, blockNumber)
|
|
|
|
}
|
|
|
|
return e.blockReader.CanonicalHash(ctx, tx, blockNumber)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remaining
|
2023-07-14 14:38:17 +00:00
|
|
|
|
2023-07-27 12:37:49 +00:00
|
|
|
func (e *EthereumExecutionModule) UpdateForkChoice(ctx context.Context, req *execution.ForkChoice) (*execution.ForkChoiceReceipt, error) {
|
2023-07-20 16:35:02 +00:00
|
|
|
type canonicalEntry struct {
|
|
|
|
hash libcommon.Hash
|
|
|
|
number uint64
|
|
|
|
}
|
2023-07-23 17:10:24 +00:00
|
|
|
if !e.semaphore.TryAcquire(1) {
|
|
|
|
return &execution.ForkChoiceReceipt{
|
|
|
|
LatestValidHash: gointerfaces.ConvertHashToH256(libcommon.Hash{}),
|
|
|
|
Status: execution.ValidationStatus_Busy,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
defer e.semaphore.Release(1)
|
|
|
|
|
2023-07-20 16:35:02 +00:00
|
|
|
tx, err := e.db.BeginRw(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
2023-07-27 12:37:49 +00:00
|
|
|
blockHash := gointerfaces.ConvertH256ToHash(req.HeadBlockHash)
|
2023-07-20 16:35:02 +00:00
|
|
|
// Step one, find reconnection point, and mark all of those headers as canonical.
|
|
|
|
fcuHeader, err := e.blockReader.HeaderByHash(ctx, tx, blockHash)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// If we dont have it, too bad
|
|
|
|
if fcuHeader == nil {
|
|
|
|
return &execution.ForkChoiceReceipt{
|
|
|
|
Status: execution.ValidationStatus_MissingSegment,
|
|
|
|
LatestValidHash: &types2.H256{},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
currentParentHash := fcuHeader.ParentHash
|
|
|
|
currentParentNumber := fcuHeader.Number.Uint64() - 1
|
|
|
|
isCanonicalHash, err := rawdb.IsCanonicalHash(tx, currentParentHash, currentParentNumber)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// Find such point, and collect all hashes
|
|
|
|
newCanonicals := make([]*canonicalEntry, 0, 2048)
|
|
|
|
newCanonicals = append(newCanonicals, &canonicalEntry{
|
|
|
|
hash: fcuHeader.Hash(),
|
|
|
|
number: fcuHeader.Number.Uint64(),
|
|
|
|
})
|
|
|
|
for !isCanonicalHash {
|
|
|
|
newCanonicals = append(newCanonicals, &canonicalEntry{
|
|
|
|
hash: currentParentHash,
|
|
|
|
number: currentParentNumber,
|
|
|
|
})
|
|
|
|
currentHeader, err := e.blockReader.Header(ctx, tx, currentParentHash, currentParentNumber)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if currentHeader == nil {
|
|
|
|
return &execution.ForkChoiceReceipt{
|
|
|
|
Status: execution.ValidationStatus_MissingSegment,
|
|
|
|
LatestValidHash: &types2.H256{},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
currentParentHash = currentHeader.ParentHash
|
|
|
|
currentParentNumber = currentHeader.Number.Uint64() - 1
|
|
|
|
isCanonicalHash, err = rawdb.IsCanonicalHash(tx, currentParentHash, currentParentNumber)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if currentParentNumber != fcuHeader.Number.Uint64()-1 {
|
|
|
|
e.executionPipeline.UnwindTo(currentParentNumber, libcommon.Hash{})
|
|
|
|
}
|
|
|
|
// Run the unwind
|
|
|
|
if err := e.executionPipeline.RunUnwind(e.db, tx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// Mark all new canonicals as canonicals
|
|
|
|
for _, canonicalSegment := range newCanonicals {
|
|
|
|
if err := rawdb.WriteCanonicalHash(tx, canonicalSegment.hash, canonicalSegment.number); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Set Progress for headers and bodies accordingly.
|
|
|
|
if err := stages.SaveStageProgress(tx, stages.Headers, fcuHeader.Number.Uint64()); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := stages.SaveStageProgress(tx, stages.Bodies, fcuHeader.Number.Uint64()); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err = rawdb.WriteHeadHeaderHash(tx, blockHash); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// Run the forkchoice
|
|
|
|
if err := e.executionPipeline.Run(e.db, tx, false); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// if head hash was set then success otherwise no
|
|
|
|
headHash := rawdb.ReadHeadBlockHash(tx)
|
|
|
|
headNumber := rawdb.ReadHeaderNumber(tx, headHash)
|
|
|
|
if headNumber != nil && e.logger != nil {
|
|
|
|
e.logger.Info("Current forkchoice", "hash", headHash, "number", *headNumber)
|
|
|
|
}
|
|
|
|
status := execution.ValidationStatus_Success
|
|
|
|
if headHash != blockHash {
|
|
|
|
status = execution.ValidationStatus_BadBlock
|
|
|
|
}
|
|
|
|
return &execution.ForkChoiceReceipt{
|
|
|
|
LatestValidHash: gointerfaces.ConvertHashToH256(headHash),
|
|
|
|
Status: status,
|
|
|
|
}, tx.Commit()
|
|
|
|
}
|
|
|
|
|
2023-07-23 17:10:24 +00:00
|
|
|
func (e *EthereumExecutionModule) ValidateChain(ctx context.Context, req *execution.ValidationRequest) (*execution.ValidationReceipt, error) {
|
|
|
|
if !e.semaphore.TryAcquire(1) {
|
|
|
|
return &execution.ValidationReceipt{
|
|
|
|
LatestValidHash: gointerfaces.ConvertHashToH256(libcommon.Hash{}),
|
|
|
|
ValidationStatus: execution.ValidationStatus_Busy,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
defer e.semaphore.Release(1)
|
|
|
|
tx, err := e.db.BeginRw(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
|
|
|
blockHash := gointerfaces.ConvertH256ToHash(req.Hash)
|
|
|
|
header, err := e.blockReader.Header(ctx, tx, blockHash, req.Number)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
body, err := e.blockReader.BodyWithTransactions(ctx, tx, blockHash, req.Number)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if header == nil || body == nil {
|
|
|
|
return &execution.ValidationReceipt{
|
|
|
|
LatestValidHash: gointerfaces.ConvertHashToH256(libcommon.Hash{}),
|
|
|
|
MissingHash: req.Hash,
|
|
|
|
ValidationStatus: execution.ValidationStatus_MissingSegment,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
status, lvh, validationError, criticalError := e.forkValidator.ValidatePayload(tx, header, body.RawBody(), false)
|
|
|
|
if criticalError != nil {
|
|
|
|
return nil, criticalError
|
|
|
|
}
|
|
|
|
validationStatus := execution.ValidationStatus_Success
|
|
|
|
if status == engine_types.AcceptedStatus {
|
|
|
|
validationStatus = execution.ValidationStatus_MissingSegment
|
|
|
|
}
|
|
|
|
if status == engine_types.InvalidStatus || status == engine_types.InvalidBlockHashStatus || validationError != nil {
|
|
|
|
e.logger.Warn("ethereumExecutionModule.ValidateChain: chain %x is invalid. reason %s", blockHash, err)
|
|
|
|
validationStatus = execution.ValidationStatus_BadBlock
|
|
|
|
}
|
|
|
|
return &execution.ValidationReceipt{
|
|
|
|
ValidationStatus: validationStatus,
|
|
|
|
LatestValidHash: gointerfaces.ConvertHashToH256(lvh),
|
|
|
|
MissingHash: gointerfaces.ConvertHashToH256(libcommon.Hash{}), // TODO: implement
|
|
|
|
}, nil
|
|
|
|
}
|