mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-14 23:08:20 +00:00
4565bd43c1
* up devel * up devel * up devel * up devel * up devel * up devel * up devel * up devel * squash * squash * squash * squash * squash * squash * squash * squash * squash * squash * squash * squash * squash * switch to master * mdbx 0.9.2 * mdbx 0.9.2 * %<<$!%%%%%%<o'{}k.### * %<<$!%%%%%%<o'{}k.### * merge * merge * merge * merge * ad * ad * ad * ad
946 lines
26 KiB
Go
946 lines
26 KiB
Go
package trie
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"time"
|
|
|
|
"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/log"
|
|
"github.com/ledgerwatch/turbo-geth/turbo/rlphacks"
|
|
)
|
|
|
|
/*
|
|
**Theoretically:** "Merkle trie root calculation" starts from state, build from state keys - trie,
|
|
on each level of trie calculates intermediate hash of underlying data.
|
|
|
|
**Practically:** It can be implemented as "Preorder trie traversal" (Preorder - visit Root, visit Left, visit Right).
|
|
But, let's make couple observations to make traversal over huge state efficient.
|
|
|
|
**Observation 1:** `CurrentStateBucket` already stores state keys in sorted way.
|
|
Iteration over this bucket will retrieve keys in same order as "Preorder trie traversal".
|
|
|
|
**Observation 2:** each Eth block - changes not big part of state - it means most of Merkle trie intermediate hashes will not change.
|
|
It means we effectively can cache them. `IntermediateTrieHashBucket` stores "Intermediate hashes of all Merkle trie levels".
|
|
It also sorted and Iteration over `IntermediateTrieHashBucket` will retrieve keys in same order as "Preorder trie traversal".
|
|
|
|
**Implementation:** by opening 1 Cursor on state and 1 more Cursor on intermediate hashes bucket - we will receive data in
|
|
order of "Preorder trie traversal". Cursors will only do "sequential reads" and "jumps forward" - been hardware-friendly.
|
|
1 stack keeps all accumulated hashes, when sub-trie traverse ends - all hashes pulled from stack -> hashed -> new hash puts on stack - it's hash of visited sub-trie (it emulates recursive nature of "Preorder trie traversal" algo).
|
|
|
|
Imagine that account with key 0000....00 (64 zeroes, 32 bytes of zeroes) changed.
|
|
Here is an example sequence which can be seen by running 2 Cursors:
|
|
```
|
|
00 // key which came from cache, can't use it - because account with this prefix changed
|
|
0000 // key which came from cache, can't use it - because account with this prefix changed
|
|
...
|
|
{30 zero bytes}00 // key which came from cache, can't use it - because account with this prefix changed
|
|
{30 zero bytes}0000 // Account which came from state, use it - calculate hash, jump to "next sub-trie"
|
|
{30 zero bytes}01 // key which came from cache, it is "next sub-trie", use it, jump to "next sub-trie"
|
|
{30 zero bytes}02 // key which came from cache, it is "next sub-trie", use it, jump to "next sub-trie"
|
|
...
|
|
{30 zero bytes}ff // key which came from cache, it is "next sub-trie", use it, jump to "next sub-trie"
|
|
{29 zero bytes}01 // key which came from cache, it is "next sub-trie" (1 byte shorter key), use it, jump to "next sub-trie"
|
|
{29 zero bytes}02 // key which came from cache, it is "next sub-trie" (1 byte shorter key), use it, jump to "next sub-trie"
|
|
...
|
|
ff // key which came from cache, it is "next sub-trie" (1 byte shorter key), use it, jump to "next sub-trie"
|
|
nil // db returned nil - means no more keys there, done
|
|
```
|
|
On practice Trie is no full - it means after account key `{30 zero bytes}0000` may come `{5 zero bytes}01` and amount of iterations will not be big.
|
|
|
|
### Attack - by delete account with huge state
|
|
|
|
It's possible to create Account with very big storage (increase storage size during many blocks).
|
|
Then delete this account (SELFDESTRUCT).
|
|
Naive storage deletion may take several minutes - depends on Disk speed - means every Eth client
|
|
will not process any incoming block that time. To protect against this attack:
|
|
PlainState, HashedState and IntermediateTrieHash buckets have "incarnations". Account entity has field "Incarnation" -
|
|
just a digit which increasing each SELFDESTRUCT or CREATE2 opcodes. Storage key formed by:
|
|
`{account_key}{incarnation}{storage_hash}`. And [trie/trie_root.go](../../trie/trie_root.go) has logic - every time
|
|
when Account visited - we save it to `accAddrHashWithInc` variable and skip any Storage or IntermediateTrieHashes with another incarnation.
|
|
*/
|
|
|
|
// FlatDBTrieLoader reads state and intermediate trie hashes in order equal to "Preorder trie traversal"
|
|
// (Preorder - visit Root, visit Left, visit Right)
|
|
//
|
|
// It produces stream of values and send this stream to `defaultReceiver`
|
|
// It skips storage with incorrect incarnations
|
|
//
|
|
// Each intermediate hash key firstly pass to RetainDecider, only if it returns "false" - such IH can be used.
|
|
type FlatDBTrieLoader struct {
|
|
logPrefix string
|
|
trace bool
|
|
itemPresent bool
|
|
itemType StreamItem
|
|
stateBucket string
|
|
intermediateHashesBucket string
|
|
rd RetainDecider
|
|
accAddrHashWithInc [40]byte // Concatenation of addrHash of the currently build account with its incarnation encoding
|
|
nextAccountKey [32]byte
|
|
k, v []byte
|
|
kHex, vHex []byte
|
|
ihK, ihV, ihSeek []byte
|
|
|
|
// Storage item buffer
|
|
storageKey []byte
|
|
storageValue []byte
|
|
|
|
// Account item buffer
|
|
accountKey []byte
|
|
accountValue accounts.Account
|
|
hashValue []byte
|
|
|
|
receiver StreamReceiver
|
|
defaultReceiver *RootHashAggregator
|
|
hc HashCollector
|
|
}
|
|
|
|
// RootHashAggregator - calculates Merkle trie root hash from incoming data stream
|
|
type RootHashAggregator struct {
|
|
trace bool
|
|
wasIH bool
|
|
wasIHStorage bool
|
|
root common.Hash
|
|
hc HashCollector
|
|
currStorage bytes.Buffer // Current key for the structure generation algorithm, as well as the input tape for the hash builder
|
|
succStorage bytes.Buffer
|
|
valueStorage []byte // Current value to be used as the value tape for the hash builder
|
|
curr bytes.Buffer // Current key for the structure generation algorithm, as well as the input tape for the hash builder
|
|
succ bytes.Buffer
|
|
value []byte // Current value to be used as the value tape for the hash builder
|
|
groups []uint16 // `groups` parameter is the map of the stack. each element of the `groups` slice is a bitmask, one bit per element currently on the stack. See `GenStructStep` docs
|
|
hb *HashBuilder
|
|
hashData GenStructStepHashData
|
|
a accounts.Account
|
|
leafData GenStructStepLeafData
|
|
accData GenStructStepAccountData
|
|
}
|
|
|
|
func NewRootHashAggregator() *RootHashAggregator {
|
|
return &RootHashAggregator{
|
|
hb: NewHashBuilder(false),
|
|
}
|
|
}
|
|
|
|
func NewFlatDBTrieLoader(logPrefix, stateBucket, intermediateHashesBucket string) *FlatDBTrieLoader {
|
|
return &FlatDBTrieLoader{
|
|
logPrefix: logPrefix,
|
|
defaultReceiver: NewRootHashAggregator(),
|
|
stateBucket: stateBucket,
|
|
intermediateHashesBucket: intermediateHashesBucket,
|
|
}
|
|
}
|
|
|
|
// Reset prepares the loader for reuse
|
|
func (l *FlatDBTrieLoader) Reset(rd RetainDecider, hc HashCollector, trace bool) error {
|
|
l.defaultReceiver.Reset(hc, trace)
|
|
l.hc = hc
|
|
l.receiver = l.defaultReceiver
|
|
l.trace = trace
|
|
l.rd = rd
|
|
l.kHex, l.vHex = nil, nil
|
|
l.itemPresent = false
|
|
if l.trace {
|
|
fmt.Printf("----------\n")
|
|
fmt.Printf("CalcTrieRoot\n")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (l *FlatDBTrieLoader) SetStreamReceiver(receiver StreamReceiver) {
|
|
l.receiver = receiver
|
|
}
|
|
|
|
// iteration moves through the database buckets and creates at most
|
|
// one stream item, which is indicated by setting the field fstl.itemPresent to true
|
|
func (l *FlatDBTrieLoader) iteration(c *StateCursor, ih *IHCursor, first bool) error {
|
|
var isIH, isIHSequence bool
|
|
var err error
|
|
if first {
|
|
if l.ihK, l.ihV, isIHSequence, err = ih.Seek([]byte{}); err != nil {
|
|
return err
|
|
}
|
|
if isIHSequence {
|
|
l.kHex = l.ihK
|
|
if len(l.kHex)%2 == 1 {
|
|
l.kHex = append(l.kHex, 0)
|
|
}
|
|
l.k = make([]byte, len(l.kHex)/2)
|
|
CompressNibbles(l.kHex, &l.k)
|
|
return nil
|
|
}
|
|
if l.k, l.kHex, l.v, err = c.Seek([]byte{}); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Skip wrong incarnation
|
|
if len(l.k) > common.HashLength {
|
|
if nextAccount(l.k, l.nextAccountKey[:]) {
|
|
if l.k, l.kHex, l.v, err = c.Seek(l.nextAccountKey[:]); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
l.k = nil
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if l.ihK == nil && l.k == nil { // loop termination
|
|
l.itemPresent = true
|
|
l.itemType = CutoffStreamItem
|
|
l.accountKey = nil
|
|
l.storageKey = nil
|
|
l.storageValue = nil
|
|
l.hashValue = nil
|
|
return nil
|
|
}
|
|
|
|
isIH, _ = keyIsBeforeOrEqual(l.ihK, l.kHex)
|
|
if !isIH {
|
|
// skip wrong incarnation
|
|
if len(l.k) > common.HashLength && !bytes.HasPrefix(l.k, l.accAddrHashWithInc[:]) {
|
|
if bytes.Compare(l.k, l.accAddrHashWithInc[:]) < 0 {
|
|
// Skip all the irrelevant storage in the middle
|
|
if l.k, l.kHex, l.v, err = c.Seek(l.accAddrHashWithInc[:]); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if nextAccount(l.k, l.nextAccountKey[:]) {
|
|
if l.k, l.kHex, l.v, err = c.Seek(l.nextAccountKey[:]); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
l.k = nil
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
l.itemPresent = true
|
|
if len(l.k) > common.HashLength {
|
|
l.itemType = StorageStreamItem
|
|
l.accountKey = nil
|
|
l.storageKey = common.CopyBytes(l.kHex) // no reason to copy, because this "pointer and data" will valid until end of transaction
|
|
l.hashValue = nil
|
|
l.storageValue = l.v
|
|
if l.k, l.kHex, l.v, err = c.Next(); err != nil {
|
|
return err
|
|
}
|
|
if l.trace {
|
|
fmt.Printf("k after storageWalker and Next: %x\n", l.k)
|
|
}
|
|
} else if len(l.k) > 0 {
|
|
l.itemType = AccountStreamItem
|
|
l.accountKey = common.CopyBytes(l.kHex)
|
|
l.storageKey = nil
|
|
l.storageValue = nil
|
|
l.hashValue = nil
|
|
if err = l.accountValue.DecodeForStorage(l.v); err != nil {
|
|
return fmt.Errorf("fail DecodeForStorage: %w", err)
|
|
}
|
|
copy(l.accAddrHashWithInc[:], l.k)
|
|
binary.BigEndian.PutUint64(l.accAddrHashWithInc[32:], l.accountValue.Incarnation)
|
|
|
|
// Now we know the correct incarnation of the account, and we can skip all irrelevant storage records
|
|
// Since 0 incarnation if 0xfff...fff, and we do not expect any records like that, this automatically
|
|
// skips over all storage items
|
|
if l.k, l.kHex, l.v, err = c.Seek(l.accAddrHashWithInc[:]); err != nil {
|
|
return err
|
|
}
|
|
|
|
if l.trace {
|
|
fmt.Printf("k after accountWalker and Seek: %x\n", l.k)
|
|
}
|
|
DecompressNibbles(l.accAddrHashWithInc[:], &l.ihSeek)
|
|
if keyIsBefore(l.ihK, l.ihSeek) {
|
|
if l.ihK, l.ihV, _, err = ih.Seek(l.ihSeek); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ih part
|
|
if l.trace {
|
|
fmt.Printf("l.ihK %x, l.accAddrHashWithInc %x\n", l.ihK, l.accAddrHashWithInc[:])
|
|
}
|
|
|
|
// Skip IH with wrong incarnation
|
|
DecompressNibbles(l.accAddrHashWithInc[:], &l.ihSeek)
|
|
if len(l.ihK) > common.HashLength*2 && !bytes.HasPrefix(l.ihK, l.ihSeek) {
|
|
if bytes.Compare(l.ihK, l.ihSeek) < 0 {
|
|
// Skip all the irrelevant storage in the middle
|
|
if l.ihK, l.ihV, _, err = ih.Seek(l.ihSeek); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if nextAccountHex(l.ihK, l.ihSeek) {
|
|
if l.ihK, l.ihV, _, err = ih.Seek(l.ihSeek); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
l.ihK = nil
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
l.itemPresent = true
|
|
if len(l.ihK) > common.HashLength*2 {
|
|
l.itemType = SHashStreamItem
|
|
l.accountKey = nil
|
|
l.storageKey = l.ihK
|
|
l.hashValue = l.ihV
|
|
l.storageValue = nil
|
|
} else {
|
|
l.itemType = AHashStreamItem
|
|
l.accountKey = l.ihK
|
|
l.storageKey = nil
|
|
l.storageValue = nil
|
|
l.hashValue = l.ihV
|
|
}
|
|
|
|
// go to Next Sub-Tree
|
|
next, ok := dbutils.NextSubtreeHex(l.ihK)
|
|
if !ok { // no siblings left
|
|
l.k, l.kHex, l.ihK, l.ihV = nil, nil, nil, nil
|
|
return nil
|
|
}
|
|
if l.trace {
|
|
fmt.Printf("next: %x\n", next)
|
|
}
|
|
|
|
if l.ihK, l.ihV, isIHSequence, err = ih.Seek(next); err != nil {
|
|
return err
|
|
}
|
|
|
|
if isIHSequence {
|
|
l.kHex = l.ihK
|
|
if len(l.kHex)%2 == 1 {
|
|
l.kHex = append(l.kHex, 0)
|
|
}
|
|
l.k = make([]byte, len(l.kHex)/2)
|
|
CompressNibbles(l.kHex, &l.k)
|
|
return nil
|
|
}
|
|
if len(next)%2 == 1 {
|
|
next = append(next, 0)
|
|
}
|
|
next2 := make([]byte, len(next)/2)
|
|
CompressNibbles(next, &next2)
|
|
if l.k, l.kHex, l.v, err = c.Seek(next2); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Skip wrong incarnation
|
|
if len(next2) <= common.HashLength && len(l.k) > common.HashLength {
|
|
// Advance past the storage to the first account
|
|
if nextAccount(l.k, l.nextAccountKey[:]) {
|
|
if l.k, l.kHex, l.v, err = c.Seek(l.nextAccountKey[:]); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
l.k = nil
|
|
}
|
|
}
|
|
if l.trace {
|
|
fmt.Printf("k after next: %x\n", l.k)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// CalcTrieRoot - spawn 2 cursors (IntermediateHashes and HashedState)
|
|
// Wrap IntermediateHashes cursor to IH class - this class will return only keys which passed RetainDecider check
|
|
// If RetainDecider check not passed, then such key must be deleted - HashCollector receiving nil for such key.
|
|
func (l *FlatDBTrieLoader) CalcTrieRoot(db ethdb.Database, quit <-chan struct{}) (common.Hash, error) {
|
|
var (
|
|
tx ethdb.Tx
|
|
)
|
|
|
|
var txDB ethdb.DbWithPendingMutations
|
|
var useExternalTx bool
|
|
|
|
// If method executed within transaction - use it, or open new read transaction
|
|
if hasTx, ok := db.(ethdb.HasTx); ok && hasTx.Tx() != nil {
|
|
txDB = hasTx.(ethdb.DbWithPendingMutations)
|
|
tx = hasTx.Tx()
|
|
useExternalTx = true
|
|
} else {
|
|
var err error
|
|
txDB, err = db.Begin(context.Background(), ethdb.RW)
|
|
if err != nil {
|
|
return EmptyRoot, err
|
|
}
|
|
|
|
defer txDB.Rollback()
|
|
tx = txDB.(ethdb.HasTx).Tx()
|
|
}
|
|
|
|
c := NewStateCursor(tx.Cursor(l.stateBucket))
|
|
var filter = func(k []byte) bool {
|
|
return !l.rd.Retain(k)
|
|
}
|
|
ih := IH(filter, tx.CursorDupSort(l.intermediateHashesBucket))
|
|
if err := l.iteration(c, ih, true /* first */); err != nil {
|
|
return EmptyRoot, err
|
|
}
|
|
logEvery := time.NewTicker(30 * time.Second)
|
|
defer logEvery.Stop()
|
|
|
|
for l.itemType != CutoffStreamItem {
|
|
if err := common.Stopped(quit); err != nil {
|
|
return EmptyRoot, err
|
|
}
|
|
|
|
for !l.itemPresent {
|
|
if err := l.iteration(c, ih, false /* first */); err != nil {
|
|
return EmptyRoot, err
|
|
}
|
|
}
|
|
|
|
if err := l.receiver.Receive(l.itemType, l.accountKey, l.storageKey, &l.accountValue, l.storageValue, l.hashValue, 0); err != nil {
|
|
return EmptyRoot, err
|
|
}
|
|
l.itemPresent = false
|
|
|
|
select {
|
|
default:
|
|
case <-logEvery.C:
|
|
l.logProgress()
|
|
}
|
|
}
|
|
|
|
if !useExternalTx {
|
|
_, err := txDB.Commit()
|
|
if err != nil {
|
|
return EmptyRoot, err
|
|
}
|
|
}
|
|
|
|
return l.receiver.Root(), nil
|
|
}
|
|
|
|
func (l *FlatDBTrieLoader) logProgress() {
|
|
var k string
|
|
if l.accountKey != nil {
|
|
k = makeCurrentKeyStr(l.accountKey)
|
|
} else if l.storageKey != nil {
|
|
k = makeCurrentKeyStr(l.storageKey)
|
|
} else if l.ihK != nil {
|
|
k = makeCurrentKeyStr(l.ihK)
|
|
}
|
|
log.Info(fmt.Sprintf("[%s] Calculating Merkle root", l.logPrefix), "current key", k)
|
|
}
|
|
|
|
func (r *RootHashAggregator) RetainNothing(_ []byte) bool {
|
|
return false
|
|
}
|
|
|
|
func (r *RootHashAggregator) Reset(hc HashCollector, trace bool) {
|
|
r.hc = hc
|
|
r.curr.Reset()
|
|
r.succ.Reset()
|
|
r.value = nil
|
|
r.groups = r.groups[:0]
|
|
r.a.Reset()
|
|
r.hb.Reset()
|
|
r.wasIH = false
|
|
r.currStorage.Reset()
|
|
r.succStorage.Reset()
|
|
r.valueStorage = nil
|
|
r.wasIHStorage = false
|
|
r.root = common.Hash{}
|
|
r.trace = trace
|
|
r.hb.trace = trace
|
|
}
|
|
|
|
func (r *RootHashAggregator) Receive(itemType StreamItem,
|
|
accountKey []byte,
|
|
storageKey []byte,
|
|
accountValue *accounts.Account,
|
|
storageValue []byte,
|
|
hash []byte,
|
|
cutoff int,
|
|
) error {
|
|
switch itemType {
|
|
case StorageStreamItem:
|
|
r.advanceKeysStorage(storageKey, true /* terminator */)
|
|
if r.currStorage.Len() > 0 {
|
|
if err := r.genStructStorage(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
r.saveValueStorage(false, storageValue, hash)
|
|
case SHashStreamItem:
|
|
r.advanceKeysStorage(storageKey, false /* terminator */)
|
|
if r.currStorage.Len() > 0 {
|
|
if err := r.genStructStorage(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
r.saveValueStorage(true, storageValue, hash)
|
|
case AccountStreamItem:
|
|
r.advanceKeysAccount(accountKey, true /* terminator */)
|
|
if r.curr.Len() > 0 && !r.wasIH {
|
|
r.cutoffKeysStorage(2 * (common.HashLength + common.IncarnationLength))
|
|
if r.currStorage.Len() > 0 {
|
|
if err := r.genStructStorage(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if r.currStorage.Len() > 0 {
|
|
if len(r.groups) >= 2*common.HashLength {
|
|
r.groups = r.groups[:2*common.HashLength-1]
|
|
}
|
|
for len(r.groups) > 0 && r.groups[len(r.groups)-1] == 0 {
|
|
r.groups = r.groups[:len(r.groups)-1]
|
|
}
|
|
r.currStorage.Reset()
|
|
r.succStorage.Reset()
|
|
r.wasIHStorage = false
|
|
// There are some storage items
|
|
r.accData.FieldSet |= AccountFieldStorageOnly
|
|
}
|
|
}
|
|
if r.curr.Len() > 0 {
|
|
if err := r.genStructAccount(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := r.saveValueAccount(false, accountValue, hash); err != nil {
|
|
return err
|
|
}
|
|
case AHashStreamItem:
|
|
r.advanceKeysAccount(accountKey, false /* terminator */)
|
|
if r.curr.Len() > 0 && !r.wasIH {
|
|
r.cutoffKeysStorage(2 * (common.HashLength + common.IncarnationLength))
|
|
if r.currStorage.Len() > 0 {
|
|
if err := r.genStructStorage(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if r.currStorage.Len() > 0 {
|
|
if len(r.groups) >= 2*common.HashLength {
|
|
r.groups = r.groups[:2*common.HashLength-1]
|
|
}
|
|
for len(r.groups) > 0 && r.groups[len(r.groups)-1] == 0 {
|
|
r.groups = r.groups[:len(r.groups)-1]
|
|
}
|
|
r.currStorage.Reset()
|
|
r.succStorage.Reset()
|
|
r.wasIHStorage = false
|
|
// There are some storage items
|
|
r.accData.FieldSet |= AccountFieldStorageOnly
|
|
}
|
|
}
|
|
if r.curr.Len() > 0 {
|
|
if err := r.genStructAccount(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := r.saveValueAccount(true, accountValue, hash); err != nil {
|
|
return err
|
|
}
|
|
case CutoffStreamItem:
|
|
if r.trace {
|
|
fmt.Printf("storage cuttoff %d\n", cutoff)
|
|
}
|
|
|
|
r.cutoffKeysAccount(cutoff)
|
|
if r.curr.Len() > 0 && !r.wasIH {
|
|
r.cutoffKeysStorage(2 * (common.HashLength + common.IncarnationLength))
|
|
if r.currStorage.Len() > 0 {
|
|
if err := r.genStructStorage(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if r.currStorage.Len() > 0 {
|
|
if len(r.groups) >= 2*common.HashLength {
|
|
r.groups = r.groups[:2*common.HashLength-1]
|
|
}
|
|
for len(r.groups) > 0 && r.groups[len(r.groups)-1] == 0 {
|
|
r.groups = r.groups[:len(r.groups)-1]
|
|
}
|
|
r.currStorage.Reset()
|
|
r.succStorage.Reset()
|
|
r.wasIHStorage = false
|
|
// There are some storage items
|
|
r.accData.FieldSet |= AccountFieldStorageOnly
|
|
}
|
|
}
|
|
if r.curr.Len() > 0 {
|
|
if err := r.genStructAccount(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if r.curr.Len() > 0 {
|
|
if len(r.groups) > cutoff {
|
|
r.groups = r.groups[:cutoff]
|
|
}
|
|
for len(r.groups) > 0 && r.groups[len(r.groups)-1] == 0 {
|
|
r.groups = r.groups[:len(r.groups)-1]
|
|
}
|
|
}
|
|
if r.hb.hasRoot() {
|
|
r.root = r.hb.rootHash()
|
|
} else {
|
|
r.root = EmptyRoot
|
|
}
|
|
r.groups = r.groups[:0]
|
|
r.hb.Reset()
|
|
r.wasIH = false
|
|
r.wasIHStorage = false
|
|
r.curr.Reset()
|
|
r.succ.Reset()
|
|
r.currStorage.Reset()
|
|
r.succStorage.Reset()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *RootHashAggregator) Result() SubTries {
|
|
panic("don't call me")
|
|
}
|
|
|
|
func (r *RootHashAggregator) Root() common.Hash {
|
|
return r.root
|
|
}
|
|
|
|
func (r *RootHashAggregator) advanceKeysStorage(k []byte, terminator bool) {
|
|
r.currStorage.Reset()
|
|
r.currStorage.Write(r.succStorage.Bytes())
|
|
r.succStorage.Reset()
|
|
// Transform k to nibbles, but skip the incarnation part in the middle
|
|
r.succStorage.Write(k)
|
|
|
|
if terminator {
|
|
r.succStorage.WriteByte(16)
|
|
}
|
|
}
|
|
|
|
func (r *RootHashAggregator) cutoffKeysStorage(cutoff int) {
|
|
r.currStorage.Reset()
|
|
r.currStorage.Write(r.succStorage.Bytes())
|
|
r.succStorage.Reset()
|
|
if r.currStorage.Len() > 0 {
|
|
r.succStorage.Write(r.currStorage.Bytes()[:cutoff-1])
|
|
r.succStorage.WriteByte(r.currStorage.Bytes()[cutoff-1] + 1) // Modify last nibble in the incarnation part of the `currStorage`
|
|
}
|
|
}
|
|
|
|
func (r *RootHashAggregator) genStructStorage() error {
|
|
var err error
|
|
var data GenStructStepData
|
|
if r.wasIHStorage {
|
|
r.hashData.Hash = common.BytesToHash(r.valueStorage)
|
|
data = &r.hashData
|
|
} else {
|
|
r.leafData.Value = rlphacks.RlpSerializableBytes(r.valueStorage)
|
|
data = &r.leafData
|
|
}
|
|
r.groups, err = GenStructStep(r.RetainNothing, r.currStorage.Bytes(), r.succStorage.Bytes(), r.hb, r.hc, data, r.groups, r.trace)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *RootHashAggregator) saveValueStorage(isIH bool, v, h []byte) {
|
|
// Remember the current value
|
|
r.wasIHStorage = isIH
|
|
r.valueStorage = nil
|
|
if isIH {
|
|
r.valueStorage = h
|
|
} else {
|
|
r.valueStorage = v
|
|
}
|
|
}
|
|
|
|
func (r *RootHashAggregator) advanceKeysAccount(k []byte, terminator bool) {
|
|
r.curr.Reset()
|
|
r.curr.Write(r.succ.Bytes())
|
|
r.succ.Reset()
|
|
r.succ.Write(k)
|
|
if terminator {
|
|
r.succ.WriteByte(16)
|
|
}
|
|
}
|
|
|
|
func (r *RootHashAggregator) cutoffKeysAccount(cutoff int) {
|
|
r.curr.Reset()
|
|
r.curr.Write(r.succ.Bytes())
|
|
r.succ.Reset()
|
|
if r.curr.Len() > 0 && cutoff > 0 {
|
|
r.succ.Write(r.curr.Bytes()[:cutoff-1])
|
|
r.succ.WriteByte(r.curr.Bytes()[cutoff-1] + 1) // Modify last nibble before the cutoff point
|
|
}
|
|
}
|
|
|
|
func (r *RootHashAggregator) genStructAccount() error {
|
|
var data GenStructStepData
|
|
if r.wasIH {
|
|
copy(r.hashData.Hash[:], r.value)
|
|
data = &r.hashData
|
|
} else {
|
|
r.accData.Balance.Set(&r.a.Balance)
|
|
if r.a.Balance.Sign() != 0 {
|
|
r.accData.FieldSet |= AccountFieldBalanceOnly
|
|
}
|
|
r.accData.Nonce = r.a.Nonce
|
|
if r.a.Nonce != 0 {
|
|
r.accData.FieldSet |= AccountFieldNonceOnly
|
|
}
|
|
r.accData.Incarnation = r.a.Incarnation
|
|
data = &r.accData
|
|
}
|
|
r.wasIHStorage = false
|
|
r.currStorage.Reset()
|
|
r.succStorage.Reset()
|
|
var err error
|
|
if r.groups, err = GenStructStep(r.RetainNothing, r.curr.Bytes(), r.succ.Bytes(), r.hb, r.hc, data, r.groups, r.trace); err != nil {
|
|
return err
|
|
}
|
|
r.accData.FieldSet = 0
|
|
return nil
|
|
}
|
|
|
|
func (r *RootHashAggregator) saveValueAccount(isIH bool, v *accounts.Account, h []byte) error {
|
|
r.wasIH = isIH
|
|
if isIH {
|
|
r.value = h
|
|
return nil
|
|
}
|
|
r.a.Copy(v)
|
|
// Place code on the stack first, the storage will follow
|
|
if !r.a.IsEmptyCodeHash() {
|
|
// the first item ends up deepest on the stack, the second item - on the top
|
|
r.accData.FieldSet |= AccountFieldCodeOnly
|
|
if err := r.hb.hash(r.a.CodeHash[:]); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type Filter func([]byte) bool // returns false - if element must be skipped
|
|
|
|
const IHDupKeyLen = 2 * (common.HashLength + common.IncarnationLength)
|
|
|
|
// IHCursor - holds logic related to iteration over IH bucket
|
|
type IHCursor struct {
|
|
c ethdb.CursorDupSort
|
|
filter Filter
|
|
}
|
|
|
|
func IH(f Filter, c ethdb.CursorDupSort) *IHCursor {
|
|
return &IHCursor{c: c, filter: f}
|
|
}
|
|
|
|
func (c *IHCursor) _seek(seek []byte) (k, v []byte, err error) {
|
|
if len(seek) > IHDupKeyLen {
|
|
k, v, err = c.c.SeekBothRange(seek[:IHDupKeyLen], seek[IHDupKeyLen:])
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
if k == nil {
|
|
k, v, err = c.c.Next()
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
}
|
|
} else {
|
|
k, v, err = c.c.Seek(seek)
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
}
|
|
if k == nil {
|
|
return nil, nil, nil
|
|
}
|
|
|
|
if len(v) > common.HashLength {
|
|
keyPart := len(v) - common.HashLength
|
|
k = append(k, v[:keyPart]...)
|
|
v = v[keyPart:]
|
|
}
|
|
|
|
if c.filter(k) { // if filter allow us, return. otherwise delete and go ahead.
|
|
return k, v, nil
|
|
}
|
|
|
|
err = c.c.DeleteCurrent()
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
return c._next()
|
|
}
|
|
|
|
func (c *IHCursor) _next() (k, v []byte, err error) {
|
|
k, v, err = c.c.Next()
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
for {
|
|
if k == nil {
|
|
return nil, nil, nil
|
|
}
|
|
|
|
if len(v) > common.HashLength {
|
|
keyPart := len(v) - common.HashLength
|
|
k = append(k, v[:keyPart]...)
|
|
v = v[keyPart:]
|
|
}
|
|
|
|
if c.filter(k) { // if filter allow us, return. otherwise delete and go ahead.
|
|
return k, v, nil
|
|
}
|
|
|
|
err = c.c.DeleteCurrent()
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
|
|
k, v, err = c.c.Next()
|
|
if err != nil {
|
|
return []byte{}, nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *IHCursor) Seek(seek []byte) ([]byte, []byte, bool, error) {
|
|
k, v, err := c._seek(seek)
|
|
if err != nil {
|
|
return []byte{}, nil, false, err
|
|
}
|
|
|
|
if k == nil {
|
|
return nil, nil, false, nil
|
|
}
|
|
|
|
return common.CopyBytes(k), common.CopyBytes(v), isSequence(seek, k), nil
|
|
}
|
|
|
|
/*
|
|
Sequence - if between 2 IH records not possible insert any state record - then they form "sequence"
|
|
Example1:
|
|
1234
|
|
1235
|
|
Example2:
|
|
12ff
|
|
13
|
|
Example3:
|
|
12ff
|
|
13000000
|
|
If 2 IH records form "sequence" then it can be consumed without moving StateCursor
|
|
*/
|
|
func isSequence(prev []byte, next []byte) bool {
|
|
isSequence := false
|
|
if bytes.HasPrefix(next, prev) {
|
|
tail := next[len(prev):] // if tail has only zeroes, then no state records can be between fstl.nextHex and fstl.ihK
|
|
isSequence = true
|
|
for _, n := range tail {
|
|
if n != 0 {
|
|
isSequence = false
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
return isSequence
|
|
}
|
|
|
|
type StateCursor struct {
|
|
c ethdb.Cursor
|
|
kHex []byte
|
|
}
|
|
|
|
func NewStateCursor(c ethdb.Cursor) *StateCursor {
|
|
return &StateCursor{c: c}
|
|
}
|
|
|
|
func (c *StateCursor) Seek(seek []byte) ([]byte, []byte, []byte, error) {
|
|
k, v, err := c.c.Seek(seek)
|
|
if err != nil {
|
|
return []byte{}, nil, nil, err
|
|
}
|
|
|
|
DecompressNibbles(k, &c.kHex)
|
|
return k, c.kHex, v, nil
|
|
}
|
|
|
|
func (c *StateCursor) Next() ([]byte, []byte, []byte, error) {
|
|
k, v, err := c.c.Next()
|
|
if err != nil {
|
|
return []byte{}, nil, nil, err
|
|
}
|
|
|
|
DecompressNibbles(k, &c.kHex)
|
|
return k, c.kHex, v, nil
|
|
}
|
|
|
|
func nextAccount(in, out []byte) bool {
|
|
copy(out, in)
|
|
for i := len(out) - 1; i >= 0; i-- {
|
|
if out[i] != 255 {
|
|
out[i]++
|
|
return true
|
|
}
|
|
out[i] = 0
|
|
}
|
|
return false
|
|
}
|
|
|
|
func nextAccountHex(in, out []byte) bool {
|
|
copy(out, in)
|
|
for i := len(out) - 1; i >= 0; i-- {
|
|
if out[i] != 15 {
|
|
out[i]++
|
|
return true
|
|
}
|
|
out[i] = 0
|
|
}
|
|
return false
|
|
}
|
|
|
|
// keyIsBefore - kind of bytes.Compare, but nil is the last key. And return
|
|
func keyIsBeforeOrEqual(k1, k2 []byte) (bool, []byte) {
|
|
if k1 == nil {
|
|
return false, k2
|
|
}
|
|
|
|
if k2 == nil {
|
|
return true, k1
|
|
}
|
|
|
|
switch bytes.Compare(k1, k2) {
|
|
case -1, 0:
|
|
return true, k1
|
|
default:
|
|
return false, k2
|
|
}
|
|
}
|
|
|
|
// keyIsBefore - kind of bytes.Compare, but nil is the last key. And return
|
|
func keyIsBefore(k1, k2 []byte) bool {
|
|
if k1 == nil {
|
|
return false
|
|
}
|
|
|
|
if k2 == nil {
|
|
return true
|
|
}
|
|
|
|
switch bytes.Compare(k1, k2) {
|
|
case -1:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|