mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-11 13:30:05 +00:00
4f4b395aa4
* introduce code node * replace codeMap with code touches * fix a comment * fixups to tests * fix compile error * fix getnodedata tests * add tests and test stubs * add more test stubs * add test method bodies * add and fix more tests on trie for new codenode * add test change code between blocks * fix crash in stateless * remove unneded files * remove comment * fix deleted account code * fix resolve set builder for code nodes
290 lines
8.2 KiB
Go
290 lines
8.2 KiB
Go
package trie
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"runtime/debug"
|
|
"strings"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
|
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
|
"github.com/ledgerwatch/turbo-geth/trie/rlphacks"
|
|
)
|
|
|
|
type hookFunction func(*ResolveRequest, node, common.Hash) error
|
|
|
|
type ResolverStateful struct {
|
|
rss []*ResolveSet
|
|
curr bytes.Buffer // Current key for the structure generation algorithm, as well as the input tape for the hash builder
|
|
succ bytes.Buffer
|
|
value bytes.Buffer // Current value to be used as the value tape for the hash builder
|
|
groups []uint16
|
|
reqIndices []int // Indices pointing back to request slice from slices returned by PrepareResolveParams
|
|
hb *HashBuilder
|
|
topLevels int // How many top levels of the trie to keep (not roll into hashes)
|
|
currentReq *ResolveRequest // Request currently being handled
|
|
currentRs *ResolveSet // ResolveSet currently being used
|
|
keyIdx int
|
|
fieldSet uint32 // fieldSet for the next invocation of genStructStep
|
|
a accounts.Account
|
|
leafData GenStructStepLeafData
|
|
accData GenStructStepAccountData
|
|
requests []*ResolveRequest
|
|
|
|
roots []node // roots of the tries that are being built
|
|
hookFunction hookFunction
|
|
}
|
|
|
|
func NewResolverStateful(topLevels int, requests []*ResolveRequest, hookFunction hookFunction) *ResolverStateful {
|
|
return &ResolverStateful{
|
|
topLevels: topLevels,
|
|
hb: NewHashBuilder(false),
|
|
reqIndices: []int{},
|
|
requests: requests,
|
|
hookFunction: hookFunction,
|
|
}
|
|
}
|
|
|
|
// Reset prepares the Resolver for reuse
|
|
func (tr *ResolverStateful) Reset(topLevels int, requests []*ResolveRequest, hookFunction hookFunction) {
|
|
tr.topLevels = topLevels
|
|
tr.requests = tr.requests[:0]
|
|
tr.reqIndices = tr.reqIndices[:0]
|
|
tr.keyIdx = 0
|
|
tr.currentReq = nil
|
|
tr.currentRs = nil
|
|
tr.fieldSet = 0
|
|
tr.rss = tr.rss[:0]
|
|
tr.requests = requests
|
|
tr.hookFunction = hookFunction
|
|
tr.curr.Reset()
|
|
tr.succ.Reset()
|
|
tr.value.Reset()
|
|
tr.groups = tr.groups[:0]
|
|
tr.a.Reset()
|
|
tr.hb.Reset()
|
|
}
|
|
|
|
func (tr *ResolverStateful) PopRoots() []node {
|
|
roots := tr.roots
|
|
tr.roots = nil
|
|
return roots
|
|
}
|
|
|
|
// PrepareResolveParams prepares information for the MultiWalk
|
|
func (tr *ResolverStateful) PrepareResolveParams() ([][]byte, []uint) {
|
|
// Remove requests strictly contained in the preceding ones
|
|
startkeys := [][]byte{}
|
|
fixedbits := []uint{}
|
|
tr.rss = tr.rss[:0]
|
|
if len(tr.requests) == 0 {
|
|
return startkeys, fixedbits
|
|
}
|
|
var prevReq *ResolveRequest
|
|
for i, req := range tr.requests {
|
|
if prevReq == nil ||
|
|
!bytes.Equal(req.contract, prevReq.contract) ||
|
|
!bytes.Equal(req.resolveHex[:req.resolvePos], prevReq.resolveHex[:prevReq.resolvePos]) {
|
|
|
|
tr.reqIndices = append(tr.reqIndices, i)
|
|
pLen := len(req.contract)
|
|
key := make([]byte, pLen+32)
|
|
copy(key[:], req.contract)
|
|
decodeNibbles(req.resolveHex[:req.resolvePos], key[pLen:])
|
|
startkeys = append(startkeys, key)
|
|
req.extResolvePos = req.resolvePos + 2*pLen
|
|
fixedbits = append(fixedbits, uint(4*req.extResolvePos))
|
|
prevReq = req
|
|
var minLength int
|
|
if req.resolvePos >= tr.topLevels {
|
|
minLength = 0
|
|
} else {
|
|
minLength = tr.topLevels - req.resolvePos
|
|
}
|
|
rs := NewResolveSet(minLength)
|
|
tr.rss = append(tr.rss, rs)
|
|
rs.AddHex(req.resolveHex[req.resolvePos:])
|
|
} else {
|
|
rs := tr.rss[len(tr.rss)-1]
|
|
rs.AddHex(req.resolveHex[req.resolvePos:])
|
|
}
|
|
}
|
|
tr.currentReq = tr.requests[tr.reqIndices[0]]
|
|
tr.currentRs = tr.rss[0]
|
|
return startkeys, fixedbits
|
|
}
|
|
|
|
func (tr *ResolverStateful) finaliseRoot() error {
|
|
tr.curr.Reset()
|
|
tr.curr.Write(tr.succ.Bytes())
|
|
tr.succ.Reset()
|
|
if tr.curr.Len() > 0 {
|
|
var err error
|
|
var data GenStructStepData
|
|
if tr.fieldSet == 0 {
|
|
tr.leafData.Value = rlphacks.RlpSerializableBytes(tr.value.Bytes())
|
|
data = &tr.leafData
|
|
} else {
|
|
tr.accData.FieldSet = tr.fieldSet
|
|
tr.accData.StorageSize = tr.a.StorageSize
|
|
tr.accData.Balance.Set(&tr.a.Balance)
|
|
tr.accData.Nonce = tr.a.Nonce
|
|
tr.accData.Incarnation = tr.a.Incarnation
|
|
data = &tr.accData
|
|
}
|
|
tr.groups, err = GenStructStep(tr.currentRs.HashOnly, tr.curr.Bytes(), tr.succ.Bytes(), tr.hb, data, tr.groups, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if tr.hb.hasRoot() {
|
|
hbRoot := tr.hb.root()
|
|
hbHash := tr.hb.rootHash()
|
|
return tr.hookFunction(tr.currentReq, hbRoot, hbHash)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tr *ResolverStateful) RebuildTrie(
|
|
db ethdb.Database,
|
|
blockNr uint64,
|
|
accounts bool,
|
|
historical bool) error {
|
|
startkeys, fixedbits := tr.PrepareResolveParams()
|
|
if db == nil {
|
|
var b strings.Builder
|
|
fmt.Fprintf(&b, "ResolveWithDb(db=nil), accounts: %t\n", accounts)
|
|
for i, sk := range startkeys {
|
|
fmt.Fprintf(&b, "sk %x, bits: %d\n", sk, fixedbits[i])
|
|
}
|
|
return fmt.Errorf("unexpected resolution: %s at %s", b.String(), debug.Stack())
|
|
}
|
|
|
|
var err error
|
|
if accounts {
|
|
if historical {
|
|
err = db.MultiWalkAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, startkeys, fixedbits, blockNr+1, tr.WalkerAccounts)
|
|
} else {
|
|
err = db.MultiWalk(dbutils.AccountsBucket, startkeys, fixedbits, tr.WalkerAccounts)
|
|
}
|
|
} else {
|
|
if historical {
|
|
err = db.MultiWalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, startkeys, fixedbits, blockNr+1, tr.WalkerStorage)
|
|
} else {
|
|
err = db.MultiWalk(dbutils.StorageBucket, startkeys, fixedbits, tr.WalkerStorage)
|
|
}
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err = tr.finaliseRoot(); err != nil {
|
|
return fmt.Errorf("error in finaliseRoot, for block %d: %w", blockNr, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tr *ResolverStateful) AttachRequestedCode(db ethdb.Getter, requests []*ResolveRequestForCode) error {
|
|
for _, req := range requests {
|
|
codeHash := req.codeHash
|
|
code, err := db.Get(dbutils.CodeBucket, codeHash[:])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := req.t.UpdateAccountCode(req.addrHash[:], codeNode(code)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (tr *ResolverStateful) WalkerAccounts(keyIdx int, k []byte, v []byte) error {
|
|
return tr.Walker(true, keyIdx, k, v)
|
|
}
|
|
|
|
func (tr *ResolverStateful) WalkerStorage(keyIdx int, k []byte, v []byte) error {
|
|
return tr.Walker(false, keyIdx, k, v)
|
|
}
|
|
|
|
// Walker - k, v - shouldn't be reused in the caller's code
|
|
func (tr *ResolverStateful) Walker(isAccount bool, keyIdx int, k []byte, v []byte) error {
|
|
//fmt.Printf("Walker: keyIdx: %d key:%x value:%x\n", keyIdx, k, v)
|
|
if keyIdx != tr.keyIdx {
|
|
if err := tr.finaliseRoot(); err != nil {
|
|
return err
|
|
}
|
|
tr.hb.Reset()
|
|
tr.groups = nil
|
|
tr.keyIdx = keyIdx
|
|
tr.currentReq = tr.requests[tr.reqIndices[keyIdx]]
|
|
tr.currentRs = tr.rss[keyIdx]
|
|
tr.curr.Reset()
|
|
}
|
|
if len(v) > 0 {
|
|
tr.curr.Reset()
|
|
tr.curr.Write(tr.succ.Bytes())
|
|
tr.succ.Reset()
|
|
skip := tr.currentReq.extResolvePos // how many first nibbles to skip
|
|
i := 0
|
|
for _, b := range k {
|
|
if i >= skip {
|
|
tr.succ.WriteByte(b / 16)
|
|
}
|
|
i++
|
|
if i >= skip {
|
|
tr.succ.WriteByte(b % 16)
|
|
}
|
|
i++
|
|
}
|
|
tr.succ.WriteByte(16)
|
|
if tr.curr.Len() > 0 {
|
|
var err error
|
|
var data GenStructStepData
|
|
if tr.fieldSet == 0 {
|
|
tr.leafData.Value = rlphacks.RlpSerializableBytes(tr.value.Bytes())
|
|
data = &tr.leafData
|
|
} else {
|
|
tr.accData.FieldSet = tr.fieldSet
|
|
tr.accData.StorageSize = tr.a.StorageSize
|
|
tr.accData.Balance.Set(&tr.a.Balance)
|
|
tr.accData.Nonce = tr.a.Nonce
|
|
tr.accData.Incarnation = tr.a.Incarnation
|
|
data = &tr.accData
|
|
}
|
|
tr.groups, err = GenStructStep(tr.currentRs.HashOnly, tr.curr.Bytes(), tr.succ.Bytes(), tr.hb, data, tr.groups, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
// Remember the current key and value
|
|
if isAccount {
|
|
if err := tr.a.DecodeForStorage(v); err != nil {
|
|
return err
|
|
}
|
|
if tr.a.IsEmptyCodeHash() && tr.a.IsEmptyRoot() {
|
|
tr.fieldSet = AccountFieldSetNotContract
|
|
} else {
|
|
if tr.a.HasStorageSize {
|
|
tr.fieldSet = AccountFieldSetContractWithSize
|
|
} else {
|
|
tr.fieldSet = AccountFieldSetContract
|
|
}
|
|
// the first item ends up deepest on the stack, the seccond item - on the top
|
|
if err := tr.hb.hash(tr.a.CodeHash[:]); err != nil {
|
|
return err
|
|
}
|
|
if err := tr.hb.hash(tr.a.Root[:]); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
} else {
|
|
tr.value.Reset()
|
|
tr.value.Write(v)
|
|
tr.fieldSet = AccountFieldSetNotAccount
|
|
}
|
|
}
|
|
return nil
|
|
}
|