| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | // 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 of | 
					
						
							|  |  |  | // 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 snapshot | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"encoding/binary" | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	"math/rand" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	"github.com/VictoriaMetrics/fastcache" | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/rawdb" | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // TestAccountIteratorBasics tests some simple single-layer iteration | 
					
						
							|  |  |  | func TestAccountIteratorBasics(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 		destructs = make(map[common.Hash]struct{}) | 
					
						
							|  |  |  | 		accounts  = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		storage   = make(map[common.Hash]map[common.Hash][]byte) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	) | 
					
						
							|  |  |  | 	// Fill up a parent | 
					
						
							|  |  |  | 	for i := 0; i < 100; i++ { | 
					
						
							|  |  |  | 		h := randomHash() | 
					
						
							|  |  |  | 		data := randomAccount() | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		accounts[h] = data | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 		if rand.Intn(4) == 0 { | 
					
						
							|  |  |  | 			destructs[h] = struct{}{} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if rand.Intn(2) == 0 { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 			accStorage := make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 			value := make([]byte, 32) | 
					
						
							|  |  |  | 			rand.Read(value) | 
					
						
							|  |  |  | 			accStorage[randomHash()] = value | 
					
						
							|  |  |  | 			storage[h] = accStorage | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Add some (identical) layers on top | 
					
						
							| 
									
										
										
										
											2020-03-04 15:06:04 +02:00
										 |  |  | 	parent := newDiffLayer(emptyLayer(), common.Hash{}, copyDestructs(destructs), copyAccounts(accounts), copyStorage(storage)) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	it := parent.AccountIterator(common.Hash{}) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	verifyIterator(t, 100, it) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type testIterator struct { | 
					
						
							|  |  |  | 	values []byte | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func newTestIterator(values ...byte) *testIterator { | 
					
						
							|  |  |  | 	return &testIterator{values} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (ti *testIterator) Seek(common.Hash) { | 
					
						
							|  |  |  | 	panic("implement me") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (ti *testIterator) Next() bool { | 
					
						
							|  |  |  | 	ti.values = ti.values[1:] | 
					
						
							| 
									
										
										
										
											2020-02-24 13:26:34 +02:00
										 |  |  | 	return len(ti.values) > 0 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (ti *testIterator) Error() error { | 
					
						
							| 
									
										
										
										
											2020-02-24 13:26:34 +02:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | func (ti *testIterator) Hash() common.Hash { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	return common.BytesToHash([]byte{ti.values[0]}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | func (ti *testIterator) Account() []byte { | 
					
						
							| 
									
										
										
										
											2020-02-24 13:26:34 +02:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | func (ti *testIterator) Release() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | func TestFastIteratorBasics(t *testing.T) { | 
					
						
							|  |  |  | 	type testCase struct { | 
					
						
							|  |  |  | 		lists   [][]byte | 
					
						
							|  |  |  | 		expKeys []byte | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for i, tc := range []testCase{ | 
					
						
							|  |  |  | 		{lists: [][]byte{{0, 1, 8}, {1, 2, 8}, {2, 9}, {4}, | 
					
						
							|  |  |  | 			{7, 14, 15}, {9, 13, 15, 16}}, | 
					
						
							|  |  |  | 			expKeys: []byte{0, 1, 2, 4, 7, 8, 9, 13, 14, 15, 16}}, | 
					
						
							|  |  |  | 		{lists: [][]byte{{0, 8}, {1, 2, 8}, {7, 14, 15}, {8, 9}, | 
					
						
							|  |  |  | 			{9, 10}, {10, 13, 15, 16}}, | 
					
						
							|  |  |  | 			expKeys: []byte{0, 1, 2, 7, 8, 9, 10, 13, 14, 15, 16}}, | 
					
						
							|  |  |  | 	} { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		var iterators []*weightedAccountIterator | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 		for i, data := range tc.lists { | 
					
						
							|  |  |  | 			it := newTestIterator(data...) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			iterators = append(iterators, &weightedAccountIterator{it, i}) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		fi := &fastAccountIterator{ | 
					
						
							|  |  |  | 			iterators: iterators, | 
					
						
							|  |  |  | 			initiated: false, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		count := 0 | 
					
						
							|  |  |  | 		for fi.Next() { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			if got, exp := fi.Hash()[31], tc.expKeys[count]; exp != got { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 				t.Errorf("tc %d, [%d]: got %d exp %d", i, count, got, exp) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			count++ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func verifyIterator(t *testing.T, expCount int, it AccountIterator) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	t.Helper() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		count = 0 | 
					
						
							|  |  |  | 		last  = common.Hash{} | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	) | 
					
						
							|  |  |  | 	for it.Next() { | 
					
						
							| 
									
										
										
										
											2020-03-04 13:38:55 +01:00
										 |  |  | 		hash := it.Hash() | 
					
						
							|  |  |  | 		if bytes.Compare(last[:], hash[:]) >= 0 { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			t.Errorf("wrong order: %x >= %x", last, hash) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-03-04 13:38:55 +01:00
										 |  |  | 		if it.Account() == nil { | 
					
						
							|  |  |  | 			t.Errorf("iterator returned nil-value for hash %x", hash) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		count++ | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	if count != expCount { | 
					
						
							|  |  |  | 		t.Errorf("iterator count mismatch: have %d, want %d", count, expCount) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err := it.Error(); err != nil { | 
					
						
							|  |  |  | 		t.Errorf("iterator failed: %v", err) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // TestAccountIteratorTraversal tests some simple multi-layer iteration. | 
					
						
							|  |  |  | func TestAccountIteratorTraversal(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create an empty base layer and a snapshot tree out of it | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Stack three diff layers on top with various overlaps | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x02"), common.HexToHash("0x01"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xaa", "0xee", "0xff", "0xf0"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x03"), common.HexToHash("0x02"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xbb", "0xdd", "0xf0"), nil) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x04"), common.HexToHash("0x03"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xcc", "0xf0", "0xff"), nil) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Verify the single and multi-layer iterators | 
					
						
							|  |  |  | 	head := snaps.Snapshot(common.HexToHash("0x04")) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	verifyIterator(t, 3, head.(snapshot).AccountIterator(common.Hash{})) | 
					
						
							|  |  |  | 	verifyIterator(t, 7, head.(*diffLayer).newBinaryAccountIterator()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ := snaps.AccountIterator(common.HexToHash("0x04"), common.Hash{}) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	verifyIterator(t, 7, it) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // TestAccountIteratorTraversalValues tests some multi-layer iteration, where we | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | // also expect the correct values to show up. | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | func TestAccountIteratorTraversalValues(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create an empty base layer and a snapshot tree out of it | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Create a batch of account sets to seed subsequent layers with | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		a = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		b = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		c = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		d = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		e = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		f = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		g = make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		h = make(map[common.Hash][]byte) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	) | 
					
						
							|  |  |  | 	for i := byte(2); i < 0xff; i++ { | 
					
						
							|  |  |  | 		a[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 0, i)) | 
					
						
							|  |  |  | 		if i > 20 && i%2 == 0 { | 
					
						
							|  |  |  | 			b[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 1, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i%4 == 0 { | 
					
						
							|  |  |  | 			c[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 2, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i%7 == 0 { | 
					
						
							|  |  |  | 			d[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 3, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i%8 == 0 { | 
					
						
							|  |  |  | 			e[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 4, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i > 50 || i < 85 { | 
					
						
							|  |  |  | 			f[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 5, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i%64 == 0 { | 
					
						
							|  |  |  | 			g[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 6, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i%128 == 0 { | 
					
						
							|  |  |  | 			h[common.Hash{i}] = []byte(fmt.Sprintf("layer-%d, key %d", 7, i)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Assemble a stack of snapshots from the account layers | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x02"), common.HexToHash("0x01"), nil, a, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x03"), common.HexToHash("0x02"), nil, b, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x04"), common.HexToHash("0x03"), nil, c, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x05"), common.HexToHash("0x04"), nil, d, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x06"), common.HexToHash("0x05"), nil, e, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x07"), common.HexToHash("0x06"), nil, f, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x08"), common.HexToHash("0x07"), nil, g, nil) | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x09"), common.HexToHash("0x08"), nil, h, nil) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	it, _ := snaps.AccountIterator(common.HexToHash("0x09"), common.Hash{}) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	head := snaps.Snapshot(common.HexToHash("0x09")) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	for it.Next() { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		hash := it.Hash() | 
					
						
							|  |  |  | 		want, err := head.AccountRLP(hash) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			t.Fatalf("failed to retrieve expected account: %v", err) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		if have := it.Account(); !bytes.Equal(want, have) { | 
					
						
							|  |  |  | 			t.Fatalf("hash %x: account mismatch: have %x, want %x", hash, have, want) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | // This testcase is notorious, all layers contain the exact same 200 accounts. | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | func TestAccountIteratorLargeTraversal(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create a custom account factory to recreate the same addresses | 
					
						
							|  |  |  | 	makeAccounts := func(num int) map[common.Hash][]byte { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		accounts := make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		for i := 0; i < num; i++ { | 
					
						
							|  |  |  | 			h := common.Hash{} | 
					
						
							|  |  |  | 			binary.BigEndian.PutUint64(h[:], uint64(i+1)) | 
					
						
							|  |  |  | 			accounts[h] = randomAccount() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return accounts | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Build up a large stack of snapshots | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for i := 1; i < 128; i++ { | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 		snaps.Update(common.HexToHash(fmt.Sprintf("0x%02x", i+1)), common.HexToHash(fmt.Sprintf("0x%02x", i)), nil, makeAccounts(200), nil) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	// Iterate the entire stack and ensure everything is hit only once | 
					
						
							|  |  |  | 	head := snaps.Snapshot(common.HexToHash("0x80")) | 
					
						
							|  |  |  | 	verifyIterator(t, 200, head.(snapshot).AccountIterator(common.Hash{})) | 
					
						
							|  |  |  | 	verifyIterator(t, 200, head.(*diffLayer).newBinaryAccountIterator()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ := snaps.AccountIterator(common.HexToHash("0x80"), common.Hash{}) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	verifyIterator(t, 200, it) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // TestAccountIteratorFlattening tests what happens when we | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | // - have a live iterator on child C (parent C1 -> C2 .. CN) | 
					
						
							|  |  |  | // - flattens C2 all the way into CN | 
					
						
							|  |  |  | // - continues iterating | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | func TestAccountIteratorFlattening(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create an empty base layer and a snapshot tree out of it | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Create a stack of diffs on top | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x02"), common.HexToHash("0x01"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xaa", "0xee", "0xff", "0xf0"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x03"), common.HexToHash("0x02"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xbb", "0xdd", "0xf0"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x04"), common.HexToHash("0x03"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xcc", "0xf0", "0xff"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Create an iterator and flatten the data from underneath it | 
					
						
							|  |  |  | 	it, _ := snaps.AccountIterator(common.HexToHash("0x04"), common.Hash{}) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err := snaps.Cap(common.HexToHash("0x04"), 1); err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("failed to flatten snapshot stack: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	//verifyIterator(t, 7, it) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | func TestAccountIteratorSeek(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create a snapshot stack with some initial data | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x02"), common.HexToHash("0x01"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xaa", "0xee", "0xff", "0xf0"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x03"), common.HexToHash("0x02"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xbb", "0xdd", "0xf0"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x04"), common.HexToHash("0x03"), nil, | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 		randomAccountSet("0xcc", "0xf0", "0xff"), nil) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Construct various iterators and ensure their tranversal is correct | 
					
						
							|  |  |  | 	it, _ := snaps.AccountIterator(common.HexToHash("0x02"), common.HexToHash("0xdd")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 3, it) // expected: ee, f0, ff | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x02"), common.HexToHash("0xaa")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 3, it) // expected: ee, f0, ff | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x02"), common.HexToHash("0xff")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 0, it) // expected: nothing | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x04"), common.HexToHash("0xbb")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 5, it) // expected: cc, dd, ee, f0, ff | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x04"), common.HexToHash("0xef")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 2, it) // expected: f0, ff | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x04"), common.HexToHash("0xf0")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 1, it) // expected: ff | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x04"), common.HexToHash("0xff")) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	verifyIterator(t, 0, it) // expected: nothing | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-04 13:38:55 +01:00
										 |  |  | // TestIteratorDeletions tests that the iterator behaves correct when there are | 
					
						
							|  |  |  | // deleted accounts (where the Account() value is nil). The iterator | 
					
						
							|  |  |  | // should not output any accounts or nil-values for those cases. | 
					
						
							|  |  |  | func TestIteratorDeletions(t *testing.T) { | 
					
						
							|  |  |  | 	// Create an empty base layer and a snapshot tree out of it | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Stack three diff layers on top with various overlaps | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x02"), common.HexToHash("0x01"), | 
					
						
							| 
									
										
										
										
											2020-03-04 15:06:04 +02:00
										 |  |  | 		nil, randomAccountSet("0x11", "0x22", "0x33"), nil) | 
					
						
							| 
									
										
										
										
											2020-03-04 13:38:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	deleted := common.HexToHash("0x22") | 
					
						
							| 
									
										
										
										
											2020-03-04 15:06:04 +02:00
										 |  |  | 	destructed := map[common.Hash]struct{}{ | 
					
						
							|  |  |  | 		deleted: struct{}{}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x03"), common.HexToHash("0x02"), | 
					
						
							|  |  |  | 		destructed, randomAccountSet("0x11", "0x33"), nil) | 
					
						
							| 
									
										
										
										
											2020-03-04 13:38:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	snaps.Update(common.HexToHash("0x04"), common.HexToHash("0x03"), | 
					
						
							| 
									
										
										
										
											2020-03-04 15:06:04 +02:00
										 |  |  | 		nil, randomAccountSet("0x33", "0x44", "0x55"), nil) | 
					
						
							| 
									
										
										
										
											2020-03-04 13:38:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// The output should be 11,33,44,55 | 
					
						
							|  |  |  | 	it, _ := snaps.AccountIterator(common.HexToHash("0x04"), common.Hash{}) | 
					
						
							|  |  |  | 	// Do a quick check | 
					
						
							|  |  |  | 	verifyIterator(t, 4, it) | 
					
						
							|  |  |  | 	it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// And a more detailed verification that we indeed do not see '0x22' | 
					
						
							|  |  |  | 	it, _ = snaps.AccountIterator(common.HexToHash("0x04"), common.Hash{}) | 
					
						
							|  |  |  | 	defer it.Release() | 
					
						
							|  |  |  | 	for it.Next() { | 
					
						
							|  |  |  | 		hash := it.Hash() | 
					
						
							|  |  |  | 		if it.Account() == nil { | 
					
						
							|  |  |  | 			t.Errorf("iterator returned nil-value for hash %x", hash) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if hash == deleted { | 
					
						
							|  |  |  | 			t.Errorf("expected deleted elem %x to not be returned by iterator", deleted) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // BenchmarkAccountIteratorTraversal is a bit a bit notorious -- all layers contain the | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | // exact same 200 accounts. That means that we need to process 2000 items, but | 
					
						
							|  |  |  | // only spit out 200 values eventually. | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | // The value-fetching benchmark is easy on the binary iterator, since it never has to reach | 
					
						
							|  |  |  | // down at any depth for retrieving the values -- all are on the toppmost layer | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // BenchmarkAccountIteratorTraversal/binary_iterator_keys-6         	    2239	    483674 ns/op | 
					
						
							|  |  |  | // BenchmarkAccountIteratorTraversal/binary_iterator_values-6       	    2403	    501810 ns/op | 
					
						
							|  |  |  | // BenchmarkAccountIteratorTraversal/fast_iterator_keys-6           	    1923	    677966 ns/op | 
					
						
							|  |  |  | // BenchmarkAccountIteratorTraversal/fast_iterator_values-6         	    1741	    649967 ns/op | 
					
						
							|  |  |  | func BenchmarkAccountIteratorTraversal(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create a custom account factory to recreate the same addresses | 
					
						
							|  |  |  | 	makeAccounts := func(num int) map[common.Hash][]byte { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		accounts := make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		for i := 0; i < num; i++ { | 
					
						
							|  |  |  | 			h := common.Hash{} | 
					
						
							|  |  |  | 			binary.BigEndian.PutUint64(h[:], uint64(i+1)) | 
					
						
							|  |  |  | 			accounts[h] = randomAccount() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return accounts | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Build up a large stack of snapshots | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for i := 1; i <= 100; i++ { | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 		snaps.Update(common.HexToHash(fmt.Sprintf("0x%02x", i+1)), common.HexToHash(fmt.Sprintf("0x%02x", i)), nil, makeAccounts(200), nil) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	// We call this once before the benchmark, so the creation of | 
					
						
							|  |  |  | 	// sorted accountlists are not included in the results. | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	head := snaps.Snapshot(common.HexToHash("0x65")) | 
					
						
							|  |  |  | 	head.(*diffLayer).newBinaryAccountIterator() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	b.Run("binary iterator keys", func(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 			got := 0 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it := head.(*diffLayer).newBinaryAccountIterator() | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 200; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	b.Run("binary iterator values", func(b *testing.B) { | 
					
						
							|  |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 			got := 0 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it := head.(*diffLayer).newBinaryAccountIterator() | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 				head.(*diffLayer).accountRLP(it.Hash(), 0) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 200; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	b.Run("fast iterator keys", func(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it, _ := snaps.AccountIterator(common.HexToHash("0x65"), common.Hash{}) | 
					
						
							|  |  |  | 			defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 			got := 0 | 
					
						
							|  |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 200; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	b.Run("fast iterator values", func(b *testing.B) { | 
					
						
							|  |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it, _ := snaps.AccountIterator(common.HexToHash("0x65"), common.Hash{}) | 
					
						
							|  |  |  | 			defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			got := 0 | 
					
						
							|  |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 				it.Account() | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 200; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // BenchmarkAccountIteratorLargeBaselayer is a pretty realistic benchmark, where | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | // the baselayer is a lot larger than the upper layer. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This is heavy on the binary iterator, which in most cases will have to | 
					
						
							|  |  |  | // call recursively 100 times for the majority of the values | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | // BenchmarkAccountIteratorLargeBaselayer/binary_iterator_(keys)-6         	     514	   1971999 ns/op | 
					
						
							|  |  |  | // BenchmarkAccountIteratorLargeBaselayer/binary_iterator_(values)-6       	      61	  18997492 ns/op | 
					
						
							|  |  |  | // BenchmarkAccountIteratorLargeBaselayer/fast_iterator_(keys)-6           	   10000	    114385 ns/op | 
					
						
							|  |  |  | // BenchmarkAccountIteratorLargeBaselayer/fast_iterator_(values)-6         	    4047	    296823 ns/op | 
					
						
							|  |  |  | func BenchmarkAccountIteratorLargeBaselayer(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Create a custom account factory to recreate the same addresses | 
					
						
							|  |  |  | 	makeAccounts := func(num int) map[common.Hash][]byte { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		accounts := make(map[common.Hash][]byte) | 
					
						
							|  |  |  | 		for i := 0; i < num; i++ { | 
					
						
							|  |  |  | 			h := common.Hash{} | 
					
						
							|  |  |  | 			binary.BigEndian.PutUint64(h[:], uint64(i+1)) | 
					
						
							|  |  |  | 			accounts[h] = randomAccount() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return accounts | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	// Build up a large stack of snapshots | 
					
						
							|  |  |  | 	base := &diskLayer{ | 
					
						
							|  |  |  | 		diskdb: rawdb.NewMemoryDatabase(), | 
					
						
							|  |  |  | 		root:   common.HexToHash("0x01"), | 
					
						
							|  |  |  | 		cache:  fastcache.New(1024 * 500), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	snaps := &Tree{ | 
					
						
							|  |  |  | 		layers: map[common.Hash]snapshot{ | 
					
						
							|  |  |  | 			base.root: base, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 	snaps.Update(common.HexToHash("0x02"), common.HexToHash("0x01"), nil, makeAccounts(2000), nil) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	for i := 2; i <= 100; i++ { | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 		snaps.Update(common.HexToHash(fmt.Sprintf("0x%02x", i+1)), common.HexToHash(fmt.Sprintf("0x%02x", i)), nil, makeAccounts(20), nil) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	// We call this once before the benchmark, so the creation of | 
					
						
							|  |  |  | 	// sorted accountlists are not included in the results. | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	head := snaps.Snapshot(common.HexToHash("0x65")) | 
					
						
							|  |  |  | 	head.(*diffLayer).newBinaryAccountIterator() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	b.Run("binary iterator (keys)", func(b *testing.B) { | 
					
						
							|  |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 			got := 0 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it := head.(*diffLayer).newBinaryAccountIterator() | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 2000; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	b.Run("binary iterator (values)", func(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 			got := 0 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it := head.(*diffLayer).newBinaryAccountIterator() | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 				v := it.Hash() | 
					
						
							|  |  |  | 				head.(*diffLayer).accountRLP(v, 0) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 2000; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	b.Run("fast iterator (keys)", func(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it, _ := snaps.AccountIterator(common.HexToHash("0x65"), common.Hash{}) | 
					
						
							|  |  |  | 			defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 			got := 0 | 
					
						
							|  |  |  | 			for it.Next() { | 
					
						
							|  |  |  | 				got++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 2000; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	b.Run("fast iterator (values)", func(b *testing.B) { | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 		for i := 0; i < b.N; i++ { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 			it, _ := snaps.AccountIterator(common.HexToHash("0x65"), common.Hash{}) | 
					
						
							|  |  |  | 			defer it.Release() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 			got := 0 | 
					
						
							|  |  |  | 			for it.Next() { | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 				it.Account() | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 				got++ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if exp := 2000; got != exp { | 
					
						
							|  |  |  | 				b.Errorf("iterator len wrong, expected %d, got %d", exp, got) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | /* | 
					
						
							|  |  |  | func BenchmarkBinaryAccountIteration(b *testing.B) { | 
					
						
							|  |  |  | 	benchmarkAccountIteration(b, func(snap snapshot) AccountIterator { | 
					
						
							|  |  |  | 		return snap.(*diffLayer).newBinaryAccountIterator() | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | func BenchmarkFastAccountIteration(b *testing.B) { | 
					
						
							|  |  |  | 	benchmarkAccountIteration(b, newFastAccountIterator) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | func benchmarkAccountIteration(b *testing.B, iterator func(snap snapshot) AccountIterator) { | 
					
						
							|  |  |  | 	// Create a diff stack and randomize the accounts across them | 
					
						
							|  |  |  | 	layers := make([]map[common.Hash][]byte, 128) | 
					
						
							|  |  |  | 	for i := 0; i < len(layers); i++ { | 
					
						
							|  |  |  | 		layers[i] = make(map[common.Hash][]byte) | 
					
						
							| 
									
										
										
										
											2019-12-05 15:37:25 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 		depth := rand.Intn(len(layers)) | 
					
						
							|  |  |  | 		layers[depth][randomHash()] = randomAccount() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	stack := snapshot(emptyLayer()) | 
					
						
							|  |  |  | 	for _, layer := range layers { | 
					
						
							| 
									
										
										
										
											2020-03-03 15:52:00 +02:00
										 |  |  | 		stack = stack.Update(common.Hash{}, layer, nil, nil) | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	// Reset the timers and report all the stats | 
					
						
							|  |  |  | 	it := iterator(stack) | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	b.ResetTimer() | 
					
						
							|  |  |  | 	b.ReportAllocs() | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | 	for it.Next() { | 
					
						
							| 
									
										
										
										
											2019-12-06 23:27:18 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-12-10 11:00:03 +02:00
										 |  |  | */ |