mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-16 23:08:45 +00:00
b81eeec374
* 1st attemp to optimize blk fetch w/ att tgt fetch * test test test test
235 lines
7.1 KiB
Go
235 lines
7.1 KiB
Go
package ssz
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"reflect"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/karlseguin/ccache"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
)
|
|
|
|
var (
|
|
// ErrNotMerkleRoot will be returned when a cache object is not a merkle root.
|
|
ErrNotMerkleRoot = errors.New("object is not a merkle root")
|
|
// maxCacheSize is 2x of the follow distance for additional cache padding.
|
|
// Requests should be only accessing blocks within recent blocks within the
|
|
// Eth1FollowDistance.
|
|
maxCacheSize = params.BeaconConfig().HashCacheSize
|
|
// Metrics
|
|
hashCacheMiss = promauto.NewCounter(prometheus.CounterOpts{
|
|
Name: "hash_cache_miss",
|
|
Help: "The number of hash requests that aren't present in the cache.",
|
|
})
|
|
hashCacheHit = promauto.NewCounter(prometheus.CounterOpts{
|
|
Name: "hash_cache_hit",
|
|
Help: "The number of hash requests that are present in the cache.",
|
|
})
|
|
hashCacheSize = promauto.NewGauge(prometheus.GaugeOpts{
|
|
Name: "hash_cache_size",
|
|
Help: "The number of hashes in the block cache",
|
|
})
|
|
)
|
|
|
|
// hashCacheS struct with one queue for looking up by hash.
|
|
type hashCacheS struct {
|
|
hashCache *ccache.Cache
|
|
}
|
|
|
|
// root specifies the hash of data in a struct
|
|
type root struct {
|
|
Hash common.Hash
|
|
MerkleRoot []byte
|
|
}
|
|
|
|
// newHashCache creates a new hash cache for storing/accessing root hashes from
|
|
// memory.
|
|
func newHashCache() *hashCacheS {
|
|
return &hashCacheS{
|
|
hashCache: ccache.New(ccache.Configure().MaxSize(maxCacheSize)),
|
|
}
|
|
}
|
|
|
|
// RootByEncodedHash fetches Root by the encoded hash of the object. Returns true with a
|
|
// reference to the root if exists. Otherwise returns false, nil.
|
|
func (b *hashCacheS) RootByEncodedHash(h common.Hash) (bool, *root, error) {
|
|
item := b.hashCache.Get(h.Hex())
|
|
if item == nil {
|
|
hashCacheMiss.Inc()
|
|
return false, nil, nil
|
|
}
|
|
hashCacheHit.Inc()
|
|
hInfo, ok := item.Value().(*root)
|
|
if !ok {
|
|
return false, nil, ErrNotMerkleRoot
|
|
}
|
|
|
|
return true, hInfo, nil
|
|
}
|
|
|
|
// TrieRootCached computes a trie root and add it to the cache.
|
|
// if the encoded hash of the object is in cache, it will be retrieved from cache.
|
|
// This method also trims the least recently added root info. if the cache size
|
|
// has reached the max cache size limit.
|
|
func (b *hashCacheS) TrieRootCached(val interface{}) ([32]byte, error) {
|
|
if val == nil {
|
|
return [32]byte{}, newHashError("untyped nil is not supported", nil)
|
|
}
|
|
rval := reflect.ValueOf(val)
|
|
hs, err := hashedEncoding(rval)
|
|
if err != nil {
|
|
return [32]byte{}, newHashError(fmt.Sprint(err), rval.Type())
|
|
}
|
|
exists, fetchedInfo, err := b.RootByEncodedHash(bytesutil.ToBytes32(hs))
|
|
if err != nil {
|
|
return [32]byte{}, newHashError(fmt.Sprint(err), rval.Type())
|
|
}
|
|
var paddedOutput [32]byte
|
|
if exists {
|
|
paddedOutput = bytesutil.ToBytes32(fetchedInfo.MerkleRoot)
|
|
} else {
|
|
sszUtils, err := cachedSSZUtils(rval.Type())
|
|
if err != nil {
|
|
return [32]byte{}, newHashError(fmt.Sprint(err), rval.Type())
|
|
}
|
|
output, err := sszUtils.hasher(rval)
|
|
if err != nil {
|
|
return [32]byte{}, newHashError(fmt.Sprint(err), rval.Type())
|
|
}
|
|
// Right-pad with 0 to make 32 bytes long, if necessary.
|
|
paddedOutput = bytesutil.ToBytes32(output)
|
|
err = b.AddRoot(bytesutil.ToBytes32(hs), paddedOutput[:])
|
|
if err != nil {
|
|
return [32]byte{}, newHashError(fmt.Sprint(err), rval.Type())
|
|
}
|
|
}
|
|
return paddedOutput, nil
|
|
}
|
|
|
|
// MerkleHashCached adds a merkle object to the cache. This method also trims the
|
|
// least recently added root info if the cache size has reached the max cache
|
|
// size limit.
|
|
func (b *hashCacheS) MerkleHashCached(byteSlice [][]byte) ([]byte, error) {
|
|
mh := []byte{}
|
|
hs, err := hashedEncoding(reflect.ValueOf(byteSlice))
|
|
if err != nil {
|
|
return mh, newHashError(fmt.Sprint(err), reflect.TypeOf(byteSlice))
|
|
}
|
|
exists, fetchedInfo, err := b.RootByEncodedHash(bytesutil.ToBytes32(hs))
|
|
if err != nil {
|
|
return mh, newHashError(fmt.Sprint(err), reflect.TypeOf(byteSlice))
|
|
}
|
|
if exists {
|
|
mh = fetchedInfo.MerkleRoot
|
|
} else {
|
|
mh, err = merkleHash(byteSlice)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mr := &root{
|
|
Hash: bytesutil.ToBytes32(hs),
|
|
MerkleRoot: mh,
|
|
}
|
|
b.hashCache.Set(mr.Hash.Hex(), mr, time.Hour)
|
|
hashCacheSize.Set(float64(b.hashCache.ItemCount()))
|
|
}
|
|
|
|
return mh, nil
|
|
}
|
|
|
|
// AddRoot adds an encodedhash of the object as key and a rootHash object to the cache.
|
|
// This method also trims the
|
|
// least recently added root info if the cache size has reached the max cache
|
|
// size limit.
|
|
func (b *hashCacheS) AddRoot(h common.Hash, rootB []byte) error {
|
|
mr := &root{
|
|
Hash: h,
|
|
MerkleRoot: rootB,
|
|
}
|
|
b.hashCache.Set(mr.Hash.Hex(), mr, time.Hour)
|
|
return nil
|
|
}
|
|
|
|
// MakeSliceHasherCache add caching mechanism to slice hasher.
|
|
func makeSliceHasherCache(typ reflect.Type) (hasher, error) {
|
|
elemSSZUtils, err := cachedSSZUtilsNoAcquireLock(typ.Elem())
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get ssz utils: %v", err)
|
|
}
|
|
hasher := func(val reflect.Value) ([]byte, error) {
|
|
hs, err := hashedEncoding(val)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to encode element of slice/array: %v", err)
|
|
}
|
|
exists, fetchedInfo, err := hashCache.RootByEncodedHash(bytesutil.ToBytes32(hs))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to encode element of slice/array: %v", err)
|
|
}
|
|
var output []byte
|
|
if exists {
|
|
output = fetchedInfo.MerkleRoot
|
|
} else {
|
|
var elemHashList [][]byte
|
|
for i := 0; i < val.Len(); i++ {
|
|
elemHash, err := elemSSZUtils.hasher(val.Index(i))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to hash element of slice/array: %v", err)
|
|
}
|
|
elemHashList = append(elemHashList, elemHash)
|
|
}
|
|
output, err = hashCache.MerkleHashCached(elemHashList)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to calculate merkle hash of element hash list: %v", err)
|
|
}
|
|
err := hashCache.AddRoot(bytesutil.ToBytes32(hs), output)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to add root to cache: %v", err)
|
|
}
|
|
hashCacheSize.Set(float64(hashCache.hashCache.ItemCount()))
|
|
|
|
}
|
|
|
|
return output, nil
|
|
}
|
|
return hasher, nil
|
|
}
|
|
|
|
func makeStructHasherCache(typ reflect.Type) (hasher, error) {
|
|
fields, err := structFields(typ)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
hasher := func(val reflect.Value) ([]byte, error) {
|
|
hs, err := hashedEncoding(val)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to encode element of slice/array: %v", err)
|
|
}
|
|
exists, fetchedInfo, err := hashCache.RootByEncodedHash(bytesutil.ToBytes32(hs))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to encode element of slice/array: %v", err)
|
|
}
|
|
var result [32]byte
|
|
if exists {
|
|
result = bytesutil.ToBytes32(fetchedInfo.MerkleRoot)
|
|
return result[:], nil
|
|
}
|
|
concatElemHash := make([]byte, 0)
|
|
for _, f := range fields {
|
|
elemHash, err := f.sszUtils.hasher(val.Field(f.index))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to hash field of struct: %v", err)
|
|
}
|
|
concatElemHash = append(concatElemHash, elemHash...)
|
|
}
|
|
result = hashutil.Hash(concatElemHash)
|
|
return result[:], nil
|
|
}
|
|
return hasher, nil
|
|
}
|