2019-11-25 13:46:36 +00:00
|
|
|
package stateless
|
2019-05-27 13:51:49 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-06-16 13:36:16 +00:00
|
|
|
"context"
|
2019-05-27 13:51:49 +00:00
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
|
2019-11-25 13:46:36 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
2020-06-16 13:36:16 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
2019-05-27 13:51:49 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/turbo-geth/crypto"
|
|
|
|
)
|
|
|
|
|
|
|
|
func countDepths() {
|
|
|
|
startTime := time.Now()
|
2020-06-16 13:36:16 +00:00
|
|
|
db := ethdb.MustOpen("/Volumes/tb41/turbo-geth-10/geth/chaindata")
|
2019-05-27 13:51:49 +00:00
|
|
|
defer db.Close()
|
|
|
|
var occups [64]int // Occupancy of the current level
|
|
|
|
var counts [64][17]int
|
|
|
|
var prev [32]byte
|
|
|
|
count := 0
|
2020-06-16 13:36:16 +00:00
|
|
|
if err := db.KV().View(context.Background(), func(tx ethdb.Tx) error {
|
2020-04-19 19:51:32 +00:00
|
|
|
b := tx.Bucket(dbutils.CurrentStateBucket)
|
2019-05-27 13:51:49 +00:00
|
|
|
if b == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
c := b.Cursor()
|
2020-06-16 13:36:16 +00:00
|
|
|
for k, _, err := c.First(); k != nil; k, _, err = c.Next() {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-04-19 19:51:32 +00:00
|
|
|
if len(k) != 32 {
|
|
|
|
continue
|
|
|
|
}
|
2019-05-27 13:51:49 +00:00
|
|
|
if count == 0 {
|
|
|
|
for i := 0; i < 64; i++ {
|
|
|
|
occups[i] = 1
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Find the first nibble where k and prev are different
|
|
|
|
var mask byte = 0xf0
|
|
|
|
var i int
|
|
|
|
for i = 0; i < 64; i++ {
|
|
|
|
idx := i >> 1
|
|
|
|
if (k[idx] & mask) != (prev[idx] & mask) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
mask ^= 0xff
|
|
|
|
}
|
|
|
|
// Finalise lower nodes
|
|
|
|
after1 := false
|
|
|
|
for j := i + 1; j < 64; j++ {
|
|
|
|
if occups[j] > 1 || !after1 {
|
|
|
|
counts[j][occups[j]]++
|
|
|
|
}
|
|
|
|
if occups[j] == 1 {
|
|
|
|
after1 = true
|
|
|
|
} else {
|
|
|
|
after1 = false
|
|
|
|
occups[j] = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
occups[i]++
|
|
|
|
}
|
|
|
|
copy(prev[:], k)
|
|
|
|
count++
|
|
|
|
if count%100000 == 0 {
|
|
|
|
fmt.Printf("Processed %d account records\n", count)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
after1 := false
|
|
|
|
for j := 0; j < 64; j++ {
|
|
|
|
if occups[j] > 1 || !after1 {
|
|
|
|
counts[j][occups[j]]++
|
|
|
|
}
|
|
|
|
if occups[j] == 1 {
|
|
|
|
after1 = true
|
|
|
|
} else {
|
|
|
|
after1 = false
|
|
|
|
occups[j] = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("Processed %d account records\n", count)
|
|
|
|
return nil
|
2020-06-16 13:36:16 +00:00
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-05-27 13:51:49 +00:00
|
|
|
nodes := 0
|
|
|
|
for i := 0; i < 64; i++ {
|
|
|
|
exists := false
|
|
|
|
for j := 1; j <= 16; j++ {
|
|
|
|
if counts[i][j] > 0 {
|
|
|
|
exists = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !exists {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
fmt.Printf("LEVEL %d ==========================\n", i)
|
|
|
|
for j := 1; j <= 16; j++ {
|
|
|
|
if counts[i][j] > 0 {
|
|
|
|
fmt.Printf("%d: %d ", j, counts[i][j])
|
|
|
|
nodes += counts[i][j]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("\n")
|
|
|
|
}
|
|
|
|
fmt.Printf("Total number of nodes: %d\n", nodes)
|
|
|
|
fmt.Printf("Count depth took %s\n", time.Since(startTime))
|
|
|
|
}
|
|
|
|
|
|
|
|
func countStorageDepths() {
|
|
|
|
startTime := time.Now()
|
2020-06-16 13:36:16 +00:00
|
|
|
db := ethdb.MustOpen("/Volumes/tb41/turbo-geth-10/geth/chaindata")
|
2019-05-27 13:51:49 +00:00
|
|
|
defer db.Close()
|
|
|
|
var occups [64]int // Occupancy of the current level
|
|
|
|
var counts [64][17]int
|
|
|
|
var prevAddr [20]byte
|
|
|
|
var prev [32]byte
|
|
|
|
var accountExists bool
|
|
|
|
var filtered int
|
|
|
|
count := 0
|
2020-06-16 13:36:16 +00:00
|
|
|
if err := db.KV().View(context.Background(), func(tx ethdb.Tx) error {
|
2020-04-19 19:51:32 +00:00
|
|
|
st := tx.Bucket(dbutils.CurrentStateBucket)
|
|
|
|
if st == nil {
|
2019-05-27 13:51:49 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-04-19 19:51:32 +00:00
|
|
|
c := st.Cursor()
|
2020-06-16 13:36:16 +00:00
|
|
|
for k, _, err := c.First(); k != nil; k, _, err = c.Next() {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-04-19 19:51:32 +00:00
|
|
|
if len(k) == 32 {
|
|
|
|
continue
|
|
|
|
}
|
2019-05-27 13:51:49 +00:00
|
|
|
addr := k[:20]
|
|
|
|
sameAddr := bytes.Equal(addr, prevAddr[:])
|
|
|
|
if !sameAddr {
|
|
|
|
copy(prevAddr[:], addr)
|
2020-04-19 19:51:32 +00:00
|
|
|
v, _ := st.Get(crypto.Keccak256(addr[:]))
|
2019-05-27 13:51:49 +00:00
|
|
|
accountExists = v != nil
|
|
|
|
if !accountExists {
|
|
|
|
filtered++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Filter out storage of non-existent accounts
|
|
|
|
if !accountExists {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
key := k[20:52]
|
|
|
|
if count == 0 {
|
|
|
|
for i := 0; i < 64; i++ {
|
|
|
|
occups[i] = 1
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if sameAddr {
|
|
|
|
// Find the first nibble where k and prev are different
|
|
|
|
var mask byte = 0xf0
|
|
|
|
var i int
|
|
|
|
for i = 0; i < 64; i++ {
|
|
|
|
idx := i >> 1
|
|
|
|
if (key[idx] & mask) != (prev[idx] & mask) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
mask ^= 0xff
|
|
|
|
}
|
|
|
|
// Finalise lower nodes
|
|
|
|
after1 := false
|
|
|
|
for j := i + 1; j < 64; j++ {
|
|
|
|
if occups[j] > 1 || !after1 {
|
|
|
|
counts[j][occups[j]]++
|
|
|
|
}
|
|
|
|
if occups[j] == 1 {
|
|
|
|
after1 = true
|
|
|
|
} else {
|
|
|
|
after1 = false
|
|
|
|
occups[j] = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
occups[i]++
|
|
|
|
} else {
|
|
|
|
after1 := false
|
|
|
|
for j := 0; j < 64; j++ {
|
|
|
|
if occups[j] > 1 || !after1 {
|
|
|
|
counts[j][occups[j]]++
|
|
|
|
}
|
|
|
|
if occups[j] == 1 {
|
|
|
|
after1 = true
|
|
|
|
} else {
|
|
|
|
after1 = false
|
|
|
|
occups[j] = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
copy(prev[:], key)
|
|
|
|
count++
|
|
|
|
if count%100000 == 0 {
|
|
|
|
fmt.Printf("Processed %d storage records, filtered accounts: %d\n", count, filtered)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
after1 := false
|
|
|
|
for j := 0; j < 64; j++ {
|
|
|
|
if occups[j] > 1 || !after1 {
|
|
|
|
counts[j][occups[j]]++
|
|
|
|
}
|
|
|
|
if occups[j] == 1 {
|
|
|
|
after1 = true
|
|
|
|
} else {
|
|
|
|
after1 = false
|
|
|
|
occups[j] = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("Processed %d storage records, filtered accounts: %d\n", count, filtered)
|
|
|
|
return nil
|
2020-06-16 13:36:16 +00:00
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-05-27 13:51:49 +00:00
|
|
|
nodes := 0
|
|
|
|
for i := 0; i < 64; i++ {
|
|
|
|
exists := false
|
|
|
|
for j := 1; j <= 16; j++ {
|
|
|
|
if counts[i][j] > 0 {
|
|
|
|
exists = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !exists {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
fmt.Printf("LEVEL %d ==========================\n", i)
|
|
|
|
for j := 1; j <= 16; j++ {
|
|
|
|
if counts[i][j] > 0 {
|
|
|
|
fmt.Printf("%d: %d ", j, counts[i][j])
|
|
|
|
nodes += counts[i][j]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("\n")
|
|
|
|
}
|
|
|
|
fmt.Printf("Total number of nodes: %d\n", nodes)
|
|
|
|
fmt.Printf("Count depth took %s\n", time.Since(startTime))
|
|
|
|
}
|