prysm-pulse/shared/ssz/hash_cache.go

235 lines
7.1 KiB
Go
Raw Normal View History

Add Caching to Tree Hashing Algorithm (#1929) * added todo to hash file in ssz * params and copy of block cache * start hash cache * Hash cache implementation * fixed some comments * fixed promatheus duplicate counter name * removed TODO * change to use special expiration cache * function name fixes junk object generator * naming changes * gazzle fix * added pruning last read data test * fixed gometallinter errors * fix benchmarks and no int64 not serializable * move struct from test * add feature flag * fix merge issues * add featureflag to beacon and validator * featureflag init for tests * added feature flag to all ssz dependent tests * remove setter func * replace k8s tweaked expiration cache to https://github.com/karlseguin/ccache * remove else * change request by preston * added init featureflags to genesis_test * Update shared/ssz/hash_cache.go add dot Co-Authored-By: shayzluf <thezluf@gmail.com> * Update shared/ssz/hash_cache.go Co-Authored-By: shayzluf <thezluf@gmail.com> * Update shared/ssz/hash_cache.go remove extra space Co-Authored-By: shayzluf <thezluf@gmail.com> * Update shared/params/config.go add dot Co-Authored-By: shayzluf <thezluf@gmail.com> * Update shared/featureconfig/config.go remove dot Co-Authored-By: shayzluf <thezluf@gmail.com> * Update shared/featureconfig/config.go remove dot Co-Authored-By: shayzluf <thezluf@gmail.com> * remove powchain from prometheus hash cache name * fixes fron change requests * fix change requests * remove faulty merge test * gazelle fix * fix fmt.sprintf * remove debug binary * fix gazelle
2019-04-24 05:39:02 +00:00
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(error.Error(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
}