mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-16 06:58:20 +00:00
d077483577
* v3 import renamings * tidy * fmt * rev * Update beacon-chain/core/epoch/precompute/reward_penalty_test.go * Update beacon-chain/core/helpers/validators_test.go * Update beacon-chain/db/alias.go * Update beacon-chain/db/alias.go * Update beacon-chain/db/alias.go * Update beacon-chain/db/iface/BUILD.bazel * Update beacon-chain/db/kv/kv.go * Update beacon-chain/db/kv/state.go * Update beacon-chain/rpc/prysm/v1alpha1/validator/attester_test.go * Update beacon-chain/rpc/prysm/v1alpha1/validator/attester_test.go * Update beacon-chain/sync/initial-sync/service.go * fix deps * fix bad replacements * fix bad replacements * change back * gohashtree version * fix deps Co-authored-by: Nishant Das <nishdas93@gmail.com> Co-authored-by: Potuz <potuz@prysmaticlabs.com>
342 lines
10 KiB
Go
342 lines
10 KiB
Go
package stateutil
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v3/config/features"
|
|
"github.com/prysmaticlabs/prysm/v3/container/trie"
|
|
"github.com/prysmaticlabs/prysm/v3/crypto/hash"
|
|
"github.com/prysmaticlabs/prysm/v3/crypto/hash/htr"
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/ssz"
|
|
"github.com/prysmaticlabs/prysm/v3/math"
|
|
)
|
|
|
|
// ReturnTrieLayer returns the representation of a merkle trie when
|
|
// provided with the elements of a fixed sized trie and the corresponding depth of
|
|
// it.
|
|
func ReturnTrieLayer(elements [][32]byte, length uint64) ([][]*[32]byte, error) {
|
|
hasher := hash.CustomSHA256Hasher()
|
|
leaves := elements
|
|
|
|
if len(leaves) == 1 {
|
|
return [][]*[32]byte{{&leaves[0]}}, nil
|
|
}
|
|
hashLayer := leaves
|
|
layers := make([][][32]byte, ssz.Depth(length)+1)
|
|
layers[0] = hashLayer
|
|
var err error
|
|
layers, _, err = MerkleizeTrieLeaves(layers, hashLayer, hasher)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
refLayers := make([][]*[32]byte, len(layers))
|
|
for i, val := range layers {
|
|
refLayers[i] = make([]*[32]byte, len(val))
|
|
for j, innerVal := range val {
|
|
newVal := innerVal
|
|
refLayers[i][j] = &newVal
|
|
}
|
|
}
|
|
return refLayers, nil
|
|
}
|
|
|
|
// ReturnTrieLayerVariable returns the representation of a merkle trie when
|
|
// provided with the elements of a variable sized trie and the corresponding depth of
|
|
// it.
|
|
func ReturnTrieLayerVariable(elements [][32]byte, length uint64) [][]*[32]byte {
|
|
hasher := hash.CustomSHA256Hasher()
|
|
depth := ssz.Depth(length)
|
|
layers := make([][]*[32]byte, depth+1)
|
|
// Return zerohash at depth
|
|
if len(elements) == 0 {
|
|
zerohash := trie.ZeroHashes[depth]
|
|
layers[len(layers)-1] = []*[32]byte{&zerohash}
|
|
return layers
|
|
}
|
|
transformedLeaves := make([]*[32]byte, len(elements))
|
|
for i := range elements {
|
|
arr := elements[i]
|
|
transformedLeaves[i] = &arr
|
|
}
|
|
layers[0] = transformedLeaves
|
|
buffer := bytes.NewBuffer([]byte{})
|
|
buffer.Grow(64)
|
|
|
|
for i := uint8(0); i < depth; i++ {
|
|
layerLen := len(layers[i])
|
|
oddNodeLength := layerLen%2 == 1
|
|
if features.Get().EnableVectorizedHTR {
|
|
if oddNodeLength {
|
|
zerohash := trie.ZeroHashes[i]
|
|
elements = append(elements, zerohash)
|
|
layerLen++
|
|
}
|
|
|
|
layers[i+1] = make([]*[32]byte, layerLen/2)
|
|
newElems := make([][32]byte, layerLen/2)
|
|
htr.VectorizedSha256(elements, newElems)
|
|
elements = newElems
|
|
for j := range elements {
|
|
layers[i+1][j] = &elements[j]
|
|
}
|
|
} else {
|
|
if oddNodeLength {
|
|
zerohash := trie.ZeroHashes[i]
|
|
layers[i] = append(layers[i], &zerohash)
|
|
}
|
|
updatedValues := make([]*[32]byte, 0, len(layers[i])/2)
|
|
for j := 0; j < len(layers[i]); j += 2 {
|
|
buffer.Write(layers[i][j][:])
|
|
buffer.Write(layers[i][j+1][:])
|
|
concat := hasher(buffer.Bytes())
|
|
updatedValues = append(updatedValues, &concat)
|
|
buffer.Reset()
|
|
}
|
|
// remove zerohash node from tree
|
|
if oddNodeLength {
|
|
layers[i] = layers[i][:len(layers[i])-1]
|
|
}
|
|
layers[i+1] = updatedValues
|
|
}
|
|
}
|
|
return layers
|
|
}
|
|
|
|
// RecomputeFromLayer recomputes specific branches of a fixed sized trie depending on the provided changed indexes.
|
|
func RecomputeFromLayer(changedLeaves [][32]byte, changedIdx []uint64, layer [][]*[32]byte) ([32]byte, [][]*[32]byte, error) {
|
|
hasher := hash.CustomSHA256Hasher()
|
|
for i, idx := range changedIdx {
|
|
layer[0][idx] = &changedLeaves[i]
|
|
}
|
|
|
|
if len(changedIdx) == 0 {
|
|
return *layer[0][0], layer, nil
|
|
}
|
|
|
|
leaves := layer[0]
|
|
|
|
// We need to ensure we recompute indices of the Merkle tree which
|
|
// changed in-between calls to this function. This check adds an offset
|
|
// to the recomputed indices to ensure we do so evenly.
|
|
maxChangedIndex := changedIdx[len(changedIdx)-1]
|
|
if int(maxChangedIndex+2) == len(leaves) && maxChangedIndex%2 != 0 {
|
|
changedIdx = append(changedIdx, maxChangedIndex+1)
|
|
}
|
|
|
|
root := *layer[0][0]
|
|
|
|
for _, idx := range changedIdx {
|
|
ii, err := math.Int(idx)
|
|
if err != nil {
|
|
return [32]byte{}, nil, err
|
|
}
|
|
root, layer, err = recomputeRootFromLayer(ii, layer, leaves, hasher)
|
|
if err != nil {
|
|
return [32]byte{}, nil, err
|
|
}
|
|
}
|
|
return root, layer, nil
|
|
}
|
|
|
|
// RecomputeFromLayerVariable recomputes specific branches of a variable sized trie depending on the provided changed indexes.
|
|
func RecomputeFromLayerVariable(changedLeaves [][32]byte, changedIdx []uint64, layer [][]*[32]byte) ([32]byte, [][]*[32]byte, error) {
|
|
hasher := hash.CustomSHA256Hasher()
|
|
if len(changedIdx) == 0 {
|
|
return *layer[0][0], layer, nil
|
|
}
|
|
root := *layer[len(layer)-1][0]
|
|
|
|
for i, idx := range changedIdx {
|
|
ii, err := math.Int(idx)
|
|
if err != nil {
|
|
return [32]byte{}, nil, err
|
|
}
|
|
root, layer, err = recomputeRootFromLayerVariable(ii, changedLeaves[i], layer, hasher)
|
|
if err != nil {
|
|
return [32]byte{}, nil, err
|
|
}
|
|
}
|
|
return root, layer, nil
|
|
}
|
|
|
|
// this method assumes that the provided trie already has all its elements included
|
|
// in the base depth.
|
|
func recomputeRootFromLayer(idx int, layers [][]*[32]byte, chunks []*[32]byte,
|
|
hasher func([]byte) [32]byte) ([32]byte, [][]*[32]byte, error) {
|
|
root := *chunks[idx]
|
|
layers[0] = chunks
|
|
// The merkle tree structure looks as follows:
|
|
// [[r1, r2, r3, r4], [parent1, parent2], [root]]
|
|
// Using information about the index which changed, idx, we recompute
|
|
// only its branch up the tree.
|
|
currentIndex := idx
|
|
// Allocate only once.
|
|
combinedChunks := [64]byte{}
|
|
for i := 0; i < len(layers)-1; i++ {
|
|
isLeft := currentIndex%2 == 0
|
|
neighborIdx := currentIndex ^ 1
|
|
|
|
neighbor := [32]byte{}
|
|
if layers[i] != nil && len(layers[i]) != 0 && neighborIdx < len(layers[i]) {
|
|
neighbor = *layers[i][neighborIdx]
|
|
}
|
|
if isLeft {
|
|
copy(combinedChunks[:32], root[:])
|
|
copy(combinedChunks[32:], neighbor[:])
|
|
} else {
|
|
copy(combinedChunks[:32], neighbor[:])
|
|
copy(combinedChunks[32:], root[:])
|
|
}
|
|
|
|
parentHash := hasher(combinedChunks[:])
|
|
root = parentHash
|
|
|
|
parentIdx := currentIndex / 2
|
|
// Update the cached layers at the parent index.
|
|
rootVal := root
|
|
if len(layers[i+1]) == 0 {
|
|
layers[i+1] = append(layers[i+1], &rootVal)
|
|
} else {
|
|
layers[i+1][parentIdx] = &rootVal
|
|
}
|
|
currentIndex = parentIdx
|
|
}
|
|
// If there is only a single leaf, we return it (the identity element).
|
|
if len(layers[0]) == 1 {
|
|
return *layers[0][0], layers, nil
|
|
}
|
|
return root, layers, nil
|
|
}
|
|
|
|
// this method assumes that the base branch does not consist of all leaves of the
|
|
// trie. Instead missing leaves are assumed to be zerohashes, following the structure
|
|
// of a sparse merkle trie.
|
|
func recomputeRootFromLayerVariable(idx int, item [32]byte, layers [][]*[32]byte,
|
|
hasher func([]byte) [32]byte) ([32]byte, [][]*[32]byte, error) {
|
|
for idx >= len(layers[0]) {
|
|
zerohash := trie.ZeroHashes[0]
|
|
layers[0] = append(layers[0], &zerohash)
|
|
}
|
|
layers[0][idx] = &item
|
|
|
|
currentIndex := idx
|
|
root := item
|
|
// Allocate only once.
|
|
neighbor := [32]byte{}
|
|
combinedChunks := [64]byte{}
|
|
|
|
for i := 0; i < len(layers)-1; i++ {
|
|
isLeft := currentIndex%2 == 0
|
|
neighborIdx := currentIndex ^ 1
|
|
|
|
if neighborIdx >= len(layers[i]) {
|
|
neighbor = trie.ZeroHashes[i]
|
|
} else {
|
|
neighbor = *layers[i][neighborIdx]
|
|
}
|
|
if isLeft {
|
|
copy(combinedChunks[:32], root[:])
|
|
copy(combinedChunks[32:], neighbor[:])
|
|
} else {
|
|
copy(combinedChunks[:32], neighbor[:])
|
|
copy(combinedChunks[32:], root[:])
|
|
}
|
|
|
|
parentHash := hasher(combinedChunks[:])
|
|
root = parentHash
|
|
|
|
parentIdx := currentIndex / 2
|
|
if len(layers[i+1]) == 0 || parentIdx >= len(layers[i+1]) {
|
|
newItem := root
|
|
layers[i+1] = append(layers[i+1], &newItem)
|
|
} else {
|
|
newItem := root
|
|
layers[i+1][parentIdx] = &newItem
|
|
}
|
|
currentIndex = parentIdx
|
|
}
|
|
return root, layers, nil
|
|
}
|
|
|
|
// AddInMixin describes a method from which a lenth mixin is added to the
|
|
// provided root.
|
|
func AddInMixin(root [32]byte, length uint64) ([32]byte, error) {
|
|
rootBuf := new(bytes.Buffer)
|
|
if err := binary.Write(rootBuf, binary.LittleEndian, length); err != nil {
|
|
return [32]byte{}, errors.Wrap(err, "could not marshal eth1data votes length")
|
|
}
|
|
// We need to mix in the length of the slice.
|
|
rootBufRoot := make([]byte, 32)
|
|
copy(rootBufRoot, rootBuf.Bytes())
|
|
return ssz.MixInLength(root, rootBufRoot), nil
|
|
}
|
|
|
|
// Merkleize 32-byte leaves into a Merkle trie for its adequate depth, returning
|
|
// the resulting layers of the trie based on the appropriate depth. This function
|
|
// pads the leaves to a length of 32.
|
|
func Merkleize(leaves [][]byte) [][][]byte {
|
|
hashFunc := hash.CustomSHA256Hasher()
|
|
layers := make([][][]byte, ssz.Depth(uint64(len(leaves)))+1)
|
|
for len(leaves) != 32 {
|
|
leaves = append(leaves, make([]byte, 32))
|
|
}
|
|
currentLayer := leaves
|
|
layers[0] = currentLayer
|
|
|
|
// We keep track of the hash layers of a Merkle trie until we reach
|
|
// the top layer of length 1, which contains the single root element.
|
|
// [Root] -> Top layer has length 1.
|
|
// [E] [F] -> This layer has length 2.
|
|
// [A] [B] [C] [D] -> The bottom layer has length 4 (needs to be a power of two).
|
|
i := 1
|
|
for len(currentLayer) > 1 && i < len(layers) {
|
|
layer := make([][]byte, 0)
|
|
for i := 0; i < len(currentLayer); i += 2 {
|
|
hashedChunk := hashFunc(append(currentLayer[i], currentLayer[i+1]...))
|
|
layer = append(layer, hashedChunk[:])
|
|
}
|
|
currentLayer = layer
|
|
layers[i] = currentLayer
|
|
i++
|
|
}
|
|
return layers
|
|
}
|
|
|
|
// MerkleizeTrieLeaves merkleize the trie leaves.
|
|
func MerkleizeTrieLeaves(layers [][][32]byte, hashLayer [][32]byte,
|
|
hasher func([]byte) [32]byte) ([][][32]byte, [][32]byte, error) {
|
|
// We keep track of the hash layers of a Merkle trie until we reach
|
|
// the top layer of length 1, which contains the single root element.
|
|
// [Root] -> Top layer has length 1.
|
|
// [E] [F] -> This layer has length 2.
|
|
// [A] [B] [C] [D] -> The bottom layer has length 4 (needs to be a power of two).
|
|
i := 1
|
|
chunkBuffer := bytes.NewBuffer([]byte{})
|
|
chunkBuffer.Grow(64)
|
|
for len(hashLayer) > 1 && i < len(layers) {
|
|
if !math.IsPowerOf2(uint64(len(hashLayer))) {
|
|
return nil, nil, errors.Errorf("hash layer is a non power of 2: %d", len(hashLayer))
|
|
}
|
|
if features.Get().EnableVectorizedHTR {
|
|
newLayer := make([][32]byte, len(hashLayer)/2)
|
|
htr.VectorizedSha256(hashLayer, newLayer)
|
|
hashLayer = newLayer
|
|
} else {
|
|
layer := make([][32]byte, len(hashLayer)/2)
|
|
for j := 0; j < len(hashLayer); j += 2 {
|
|
chunkBuffer.Write(hashLayer[j][:])
|
|
chunkBuffer.Write(hashLayer[j+1][:])
|
|
hashedChunk := hasher(chunkBuffer.Bytes())
|
|
layer[j/2] = hashedChunk
|
|
chunkBuffer.Reset()
|
|
}
|
|
hashLayer = layer
|
|
}
|
|
layers[i] = hashLayer
|
|
i++
|
|
}
|
|
return layers, hashLayer, nil
|
|
}
|