forked from forks/go-ethereum
This PR uses various tweaks and tricks to make the stacktrie near
alloc-free.
```
[user@work go-ethereum]$ benchstat stacktrie.1 stacktrie.7
goos: linux
goarch: amd64
pkg: github.com/ethereum/go-ethereum/trie
cpu: 12th Gen Intel(R) Core(TM) i7-1270P
│ stacktrie.1 │ stacktrie.7 │
│ sec/op │ sec/op vs base │
Insert100K-8 106.97m ± 8% 88.21m ± 34% -17.54% (p=0.000 n=10)
│ stacktrie.1 │ stacktrie.7 │
│ B/op │ B/op vs base │
Insert100K-8 13199.608Ki ± 0% 3.424Ki ± 3% -99.97% (p=0.000 n=10)
│ stacktrie.1 │ stacktrie.7 │
│ allocs/op │ allocs/op vs base │
Insert100K-8 553428.50 ± 0% 22.00 ± 5% -100.00% (p=0.000 n=10)
```
Also improves derivesha:
```
goos: linux
goarch: amd64
pkg: github.com/ethereum/go-ethereum/core/types
cpu: 12th Gen Intel(R) Core(TM) i7-1270P
│ derivesha.1 │ derivesha.2 │
│ sec/op │ sec/op vs base │
DeriveSha200/stack_trie-8 477.8µ ± 2% 430.0µ ± 12% -10.00% (p=0.000 n=10)
│ derivesha.1 │ derivesha.2 │
│ B/op │ B/op vs base │
DeriveSha200/stack_trie-8 45.17Ki ± 0% 25.65Ki ± 0% -43.21% (p=0.000 n=10)
│ derivesha.1 │ derivesha.2 │
│ allocs/op │ allocs/op vs base │
DeriveSha200/stack_trie-8 1259.0 ± 0% 232.0 ± 0% -81.57% (p=0.000 n=10)
```
---------
Co-authored-by: Gary Rong <garyrong0905@gmail.com>
95 lines
2.2 KiB
Go
95 lines
2.2 KiB
Go
// Copyright 2022 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 trie
|
|
|
|
import (
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
)
|
|
|
|
func nodeToBytes(n node) []byte {
|
|
w := rlp.NewEncoderBuffer(nil)
|
|
n.encode(w)
|
|
result := w.ToBytes()
|
|
w.Flush()
|
|
return result
|
|
}
|
|
|
|
func (n *fullNode) encode(w rlp.EncoderBuffer) {
|
|
offset := w.List()
|
|
for _, c := range n.Children {
|
|
if c != nil {
|
|
c.encode(w)
|
|
} else {
|
|
w.Write(rlp.EmptyString)
|
|
}
|
|
}
|
|
w.ListEnd(offset)
|
|
}
|
|
|
|
func (n *fullnodeEncoder) encode(w rlp.EncoderBuffer) {
|
|
offset := w.List()
|
|
for _, c := range n.Children {
|
|
if c == nil {
|
|
w.Write(rlp.EmptyString)
|
|
} else if len(c) < 32 {
|
|
w.Write(c) // rawNode
|
|
} else {
|
|
w.WriteBytes(c) // hashNode
|
|
}
|
|
}
|
|
w.ListEnd(offset)
|
|
}
|
|
|
|
func (n *shortNode) encode(w rlp.EncoderBuffer) {
|
|
offset := w.List()
|
|
w.WriteBytes(n.Key)
|
|
if n.Val != nil {
|
|
n.Val.encode(w)
|
|
} else {
|
|
w.Write(rlp.EmptyString)
|
|
}
|
|
w.ListEnd(offset)
|
|
}
|
|
|
|
func (n *extNodeEncoder) encode(w rlp.EncoderBuffer) {
|
|
offset := w.List()
|
|
w.WriteBytes(n.Key)
|
|
|
|
if n.Val == nil {
|
|
w.Write(rlp.EmptyString)
|
|
} else if len(n.Val) < 32 {
|
|
w.Write(n.Val) // rawNode
|
|
} else {
|
|
w.WriteBytes(n.Val) // hashNode
|
|
}
|
|
w.ListEnd(offset)
|
|
}
|
|
|
|
func (n *leafNodeEncoder) encode(w rlp.EncoderBuffer) {
|
|
offset := w.List()
|
|
w.WriteBytes(n.Key) // Compact format key
|
|
w.WriteBytes(n.Val) // Value node, must be non-nil
|
|
w.ListEnd(offset)
|
|
}
|
|
|
|
func (n hashNode) encode(w rlp.EncoderBuffer) {
|
|
w.WriteBytes(n)
|
|
}
|
|
|
|
func (n valueNode) encode(w rlp.EncoderBuffer) {
|
|
w.WriteBytes(n)
|
|
}
|