erigon-pulse/cmd/state/stateless/naked_storage.go
ledgerwatch cf799157cc
Jumpdest skip optimisation (#851)
* Jumpdest skipping optimisation

* Fix formatting

* Move skipAnalysis into vmConfig, introduce tracing ability

* Improve detection logging

* Added release instructions

* Fix lint
2020-08-01 17:56:57 +01:00

551 lines
16 KiB
Go

package stateless
import (
"bufio"
"bytes"
"encoding/csv"
"fmt"
"io/ioutil"
"math/big"
"os"
"os/signal"
"runtime"
"syscall"
"time"
"github.com/wcharczuk/go-chart"
"github.com/wcharczuk/go-chart/drawing"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/consensus/ethash"
"github.com/ledgerwatch/turbo-geth/core"
"github.com/ledgerwatch/turbo-geth/core/state"
"github.com/ledgerwatch/turbo-geth/core/types"
"github.com/ledgerwatch/turbo-geth/core/vm"
"github.com/ledgerwatch/turbo-geth/core/vm/stack"
"github.com/ledgerwatch/turbo-geth/ethdb"
"github.com/ledgerwatch/turbo-geth/params"
)
type StorageTracer struct {
loaded map[common.Address]map[common.Hash]struct{}
totalSstores int
nakedSstores int
totalSloads int
nakedSloads int
}
func NewStorageTracer() *StorageTracer {
return &StorageTracer{
loaded: make(map[common.Address]map[common.Hash]struct{}),
}
}
func (st *StorageTracer) CaptureStart(depth int, from common.Address, to common.Address, call bool, input []byte, gas uint64, value *big.Int) error {
return nil
}
func (st *StorageTracer) CaptureState(env *vm.EVM, pc uint64, op vm.OpCode, gas, cost uint64, memory *vm.Memory, stack *stack.Stack, _ *stack.ReturnStack, contract *vm.Contract, depth int, err error) error {
if op == vm.SSTORE {
addr := contract.Address()
if stack.Len() == 0 {
return nil
}
loc := common.Hash(stack.Back(0).Bytes32())
if l1, ok1 := st.loaded[addr]; ok1 {
if _, ok2 := l1[loc]; !ok2 {
st.nakedSstores++
l1[loc] = struct{}{}
}
} else {
st.nakedSstores++
l2 := make(map[common.Hash]struct{})
l2[loc] = struct{}{}
st.loaded[addr] = l2
}
st.totalSstores++
} else if op == vm.SLOAD {
addr := contract.Address()
if stack.Len() == 0 {
return nil
}
loc := common.Hash(stack.Back(0).Bytes32())
if l1, ok1 := st.loaded[addr]; ok1 {
if _, ok2 := l1[loc]; !ok2 {
st.nakedSloads++
l1[loc] = struct{}{}
}
} else {
st.nakedSloads++
l2 := make(map[common.Hash]struct{})
l2[loc] = struct{}{}
st.loaded[addr] = l2
}
st.totalSloads++
}
return nil
}
func (st *StorageTracer) CaptureFault(env *vm.EVM, pc uint64, op vm.OpCode, gas, cost uint64, memory *vm.Memory, stack *stack.Stack, _ *stack.ReturnStack, contract *vm.Contract, depth int, err error) error {
return nil
}
func (st *StorageTracer) CaptureEnd(depth int, output []byte, gasUsed uint64, t time.Duration, err error) error {
return nil
}
func (st *StorageTracer) CaptureCreate(creator common.Address, creation common.Address) error {
return nil
}
func (st *StorageTracer) CaptureAccountRead(account common.Address) error {
return nil
}
func (st *StorageTracer) CaptureAccountWrite(account common.Address) error {
return nil
}
//nolint:deadcode,unused
func storageReadWrites(blockNum uint64) {
startTime := time.Now()
sigs := make(chan os.Signal, 1)
interruptCh := make(chan bool, 1)
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
go func() {
<-sigs
interruptCh <- true
}()
ethDb := ethdb.MustOpen("/Volumes/tb41/turbo-geth-10/geth/chaindata")
defer ethDb.Close()
chainConfig := params.MainnetChainConfig
srwFile, err := os.OpenFile("/Volumes/tb41/turbo-geth/storage_read_writes.csv", os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
check(err)
defer srwFile.Close()
w := bufio.NewWriter(srwFile)
defer w.Flush()
st := NewStorageTracer()
vmConfig := vm.Config{Tracer: st, Debug: true}
txCacher := core.NewTxSenderCacher(runtime.NumCPU())
bc, err := core.NewBlockChain(ethDb, nil, chainConfig, ethash.NewFaker(), vmConfig, nil, txCacher)
check(err)
defer bc.Stop()
interrupt := false
totalSstores := 0
nakedSstores := 0
totalSloads := 0
nakedSloads := 0
for !interrupt {
block := bc.GetBlockByNumber(blockNum)
if block == nil {
break
}
dbstate := state.NewDbState(ethDb.KV(), block.NumberU64()-1)
statedb := state.New(dbstate)
signer := types.MakeSigner(chainConfig, block.Number())
st.loaded = make(map[common.Address]map[common.Hash]struct{})
st.totalSstores = 0
st.nakedSstores = 0
st.totalSloads = 0
st.nakedSloads = 0
for _, tx := range block.Transactions() {
// Assemble the transaction call message and return if the requested offset
msg, _ := tx.AsMessage(signer)
context := core.NewEVMContext(msg, block.Header(), bc, nil)
// Not yet the searched for transaction, execute on top of the current state
vmenv := vm.NewEVM(context, statedb, chainConfig, vmConfig)
if _, err := core.ApplyMessage(vmenv, msg, new(core.GasPool).AddGas(tx.Gas())); err != nil {
panic(fmt.Errorf("tx %x failed: %v", tx.Hash(), err))
}
}
fmt.Fprintf(w, "%d,%d,%d,%d,%d\n", blockNum, st.totalSstores, st.nakedSstores, st.totalSloads, st.nakedSloads)
totalSstores += st.totalSstores
nakedSstores += st.nakedSstores
totalSloads += st.totalSloads
nakedSloads += st.nakedSloads
blockNum++
if blockNum%1000 == 0 {
fmt.Printf("Processed %d blocks, totalSstores %d, nakedSstores %d, totalSloads %d, nakedSloads %d\n", blockNum, totalSstores, nakedSstores, totalSloads, nakedSloads)
}
// Check for interrupts
select {
case interrupt = <-interruptCh:
fmt.Println("interrupted, please wait for cleanup...")
default:
}
}
fmt.Printf("Processed %d blocks, totalSstores %d, nakedSstores %d, totalSloads %d, nakedSloads %d\n", blockNum, totalSstores, nakedSstores, totalSloads, nakedSloads)
fmt.Printf("Next time specify -block %d\n", blockNum)
fmt.Printf("Storage read/write analysis took %s\n", time.Since(startTime))
}
func operations() []chart.GridLine {
return []chart.GridLine{
{Value: 5.0},
{Value: 10.0},
{Value: 15.0},
{Value: 20.0},
{Value: 25.0},
{Value: 30.0},
{Value: 35.0},
{Value: 40.0},
{Value: 45.0},
}
}
func nakedSstoreChart() {
swrFile, err := os.Open("/Volumes/tb4/turbo-geth/storage_read_writes.csv")
check(err)
defer swrFile.Close()
swrReader := csv.NewReader(bufio.NewReader(swrFile))
var blocks []float64
var totalSstores []float64
var nakedSstores []float64
for records, _ := swrReader.Read(); records != nil; records, _ = swrReader.Read() {
blocks = append(blocks, parseFloat64(records[0])/1000000.0)
totalSstores = append(totalSstores, parseFloat64(records[1]))
nakedSstores = append(nakedSstores, parseFloat64(records[2]))
}
var totalSstoresGroup float64 = 0
var nakedSstoresGroup float64 = 0
var i int
var window int = 1024
b := make([]float64, len(blocks)-window+1)
ts := make([]float64, len(blocks)-window+1)
ns := make([]float64, len(blocks)-window+1)
for i = 0; i < len(blocks); i++ {
totalSstoresGroup += totalSstores[i]
nakedSstoresGroup += nakedSstores[i]
if i >= window {
totalSstoresGroup -= totalSstores[i-window]
nakedSstoresGroup -= nakedSstores[i-window]
}
if i >= window-1 {
b[i-window+1] = blocks[i]
ts[i-window+1] = totalSstoresGroup / float64(window)
ns[i-window+1] = nakedSstoresGroup / float64(window)
}
}
check(err)
totalSeries := &chart.ContinuousSeries{
Name: fmt.Sprintf("Total SSTOREs per block, moving average with window %d", window),
Style: chart.Style{
Show: true,
StrokeColor: chart.ColorBlue,
FillColor: chart.ColorBlue.WithAlpha(100),
},
XValues: b,
YValues: ts,
}
nakedSeries := &chart.ContinuousSeries{
Name: fmt.Sprintf("Naked SSTOREs per block, moving average with window %d", window),
Style: chart.Style{
Show: true,
StrokeColor: chart.ColorBlack,
},
XValues: b,
YValues: ns,
}
graph1 := chart.Chart{
Width: 1280,
Height: 720,
Background: chart.Style{
Padding: chart.Box{
Top: 50,
},
},
YAxis: chart.YAxis{
Name: "operations",
NameStyle: chart.StyleShow(),
Style: chart.StyleShow(),
TickStyle: chart.Style{
TextRotationDegrees: 45.0,
},
ValueFormatter: func(v interface{}) string {
return fmt.Sprintf("%.1f", v.(float64))
},
GridMajorStyle: chart.Style{
Show: true,
StrokeColor: chart.ColorAlternateGray,
StrokeWidth: 1.0,
},
//GridLines: operations(),
},
XAxis: chart.XAxis{
Name: "Blocks, million",
Style: chart.Style{
Show: true,
},
ValueFormatter: func(v interface{}) string {
return fmt.Sprintf("%.3fm", v.(float64))
},
GridMajorStyle: chart.Style{
Show: true,
StrokeColor: chart.ColorAlternateGray,
StrokeWidth: 1.0,
},
GridLines: blockMillions(),
},
Series: []chart.Series{
totalSeries,
nakedSeries,
},
}
graph1.Elements = []chart.Renderable{chart.LegendThin(&graph1)}
buffer := bytes.NewBuffer([]byte{})
err = graph1.Render(chart.PNG, buffer)
check(err)
err = ioutil.WriteFile("naked_sstores.png", buffer.Bytes(), 0644)
check(err)
}
func nakedSloadChart() {
swrFile, err := os.Open("/Volumes/tb4/turbo-geth/storage_read_writes.csv")
check(err)
defer swrFile.Close()
swrReader := csv.NewReader(bufio.NewReader(swrFile))
var blocks []float64
var totalSloads []float64
var nakedSloads []float64
for records, _ := swrReader.Read(); records != nil; records, _ = swrReader.Read() {
blocks = append(blocks, parseFloat64(records[0])/1000000.0)
totalSloads = append(totalSloads, parseFloat64(records[3]))
nakedSloads = append(nakedSloads, parseFloat64(records[4]))
}
var totalSloadsGroup float64 = 0
var nakedSloadsGroup float64 = 0
var i int
var window int = 1024
b := make([]float64, len(blocks)-window+1)
ts := make([]float64, len(blocks)-window+1)
ns := make([]float64, len(blocks)-window+1)
for i = 0; i < len(blocks); i++ {
totalSloadsGroup += totalSloads[i]
nakedSloadsGroup += nakedSloads[i]
if i >= window {
totalSloadsGroup -= totalSloads[i-window]
nakedSloadsGroup -= nakedSloads[i-window]
}
if i >= window-1 {
b[i-window+1] = blocks[i]
ts[i-window+1] = totalSloadsGroup / float64(window)
ns[i-window+1] = nakedSloadsGroup / float64(window)
}
}
check(err)
totalSeries := &chart.ContinuousSeries{
Name: fmt.Sprintf("Total SLOADs per block, moving average with window %d", window),
Style: chart.Style{
Show: true,
StrokeColor: chart.ColorGreen,
FillColor: chart.ColorGreen.WithAlpha(100),
},
XValues: b,
YValues: ts,
}
nakedSeries := &chart.ContinuousSeries{
Name: fmt.Sprintf("Naked SLOADs per block, moving average with window %d", window),
Style: chart.Style{
Show: true,
StrokeColor: chart.ColorBlack,
},
XValues: b,
YValues: ns,
}
graph1 := chart.Chart{
Width: 1280,
Height: 720,
Background: chart.Style{
Padding: chart.Box{
Top: 50,
},
},
YAxis: chart.YAxis{
Name: "operations",
NameStyle: chart.StyleShow(),
Style: chart.StyleShow(),
TickStyle: chart.Style{
TextRotationDegrees: 45.0,
},
ValueFormatter: func(v interface{}) string {
return fmt.Sprintf("%.1f", v.(float64))
},
GridMajorStyle: chart.Style{
Show: true,
StrokeColor: chart.ColorAlternateGray,
StrokeWidth: 1.0,
},
//GridLines: operations(),
},
XAxis: chart.XAxis{
Name: "Blocks, million",
Style: chart.Style{
Show: true,
},
ValueFormatter: func(v interface{}) string {
return fmt.Sprintf("%.3fm", v.(float64))
},
GridMajorStyle: chart.Style{
Show: true,
StrokeColor: chart.ColorAlternateGray,
StrokeWidth: 1.0,
},
GridLines: blockMillions(),
},
Series: []chart.Series{
totalSeries,
nakedSeries,
},
}
graph1.Elements = []chart.Renderable{chart.LegendThin(&graph1)}
buffer := bytes.NewBuffer([]byte{})
err = graph1.Render(chart.PNG, buffer)
check(err)
err = ioutil.WriteFile("naked_sloads.png", buffer.Bytes(), 0644)
check(err)
}
func naked_storage_vs_blockproof() {
swrFile, err := os.Open("/Volumes/tb41/turbo-geth/storage_read_writes.csv")
check(err)
defer swrFile.Close()
swrReader := csv.NewReader(bufio.NewReader(swrFile))
var blocks []float64
var totalSstores []float64
var nakedSstores []float64
var totalSloads []float64
var nakedSloads []float64
var totalNaked []float64
for records, _ := swrReader.Read(); records != nil; records, _ = swrReader.Read() {
blocks = append(blocks, parseFloat64(records[0])/1000000.0)
totalSstores = append(totalSstores, parseFloat64(records[1]))
nakedSstores = append(nakedSstores, parseFloat64(records[2]))
totalSloads = append(totalSloads, parseFloat64(records[3]))
nakedSloads = append(nakedSloads, parseFloat64(records[4]))
totalNaked = append(totalNaked, parseFloat64(records[2])+parseFloat64(records[4]))
}
file, err := os.Open("/Volumes/tb41/turbo-geth/stateless.csv")
check(err)
defer file.Close()
reader := csv.NewReader(bufio.NewReader(file))
var blocks2 []float64
var vals [18][]float64
for records, _ := reader.Read(); records != nil; records, _ = reader.Read() {
if len(records) < 16 {
break
}
blocks2 = append(blocks2, parseFloat64(records[0])/1000000.0)
for i := 0; i < 18; i++ {
cProofs := 4.0*parseFloat64(records[2]) + 32.0*parseFloat64(records[3]) + parseFloat64(records[11]) + parseFloat64(records[12])
proofs := 4.0*parseFloat64(records[7]) + 32.0*parseFloat64(records[8]) + parseFloat64(records[14]) + parseFloat64(records[15])
switch i {
case 1, 6:
vals[i] = append(vals[i], 4.0*parseFloat64(records[i+1]))
case 2, 7:
vals[i] = append(vals[i], 32.0*parseFloat64(records[i+1]))
case 15:
vals[i] = append(vals[i], cProofs)
case 16:
vals[i] = append(vals[i], proofs)
case 17:
vals[i] = append(vals[i], cProofs+proofs+parseFloat64(records[13]))
default:
vals[i] = append(vals[i], parseFloat64(records[i+1]))
}
}
}
limit := len(blocks)
if len(blocks2) < limit {
limit = len(blocks2)
}
var min float64 = 100000000.0
var max float64
for i := 0; i < limit; i++ {
if totalNaked[i] < min {
min = totalNaked[i]
}
if totalNaked[i] > max {
max = totalNaked[i]
}
}
fmt.Printf("limit: %d, min total naked: %f, max total naked: %f\n", limit, min, max)
colorByBlock := func(xr, yr chart.Range, index int, x, y float64) drawing.Color {
if index < 1000000 {
return chart.ColorGreen.WithAlpha(100)
} else if index < 2000000 {
return chart.ColorBlue.WithAlpha(100)
} else if index < 3000000 {
return chart.ColorRed.WithAlpha(100)
} else if index < 4000000 {
return chart.ColorBlack.WithAlpha(100)
} else if index < 5000000 {
return chart.ColorYellow.WithAlpha(100)
} else if index < 6000000 {
return chart.ColorOrange.WithAlpha(100)
} else {
return chart.ColorCyan.WithAlpha(100)
}
}
hashSeries := &chart.ContinuousSeries{
Name: "Block proof hashes (for contracts) vs naked SLOADs and naked SSTOREs",
Style: chart.Style{
Show: true,
StrokeWidth: chart.Disabled,
DotWidth: 1,
DotColorProvider: colorByBlock,
},
XValues: totalNaked[:limit],
YValues: vals[2][:limit],
}
graph1 := chart.Chart{
Width: 1280,
Height: 720,
Background: chart.Style{
Padding: chart.Box{
Top: 50,
},
},
XAxis: chart.XAxis{
Name: "operations",
NameStyle: chart.StyleShow(),
Style: chart.StyleShow(),
TickStyle: chart.Style{
TextRotationDegrees: 45.0,
},
ValueFormatter: func(v interface{}) string {
return fmt.Sprintf("%.1f", v.(float64))
},
GridMajorStyle: chart.Style{
Show: true,
StrokeColor: chart.ColorAlternateGray,
StrokeWidth: 1.0,
},
},
YAxis: chart.YAxis{
Name: "block proof hashes (contracts)",
Style: chart.Style{
Show: true,
},
ValueFormatter: func(v interface{}) string {
return fmt.Sprintf("%d kB", int(v.(float64)/1024.0))
},
GridMajorStyle: chart.Style{
Show: true,
StrokeColor: chart.ColorAlternateGray,
StrokeWidth: 1.0,
},
},
Series: []chart.Series{
hashSeries,
},
}
graph1.Elements = []chart.Renderable{chart.LegendThin(&graph1)}
buffer := bytes.NewBuffer([]byte{})
err = graph1.Render(chart.PNG, buffer)
check(err)
err = ioutil.WriteFile("naked_storage_vs_blockproof.png", buffer.Bytes(), 0644)
check(err)
}