erigon-pulse/cl/merkle_tree/list.go
Giulio rebuffo faf62a0c11
Caplin: Adding SSZ generics (#7542)
This PR makes the SSZ encodings less demented and more compact/flexible
overall
2023-05-23 20:58:34 +02:00

128 lines
3.2 KiB
Go

package merkle_tree
import (
"math/bits"
"github.com/prysmaticlabs/gohashtree"
"github.com/ledgerwatch/erigon-lib/common"
"github.com/ledgerwatch/erigon-lib/common/length"
"github.com/ledgerwatch/erigon-lib/types/ssz"
"github.com/ledgerwatch/erigon/cl/utils"
)
// MerkleizeVector uses our optimized routine to hash a list of 32-byte
// elements.
func MerkleizeVector(elements [][32]byte, length uint64) ([32]byte, error) {
depth := getDepth(length)
// Return zerohash at depth
if len(elements) == 0 {
return ZeroHashes[depth], nil
}
for i := uint8(0); i < depth; i++ {
// Sequential
layerLen := len(elements)
if layerLen%2 == 1 {
elements = append(elements, ZeroHashes[i])
}
outputLen := len(elements) / 2
if err := gohashtree.Hash(elements, elements); err != nil {
return [32]byte{}, err
}
elements = elements[:outputLen]
}
return elements[0], nil
}
// MerkleizeVector uses our optimized routine to hash a list of 32-byte
// elements.
func MerkleizeVectorFlat(in []byte, limit uint64) ([32]byte, error) {
elements := make([]byte, len(in))
copy(elements, in)
for i := uint8(0); i < getDepth(limit); i++ {
// Sequential
layerLen := len(elements)
if layerLen%64 == 32 {
elements = append(elements, ZeroHashes[i][:]...)
}
outputLen := len(elements) / 2
if err := HashByteSlice(elements, elements); err != nil {
return [32]byte{}, err
}
elements = elements[:outputLen]
}
return common.BytesToHash(elements[:length.Hash]), nil
}
// BitlistRootWithLimit computes the HashSSZ merkleization of
// participation roots.
func BitlistRootWithLimit(bits []byte, limit uint64) ([32]byte, error) {
var (
unpackedRoots []byte
size uint64
)
unpackedRoots, size = parseBitlist(unpackedRoots, bits)
roots := packBits(unpackedRoots)
base, err := MerkleizeVector(roots, (limit+255)/256)
if err != nil {
return [32]byte{}, err
}
lengthRoot := Uint64Root(size)
return utils.Keccak256(base[:], lengthRoot[:]), nil
}
func packBits(bytes []byte) [][32]byte {
var chunks [][32]byte
for i := 0; i < len(bytes); i += 32 {
var chunk [32]byte
copy(chunk[:], bytes[i:])
chunks = append(chunks, chunk)
}
return chunks
}
func parseBitlist(dst, buf []byte) ([]byte, uint64) {
msb := uint8(bits.Len8(buf[len(buf)-1])) - 1
size := uint64(8*(len(buf)-1) + int(msb))
dst = append(dst, buf...)
dst[len(dst)-1] &^= uint8(1 << msb)
newLen := len(dst)
for i := len(dst) - 1; i >= 0; i-- {
if dst[i] != 0x00 {
break
}
newLen = i
}
res := dst[:newLen]
return res, size
}
func TransactionsListRoot(transactions [][]byte) ([32]byte, error) {
return globalHasher.transactionsListRoot(transactions)
}
func ListObjectSSZRoot[T ssz.HashableSSZ](list []T, limit uint64) ([32]byte, error) {
globalHasher.mu2.Lock()
defer globalHasher.mu2.Unlock()
// due to go generics we cannot make a method for global hasher.
subLeaves := globalHasher.getBufferForSSZList(len(list))
for i, element := range list {
subLeaf, err := element.HashSSZ()
if err != nil {
return [32]byte{}, err
}
subLeaves[i] = subLeaf
}
vectorLeaf, err := MerkleizeVector(subLeaves, limit)
if err != nil {
return [32]byte{}, err
}
lenLeaf := Uint64Root(uint64(len(list)))
return utils.Keccak256(vectorLeaf[:], lenLeaf[:]), nil
}