erigon-pulse/core/state/recon_state.go
ledgerwatch b5a7faa8db
[e3] Incremental reconstitution (#6270)
Co-authored-by: Alex Sharp <alexsharp@Alexs-MacBook-Pro-2.local>
Co-authored-by: Alexey Sharp <alexeysharp@Alexeys-iMac.local>
2022-12-10 09:34:17 +00:00

270 lines
6.4 KiB
Go

package state
import (
//"fmt"
"bytes"
"container/heap"
"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 uint64
}
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 += btreeOverhead + uint64(len(key1)) + uint64(len(key2)) + uint64(len(val))
if ok {
rs.sizeEstimate -= btreeOverhead + uint64(len(old.key1)) + uint64(len(old.key2)) + uint64(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 += btreeOverhead + uint64(len(key1)) + uint64(len(key2))
if ok {
rs.sizeEstimate -= btreeOverhead + uint64(len(old.key1)) + uint64(len(old.key2)) + uint64(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.GetHint(reconPair{txNum: txNum, key1: key1, key2: key2}, rs.hints[table])
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() (*exec22.TxTask, bool) {
rs.lock.Lock()
defer rs.lock.Unlock()
for rs.queue.Len() < 16 {
txTask, ok := <-rs.workCh
if !ok {
// No more work, channel is closed
break
}
heap.Push(&rs.queue, txTask)
}
if rs.queue.Len() > 0 {
return heap.Pop(&rs.queue).(*exec22.TxTask), true
}
return nil, false
}
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 rs.sizeEstimate
}