mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2024-12-31 16:21:21 +00:00
276 lines
6.4 KiB
Go
276 lines
6.4 KiB
Go
package state
|
|
|
|
import (
|
|
//"fmt"
|
|
|
|
"bytes"
|
|
"container/heap"
|
|
"context"
|
|
"encoding/binary"
|
|
"sync"
|
|
|
|
"github.com/RoaringBitmap/roaring/roaring64"
|
|
"github.com/google/btree"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/cmd/state/exec22"
|
|
btree2 "github.com/tidwall/btree"
|
|
)
|
|
|
|
type reconPair struct {
|
|
txNum uint64 // txNum where the item has been created
|
|
key1, key2 []byte
|
|
val []byte
|
|
}
|
|
|
|
func (i reconPair) Less(than btree.Item) bool { return ReconnLess(i, than.(reconPair)) }
|
|
|
|
func ReconnLess(i, thanItem reconPair) bool {
|
|
if i.txNum == thanItem.txNum {
|
|
c1 := bytes.Compare(i.key1, thanItem.key1)
|
|
if c1 == 0 {
|
|
c2 := bytes.Compare(i.key2, thanItem.key2)
|
|
return c2 < 0
|
|
}
|
|
return c1 < 0
|
|
}
|
|
return i.txNum < thanItem.txNum
|
|
}
|
|
|
|
type ReconnWork struct {
|
|
lock sync.RWMutex
|
|
doneBitmap roaring64.Bitmap
|
|
triggers map[uint64][]*exec22.TxTask
|
|
workCh chan *exec22.TxTask
|
|
queue exec22.TxTaskQueue
|
|
rollbackCount uint64
|
|
maxTxNum uint64
|
|
}
|
|
|
|
// ReconState is the accumulator of changes to the state
|
|
type ReconState struct {
|
|
*ReconnWork //has it's own mutex. allow avoid lock-contention between state.Get() and work.Done() methods
|
|
|
|
lock sync.RWMutex
|
|
changes map[string]*btree2.BTreeG[reconPair] // table => [] (txNum; key1; key2; val)
|
|
hints map[string]*btree2.PathHint
|
|
sizeEstimate int
|
|
}
|
|
|
|
func NewReconState(workCh chan *exec22.TxTask) *ReconState {
|
|
rs := &ReconState{
|
|
ReconnWork: &ReconnWork{
|
|
workCh: workCh,
|
|
triggers: map[uint64][]*exec22.TxTask{},
|
|
},
|
|
changes: map[string]*btree2.BTreeG[reconPair]{},
|
|
hints: map[string]*btree2.PathHint{},
|
|
}
|
|
return rs
|
|
}
|
|
|
|
func (rs *ReconState) Reset(workCh chan *exec22.TxTask) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
rs.workCh = workCh
|
|
rs.triggers = map[uint64][]*exec22.TxTask{}
|
|
rs.rollbackCount = 0
|
|
rs.queue = rs.queue[:cap(rs.queue)]
|
|
for i := 0; i < len(rs.queue); i++ {
|
|
rs.queue[i] = nil
|
|
}
|
|
rs.queue = rs.queue[:0]
|
|
}
|
|
|
|
func (rs *ReconState) Put(table string, key1, key2, val []byte, txNum uint64) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
t, ok := rs.changes[table]
|
|
if !ok {
|
|
t = btree2.NewBTreeGOptions[reconPair](ReconnLess, btree2.Options{Degree: 128, NoLocks: true})
|
|
rs.changes[table] = t
|
|
rs.hints[table] = &btree2.PathHint{}
|
|
}
|
|
item := reconPair{key1: key1, key2: key2, val: val, txNum: txNum}
|
|
old, ok := t.SetHint(item, rs.hints[table])
|
|
rs.sizeEstimate += len(key1) + len(key2) + len(val)
|
|
if ok {
|
|
rs.sizeEstimate -= len(old.key1) + len(old.key2) + len(old.val)
|
|
}
|
|
}
|
|
|
|
func (rs *ReconState) Delete(table string, key1, key2 []byte, txNum uint64) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
t, ok := rs.changes[table]
|
|
if !ok {
|
|
t = btree2.NewBTreeGOptions[reconPair](ReconnLess, btree2.Options{Degree: 128, NoLocks: true})
|
|
rs.changes[table] = t
|
|
rs.hints[table] = &btree2.PathHint{}
|
|
}
|
|
item := reconPair{key1: key1, key2: key2, val: nil, txNum: txNum}
|
|
old, ok := t.SetHint(item, rs.hints[table])
|
|
rs.sizeEstimate += len(key1) + len(key2)
|
|
if ok {
|
|
rs.sizeEstimate -= len(old.key1) + len(old.key2) + len(old.val)
|
|
}
|
|
}
|
|
|
|
func (rs *ReconState) RemoveAll(table string, key1 []byte) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
t, ok := rs.changes[table]
|
|
if !ok {
|
|
return
|
|
}
|
|
t.Ascend(reconPair{key1: key1, key2: nil}, func(item reconPair) bool {
|
|
if !bytes.Equal(item.key1, key1) {
|
|
return false
|
|
}
|
|
if item.key2 == nil {
|
|
return true
|
|
}
|
|
t.Delete(item)
|
|
return true
|
|
})
|
|
}
|
|
|
|
func (rs *ReconState) Get(table string, key1, key2 []byte, txNum uint64) []byte {
|
|
rs.lock.RLock()
|
|
defer rs.lock.RUnlock()
|
|
t, ok := rs.changes[table]
|
|
if !ok {
|
|
return nil
|
|
}
|
|
i, ok := t.Get(reconPair{txNum: txNum, key1: key1, key2: key2})
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return i.val
|
|
}
|
|
|
|
func (rs *ReconState) Flush(rwTx kv.RwTx) error {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
for table, t := range rs.changes {
|
|
var err error
|
|
t.Walk(func(items []reconPair) bool {
|
|
for _, item := range items {
|
|
var composite []byte
|
|
if item.key2 == nil {
|
|
composite = make([]byte, 8+len(item.key1))
|
|
} else {
|
|
composite = make([]byte, 8+len(item.key1)+8+len(item.key2))
|
|
binary.BigEndian.PutUint64(composite[8+len(item.key1):], FirstContractIncarnation)
|
|
copy(composite[8+len(item.key1)+8:], item.key2)
|
|
}
|
|
binary.BigEndian.PutUint64(composite, item.txNum)
|
|
copy(composite[8:], item.key1)
|
|
if len(item.val) == 0 {
|
|
if err = rwTx.Put(table, composite[:8], composite[8:]); err != nil {
|
|
return false
|
|
}
|
|
} else {
|
|
if err = rwTx.Put(table, composite, item.val); err != nil {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
return true
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
t.Clear()
|
|
}
|
|
rs.sizeEstimate = 0
|
|
return nil
|
|
}
|
|
|
|
func (rs *ReconnWork) Schedule(ctx context.Context) (*exec22.TxTask, bool, error) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
Loop:
|
|
for rs.queue.Len() < 16 {
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, false, ctx.Err()
|
|
case txTask, ok := <-rs.workCh:
|
|
if !ok {
|
|
// No more work, channel is closed
|
|
break Loop
|
|
}
|
|
heap.Push(&rs.queue, txTask)
|
|
}
|
|
}
|
|
if rs.queue.Len() > 0 {
|
|
return heap.Pop(&rs.queue).(*exec22.TxTask), true, nil
|
|
}
|
|
return nil, false, nil
|
|
}
|
|
|
|
func (rs *ReconnWork) CommitTxNum(txNum uint64) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
if tt, ok := rs.triggers[txNum]; ok {
|
|
for _, t := range tt {
|
|
heap.Push(&rs.queue, t)
|
|
}
|
|
delete(rs.triggers, txNum)
|
|
}
|
|
rs.doneBitmap.Add(txNum)
|
|
if txNum > rs.maxTxNum {
|
|
rs.maxTxNum = txNum
|
|
}
|
|
}
|
|
|
|
func (rs *ReconnWork) RollbackTx(txTask *exec22.TxTask, dependency uint64) {
|
|
rs.lock.Lock()
|
|
defer rs.lock.Unlock()
|
|
if rs.doneBitmap.Contains(dependency) {
|
|
heap.Push(&rs.queue, txTask)
|
|
} else {
|
|
tt := rs.triggers[dependency]
|
|
tt = append(tt, txTask)
|
|
rs.triggers[dependency] = tt
|
|
}
|
|
rs.rollbackCount++
|
|
}
|
|
|
|
func (rs *ReconnWork) Done(txNum uint64) bool {
|
|
rs.lock.RLock()
|
|
c := rs.doneBitmap.Contains(txNum)
|
|
rs.lock.RUnlock()
|
|
return c
|
|
}
|
|
|
|
func (rs *ReconnWork) DoneCount() uint64 {
|
|
rs.lock.RLock()
|
|
c := rs.doneBitmap.GetCardinality()
|
|
rs.lock.RUnlock()
|
|
return c
|
|
}
|
|
|
|
func (rs *ReconnWork) MaxTxNum() uint64 {
|
|
rs.lock.RLock()
|
|
defer rs.lock.RUnlock()
|
|
return rs.maxTxNum
|
|
}
|
|
|
|
func (rs *ReconnWork) RollbackCount() uint64 {
|
|
rs.lock.RLock()
|
|
defer rs.lock.RUnlock()
|
|
return rs.rollbackCount
|
|
}
|
|
|
|
func (rs *ReconnWork) QueueLen() int {
|
|
rs.lock.RLock()
|
|
defer rs.lock.RUnlock()
|
|
return rs.queue.Len()
|
|
}
|
|
|
|
func (rs *ReconState) SizeEstimate() uint64 {
|
|
rs.lock.RLock()
|
|
defer rs.lock.RUnlock()
|
|
return uint64(rs.sizeEstimate)
|
|
}
|