mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-05 10:32:19 +00:00
221 lines
4.8 KiB
Go
221 lines
4.8 KiB
Go
package engineapi
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"sync/atomic"
|
|
|
|
"github.com/emirpasic/gods/maps/treemap"
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces/remote"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
)
|
|
|
|
// This is the status of a newly execute block.
|
|
// Hash: Block hash
|
|
// Status: block's status
|
|
type PayloadStatus struct {
|
|
Status remote.EngineStatus
|
|
LatestValidHash libcommon.Hash
|
|
ValidationError error
|
|
CriticalError error
|
|
}
|
|
|
|
// The message we are going to send to the stage sync in ForkchoiceUpdated
|
|
type ForkChoiceMessage struct {
|
|
HeadBlockHash libcommon.Hash
|
|
SafeBlockHash libcommon.Hash
|
|
FinalizedBlockHash libcommon.Hash
|
|
}
|
|
|
|
type RequestStatus int
|
|
|
|
const ( // RequestStatus values
|
|
New = iota
|
|
DataWasMissing
|
|
)
|
|
|
|
type RequestWithStatus struct {
|
|
Message interface{} // *Block or *ForkChoiceMessage
|
|
Status RequestStatus
|
|
}
|
|
|
|
type Interrupt int
|
|
|
|
const ( // Interrupt values
|
|
None = iota
|
|
Synced
|
|
Stopping
|
|
)
|
|
|
|
type RequestList struct {
|
|
requestId int
|
|
requests *treemap.Map // map[int]*RequestWithStatus
|
|
interrupt Interrupt
|
|
waiting atomic.Uint32
|
|
syncCond *sync.Cond
|
|
|
|
// waiterMtx is used to place locks around `waiters` so the slice can be read/modified safely
|
|
// and also serves as a lock around the `waiting` field so that this can't be modified whilst it
|
|
// is being read. waiters is modified as part of updating waiter so we need to ensure these
|
|
// two things lock at the same time
|
|
waiterMtx sync.Mutex
|
|
|
|
waiters []chan struct{}
|
|
}
|
|
|
|
func NewRequestList() *RequestList {
|
|
rl := &RequestList{
|
|
requests: treemap.NewWithIntComparator(),
|
|
syncCond: sync.NewCond(&sync.Mutex{}),
|
|
waiterMtx: sync.Mutex{},
|
|
waiters: make([]chan struct{}, 0),
|
|
waiting: atomic.Uint32{},
|
|
}
|
|
return rl
|
|
}
|
|
|
|
func (rl *RequestList) AddPayloadRequest(message *types.Block) {
|
|
rl.syncCond.L.Lock()
|
|
defer rl.syncCond.L.Unlock()
|
|
|
|
rl.requestId++
|
|
|
|
rl.requests.Put(rl.requestId, &RequestWithStatus{
|
|
Message: message,
|
|
Status: New,
|
|
})
|
|
|
|
rl.syncCond.Broadcast()
|
|
}
|
|
|
|
func (rl *RequestList) AddForkChoiceRequest(message *ForkChoiceMessage) {
|
|
rl.syncCond.L.Lock()
|
|
defer rl.syncCond.L.Unlock()
|
|
|
|
rl.requestId++
|
|
|
|
// purge previous fork choices that are still syncing
|
|
rl.requests = rl.requests.Select(func(key interface{}, value interface{}) bool {
|
|
req := value.(*RequestWithStatus)
|
|
_, isForkChoice := req.Message.(*ForkChoiceMessage)
|
|
return req.Status == New || !isForkChoice
|
|
})
|
|
// TODO(yperbasis): potentially purge some non-syncing old fork choices?
|
|
|
|
rl.requests.Put(rl.requestId, &RequestWithStatus{
|
|
Message: message,
|
|
Status: New,
|
|
})
|
|
|
|
rl.syncCond.Broadcast()
|
|
}
|
|
|
|
func (rl *RequestList) firstRequest(onlyNew bool) (id int, request *RequestWithStatus) {
|
|
foundKey, foundValue := rl.requests.Min()
|
|
if onlyNew {
|
|
foundKey, foundValue = rl.requests.Find(func(key interface{}, value interface{}) bool {
|
|
return value.(*RequestWithStatus).Status == New
|
|
})
|
|
}
|
|
if foundKey != nil {
|
|
return foundKey.(int), foundValue.(*RequestWithStatus)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func (rl *RequestList) WaitForRequest(onlyNew bool, noWait bool) (interrupt Interrupt, id int, request *RequestWithStatus) {
|
|
rl.syncCond.L.Lock()
|
|
defer rl.syncCond.L.Unlock()
|
|
|
|
rl.waiterMtx.Lock()
|
|
rl.updateWaiting(1)
|
|
rl.waiterMtx.Unlock()
|
|
defer func() {
|
|
rl.waiterMtx.Lock()
|
|
rl.updateWaiting(0)
|
|
rl.waiterMtx.Unlock()
|
|
}()
|
|
|
|
for {
|
|
interrupt = rl.interrupt
|
|
if interrupt != None {
|
|
if interrupt != Stopping {
|
|
// clear the interrupt
|
|
rl.interrupt = None
|
|
}
|
|
return
|
|
}
|
|
id, request = rl.firstRequest(onlyNew)
|
|
if request != nil || noWait {
|
|
return
|
|
}
|
|
rl.syncCond.Wait()
|
|
}
|
|
}
|
|
|
|
func (rl *RequestList) IsWaiting() bool {
|
|
return rl.waiting.Load() != 0
|
|
}
|
|
|
|
// update waiting should always be called from a locked context using rl.waiterMtx as it
|
|
// updates rl.waiters in certain scenarios
|
|
func (rl *RequestList) updateWaiting(val uint32) {
|
|
rl.waiting.Store(val)
|
|
|
|
if val == 1 {
|
|
for i, c := range rl.waiters {
|
|
close(c)
|
|
rl.waiters[i] = nil
|
|
}
|
|
rl.waiters = rl.waiters[:0]
|
|
}
|
|
}
|
|
|
|
func (rl *RequestList) WaitForWaiting(ctx context.Context) (chan struct{}, bool) {
|
|
rl.waiterMtx.Lock()
|
|
defer rl.waiterMtx.Unlock()
|
|
|
|
isWaiting := rl.waiting.Load()
|
|
|
|
if isWaiting == 1 {
|
|
// we are already waiting so just return
|
|
return nil, true
|
|
} else {
|
|
c := make(chan struct{}, 1)
|
|
rl.waiters = append(rl.waiters, c)
|
|
return c, false
|
|
}
|
|
}
|
|
|
|
func (rl *RequestList) Interrupt(kind Interrupt) {
|
|
rl.syncCond.L.Lock()
|
|
defer rl.syncCond.L.Unlock()
|
|
|
|
rl.interrupt = kind
|
|
|
|
rl.syncCond.Broadcast()
|
|
}
|
|
|
|
func (rl *RequestList) Remove(id int) {
|
|
rl.syncCond.L.Lock()
|
|
defer rl.syncCond.L.Unlock()
|
|
|
|
rl.requests.Remove(id)
|
|
// no need to broadcast
|
|
}
|
|
|
|
func (rl *RequestList) SetStatus(id int, status RequestStatus) {
|
|
rl.syncCond.L.Lock()
|
|
defer rl.syncCond.L.Unlock()
|
|
|
|
value, found := rl.requests.Get(id)
|
|
if found {
|
|
value.(*RequestWithStatus).Status = status
|
|
}
|
|
|
|
rl.syncCond.Broadcast()
|
|
}
|