2021-01-25 19:45:48 +01:00
|
|
|
# beacon_chain
|
2022-01-06 11:25:35 +00:00
|
|
|
# Copyright (c) 2018-2022 Status Research & Development GmbH
|
2021-01-25 19:45:48 +01:00
|
|
|
# Licensed and distributed under either of
|
|
|
|
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
|
|
|
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
2022-07-29 12:53:42 +02:00
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
else:
|
|
|
|
{.push raises: [].}
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2021-12-09 13:56:54 +01:00
|
|
|
## This module contains signature verification helpers corresponding to those
|
|
|
|
## in signatures.nim, for use with signature sets / batch signature verification
|
|
|
|
## The functions follow the same structure and use the same arguments, except
|
|
|
|
## that the flow is split into separate collection and verification steps.
|
|
|
|
|
2021-01-25 19:45:48 +01:00
|
|
|
import
|
|
|
|
# Status lib
|
|
|
|
blscurve,
|
2021-06-01 13:13:40 +02:00
|
|
|
stew/[byteutils, results],
|
2021-12-06 10:49:01 +01:00
|
|
|
taskpools,
|
2022-06-21 10:29:16 +02:00
|
|
|
bearssl/rand,
|
2021-01-25 19:45:48 +01:00
|
|
|
# Internal
|
2021-12-09 13:56:54 +01:00
|
|
|
"."/[helpers, beaconstate, forks, signatures],
|
2022-01-06 11:25:35 +00:00
|
|
|
"."/datatypes/[altair, bellatrix, phase0]
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2022-06-21 10:29:16 +02:00
|
|
|
export results, rand, altair, phase0, taskpools, signatures
|
2021-12-06 10:49:01 +01:00
|
|
|
|
|
|
|
type
|
2022-04-08 18:22:49 +02:00
|
|
|
TaskPoolPtr* = Taskpool
|
2021-12-06 10:49:01 +01:00
|
|
|
|
|
|
|
BatchVerifier* = object
|
|
|
|
sigVerifCache*: BatchedBLSVerifierCache ##\
|
|
|
|
## A cache for batch BLS signature verification contexts
|
2022-06-21 10:29:16 +02:00
|
|
|
rng*: ref HmacDrbgContext ##\
|
2021-12-06 10:49:01 +01:00
|
|
|
## A reference to the Nimbus application-wide RNG
|
|
|
|
|
|
|
|
taskpool*: TaskPoolPtr
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
func `$`*(s: SignatureSet): string =
|
|
|
|
"(pubkey: 0x" & s.pubkey.toHex() &
|
|
|
|
", signing_root: 0x" & s.message.toHex() &
|
|
|
|
", signature: 0x" & s.signature.toHex() & ')'
|
|
|
|
|
|
|
|
# Important:
|
|
|
|
# - Due to lazy loading, when we do crypto verification
|
|
|
|
# and only then state-transition verification,
|
|
|
|
# there is no guarantee that pubkeys and signatures received are valid
|
|
|
|
# unlike when Nimbus did eager loading which ensured they were correct beforehand
|
|
|
|
|
2021-06-01 13:13:40 +02:00
|
|
|
template loadOrExit(signature: ValidatorSig, error: cstring):
|
|
|
|
untyped =
|
2021-01-25 19:45:48 +01:00
|
|
|
## Load a BLS signature from a raw signature
|
2021-04-09 12:59:24 +00:00
|
|
|
## Exits the **caller** with false if the signature is invalid
|
2021-01-25 19:45:48 +01:00
|
|
|
let sig = signature.load()
|
|
|
|
if sig.isNone:
|
2021-06-01 13:13:40 +02:00
|
|
|
return err(error) # this exits the calling scope, as templates are inlined.
|
2021-01-25 19:45:48 +01:00
|
|
|
sig.unsafeGet()
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
func init(T: type SignatureSet,
|
|
|
|
pubkey: CookedPubKey, signing_root: Eth2Digest,
|
|
|
|
signature: CookedSig): T =
|
2021-01-25 19:45:48 +01:00
|
|
|
## Add a new signature set triplet (pubkey, message, signature)
|
|
|
|
## to a collection of signature sets for batch verification.
|
2021-12-29 15:28:40 +01:00
|
|
|
(
|
2021-06-01 13:13:40 +02:00
|
|
|
blscurve.PublicKey(pubkey),
|
2021-12-09 13:56:54 +01:00
|
|
|
signing_root.data,
|
2021-04-26 22:39:44 +02:00
|
|
|
blscurve.Signature(signature)
|
2021-12-29 15:28:40 +01:00
|
|
|
)
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
proc aggregateAttesters(
|
2021-12-09 13:56:54 +01:00
|
|
|
validatorIndices: openArray[uint64|ValidatorIndex],
|
2021-06-10 09:37:02 +02:00
|
|
|
validatorKeys: auto,
|
2021-06-01 13:13:40 +02:00
|
|
|
): Result[CookedPubKey, cstring] =
|
|
|
|
if validatorIndices.len == 0:
|
|
|
|
# Aggregation spec requires non-empty collection
|
|
|
|
# - https://tools.ietf.org/html/draft-irtf-cfrg-bls-signature-04
|
|
|
|
# Eth2 spec requires at least one attesting index in attestation
|
2022-11-08 02:37:28 +00:00
|
|
|
# - https://github.com/ethereum/consensus-specs/blob/v1.3.0-alpha.0/specs/phase0/beacon-chain.md#is_valid_indexed_attestation
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("aggregateAttesters: no attesting indices")
|
|
|
|
|
2021-06-10 09:37:02 +02:00
|
|
|
let
|
|
|
|
firstKey = validatorKeys.load(validatorIndices[0])
|
|
|
|
|
|
|
|
if not firstKey.isSome():
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("aggregateAttesters: invalid attesting index")
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2022-02-16 22:24:44 +00:00
|
|
|
var attestersAgg{.noinit.}: AggregatePublicKey
|
2021-06-10 09:37:02 +02:00
|
|
|
|
|
|
|
attestersAgg.init(firstKey.get())
|
2021-06-01 13:13:40 +02:00
|
|
|
for i in 1 ..< validatorIndices.len:
|
2021-06-10 09:37:02 +02:00
|
|
|
let key = validatorKeys.load(validatorIndices[i])
|
|
|
|
if not key.isSome():
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("aggregateAttesters: invalid attesting index")
|
2021-06-10 09:37:02 +02:00
|
|
|
attestersAgg.aggregate(key.get())
|
2021-04-02 16:36:43 +02:00
|
|
|
|
2021-06-01 13:13:40 +02:00
|
|
|
ok(finish(attestersAgg))
|
2021-04-02 16:36:43 +02:00
|
|
|
|
2021-12-09 13:56:54 +01:00
|
|
|
proc aggregateAttesters(
|
|
|
|
validatorIndices: openArray[uint64|ValidatorIndex],
|
|
|
|
bits: auto,
|
2021-06-10 09:37:02 +02:00
|
|
|
validatorKeys: auto,
|
2021-12-09 13:56:54 +01:00
|
|
|
): Result[CookedPubKey, cstring] =
|
|
|
|
if validatorIndices.len == 0:
|
|
|
|
# Aggregation spec requires non-empty collection
|
|
|
|
# - https://tools.ietf.org/html/draft-irtf-cfrg-bls-signature-04
|
|
|
|
# Eth2 spec requires at least one attesting index in attestation
|
2022-11-08 02:37:28 +00:00
|
|
|
# - https://github.com/ethereum/consensus-specs/blob/v1.3.0-alpha.0/specs/phase0/beacon-chain.md#is_valid_indexed_attestation
|
2021-12-09 13:56:54 +01:00
|
|
|
return err("aggregateAttesters: no attesting indices")
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2022-02-16 22:24:44 +00:00
|
|
|
var attestersAgg{.noinit.}: AggregatePublicKey
|
2021-06-11 17:51:46 +00:00
|
|
|
|
2021-04-26 22:39:44 +02:00
|
|
|
var inited = false
|
2021-12-09 13:56:54 +01:00
|
|
|
for i in 0..<bits.len:
|
|
|
|
if bits[i]:
|
|
|
|
let key = validatorKeys.load(validatorIndices[i])
|
|
|
|
if not key.isSome():
|
|
|
|
return err("aggregateAttesters: invalid attesting index")
|
|
|
|
if inited:
|
|
|
|
attestersAgg.aggregate(key.get())
|
|
|
|
else:
|
|
|
|
attestersAgg = AggregatePublicKey.init(key.get)
|
|
|
|
inited = true
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2021-04-26 22:39:44 +02:00
|
|
|
if not inited:
|
2021-12-09 13:56:54 +01:00
|
|
|
return err("aggregateAttesters:no attesting indices")
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2021-12-09 13:56:54 +01:00
|
|
|
ok(finish(attestersAgg))
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2021-04-02 16:36:43 +02:00
|
|
|
# Public API
|
|
|
|
# ------------------------------------------------------
|
|
|
|
|
2021-12-09 13:56:54 +01:00
|
|
|
# See also: verify_slot_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc slot_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest, slot: Slot,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_slot_signing_root(
|
|
|
|
fork, genesis_validators_root, slot)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_epoch_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc epoch_signature_set*(
|
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest, epoch: Epoch,
|
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_epoch_signing_root(
|
|
|
|
fork, genesis_validators_root, epoch)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_block_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc block_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest, slot: Slot,
|
2022-01-30 17:51:04 +01:00
|
|
|
blck: Eth2Digest | SomeForkyBeaconBlock | BeaconBlockHeader,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_block_signing_root(
|
|
|
|
fork, genesis_validators_root, slot, blck)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_aggregate_and_proof_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc aggregate_and_proof_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
|
|
|
aggregate_and_proof: AggregateAndProof,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_aggregate_and_proof_signing_root(
|
|
|
|
fork, genesis_validators_root, aggregate_and_proof)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_attestation_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc attestation_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
|
|
|
attestation_data: AttestationData,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_attestation_signing_root(
|
|
|
|
fork, genesis_validators_root, attestation_data)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_voluntary_exit_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc voluntary_exit_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
|
|
|
voluntary_exit: VoluntaryExit,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_voluntary_exit_signing_root(
|
|
|
|
fork, genesis_validators_root, voluntary_exit)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_sync_committee_message_signature
|
2021-12-29 15:28:40 +01:00
|
|
|
proc sync_committee_message_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
|
|
|
slot: Slot, block_root: Eth2Digest,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_sync_committee_message_signing_root(
|
|
|
|
fork, genesis_validators_root, slot, block_root)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
|
|
|
# See also: verify_sync_committee_selection_proof
|
2021-12-29 15:28:40 +01:00
|
|
|
proc sync_committee_selection_proof_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
2022-05-10 13:03:40 +03:00
|
|
|
slot: Slot, subcommittee_index: SyncSubcommitteeIndex,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
|
|
|
fork, genesis_validators_root, slot, subcommittee_index)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-12-09 13:56:54 +01:00
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
proc contribution_and_proof_signature_set*(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
|
|
|
msg: ContributionAndProof,
|
2021-12-29 15:28:40 +01:00
|
|
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
2021-12-09 13:56:54 +01:00
|
|
|
let signing_root = compute_contribution_and_proof_signing_root(
|
|
|
|
fork, genesis_validators_root, msg)
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
SignatureSet.init(pubkey, signing_root, signature)
|
2021-04-26 22:39:44 +02:00
|
|
|
|
2021-01-25 19:45:48 +01:00
|
|
|
proc collectSignatureSets*(
|
|
|
|
sigs: var seq[SignatureSet],
|
2021-11-05 08:34:34 +01:00
|
|
|
signed_block: ForkySignedBeaconBlock,
|
2021-06-10 09:37:02 +02:00
|
|
|
validatorKeys: auto,
|
2021-06-11 17:51:46 +00:00
|
|
|
state: ForkedHashedBeaconState,
|
2021-06-01 13:13:40 +02:00
|
|
|
cache: var StateCache): Result[void, cstring] =
|
2021-12-09 13:56:54 +01:00
|
|
|
## Collect all signature verifications that process_block would normally do
|
|
|
|
## except deposits, in one go.
|
|
|
|
##
|
2021-01-25 19:45:48 +01:00
|
|
|
## This includes
|
|
|
|
## - Block proposer
|
|
|
|
## - Randao Reaveal
|
|
|
|
## - Proposer slashings
|
|
|
|
## - Attester slashings
|
|
|
|
## - Attestations
|
|
|
|
## - VoluntaryExits
|
2021-12-09 13:56:54 +01:00
|
|
|
## - SyncCommittee (altair+)
|
2021-01-25 19:45:48 +01:00
|
|
|
##
|
2021-12-09 13:56:54 +01:00
|
|
|
## We do not include deposits as they can be invalid while still leaving the
|
|
|
|
## block valid
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# Metadata
|
|
|
|
# ----------------------------------------------------
|
2021-06-10 09:37:02 +02:00
|
|
|
mixin load
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2021-12-09 13:56:54 +01:00
|
|
|
let
|
|
|
|
fork = getStateField(state, fork)
|
|
|
|
genesis_validators_root = getStateField(state, genesis_validators_root)
|
|
|
|
|
2021-01-25 19:45:48 +01:00
|
|
|
let
|
|
|
|
proposer_index = signed_block.message.proposer_index
|
2021-06-10 09:37:02 +02:00
|
|
|
proposer_key = validatorKeys.load(proposer_index)
|
|
|
|
if not proposer_key.isSome():
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("collectSignatureSets: invalid proposer index")
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2022-01-11 11:01:54 +01:00
|
|
|
let epoch = signed_block.message.slot.epoch()
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# 1. Block proposer
|
|
|
|
# ----------------------------------------------------
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add block_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root,
|
|
|
|
signed_block.message.slot, signed_block.root,
|
|
|
|
proposer_key.get(), signed_block.signature.loadOrExit(
|
|
|
|
"collectSignatureSets: cannot load signature"))
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# 2. Randao Reveal
|
|
|
|
# ----------------------------------------------------
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add epoch_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, epoch, proposer_key.get(),
|
|
|
|
signed_block.message.body.randao_reveal.loadOrExit(
|
|
|
|
"collectSignatureSets: cannot load randao"))
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# 3. Proposer slashings
|
|
|
|
# ----------------------------------------------------
|
|
|
|
# Denial-of-service:
|
|
|
|
# SSZ deserialization guarantees that blocks received from random sources
|
|
|
|
# including peer or RPC
|
|
|
|
# have at most MAX_PROPOSER_SLASHINGS proposer slashings.
|
|
|
|
for i in 0 ..< signed_block.message.body.proposer_slashings.len:
|
|
|
|
# don't use "items" for iterating over large type
|
|
|
|
# due to https://github.com/nim-lang/Nim/issues/14421
|
|
|
|
# fixed in 1.4.2
|
|
|
|
|
|
|
|
# Alias
|
|
|
|
template slashing: untyped = signed_block.message.body.proposer_slashings[i]
|
|
|
|
|
|
|
|
# Proposed block 1
|
|
|
|
block:
|
2021-06-10 09:37:02 +02:00
|
|
|
let
|
2021-12-09 13:56:54 +01:00
|
|
|
header = slashing.signed_header_1
|
|
|
|
key = validatorKeys.load(header.message.proposer_index)
|
|
|
|
if not key.isSome():
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("collectSignatureSets: invalid slashing proposer index 1")
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add block_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, header.message.slot, header.message,
|
|
|
|
key.get(), header.signature.loadOrExit(
|
|
|
|
"collectSignatureSets: cannot load proposer slashing 1 signature"))
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# Conflicting block 2
|
|
|
|
block:
|
2021-06-10 09:37:02 +02:00
|
|
|
let
|
2021-12-09 13:56:54 +01:00
|
|
|
header = slashing.signed_header_2
|
|
|
|
key = validatorKeys.load(header.message.proposer_index)
|
|
|
|
if not key.isSome():
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("collectSignatureSets: invalid slashing proposer index 2")
|
2021-12-09 13:56:54 +01:00
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add block_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, header.message.slot, header.message,
|
|
|
|
key.get(), header.signature.loadOrExit(
|
|
|
|
"collectSignatureSets: cannot load proposer slashing 2 signature"))
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# 4. Attester slashings
|
|
|
|
# ----------------------------------------------------
|
|
|
|
# Denial-of-service:
|
|
|
|
# SSZ deserialization guarantees that blocks received from random sources
|
|
|
|
# including peer or RPC
|
|
|
|
# have at most MAX_ATTESTER_SLASHINGS attester slashings.
|
|
|
|
for i in 0 ..< signed_block.message.body.attester_slashings.len:
|
|
|
|
# don't use "items" for iterating over large type
|
|
|
|
# due to https://github.com/nim-lang/Nim/issues/14421
|
|
|
|
# fixed in 1.4.2
|
|
|
|
|
|
|
|
# Alias
|
|
|
|
template slashing: untyped = signed_block.message.body.attester_slashings[i]
|
|
|
|
|
|
|
|
# Attestation 1
|
2021-12-09 13:56:54 +01:00
|
|
|
block:
|
|
|
|
let
|
|
|
|
key = ? aggregateAttesters(
|
|
|
|
slashing.attestation_1.attesting_indices.asSeq(), validatorKeys)
|
|
|
|
sig = slashing.attestation_1.signature.loadOrExit("")
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add attestation_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, slashing.attestation_1.data, key, sig)
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# Conflicting attestation 2
|
2021-12-09 13:56:54 +01:00
|
|
|
block:
|
|
|
|
let
|
|
|
|
key = ? aggregateAttesters(
|
|
|
|
slashing.attestation_2.attesting_indices.asSeq(), validatorKeys)
|
|
|
|
sig = slashing.attestation_2.signature.loadOrExit("")
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add attestation_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, slashing.attestation_2.data, key, sig)
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# 5. Attestations
|
|
|
|
# ----------------------------------------------------
|
|
|
|
# Denial-of-service:
|
|
|
|
# SSZ deserialization guarantees that blocks received from random sources
|
|
|
|
# including peer or RPC
|
|
|
|
# have at most MAX_ATTESTATIONS attestations.
|
|
|
|
for i in 0 ..< signed_block.message.body.attestations.len:
|
|
|
|
# don't use "items" for iterating over large type
|
|
|
|
# due to https://github.com/nim-lang/Nim/issues/14421
|
|
|
|
# fixed in 1.4.2
|
2021-12-09 13:56:54 +01:00
|
|
|
template attestation: untyped = signed_block.message.body.attestations[i]
|
|
|
|
|
|
|
|
let
|
|
|
|
key = ? aggregateAttesters(
|
|
|
|
get_attesting_indices(
|
|
|
|
state, attestation.data, attestation.aggregation_bits, cache),
|
|
|
|
validatorKeys)
|
|
|
|
sig = attestation.signature.loadOrExit("")
|
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add attestation_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, attestation.data, key, sig)
|
2021-01-25 19:45:48 +01:00
|
|
|
|
|
|
|
# 6. VoluntaryExits
|
|
|
|
# ----------------------------------------------------
|
|
|
|
# Denial-of-service:
|
|
|
|
# SSZ deserialization guarantees that blocks received from random sources
|
|
|
|
# including peer or RPC
|
|
|
|
# have at most MAX_VOLUNTARY_EXITS voluntary exits.
|
|
|
|
for i in 0 ..< signed_block.message.body.voluntary_exits.len:
|
|
|
|
# don't use "items" for iterating over large type
|
|
|
|
# due to https://github.com/nim-lang/Nim/issues/14421
|
|
|
|
# fixed in 1.4.2
|
|
|
|
template volex: untyped = signed_block.message.body.voluntary_exits[i]
|
2021-06-10 09:37:02 +02:00
|
|
|
let key = validatorKeys.load(volex.message.validator_index)
|
|
|
|
if not key.isSome():
|
2021-06-01 13:13:40 +02:00
|
|
|
return err("collectSignatureSets: invalid voluntary exit")
|
2021-01-25 19:45:48 +01:00
|
|
|
|
2021-12-29 15:28:40 +01:00
|
|
|
sigs.add voluntary_exit_signature_set(
|
2021-12-09 13:56:54 +01:00
|
|
|
fork, genesis_validators_root, volex.message, key.get(),
|
|
|
|
volex.signature.loadOrExit(
|
|
|
|
"collectSignatureSets: cannot load voluntary exit signature"))
|
2021-01-25 19:45:48 +01:00
|
|
|
|
Speed up altair block processing 2x (#3115)
* Speed up altair block processing >2x
Like #3089, this PR drastially speeds up historical REST queries and
other long state replays.
* cache sync committee validator indices
* use ~80mb less memory for validator pubkey mappings
* batch-verify sync aggregate signature (fixes #2985)
* document sync committee hack with head block vs sync message block
* add batch signature verification failure tests
Before:
```
../env.sh nim c -d:release -r ncli_db --db:mainnet_0/db bench --start-slot:-1000
All time are ms
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
5830.675, 0.000, 5830.675, 5830.675, 1, Initialize DB
0.481, 1.878, 0.215, 59.167, 981, Load block from database
8422.566, 0.000, 8422.566, 8422.566, 1, Load state from database
6.996, 1.678, 0.042, 14.385, 969, Advance slot, non-epoch
93.217, 8.318, 84.192, 122.209, 32, Advance slot, epoch
20.513, 23.665, 11.510, 201.561, 981, Apply block, no slot processing
0.000, 0.000, 0.000, 0.000, 0, Database load
0.000, 0.000, 0.000, 0.000, 0, Database store
```
After:
```
7081.422, 0.000, 7081.422, 7081.422, 1, Initialize DB
0.553, 2.122, 0.175, 66.692, 981, Load block from database
5439.446, 0.000, 5439.446, 5439.446, 1, Load state from database
6.829, 1.575, 0.043, 12.156, 969, Advance slot, non-epoch
94.716, 2.749, 88.395, 100.026, 32, Advance slot, epoch
11.636, 23.766, 4.889, 205.250, 981, Apply block, no slot processing
0.000, 0.000, 0.000, 0.000, 0, Database load
0.000, 0.000, 0.000, 0.000, 0, Database store
```
* add comment
2021-11-24 13:43:50 +01:00
|
|
|
block:
|
2022-09-07 18:45:46 +02:00
|
|
|
when signed_block is phase0.SignedBeaconBlock:
|
|
|
|
discard
|
|
|
|
else:
|
|
|
|
# 7. SyncAggregate
|
|
|
|
# ----------------------------------------------------
|
|
|
|
withState(state):
|
|
|
|
when stateFork >= BeaconStateFork.Altair:
|
|
|
|
if signed_block.message.body.sync_aggregate.sync_committee_bits.countOnes() == 0:
|
|
|
|
if signed_block.message.body.sync_aggregate.sync_committee_signature != ValidatorSig.infinity():
|
|
|
|
return err("collectSignatureSets: empty sync aggregates need signature of point at infinity")
|
|
|
|
else:
|
|
|
|
let
|
|
|
|
current_sync_committee =
|
2022-09-10 06:12:07 +00:00
|
|
|
forkyState.data.get_sync_committee_cache(cache).current_sync_committee
|
|
|
|
previous_slot = max(forkyState.data.slot, Slot(1)) - 1
|
|
|
|
beacon_block_root = get_block_root_at_slot(forkyState.data, previous_slot)
|
2022-09-07 18:45:46 +02:00
|
|
|
pubkey = ? aggregateAttesters(
|
|
|
|
current_sync_committee,
|
|
|
|
signed_block.message.body.sync_aggregate.sync_committee_bits,
|
|
|
|
validatorKeys)
|
|
|
|
|
|
|
|
sigs.add sync_committee_message_signature_set(
|
|
|
|
fork, genesis_validators_root, previous_slot, beacon_block_root,
|
|
|
|
pubkey,
|
|
|
|
signed_block.message.body.sync_aggregate.sync_committee_signature.loadOrExit(
|
|
|
|
"collectSignatureSets: cannot load signature"))
|
Speed up altair block processing 2x (#3115)
* Speed up altair block processing >2x
Like #3089, this PR drastially speeds up historical REST queries and
other long state replays.
* cache sync committee validator indices
* use ~80mb less memory for validator pubkey mappings
* batch-verify sync aggregate signature (fixes #2985)
* document sync committee hack with head block vs sync message block
* add batch signature verification failure tests
Before:
```
../env.sh nim c -d:release -r ncli_db --db:mainnet_0/db bench --start-slot:-1000
All time are ms
Average, StdDev, Min, Max, Samples, Test
Validation is turned off meaning that no BLS operations are performed
5830.675, 0.000, 5830.675, 5830.675, 1, Initialize DB
0.481, 1.878, 0.215, 59.167, 981, Load block from database
8422.566, 0.000, 8422.566, 8422.566, 1, Load state from database
6.996, 1.678, 0.042, 14.385, 969, Advance slot, non-epoch
93.217, 8.318, 84.192, 122.209, 32, Advance slot, epoch
20.513, 23.665, 11.510, 201.561, 981, Apply block, no slot processing
0.000, 0.000, 0.000, 0.000, 0, Database load
0.000, 0.000, 0.000, 0.000, 0, Database store
```
After:
```
7081.422, 0.000, 7081.422, 7081.422, 1, Initialize DB
0.553, 2.122, 0.175, 66.692, 981, Load block from database
5439.446, 0.000, 5439.446, 5439.446, 1, Load state from database
6.829, 1.575, 0.043, 12.156, 969, Advance slot, non-epoch
94.716, 2.749, 88.395, 100.026, 32, Advance slot, epoch
11.636, 23.766, 4.889, 205.250, 981, Apply block, no slot processing
0.000, 0.000, 0.000, 0.000, 0, Database load
0.000, 0.000, 0.000, 0.000, 0, Database store
```
* add comment
2021-11-24 13:43:50 +01:00
|
|
|
|
2021-06-01 13:13:40 +02:00
|
|
|
ok()
|
2021-12-06 10:49:01 +01:00
|
|
|
|
|
|
|
proc batchVerify*(verifier: var BatchVerifier, sigs: openArray[SignatureSet]): bool =
|
2022-06-21 10:29:16 +02:00
|
|
|
let bytes = verifier.rng[].generate(array[32, byte])
|
2021-12-06 10:49:01 +01:00
|
|
|
try:
|
|
|
|
verifier.taskpool.batchVerify(verifier.sigVerifCache, sigs, bytes)
|
|
|
|
except Exception as exc:
|
|
|
|
raiseAssert exc.msg # Shouldn't happen
|