erigon-pulse/trie/stream.go
Andrew Ashikhmin 05a3483cb1
Fix hash map leaks (#367)
* Enable testGetNodeData by default

* Crude TestHashMapLeak

* nodeFlag ->nodeRef

* linter

* accountNode shouln't be in hashMap since only branch and short nodes are the standart ones

* Finalize hash eviction logic

* small changes to TestHashMapLeak

* Fix for the incorrect hash

Co-authored-by: ledgerwatch <akhounov@gmail.com>
2020-02-12 11:34:44 +00:00

818 lines
24 KiB
Go

// Copyright 2019 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty off
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
package trie
import (
"bytes"
"fmt"
"os"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
"github.com/ledgerwatch/turbo-geth/trie/rlphacks"
)
// StreamItem is an enum type for values that help distinguish different
// types of key-value pairs in the stream of values produce by function `ToStream`
type StreamItem uint8
const (
// NoItem is used to signal the end of iterator
NoItem StreamItem = iota
// AccountStreamItem used for marking a key-value pair in the stream as belonging to an account
AccountStreamItem
// StorageStreamItem used for marking a key-value pair in the stream as belonging to a storage item leaf
StorageStreamItem
// AHashStreamItem used for marking a key-value pair in the stream as belonging to an intermediate hash
// within the accounts (main state trie)
AHashStreamItem
// SHashStreamItem used for marking a key-value pair in the stream as belonging to an intermediate hash
// within the storage items (storage tries)
SHashStreamItem
)
// Stream represents the collection of key-value pairs, sorted by keys, where values may belong
// to three different types - accounts, strage item leaves, and intermediate hashes
type Stream struct {
keyBytes []byte
keySizes []uint8
itemTypes []StreamItem
aValues []*accounts.Account
sValues [][]byte
hashes []byte
}
// Reset sets all slices to zero sizes, without de-allocating
func (s *Stream) Reset() {
if len(s.keyBytes) > 0 {
s.keyBytes = s.keyBytes[:0]
}
if len(s.keySizes) > 0 {
s.keySizes = s.keySizes[:0]
}
if len(s.itemTypes) > 0 {
s.itemTypes = s.itemTypes[:0]
}
if len(s.aValues) > 0 {
s.aValues = s.aValues[:0]
}
if len(s.sValues) > 0 {
s.sValues = s.sValues[:0]
}
if len(s.hashes) > 0 {
s.hashes = s.hashes[:0]
}
}
// Iterator helps iterate over a trie according to a given resolve set
type Iterator struct {
rs *ResolveSet
hex []byte
nodeStack []node
iStack []int
goDeepStack []bool
lenStack []int
accountStack []bool
top int // Top of the stack
trace bool
}
// NewIterator creates a new iterator from scratch from a given trie and resolve set
func NewIterator(t *Trie, rs *ResolveSet, trace bool) *Iterator {
return &Iterator{
rs: rs,
hex: []byte{},
nodeStack: []node{t.root},
iStack: []int{0},
goDeepStack: []bool{true},
lenStack: []int{0},
accountStack: []bool{true},
top: 1,
trace: trace,
}
}
// Reset prepares iterator to be reused
func (it *Iterator) Reset(t *Trie, rs *ResolveSet, trace bool) {
it.rs = rs
it.hex = it.hex[:0]
if len(it.nodeStack) > 0 {
it.nodeStack = it.nodeStack[:0]
}
it.nodeStack = append(it.nodeStack, t.root)
if len(it.iStack) > 0 {
it.iStack = it.iStack[:0]
}
it.lenStack = append(it.lenStack, 0)
if len(it.goDeepStack) > 0 {
it.goDeepStack = it.goDeepStack[:0]
}
it.goDeepStack = append(it.goDeepStack, true)
if len(it.accountStack) > 0 {
it.accountStack = it.accountStack[:0]
}
it.accountStack = append(it.accountStack, true)
it.top = 1
it.trace = trace
}
// Next delivers the next item from the iterator
func (it *Iterator) Next() (itemType StreamItem, hex1 []byte, aValue *accounts.Account, hash []byte, value []byte) {
for {
if it.top == 0 {
return NoItem, nil, nil, nil, nil
}
l := it.top - 1
nd := it.nodeStack[l]
hexLen := it.lenStack[l]
hex := it.hex[:hexLen]
goDeep := it.goDeepStack[l]
accounts := it.accountStack[l]
switch n := nd.(type) {
case nil:
return NoItem, nil, nil, nil, nil
case valueNode:
if it.trace {
fmt.Printf("valueNode %x\n", hex)
}
it.top--
return StorageStreamItem, hex, nil, nil, []byte(n)
case *shortNode:
if it.trace {
fmt.Printf("shortNode %x\n", hex)
}
hex = append(hex, n.Key...)
switch v := n.Val.(type) {
case hashNode:
it.top--
if accounts {
return AHashStreamItem, hex, nil, []byte(v), nil
}
return SHashStreamItem, hex, nil, []byte(v), nil
case valueNode:
it.top--
return StorageStreamItem, hex, nil, nil, []byte(v)
case *accountNode:
if !it.rs.HashOnly(hex) && v.storage != nil {
it.hex = hex
it.nodeStack[l] = v.storage
it.iStack[l] = 0
it.goDeepStack[l] = true
it.lenStack[l] = len(hex)
it.accountStack[l] = false
} else {
it.top--
}
return AccountStreamItem, hex, &v.Account, nil, nil
}
it.hex = hex
it.nodeStack[l] = n.Val
it.iStack[l] = 0
it.goDeepStack[l] = !it.rs.HashOnly(hex)
it.lenStack[l] = len(hex)
it.accountStack[l] = accounts
case *duoNode:
if it.trace {
fmt.Printf("duoNode %x\n", hex)
}
if !goDeep {
it.top--
if accounts {
return AHashStreamItem, hex, nil, n.reference(), nil
}
return SHashStreamItem, hex, nil, n.reference(), nil
}
i1, i2 := n.childrenIdx()
index := it.iStack[l]
if index <= int(i1) {
it.iStack[l] = int(i2)
hex = append(hex, i1)
var childGoDeep bool
switch v := n.child1.(type) {
case hashNode:
if accounts {
return AHashStreamItem, hex, nil, []byte(v), nil
}
return SHashStreamItem, hex, nil, []byte(v), nil
case *duoNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
case *fullNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
}
it.hex = hex
l++
it.top++
if l >= len(it.nodeStack) {
it.nodeStack = append(it.nodeStack, n.child1)
it.iStack = append(it.iStack, 0)
it.goDeepStack = append(it.goDeepStack, childGoDeep)
it.lenStack = append(it.lenStack, len(hex))
it.accountStack = append(it.accountStack, accounts)
} else {
it.nodeStack[l] = n.child1
it.iStack[l] = 0
it.goDeepStack[l] = childGoDeep
it.lenStack[l] = len(hex)
it.accountStack[l] = accounts
}
} else {
hex = append(hex, i2)
var childGoDeep bool
switch v := n.child2.(type) {
case hashNode:
it.top--
if accounts {
return AHashStreamItem, hex, nil, []byte(v), nil
}
return SHashStreamItem, hex, nil, []byte(v), nil
case *duoNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
it.top--
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
case *fullNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
it.top--
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
}
it.hex = hex
it.nodeStack[l] = n.child2
it.iStack[l] = 0
it.goDeepStack[l] = childGoDeep
it.lenStack[l] = len(hex)
it.accountStack[l] = accounts
}
case *fullNode:
if it.trace {
fmt.Printf("fullNode %x\n", hex)
}
if !goDeep {
it.top--
if accounts {
return AHashStreamItem, hex, nil, n.reference(), nil
}
return SHashStreamItem, hex, nil, n.reference(), nil
}
var i1, i2 int
i1Found := false
i2Found := false
index := it.iStack[l]
for i := index; i < len(n.Children); i++ {
if n.Children[i] != nil {
if i1Found {
i2 = i
i2Found = true
break
} else {
i1 = i
i1Found = true
}
}
}
if i2Found {
it.iStack[l] = i2
hex = append(hex, byte(i1))
var childGoDeep bool
switch v := n.Children[i1].(type) {
case hashNode:
if accounts {
return AHashStreamItem, hex, nil, []byte(v), nil
}
return SHashStreamItem, hex, nil, []byte(v), nil
case *duoNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
case *fullNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
}
it.hex = hex
l++
it.top++
if l >= len(it.nodeStack) {
it.nodeStack = append(it.nodeStack, n.Children[i1])
it.iStack = append(it.iStack, 0)
it.goDeepStack = append(it.goDeepStack, childGoDeep)
it.lenStack = append(it.lenStack, len(hex))
it.accountStack = append(it.accountStack, accounts)
} else {
it.nodeStack[l] = n.Children[i1]
it.iStack[l] = 0
it.goDeepStack[l] = childGoDeep
it.lenStack[l] = len(hex)
it.accountStack[l] = accounts
}
} else {
hex = append(hex, byte(i1))
var childGoDeep bool
switch v := n.Children[i1].(type) {
case hashNode:
it.top--
if accounts {
return AHashStreamItem, hex, nil, []byte(v), nil
}
return SHashStreamItem, hex, nil, []byte(v), nil
case *duoNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
it.top--
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
case *fullNode:
childGoDeep = !it.rs.HashOnly(hex)
if !childGoDeep {
it.top--
if accounts {
return AHashStreamItem, hex, nil, v.reference(), nil
}
return SHashStreamItem, hex, nil, v.reference(), nil
}
}
it.hex = hex
it.nodeStack[l] = n.Children[i1]
it.iStack[l] = 0
it.goDeepStack[l] = childGoDeep
it.lenStack[l] = len(hex)
it.accountStack[l] = accounts
}
case *accountNode:
if it.trace {
fmt.Printf("accountNode %x\n", hex)
}
if !it.rs.HashOnly(hex) && n.storage != nil {
it.hex = hex
it.nodeStack[l] = n.storage
it.iStack[l] = 0
it.goDeepStack[l] = true
it.lenStack[l] = len(hex)
it.accountStack[l] = false
} else {
it.top--
}
return AccountStreamItem, hex, &n.Account, nil, nil
case hashNode:
if it.trace {
fmt.Printf("hashNode %x\n", hex)
}
it.top--
if accounts {
return AHashStreamItem, hex, nil, []byte(n), nil
}
return SHashStreamItem, hex, nil, []byte(n), nil
default:
panic(fmt.Errorf("unexpected node: %T", nd))
}
}
}
// StreamMergeIterator merges an Iterator and a Stream
type StreamMergeIterator struct {
it *Iterator
s *Stream
trace bool
ki, ai, si int
offset int
hex []byte
itemType StreamItem
oldItemType StreamItem
oldHex []byte
oldHexCopy []byte
oldAVal *accounts.Account
oldHash []byte
oldHashCopy []byte
oldVal []byte
}
// NewStreamMergeIterator create a brand new StreamMergeIterator
func NewStreamMergeIterator(it *Iterator, s *Stream, trace bool) *StreamMergeIterator {
smi := &StreamMergeIterator{
it: it,
s: s,
trace: trace,
}
smi.oldItemType, smi.oldHex, smi.oldAVal, smi.oldHash, smi.oldVal = it.Next()
return smi
}
// Reset prepares StreamMergeIterator for reuse
func (smi *StreamMergeIterator) Reset(it *Iterator, s *Stream, trace bool) {
smi.it = it
smi.s = s
smi.trace = trace
smi.ki = 0
smi.ai = 0
smi.si = 0
smi.offset = 0
smi.oldItemType, smi.oldHex, smi.oldAVal, smi.oldHash, smi.oldVal = it.Next()
}
// Next returns the next item in the merged iterator
func (smi *StreamMergeIterator) Next() (itemType1 StreamItem, hex1 []byte, aValue *accounts.Account, hash []byte, value []byte) {
for {
if smi.trace {
fmt.Printf("smi.hex %x, smi.ki %d, len(smi.s.keySizes) %d, smi.oldItemType %d smi.oldHex %x\n", smi.hex, smi.ki, len(smi.s.keySizes), smi.oldItemType, smi.oldHex)
}
if smi.hex == nil && smi.ki >= len(smi.s.keySizes) && smi.oldItemType == NoItem {
return NoItem, nil, nil, nil, nil
}
if smi.hex == nil && smi.ki < len(smi.s.keySizes) {
size := int(smi.s.keySizes[smi.ki])
smi.hex = smi.s.keyBytes[smi.offset : smi.offset+size]
smi.itemType = smi.s.itemTypes[smi.ki]
smi.ki++
smi.offset += size
}
hex := smi.hex // Save it to be able to use it even after assigning to nil
if smi.oldHex == nil {
smi.hex = nil // To be consumed
switch smi.itemType {
case AccountStreamItem:
ai := smi.ai
smi.ai++
if smi.s.aValues[ai] != nil {
return AccountStreamItem, hex, smi.s.aValues[ai], nil, nil
}
case StorageStreamItem:
si := smi.si
smi.si++
if len(smi.s.sValues[si]) > 0 {
return StorageStreamItem, hex, nil, nil, smi.s.sValues[si]
}
default:
panic(fmt.Errorf("unexpected stream item type (oldHex == nil): %d", smi.itemType))
}
} else if hex == nil {
if len(smi.oldHexCopy) > 0 {
smi.oldHexCopy = smi.oldHexCopy[:0]
}
smi.oldHexCopy = append(smi.oldHexCopy, smi.oldHex...)
oldItemType := smi.oldItemType
oldAVal := smi.oldAVal
if len(smi.oldHashCopy) > 0 {
smi.oldHashCopy = smi.oldHashCopy[:0]
}
smi.oldHashCopy = append(smi.oldHashCopy, smi.oldHash...)
oldVal := smi.oldVal
smi.oldItemType, smi.oldHex, smi.oldAVal, smi.oldHash, smi.oldVal = smi.it.Next()
return oldItemType, smi.oldHexCopy, oldAVal, smi.oldHashCopy, oldVal
} else {
// Special case - account gets deleted
if smi.itemType == AccountStreamItem && smi.s.aValues[smi.ai] == nil && bytes.HasPrefix(smi.oldHex, hex) {
smi.oldItemType, smi.oldHex, smi.oldAVal, smi.oldHash, smi.oldVal = smi.it.Next()
} else {
switch bytes.Compare(smi.oldHex, hex) {
case -1:
if len(smi.oldHexCopy) > 0 {
smi.oldHexCopy = smi.oldHexCopy[:0]
}
smi.oldHexCopy = append(smi.oldHexCopy, smi.oldHex...)
oldItemType := smi.oldItemType
oldAVal := smi.oldAVal
if len(smi.oldHashCopy) > 0 {
smi.oldHashCopy = smi.oldHashCopy[:0]
}
smi.oldHashCopy = append(smi.oldHashCopy, smi.oldHash...)
oldVal := smi.oldVal
smi.oldItemType, smi.oldHex, smi.oldAVal, smi.oldHash, smi.oldVal = smi.it.Next()
return oldItemType, smi.oldHexCopy, oldAVal, smi.oldHashCopy, oldVal
case 1:
smi.hex = nil // To be consumed
switch smi.itemType {
case AccountStreamItem:
ai := smi.ai
smi.ai++
if smi.s.aValues[ai] != nil {
return AccountStreamItem, hex, smi.s.aValues[ai], nil, nil
}
case StorageStreamItem:
si := smi.si
smi.si++
if len(smi.s.sValues[si]) > 0 {
return StorageStreamItem, hex, nil, nil, smi.s.sValues[si]
}
default:
panic(fmt.Errorf("unexpected stream item type (oldHex == nil): %d", smi.itemType))
}
case 0:
smi.hex = nil // To be consumed
smi.oldItemType, smi.oldHex, smi.oldAVal, smi.oldHash, smi.oldVal = smi.it.Next()
switch smi.itemType {
case AccountStreamItem:
ai := smi.ai
smi.ai++
if smi.s.aValues[ai] != nil {
return AccountStreamItem, hex, smi.s.aValues[ai], nil, nil
}
case StorageStreamItem:
si := smi.si
smi.si++
if len(smi.s.sValues[si]) > 0 {
return StorageStreamItem, hex, nil, nil, smi.s.sValues[si]
}
default:
panic(fmt.Errorf("unexpected stream item type (oldHex == nil): %d", smi.itemType))
}
}
}
}
}
}
// StreamHash computes the hash of a stream, as if it was a trie
func StreamHash(it *StreamMergeIterator, storagePrefixLen int, hb *HashBuilder, trace bool) (common.Hash, error) {
var succ bytes.Buffer
var curr bytes.Buffer
var succStorage bytes.Buffer
var currStorage bytes.Buffer
var value bytes.Buffer
var hashBuf common.Hash
var hashBufStorage common.Hash
var hashRef []byte
var hashRefStorage []byte
var groups, sGroups []uint16 // Separate groups slices for storage items and for accounts
var aRoot common.Hash
var aEmptyRoot = true
var fieldSet uint32
var itemType, sItemType StreamItem
var hashData GenStructStepHashData
var leafData GenStructStepLeafData
var accData GenStructStepAccountData
hb.Reset()
curr.Reset()
currStorage.Reset()
makeData := func(fieldSet uint32, hashRef []byte) GenStructStepData {
if hashRef != nil {
copy(hashData.Hash[:], hashRef)
return &hashData
} else if fieldSet == AccountFieldSetNotAccount {
leafData.Value = rlphacks.RlpSerializableBytes(value.Bytes())
return &leafData
} else {
accData.FieldSet = fieldSet
return &accData
}
}
hashOnly := func(_ []byte) bool { return !trace }
for newItemType, hex, aVal, hash, val := it.Next(); newItemType != NoItem; newItemType, hex, aVal, hash, val = it.Next() {
if newItemType == AccountStreamItem || newItemType == AHashStreamItem {
// If there was an open storage "sub-stream", close it and set the storage flag on
if succStorage.Len() > 0 {
currStorage.Reset()
currStorage.Write(succStorage.Bytes())
succStorage.Reset()
if currStorage.Len() > 0 {
var err error
sGroups, err = GenStructStep(hashOnly, currStorage.Bytes(), succStorage.Bytes(), hb, makeData(AccountFieldSetNotAccount, hashRefStorage), sGroups, trace)
if err != nil {
return common.Hash{}, err
}
currStorage.Reset()
fieldSet += AccountFieldRootOnly
}
} else if itemType == AccountStreamItem && !aEmptyRoot {
if err := hb.hash(aRoot[:]); err != nil {
return common.Hash{}, err
}
fieldSet += AccountFieldRootOnly
}
curr.Reset()
curr.Write(succ.Bytes())
succ.Reset()
succ.Write(hex)
if curr.Len() > 0 {
var err error
groups, err = GenStructStep(hashOnly, curr.Bytes(), succ.Bytes(), hb, makeData(fieldSet, hashRef), groups, trace)
if err != nil {
return common.Hash{}, err
}
}
itemType = newItemType
switch itemType {
case AccountStreamItem:
var a *accounts.Account = aVal
accData.StorageSize = a.StorageSize
accData.Balance.Set(&a.Balance)
accData.Nonce = a.Nonce
accData.Incarnation = a.Incarnation
aEmptyRoot = a.IsEmptyRoot()
copy(aRoot[:], a.Root[:])
fieldSet = AccountFieldSetNotContract // base level - nonce and balance
if a.HasStorageSize {
fieldSet += AccountFieldSSizeOnly
}
if !a.IsEmptyCodeHash() {
fieldSet += AccountFieldCodeHashOnly
if err := hb.hash(a.CodeHash[:]); err != nil {
return common.Hash{}, err
}
}
hashRef = nil
case AHashStreamItem:
copy(hashBuf[:], hash)
hashRef = hashBuf[:]
}
} else {
currStorage.Reset()
currStorage.Write(succStorage.Bytes())
succStorage.Reset()
succStorage.Write(hex[2*storagePrefixLen+1:])
if currStorage.Len() > 0 {
var err error
sGroups, err = GenStructStep(hashOnly, currStorage.Bytes(), succStorage.Bytes(), hb, makeData(AccountFieldSetNotAccount, hashRefStorage), sGroups, trace)
if err != nil {
return common.Hash{}, err
}
}
sItemType = newItemType
switch sItemType {
case StorageStreamItem:
value.Reset()
value.Write(val)
hashRefStorage = nil
case SHashStreamItem:
copy(hashBufStorage[:], hash)
hashRefStorage = hashBufStorage[:]
}
}
}
// If there was an open storage "sub-stream", close it and set the storage flag on
if succStorage.Len() > 0 {
currStorage.Reset()
currStorage.Write(succStorage.Bytes())
succStorage.Reset()
if currStorage.Len() > 0 {
var err error
_, err = GenStructStep(hashOnly, currStorage.Bytes(), succStorage.Bytes(), hb, makeData(AccountFieldSetNotAccount, hashRefStorage), sGroups, trace)
if err != nil {
return common.Hash{}, err
}
currStorage.Reset()
fieldSet += AccountFieldRootOnly
}
} else if itemType == AccountStreamItem && !aEmptyRoot {
if err := hb.hash(aRoot[:]); err != nil {
return common.Hash{}, err
}
fieldSet += AccountFieldRootOnly
}
curr.Reset()
curr.Write(succ.Bytes())
succ.Reset()
if curr.Len() > 0 {
var err error
_, err = GenStructStep(hashOnly, curr.Bytes(), succ.Bytes(), hb, makeData(fieldSet, hashRef), groups, trace)
if err != nil {
return common.Hash{}, err
}
}
if trace {
filename := "root.txt"
f, err1 := os.Create(filename)
if err1 == nil {
defer f.Close()
hb.root().print(f)
}
}
if hb.hasRoot() {
return hb.rootHash(), nil
}
return EmptyRoot, nil
}
// HashWithModifications computes the hash of the would-be modified trie, but without any modifications
func HashWithModifications(
t *Trie,
aKeys common.Hashes, aValues []*accounts.Account,
sKeys common.StorageKeys, sValues [][]byte,
storagePrefixLen int,
newStream *Stream, // Streams that will be reused for old and new stream
hb *HashBuilder, // HashBuilder will be reused
trace bool,
) (common.Hash, error) {
keyCount := len(aKeys) + len(sKeys)
var stream = Stream{
keyBytes: make([]byte, len(aKeys)*(2*common.HashLength+1)+len(sKeys)*(4*common.HashLength+2)),
keySizes: make([]uint8, keyCount),
itemTypes: make([]StreamItem, keyCount),
aValues: aValues,
sValues: sValues,
}
var accountKeyHex []byte
var storageKeyHex []byte
var ki, ai, si int
// Convert all account keys and storage keys to HEX encoding and merge them into one sorted list
// when we merge, the keys are never equal
offset := 0
for ki < keyCount {
if accountKeyHex == nil && ai < len(aKeys) {
accountKeyHex = keybytesToHex(aKeys[ai][:])
ai++
}
if storageKeyHex == nil && si < len(sKeys) {
storageKeyHex = concat(keybytesToHex(sKeys[si][:storagePrefixLen]), keybytesToHex(sKeys[si][storagePrefixLen:])...)
si++
}
if accountKeyHex == nil {
copy(stream.keyBytes[offset:], storageKeyHex)
stream.keySizes[ki] = uint8(len(storageKeyHex))
stream.itemTypes[ki] = StorageStreamItem
ki++
offset += len(storageKeyHex)
storageKeyHex = nil // consumed
} else if storageKeyHex == nil {
copy(stream.keyBytes[offset:], accountKeyHex)
stream.keySizes[ki] = uint8(len(accountKeyHex))
stream.itemTypes[ki] = AccountStreamItem
ki++
offset += len(accountKeyHex)
accountKeyHex = nil // consumed
} else if bytes.Compare(accountKeyHex, storageKeyHex) < 0 {
copy(stream.keyBytes[offset:], accountKeyHex)
stream.keySizes[ki] = uint8(len(accountKeyHex))
stream.itemTypes[ki] = AccountStreamItem
ki++
offset += len(accountKeyHex)
accountKeyHex = nil // consumed
} else {
copy(stream.keyBytes[offset:], storageKeyHex)
stream.keySizes[ki] = uint8(len(storageKeyHex))
stream.itemTypes[ki] = StorageStreamItem
ki++
offset += len(storageKeyHex)
storageKeyHex = nil // consumed
}
}
if trace {
fmt.Printf("len(stream.hexes)=%d\n", keyCount)
printOffset := 0
for _, size := range stream.keySizes {
fmt.Printf("%x\n", stream.keyBytes[printOffset:printOffset+int(size)])
printOffset += int(size)
}
}
rs := NewResolveSet(0)
offset = 0
for _, size := range stream.keySizes {
rs.AddHex(stream.keyBytes[offset : offset+int(size)])
offset += int(size)
}
// Now we merge old and new streams, preferring the new
newStream.Reset()
oldIt := NewIterator(t, rs, trace)
it := NewStreamMergeIterator(oldIt, &stream, trace)
return StreamHash(it, storagePrefixLen, hb, trace)
}