2021-03-18 23:29:06 +00:00
|
|
|
package stateutil
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2022-08-16 12:20:13 +00:00
|
|
|
fieldparams "github.com/prysmaticlabs/prysm/v3/config/fieldparams"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/crypto/hash"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/ssz"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1"
|
2021-03-18 23:29:06 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// ValidatorRootWithHasher describes a method from which the hash tree root
|
|
|
|
// of a validator is returned.
|
2021-09-21 15:02:48 +00:00
|
|
|
func ValidatorRootWithHasher(hasher ssz.HashFn, validator *ethpb.Validator) ([32]byte, error) {
|
2022-02-28 13:56:12 +00:00
|
|
|
fieldRoots, err := ValidatorFieldRoots(hasher, validator)
|
|
|
|
if err != nil {
|
|
|
|
return [32]byte{}, err
|
|
|
|
}
|
2022-03-04 15:19:07 +00:00
|
|
|
return ssz.BitwiseMerkleize(hasher, fieldRoots, uint64(len(fieldRoots)), uint64(len(fieldRoots)))
|
2022-02-28 13:56:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ValidatorFieldRoots describes a method from which the hash tree root
|
|
|
|
// of a validator is returned.
|
|
|
|
func ValidatorFieldRoots(hasher ssz.HashFn, validator *ethpb.Validator) ([][32]byte, error) {
|
2021-03-18 23:29:06 +00:00
|
|
|
var fieldRoots [][32]byte
|
|
|
|
if validator != nil {
|
|
|
|
pubkey := bytesutil.ToBytes48(validator.PublicKey)
|
|
|
|
withdrawCreds := bytesutil.ToBytes32(validator.WithdrawalCredentials)
|
2022-12-22 09:20:10 +00:00
|
|
|
var effectiveBalanceBuf [32]byte
|
2021-03-18 23:29:06 +00:00
|
|
|
binary.LittleEndian.PutUint64(effectiveBalanceBuf[:8], validator.EffectiveBalance)
|
|
|
|
// Slashed.
|
2022-12-22 09:20:10 +00:00
|
|
|
var slashBuf [32]byte
|
2021-03-18 23:29:06 +00:00
|
|
|
if validator.Slashed {
|
|
|
|
slashBuf[0] = uint8(1)
|
|
|
|
} else {
|
|
|
|
slashBuf[0] = uint8(0)
|
|
|
|
}
|
2022-12-22 09:20:10 +00:00
|
|
|
var activationEligibilityBuf [32]byte
|
2021-03-18 23:29:06 +00:00
|
|
|
binary.LittleEndian.PutUint64(activationEligibilityBuf[:8], uint64(validator.ActivationEligibilityEpoch))
|
|
|
|
|
2022-12-22 09:20:10 +00:00
|
|
|
var activationBuf [32]byte
|
2021-03-18 23:29:06 +00:00
|
|
|
binary.LittleEndian.PutUint64(activationBuf[:8], uint64(validator.ActivationEpoch))
|
|
|
|
|
2022-12-22 09:20:10 +00:00
|
|
|
var exitBuf [32]byte
|
2021-03-18 23:29:06 +00:00
|
|
|
binary.LittleEndian.PutUint64(exitBuf[:8], uint64(validator.ExitEpoch))
|
|
|
|
|
2022-12-22 09:20:10 +00:00
|
|
|
var withdrawalBuf [32]byte
|
2021-03-18 23:29:06 +00:00
|
|
|
binary.LittleEndian.PutUint64(withdrawalBuf[:8], uint64(validator.WithdrawableEpoch))
|
|
|
|
|
|
|
|
// Public key.
|
2022-03-04 15:19:07 +00:00
|
|
|
pubKeyRoot, err := merkleizePubkey(hasher, pubkey[:])
|
2021-03-18 23:29:06 +00:00
|
|
|
if err != nil {
|
2022-02-28 13:56:12 +00:00
|
|
|
return [][32]byte{}, err
|
2021-03-18 23:29:06 +00:00
|
|
|
}
|
|
|
|
fieldRoots = [][32]byte{pubKeyRoot, withdrawCreds, effectiveBalanceBuf, slashBuf, activationEligibilityBuf,
|
|
|
|
activationBuf, exitBuf, withdrawalBuf}
|
|
|
|
}
|
2022-02-28 13:56:12 +00:00
|
|
|
return fieldRoots, nil
|
2021-03-18 23:29:06 +00:00
|
|
|
}
|
|
|
|
|
2021-03-23 21:17:09 +00:00
|
|
|
// Uint64ListRootWithRegistryLimit computes the HashTreeRoot Merkleization of
|
|
|
|
// a list of uint64 and mixed with registry limit.
|
|
|
|
func Uint64ListRootWithRegistryLimit(balances []uint64) ([32]byte, error) {
|
2021-09-15 22:55:11 +00:00
|
|
|
hasher := hash.CustomSHA256Hasher()
|
2022-06-07 06:34:13 +00:00
|
|
|
balancesChunks, err := PackUint64IntoChunks(balances)
|
2021-03-18 23:29:06 +00:00
|
|
|
if err != nil {
|
|
|
|
return [32]byte{}, errors.Wrap(err, "could not pack balances into chunks")
|
|
|
|
}
|
2022-08-16 16:19:01 +00:00
|
|
|
balancesRootsRoot, err := ssz.BitwiseMerkleize(hasher, balancesChunks, uint64(len(balancesChunks)), ValidatorLimitForBalancesChunks())
|
2021-03-18 23:29:06 +00:00
|
|
|
if err != nil {
|
|
|
|
return [32]byte{}, errors.Wrap(err, "could not compute balances merkleization")
|
|
|
|
}
|
2021-05-10 22:39:53 +00:00
|
|
|
|
|
|
|
balancesLengthRoot := make([]byte, 32)
|
|
|
|
binary.LittleEndian.PutUint64(balancesLengthRoot, uint64(len(balances)))
|
2021-09-21 15:02:48 +00:00
|
|
|
return ssz.MixInLength(balancesRootsRoot, balancesLengthRoot), nil
|
2021-03-18 23:29:06 +00:00
|
|
|
}
|
2022-06-07 06:34:13 +00:00
|
|
|
|
2022-08-16 16:19:01 +00:00
|
|
|
// ValidatorLimitForBalancesChunks returns the limit of validators after going through the chunking process.
|
|
|
|
func ValidatorLimitForBalancesChunks() uint64 {
|
|
|
|
maxValidatorLimit := uint64(fieldparams.ValidatorRegistryLimit)
|
|
|
|
bytesInUint64 := uint64(8)
|
|
|
|
return (maxValidatorLimit*bytesInUint64 + 31) / 32 // round to nearest chunk
|
|
|
|
}
|
|
|
|
|
2022-06-07 06:34:13 +00:00
|
|
|
// PackUint64IntoChunks packs a list of uint64 values into 32 byte roots.
|
|
|
|
func PackUint64IntoChunks(vals []uint64) ([][32]byte, error) {
|
|
|
|
// Initialize how many uint64 values we can pack
|
|
|
|
// into a single chunk(32 bytes). Each uint64 value
|
|
|
|
// would take up 8 bytes.
|
|
|
|
numOfElems := 4
|
|
|
|
sizeOfElem := 32 / numOfElems
|
|
|
|
// Determine total number of chunks to be
|
|
|
|
// allocated to provided list of unsigned
|
|
|
|
// 64-bit integers.
|
|
|
|
numOfChunks := len(vals) / numOfElems
|
|
|
|
// Add an extra chunk if the list size
|
|
|
|
// is not a perfect multiple of the number
|
|
|
|
// of elements.
|
|
|
|
if len(vals)%numOfElems != 0 {
|
|
|
|
numOfChunks++
|
|
|
|
}
|
|
|
|
chunkList := make([][32]byte, numOfChunks)
|
|
|
|
for idx, b := range vals {
|
|
|
|
// In order to determine how to pack in the uint64 value by index into
|
|
|
|
// our chunk list we need to determine a few things.
|
|
|
|
// 1) The chunk which the particular uint64 value corresponds to.
|
|
|
|
// 2) The position of the value in the chunk itself.
|
|
|
|
//
|
|
|
|
// Once we have determined these 2 values we can simply find the correct
|
|
|
|
// section of contiguous bytes to insert the value in the chunk.
|
|
|
|
chunkIdx := idx / numOfElems
|
|
|
|
idxInChunk := idx % numOfElems
|
|
|
|
chunkPos := idxInChunk * sizeOfElem
|
|
|
|
binary.LittleEndian.PutUint64(chunkList[chunkIdx][chunkPos:chunkPos+sizeOfElem], b)
|
|
|
|
}
|
|
|
|
return chunkList, nil
|
|
|
|
}
|