2023-04-08 01:01:10 +00:00
|
|
|
package forkchoice
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-06-14 03:01:00 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
2023-06-04 23:52:55 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/freezer"
|
2023-05-13 21:44:07 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/phase1/forkchoice/fork_graph"
|
2023-06-11 21:50:02 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/transition/impl/eth2/statechange"
|
2023-04-08 01:01:10 +00:00
|
|
|
)
|
|
|
|
|
2023-04-26 13:33:21 +00:00
|
|
|
func (f *ForkChoiceStore) OnBlock(block *cltypes.SignedBeaconBlock, newPayload, fullValidation bool) error {
|
2023-04-08 01:01:10 +00:00
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
blockRoot, err := block.Block.HashSSZ()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-05-02 14:19:22 +00:00
|
|
|
if f.Slot() < block.Block.Slot {
|
2023-05-28 20:48:17 +00:00
|
|
|
return fmt.Errorf("block is too early compared to current_slot")
|
2023-05-02 14:19:22 +00:00
|
|
|
}
|
2023-04-08 01:01:10 +00:00
|
|
|
// Check that block is later than the finalized epoch slot (optimization to reduce calls to get_ancestor)
|
2023-05-14 22:12:24 +00:00
|
|
|
finalizedSlot := f.computeStartSlotAtEpoch(f.finalizedCheckpoint.Epoch())
|
2023-04-08 01:01:10 +00:00
|
|
|
if block.Block.Slot <= finalizedSlot {
|
2023-05-28 20:48:17 +00:00
|
|
|
return nil
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
config := f.forkGraph.Config()
|
2023-04-20 20:47:58 +00:00
|
|
|
lastProcessedState, status, err := f.forkGraph.AddChainSegment(block, fullValidation)
|
2023-05-17 23:06:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
switch status {
|
|
|
|
case fork_graph.PreValidated:
|
2023-04-17 18:06:50 +00:00
|
|
|
return nil
|
2023-05-17 23:06:25 +00:00
|
|
|
case fork_graph.Success:
|
2023-06-02 12:54:40 +00:00
|
|
|
case fork_graph.BelowAnchor:
|
|
|
|
log.Debug("replay block", "code", status)
|
2023-07-18 08:47:38 +00:00
|
|
|
return nil
|
2023-05-17 23:06:25 +00:00
|
|
|
default:
|
|
|
|
return fmt.Errorf("replay block, code: %+v", status)
|
2023-04-17 18:06:50 +00:00
|
|
|
}
|
2023-07-30 21:35:55 +00:00
|
|
|
if block.Block.Body.ExecutionPayload != nil {
|
|
|
|
f.eth2Roots.Add(blockRoot, block.Block.Body.ExecutionPayload.BlockHash)
|
|
|
|
}
|
2023-07-08 14:42:30 +00:00
|
|
|
var invalidBlock bool
|
2023-04-26 13:33:21 +00:00
|
|
|
if newPayload && f.engine != nil {
|
2023-08-01 10:31:26 +00:00
|
|
|
if invalidBlock, err = f.engine.NewPayload(block.Block.Body.ExecutionPayload, &block.Block.ParentRoot); err != nil {
|
2023-04-26 13:33:21 +00:00
|
|
|
log.Warn("newPayload failed", "err", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2023-07-08 14:42:30 +00:00
|
|
|
if invalidBlock {
|
|
|
|
f.forkGraph.MarkHeaderAsInvalid(blockRoot)
|
|
|
|
}
|
2023-07-30 21:35:55 +00:00
|
|
|
|
2023-04-17 18:06:50 +00:00
|
|
|
if block.Block.Slot > f.highestSeen {
|
|
|
|
f.highestSeen = block.Block.Slot
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
// Add proposer score boost if the block is timely
|
|
|
|
timeIntoSlot := (f.time - f.forkGraph.GenesisTime()) % lastProcessedState.BeaconConfig().SecondsPerSlot
|
|
|
|
isBeforeAttestingInterval := timeIntoSlot < config.SecondsPerSlot/config.IntervalsPerSlot
|
|
|
|
if f.Slot() == block.Block.Slot && isBeforeAttestingInterval {
|
|
|
|
f.proposerBoostRoot = blockRoot
|
|
|
|
}
|
2023-06-04 23:52:55 +00:00
|
|
|
if lastProcessedState.Slot()%f.forkGraph.Config().SlotsPerEpoch == 0 {
|
|
|
|
if err := freezer.PutObjectSSZIntoFreezer("beaconState", "caplin_core", lastProcessedState.Slot(), lastProcessedState, f.recorder); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2023-04-08 01:01:10 +00:00
|
|
|
// Update checkpoints
|
|
|
|
f.updateCheckpoints(lastProcessedState.CurrentJustifiedCheckpoint().Copy(), lastProcessedState.FinalizedCheckpoint().Copy())
|
2023-04-20 20:47:58 +00:00
|
|
|
// First thing save previous values of the checkpoints (avoid memory copy of all states and ensure easy revert)
|
|
|
|
var (
|
|
|
|
previousJustifiedCheckpoint = lastProcessedState.PreviousJustifiedCheckpoint().Copy()
|
|
|
|
currentJustifiedCheckpoint = lastProcessedState.CurrentJustifiedCheckpoint().Copy()
|
|
|
|
finalizedCheckpoint = lastProcessedState.FinalizedCheckpoint().Copy()
|
|
|
|
justificationBits = lastProcessedState.JustificationBits().Copy()
|
|
|
|
)
|
2023-04-08 01:01:10 +00:00
|
|
|
// Eagerly compute unrealized justification and finality
|
2023-06-11 21:50:02 +00:00
|
|
|
if err := statechange.ProcessJustificationBitsAndFinality(lastProcessedState); err != nil {
|
2023-04-08 01:01:10 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
f.updateUnrealizedCheckpoints(lastProcessedState.CurrentJustifiedCheckpoint().Copy(), lastProcessedState.FinalizedCheckpoint().Copy())
|
2023-04-20 20:47:58 +00:00
|
|
|
// Set the changed value pre-simulation
|
|
|
|
lastProcessedState.SetPreviousJustifiedCheckpoint(previousJustifiedCheckpoint)
|
|
|
|
lastProcessedState.SetCurrentJustifiedCheckpoint(currentJustifiedCheckpoint)
|
|
|
|
lastProcessedState.SetFinalizedCheckpoint(finalizedCheckpoint)
|
|
|
|
lastProcessedState.SetJustificationBits(justificationBits)
|
2023-04-08 01:01:10 +00:00
|
|
|
// If the block is from a prior epoch, apply the realized values
|
|
|
|
blockEpoch := f.computeEpochAtSlot(block.Block.Slot)
|
|
|
|
currentEpoch := f.computeEpochAtSlot(f.Slot())
|
|
|
|
if blockEpoch < currentEpoch {
|
|
|
|
f.updateCheckpoints(lastProcessedState.CurrentJustifiedCheckpoint().Copy(), lastProcessedState.FinalizedCheckpoint().Copy())
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|