mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-20 19:40:52 +00:00
Remove get_crosslink_committees_at_slot and fix research/state_sim (#291)
* migrate attestation pool tests from get_crosslink_committees_at_slot(...) to get_crosslink_committee(...) * rm obsolete, unused get_crosslink_committees_at_slot_cached(...) and migrate tests/test_state_transition from get_crosslink_committees_at_slot(...) to get_crosslink_committee(...) * migrate tests/testutil from get_crosslink_committees_at_slot(...) to get_crosslink_committee(...) * use more pervasive caching infrastructure, initially of compute_committee; remove buggy (and per-index) shuffling to fix research/state_sim, which was noticing validators on multiple shard committees; rm now-unused specific get_attesting_balance_cached * add some get_active_validator_index caching * rm obsolete/unused get_attesting_indices_cached * rm get_crosslink_committees_at_slot(...) * some more 0.7 changes -- some of which can't be completed pending some data structure updates -- and shard handling changes to offset with epoch start shards
This commit is contained in:
parent
c751285112
commit
c7e06374f4
@ -260,6 +260,7 @@ proc getAttestationsForBlock*(
|
|||||||
newBlockSlot = humaneSlotNum(newBlockSlot)
|
newBlockSlot = humaneSlotNum(newBlockSlot)
|
||||||
return
|
return
|
||||||
|
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
let
|
let
|
||||||
# TODO in theory we could include attestations from other slots also, but
|
# TODO in theory we could include attestations from other slots also, but
|
||||||
# we're currently not tracking which attestations have already been included
|
# we're currently not tracking which attestations have already been included
|
||||||
@ -296,7 +297,8 @@ proc getAttestationsForBlock*(
|
|||||||
# attestations into the pool in general is an open question that needs
|
# attestations into the pool in general is an open question that needs
|
||||||
# revisiting - for example, when attestations are added, against which
|
# revisiting - for example, when attestations are added, against which
|
||||||
# state should they be validated, if at all?
|
# state should they be validated, if at all?
|
||||||
if not checkAttestation(state, attestation, {skipValidation, nextSlot}):
|
if not checkAttestation(
|
||||||
|
state, attestation, {skipValidation, nextSlot}, cache):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
for v in a.validations[1..^1]:
|
for v in a.validations[1..^1]:
|
||||||
|
@ -468,13 +468,20 @@ proc handleAttestations(node: BeaconNode, head: BlockRef, slot: Slot) =
|
|||||||
# In case blocks went missing, this means advancing past the latest block
|
# In case blocks went missing, this means advancing past the latest block
|
||||||
# using empty slots as fillers.
|
# using empty slots as fillers.
|
||||||
node.blockPool.withState(node.stateCache, attestationHead):
|
node.blockPool.withState(node.stateCache, attestationHead):
|
||||||
for crosslink_committee in get_crosslink_committees_at_slot(state, slot):
|
var cache = get_empty_per_epoch_cache()
|
||||||
for i, validatorIdx in crosslink_committee.committee:
|
let epoch = slot_to_epoch(slot)
|
||||||
|
for committee_index in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
||||||
|
## TODO verify that this is the correct mapping; it's consistent with
|
||||||
|
## other code
|
||||||
|
let
|
||||||
|
shard = committee_index mod SHARD_COUNT
|
||||||
|
committee = get_crosslink_committee(state, epoch, shard, cache)
|
||||||
|
for i, validatorIdx in committee:
|
||||||
let validator = node.getAttachedValidator(state, validatorIdx)
|
let validator = node.getAttachedValidator(state, validatorIdx)
|
||||||
if validator != nil:
|
if validator != nil:
|
||||||
attestations.add (
|
attestations.add (
|
||||||
makeAttestationData(state, crosslink_committee.shard, blck.root),
|
makeAttestationData(state, shard, blck.root),
|
||||||
crosslink_committee.committee.len, i, validator)
|
committee.len, i, validator)
|
||||||
|
|
||||||
for a in attestations:
|
for a in attestations:
|
||||||
traceAsyncErrors sendAttestation(
|
traceAsyncErrors sendAttestation(
|
||||||
@ -489,9 +496,10 @@ proc handleProposal(node: BeaconNode, head: BlockRef, slot: Slot):
|
|||||||
# proposing for it - basically, we're selecting proposer based on an
|
# proposing for it - basically, we're selecting proposer based on an
|
||||||
# empty slot.. wait for the committee selection to settle, then
|
# empty slot.. wait for the committee selection to settle, then
|
||||||
# revisit this - we should be able to advance behind
|
# revisit this - we should be able to advance behind
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
node.blockPool.withState(node.stateCache, BlockSlot(blck: head, slot: slot)):
|
node.blockPool.withState(node.stateCache, BlockSlot(blck: head, slot: slot)):
|
||||||
let
|
let
|
||||||
proposerIdx = get_beacon_proposer_index(state)
|
proposerIdx = get_beacon_proposer_index(state, cache)
|
||||||
validator = node.getAttachedValidator(state, proposerIdx)
|
validator = node.getAttachedValidator(state, proposerIdx)
|
||||||
|
|
||||||
if validator != nil:
|
if validator != nil:
|
||||||
|
@ -206,10 +206,9 @@ type
|
|||||||
|
|
||||||
StateCache* = object
|
StateCache* = object
|
||||||
crosslink_committee_cache*:
|
crosslink_committee_cache*:
|
||||||
Table[tuple[a: uint64, b: bool], seq[CrosslinkCommittee]]
|
Table[tuple[a: int, b: Eth2Digest], seq[ValidatorIndex]]
|
||||||
|
active_validator_indices_cache*:
|
||||||
winning_root_participants_cache*:
|
Table[Epoch, seq[ValidatorIndex]]
|
||||||
Table[Shard, HashSet[ValidatorIndex]]
|
|
||||||
|
|
||||||
BlockSlot* = object
|
BlockSlot* = object
|
||||||
## Unique identifier for a particular fork in the block chain - normally,
|
## Unique identifier for a particular fork in the block chain - normally,
|
||||||
|
@ -148,7 +148,8 @@ func initiate_validator_exit*(state: var BeaconState,
|
|||||||
validator.exit_epoch + MIN_VALIDATOR_WITHDRAWABILITY_DELAY
|
validator.exit_epoch + MIN_VALIDATOR_WITHDRAWABILITY_DELAY
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#slash_validator
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#slash_validator
|
||||||
func slash_validator*(state: var BeaconState, slashed_index: ValidatorIndex) =
|
func slash_validator*(state: var BeaconState, slashed_index: ValidatorIndex,
|
||||||
|
stateCache: var StateCache) =
|
||||||
# Slash the validator with index ``index``.
|
# Slash the validator with index ``index``.
|
||||||
let current_epoch = get_current_epoch(state)
|
let current_epoch = get_current_epoch(state)
|
||||||
initiate_validator_exit(state, slashed_index)
|
initiate_validator_exit(state, slashed_index)
|
||||||
@ -161,7 +162,7 @@ func slash_validator*(state: var BeaconState, slashed_index: ValidatorIndex) =
|
|||||||
slashed_balance
|
slashed_balance
|
||||||
|
|
||||||
let
|
let
|
||||||
proposer_index = get_beacon_proposer_index(state)
|
proposer_index = get_beacon_proposer_index(state, stateCache)
|
||||||
# Spec has whistleblower_index as optional param, but it's never used.
|
# Spec has whistleblower_index as optional param, but it's never used.
|
||||||
whistleblower_index = proposer_index
|
whistleblower_index = proposer_index
|
||||||
whistleblowing_reward = slashed_balance div WHISTLEBLOWING_REWARD_QUOTIENT
|
whistleblowing_reward = slashed_balance div WHISTLEBLOWING_REWARD_QUOTIENT
|
||||||
@ -321,7 +322,9 @@ func get_total_balance*(state: BeaconState, validators: auto): Gwei =
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#registry-updates
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#registry-updates
|
||||||
func process_registry_updates*(state: var BeaconState) =
|
func process_registry_updates*(state: var BeaconState) =
|
||||||
# Process activation eligibility and ejections
|
## Process activation eligibility and ejections
|
||||||
|
## Try to avoid caching here, since this could easily become undefined
|
||||||
|
|
||||||
for index, validator in state.validator_registry:
|
for index, validator in state.validator_registry:
|
||||||
if validator.activation_eligibility_epoch == FAR_FUTURE_EPOCH and
|
if validator.activation_eligibility_epoch == FAR_FUTURE_EPOCH and
|
||||||
validator.effective_balance >= MAX_EFFECTIVE_BALANCE:
|
validator.effective_balance >= MAX_EFFECTIVE_BALANCE:
|
||||||
@ -411,7 +414,9 @@ func validate_indexed_attestation*(
|
|||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_attesting_indices
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_attesting_indices
|
||||||
func get_attesting_indices*(state: BeaconState,
|
func get_attesting_indices*(state: BeaconState,
|
||||||
attestation_data: AttestationData,
|
attestation_data: AttestationData,
|
||||||
bitfield: BitField): HashSet[ValidatorIndex] =
|
bitfield: BitField,
|
||||||
|
stateCache: var StateCache):
|
||||||
|
HashSet[ValidatorIndex] =
|
||||||
## Return the sorted attesting indices corresponding to ``attestation_data``
|
## Return the sorted attesting indices corresponding to ``attestation_data``
|
||||||
## and ``bitfield``.
|
## and ``bitfield``.
|
||||||
## The spec goes through a lot of hoops to sort things, and sometimes
|
## The spec goes through a lot of hoops to sort things, and sometimes
|
||||||
@ -421,7 +426,7 @@ func get_attesting_indices*(state: BeaconState,
|
|||||||
result = initSet[ValidatorIndex]()
|
result = initSet[ValidatorIndex]()
|
||||||
let committee =
|
let committee =
|
||||||
get_crosslink_committee(state, attestation_data.target_epoch,
|
get_crosslink_committee(state, attestation_data.target_epoch,
|
||||||
attestation_data.shard)
|
attestation_data.shard, stateCache)
|
||||||
doAssert verify_bitfield(bitfield, len(committee))
|
doAssert verify_bitfield(bitfield, len(committee))
|
||||||
for i, index in committee:
|
for i, index in committee:
|
||||||
if get_bitfield_bit(bitfield, i):
|
if get_bitfield_bit(bitfield, i):
|
||||||
@ -430,25 +435,29 @@ func get_attesting_indices*(state: BeaconState,
|
|||||||
func get_attesting_indices_seq*(
|
func get_attesting_indices_seq*(
|
||||||
state: BeaconState, attestation_data: AttestationData, bitfield: BitField):
|
state: BeaconState, attestation_data: AttestationData, bitfield: BitField):
|
||||||
seq[ValidatorIndex] =
|
seq[ValidatorIndex] =
|
||||||
toSeq(items(get_attesting_indices(state, attestation_data, bitfield)))
|
var cache = get_empty_per_epoch_cache()
|
||||||
|
toSeq(items(get_attesting_indices(
|
||||||
|
state, attestation_data, bitfield, cache)))
|
||||||
|
|
||||||
# TODO legacy function name; rename, reimplement caching if useful, blob/v0.6.2
|
# TODO legacy function name; rename, reimplement caching if useful, blob/v0.6.2
|
||||||
iterator get_attestation_participants_cached*(
|
iterator get_attestation_participants_cached*(
|
||||||
state: BeaconState, attestation_data: AttestationData, bitfield: BitField,
|
state: BeaconState, attestation_data: AttestationData, bitfield: BitField,
|
||||||
cache: var StateCache): ValidatorIndex =
|
cache: var StateCache): ValidatorIndex =
|
||||||
for participant in get_attesting_indices(state, attestation_data, bitfield):
|
for participant in get_attesting_indices(
|
||||||
|
state, attestation_data, bitfield, cache):
|
||||||
yield participant
|
yield participant
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#convert_to_indexed
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#convert_to_indexed
|
||||||
func convert_to_indexed(state: BeaconState, attestation: Attestation): IndexedAttestation =
|
func convert_to_indexed(state: BeaconState, attestation: Attestation,
|
||||||
|
stateCache: var StateCache): IndexedAttestation =
|
||||||
# Convert ``attestation`` to (almost) indexed-verifiable form.
|
# Convert ``attestation`` to (almost) indexed-verifiable form.
|
||||||
let
|
let
|
||||||
attesting_indices =
|
attesting_indices =
|
||||||
get_attesting_indices(
|
get_attesting_indices(
|
||||||
state, attestation.data, attestation.aggregation_bitfield)
|
state, attestation.data, attestation.aggregation_bitfield, stateCache)
|
||||||
custody_bit_1_indices =
|
custody_bit_1_indices =
|
||||||
get_attesting_indices(
|
get_attesting_indices(
|
||||||
state, attestation.data, attestation.custody_bitfield)
|
state, attestation.data, attestation.custody_bitfield, stateCache)
|
||||||
|
|
||||||
## TODO quadratic, .items, but first-class iterators, etc
|
## TODO quadratic, .items, but first-class iterators, etc
|
||||||
## filterIt can't work on HashSets directly because it is
|
## filterIt can't work on HashSets directly because it is
|
||||||
@ -486,7 +495,8 @@ func convert_to_indexed(state: BeaconState, attestation: Attestation): IndexedAt
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#attestations
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#attestations
|
||||||
proc checkAttestation*(
|
proc checkAttestation*(
|
||||||
state: BeaconState, attestation: Attestation, flags: UpdateFlags): bool =
|
state: BeaconState, attestation: Attestation, flags: UpdateFlags,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
## Process ``Attestation`` operation.
|
## Process ``Attestation`` operation.
|
||||||
## Check that an attestation follows the rules of being included in the state
|
## Check that an attestation follows the rules of being included in the state
|
||||||
## at the current slot. When acting as a proposer, the same rules need to
|
## at the current slot. When acting as a proposer, the same rules need to
|
||||||
@ -496,7 +506,10 @@ proc checkAttestation*(
|
|||||||
if nextSlot in flags: state.slot + 1
|
if nextSlot in flags: state.slot + 1
|
||||||
else: state.slot
|
else: state.slot
|
||||||
|
|
||||||
let attestation_slot = get_attestation_slot(state, attestation)
|
let
|
||||||
|
data = attestation.data
|
||||||
|
attestation_slot = get_attestation_slot(state, attestation)
|
||||||
|
|
||||||
if not (attestation_slot + MIN_ATTESTATION_INCLUSION_DELAY <= stateSlot):
|
if not (attestation_slot + MIN_ATTESTATION_INCLUSION_DELAY <= stateSlot):
|
||||||
warn("Attestation too new",
|
warn("Attestation too new",
|
||||||
attestation_slot = humaneSlotNum(attestation_slot),
|
attestation_slot = humaneSlotNum(attestation_slot),
|
||||||
@ -509,29 +522,66 @@ proc checkAttestation*(
|
|||||||
state_slot = humaneSlotNum(stateSlot))
|
state_slot = humaneSlotNum(stateSlot))
|
||||||
return
|
return
|
||||||
|
|
||||||
|
let pending_attestation = PendingAttestation(
|
||||||
|
data: data,
|
||||||
|
aggregation_bitfield: attestation.aggregation_bitfield,
|
||||||
|
inclusion_delay: state.slot - attestation_slot,
|
||||||
|
proposer_index: get_beacon_proposer_index(state, stateCache),
|
||||||
|
)
|
||||||
|
|
||||||
# Check target epoch, source epoch, source root, and source crosslink
|
# Check target epoch, source epoch, source root, and source crosslink
|
||||||
let data = attestation.data
|
if not (data.target_epoch == get_previous_epoch(state) or
|
||||||
if not (
|
data.target_epoch == get_current_epoch(state)):
|
||||||
(data.target_epoch, data.source_epoch, data.source_root, data.previous_crosslink_root) ==
|
warn("Target epoch not current or previous epoch")
|
||||||
(get_current_epoch(state), state.current_justified_epoch,
|
|
||||||
state.current_justified_root,
|
|
||||||
hash_tree_root(state.current_crosslinks[data.shard])) or
|
|
||||||
(data.target_epoch, data.source_epoch, data.source_root, data.previous_crosslink_root) ==
|
|
||||||
(get_previous_epoch(state), state.previous_justified_epoch,
|
|
||||||
state.previous_justified_root,
|
|
||||||
hash_tree_root(state.previous_crosslinks[data.shard]))):
|
|
||||||
warn("checkAttestation: target epoch, source epoch, source root, or source crosslink invalid")
|
|
||||||
return
|
return
|
||||||
|
|
||||||
## Check crosslink data root
|
# Check FFG data, crosslink data, and signature
|
||||||
## [to be removed in phase 1]
|
let ffg_check_data = (data.source_epoch, data.source_root, data.target_epoch)
|
||||||
if attestation.data.crosslink_data_root != ZERO_HASH:
|
|
||||||
warn("Invalid crosslink data root")
|
if data.target_epoch == get_current_epoch(state):
|
||||||
|
if not (ffg_check_data == (state.current_justified_epoch,
|
||||||
|
state.current_justified_root, get_current_epoch(state))):
|
||||||
|
warn("FFG data not matching current justified epoch")
|
||||||
return
|
return
|
||||||
|
|
||||||
|
#if not (data.crosslink.parent_root ==
|
||||||
|
# hash_tree_root(state.current_crosslinks[data.crosslink.shard])):
|
||||||
|
# warn("Crosslink shard's current crosslinks not matching crosslink parent root")
|
||||||
|
# return
|
||||||
|
|
||||||
|
#state.current_epoch_attestations.add(pending_attestation)
|
||||||
|
else:
|
||||||
|
if not (ffg_check_data == (state.previous_justified_epoch,
|
||||||
|
state.previous_justified_root, get_previous_epoch(state))):
|
||||||
|
warn("FFG data not matching current justified epoch")
|
||||||
|
return
|
||||||
|
|
||||||
|
#if not (data.crosslink.parent_root ==
|
||||||
|
# hash_tree_root(state.previous_crosslinks[data.crosslink.shard])):
|
||||||
|
# warn("Crosslink shard's previous crosslinks not matching crosslink parent root")
|
||||||
|
# return
|
||||||
|
|
||||||
|
#state.previous_epoch_attestations.add(pending_attestation)
|
||||||
|
|
||||||
|
# TODO un-comment when changed Crosslink to 0.7 structure
|
||||||
|
#if not (data.crosslink.start_epoch == parent_crosslink.end_epoch):
|
||||||
|
# warn("Crosslink start and end epochs not the same")
|
||||||
|
# return
|
||||||
|
|
||||||
|
#if not (data.crosslink.end_epoch == min(data.target_epoch, parent_crosslink.end_epoch + MAX_EPOCHS_PER_CROSSLINK)):
|
||||||
|
# warn("Crosslink end epoch incorrect")
|
||||||
|
# return
|
||||||
|
|
||||||
|
# Simlarly, these depend on 0.7 data structures
|
||||||
|
#assert data.crosslink.parent_root == hash_tree_root(parent_crosslink)
|
||||||
|
|
||||||
|
#if not (data.crosslink.data_root == ZERO_HASH): # [to be removed in phase 1]
|
||||||
|
# warn("Crosslink data root not zero")
|
||||||
|
# return
|
||||||
|
|
||||||
# Check signature and bitfields
|
# Check signature and bitfields
|
||||||
if not validate_indexed_attestation(
|
if not validate_indexed_attestation(
|
||||||
state, convert_to_indexed(state, attestation)):
|
state, convert_to_indexed(state, attestation, stateCache)):
|
||||||
warn("checkAttestation: signature or bitfields incorrect")
|
warn("checkAttestation: signature or bitfields incorrect")
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -552,7 +602,8 @@ proc makeAttestationData*(
|
|||||||
|
|
||||||
AttestationData(
|
AttestationData(
|
||||||
slot: state.slot,
|
slot: state.slot,
|
||||||
shard: shard,
|
# Alternative is to put this offset into all callers
|
||||||
|
shard: shard + get_epoch_start_shard(state, slot_to_epoch(state.slot)),
|
||||||
beacon_block_root: beacon_block_root,
|
beacon_block_root: beacon_block_root,
|
||||||
target_root: target_root,
|
target_root: target_root,
|
||||||
crosslink_data_root: Eth2Digest(), # Stub in phase0
|
crosslink_data_root: Eth2Digest(), # Stub in phase0
|
||||||
|
@ -363,9 +363,6 @@ type
|
|||||||
Activation = 0
|
Activation = 0
|
||||||
Exit = 1
|
Exit = 1
|
||||||
|
|
||||||
# TODO: not in spec
|
|
||||||
CrosslinkCommittee* = tuple[committee: seq[ValidatorIndex], shard: uint64]
|
|
||||||
|
|
||||||
# TODO to be replaced with some magic hash caching
|
# TODO to be replaced with some magic hash caching
|
||||||
HashedBeaconState* = object
|
HashedBeaconState* = object
|
||||||
data*: BeaconState
|
data*: BeaconState
|
||||||
|
@ -7,8 +7,7 @@
|
|||||||
# Helpers and functions pertaining to managing the validator set
|
# Helpers and functions pertaining to managing the validator set
|
||||||
|
|
||||||
import
|
import
|
||||||
options, nimcrypto, sequtils, math, chronicles,
|
options, nimcrypto, sequtils, math, tables, chronicles,
|
||||||
eth/common,
|
|
||||||
../ssz, ../beacon_node_types,
|
../ssz, ../beacon_node_types,
|
||||||
./crypto, ./datatypes, ./digest, ./helpers
|
./crypto, ./datatypes, ./digest, ./helpers
|
||||||
|
|
||||||
@ -80,44 +79,6 @@ func get_shuffled_seq*(seed: Eth2Digest,
|
|||||||
|
|
||||||
result = shuffled_active_validator_indices
|
result = shuffled_active_validator_indices
|
||||||
|
|
||||||
func get_shuffled_index(index: ValidatorIndex, index_count: uint64, seed: Eth2Digest): uint64 =
|
|
||||||
## Return the shuffled validator index corresponding to ``seed`` (and
|
|
||||||
## ``index_count``).
|
|
||||||
## https://github.com/status-im/nim-beacon-chain/blob/f77016af6818ad2c853f6c9e2751b17548e0222e/beacon_chain/spec/validator.nim#L15
|
|
||||||
|
|
||||||
doAssert index.uint64 < index_count
|
|
||||||
doAssert index_count <= 2'u64^40
|
|
||||||
|
|
||||||
result = index
|
|
||||||
var pivot_buffer: array[(32+1), byte]
|
|
||||||
var source_buffer: array[(32+1+4), byte]
|
|
||||||
|
|
||||||
# Swap or not (https://link.springer.com/content/pdf/10.1007%2F978-3-642-32009-5_1.pdf)
|
|
||||||
# See the 'generalized domain' algorithm on page 3
|
|
||||||
for round in 0 ..< SHUFFLE_ROUND_COUNT:
|
|
||||||
pivot_buffer[0..31] = seed.data
|
|
||||||
let round_bytes1 = int_to_bytes1(round)[0]
|
|
||||||
pivot_buffer[32] = round_bytes1
|
|
||||||
|
|
||||||
let
|
|
||||||
pivot = bytes_to_int(eth2hash(pivot_buffer).data[0..7]) mod index_count
|
|
||||||
flip = (pivot - index) mod index_count
|
|
||||||
position = max(index.uint64, flip)
|
|
||||||
|
|
||||||
## Tradeoff between slicing (if reusing one larger buffer) and additional
|
|
||||||
## copies here of seed and `int_to_bytes1(round)`.
|
|
||||||
source_buffer[0..31] = seed.data
|
|
||||||
source_buffer[32] = round_bytes1
|
|
||||||
source_buffer[33..36] = int_to_bytes4(position div 256)
|
|
||||||
|
|
||||||
let
|
|
||||||
source = eth2hash(source_buffer).data
|
|
||||||
byte_value = source[(position mod 256) div 8]
|
|
||||||
bit = (byte_value shr (position mod 8)) mod 2
|
|
||||||
|
|
||||||
if bit != 0:
|
|
||||||
result = flip
|
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_previous_epoch
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_previous_epoch
|
||||||
func get_previous_epoch*(state: BeaconState): Epoch =
|
func get_previous_epoch*(state: BeaconState): Epoch =
|
||||||
## Return the previous epoch of the given ``state``.
|
## Return the previous epoch of the given ``state``.
|
||||||
@ -151,69 +112,63 @@ func get_epoch_start_shard*(state: BeaconState, epoch: Epoch): Shard =
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#compute_committee
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#compute_committee
|
||||||
func compute_committee(indices: seq[ValidatorIndex], seed: Eth2Digest,
|
func compute_committee(indices: seq[ValidatorIndex], seed: Eth2Digest,
|
||||||
index: uint64, count: uint64): seq[ValidatorIndex] =
|
index: uint64, count: uint64, stateCache: var StateCache): seq[ValidatorIndex] =
|
||||||
|
|
||||||
let
|
let
|
||||||
start = (len(indices).uint64 * index) div count
|
start = (len(indices).uint64 * index) div count
|
||||||
endIdx = (len(indices).uint64 * (index + 1)) div count
|
endIdx = (len(indices).uint64 * (index + 1)) div count
|
||||||
|
key = (indices.len, seed)
|
||||||
doAssert endIdx.int - start.int > 0
|
doAssert endIdx.int - start.int > 0
|
||||||
|
|
||||||
|
if key notin stateCache.crosslink_committee_cache:
|
||||||
|
stateCache.crosslink_committee_cache[key] =
|
||||||
|
get_shuffled_seq(seed, len(indices).uint64)
|
||||||
|
|
||||||
|
# These assertions from get_shuffled_index(...)
|
||||||
|
let index_count = indices.len().uint64
|
||||||
|
doAssert endIdx <= index_count
|
||||||
|
doAssert index_count <= 2'u64^40
|
||||||
|
|
||||||
|
# In spec, this calls get_shuffled_index() every time, but that's wasteful
|
||||||
mapIt(
|
mapIt(
|
||||||
start.int .. (endIdx.int-1),
|
start.int .. (endIdx.int-1),
|
||||||
indices[
|
indices[
|
||||||
get_shuffled_index(it.ValidatorIndex, len(indices).uint64, seed).int])
|
stateCache.crosslink_committee_cache[key][it]])
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_crosslink_committee
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_crosslink_committee
|
||||||
func get_crosslink_committee*(state: BeaconState, epoch: Epoch, shard: Shard):
|
func get_crosslink_committee*(state: BeaconState, epoch: Epoch, shard: Shard,
|
||||||
seq[ValidatorIndex] =
|
stateCache: var StateCache): seq[ValidatorIndex] =
|
||||||
|
|
||||||
|
doAssert shard >= 0'u64
|
||||||
|
# This seems to be required, basically, to be true? But I'm not entirely sure
|
||||||
|
#doAssert shard >= get_epoch_start_shard(state, epoch)
|
||||||
|
doAssert shard < SHARD_COUNT
|
||||||
|
|
||||||
|
## This is a somewhat more fragile, but high-ROI, caching setup --
|
||||||
|
## get_active_validator_indices() is slow to run in a loop and only
|
||||||
|
## changes once per epoch.
|
||||||
|
if epoch notin stateCache.active_validator_indices_cache:
|
||||||
|
stateCache.active_validator_indices_cache[epoch] =
|
||||||
|
get_active_validator_indices(state, epoch)
|
||||||
|
|
||||||
compute_committee(
|
compute_committee(
|
||||||
get_active_validator_indices(state, epoch),
|
stateCache.active_validator_indices_cache[epoch],
|
||||||
generate_seed(state, epoch),
|
generate_seed(state, epoch),
|
||||||
(shard + SHARD_COUNT - get_epoch_start_shard(state, epoch)) mod SHARD_COUNT,
|
(shard + SHARD_COUNT - get_epoch_start_shard(state, epoch)) mod SHARD_COUNT,
|
||||||
get_epoch_committee_count(state, epoch),
|
get_epoch_committee_count(state, epoch),
|
||||||
|
stateCache
|
||||||
)
|
)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#get_crosslink_committees_at_slot
|
# Not from spec
|
||||||
func get_crosslink_committees_at_slot*(state: BeaconState, slot: Slot|uint64):
|
func get_empty_per_epoch_cache*(): StateCache =
|
||||||
seq[CrosslinkCommittee] =
|
result.crosslink_committee_cache =
|
||||||
## Returns the list of ``(committee, shard)`` tuples for the ``slot``.
|
initTable[tuple[a: int, b: Eth2Digest], seq[ValidatorIndex]]()
|
||||||
|
result.active_validator_indices_cache =
|
||||||
let
|
initTable[Epoch, seq[ValidatorIndex]]()
|
||||||
epoch = slot_to_epoch(slot) # TODO, enforce slot to be a Slot
|
|
||||||
current_epoch = get_current_epoch(state)
|
|
||||||
previous_epoch = get_previous_epoch(state)
|
|
||||||
next_epoch = current_epoch + 1
|
|
||||||
|
|
||||||
doAssert previous_epoch <= epoch,
|
|
||||||
"Previous epoch: " & $humaneEpochNum(previous_epoch) &
|
|
||||||
", epoch: " & $humaneEpochNum(epoch) &
|
|
||||||
" (slot: " & $humaneSlotNum(slot.Slot) & ")" &
|
|
||||||
", Next epoch: " & $humaneEpochNum(next_epoch)
|
|
||||||
|
|
||||||
doAssert epoch <= next_epoch,
|
|
||||||
"Previous epoch: " & $humaneEpochNum(previous_epoch) &
|
|
||||||
", epoch: " & $humaneEpochNum(epoch) &
|
|
||||||
" (slot: " & $humaneSlotNum(slot.Slot) & ")" &
|
|
||||||
", Next epoch: " & $humaneEpochNum(next_epoch)
|
|
||||||
|
|
||||||
for i in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
|
||||||
let shard = i mod SHARD_COUNT
|
|
||||||
result.add (
|
|
||||||
get_crosslink_committee(state, epoch, shard),
|
|
||||||
shard
|
|
||||||
)
|
|
||||||
|
|
||||||
iterator get_crosslink_committees_at_slot_cached*(
|
|
||||||
state: BeaconState, slot: Slot|uint64, cache: var StateCache):
|
|
||||||
CrosslinkCommittee =
|
|
||||||
let key = (slot_to_epoch(slot).uint64, false)
|
|
||||||
if key in cache.crosslink_committee_cache:
|
|
||||||
for v in cache.crosslink_committee_cache[key]: yield v
|
|
||||||
#debugEcho "get_crosslink_committees_at_slot_cached: MISS"
|
|
||||||
let result = get_crosslink_committees_at_slot(state, slot)
|
|
||||||
cache.crosslink_committee_cache[key] = result
|
|
||||||
for v in result: yield v
|
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_beacon_proposer_index
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#get_beacon_proposer_index
|
||||||
func get_beacon_proposer_index*(state: BeaconState): ValidatorIndex =
|
func get_beacon_proposer_index*(state: BeaconState, stateCache: var StateCache):
|
||||||
|
ValidatorIndex =
|
||||||
# Return the current beacon proposer index.
|
# Return the current beacon proposer index.
|
||||||
const
|
const
|
||||||
MAX_RANDOM_BYTE = 255
|
MAX_RANDOM_BYTE = 255
|
||||||
@ -224,7 +179,7 @@ func get_beacon_proposer_index*(state: BeaconState): ValidatorIndex =
|
|||||||
get_epoch_committee_count(state, epoch) div SLOTS_PER_EPOCH
|
get_epoch_committee_count(state, epoch) div SLOTS_PER_EPOCH
|
||||||
offset = committees_per_slot * (state.slot mod SLOTS_PER_EPOCH)
|
offset = committees_per_slot * (state.slot mod SLOTS_PER_EPOCH)
|
||||||
shard = (get_epoch_start_shard(state, epoch) + offset) mod SHARD_COUNT
|
shard = (get_epoch_start_shard(state, epoch) + offset) mod SHARD_COUNT
|
||||||
first_committee = get_crosslink_committee(state, epoch, shard)
|
first_committee = get_crosslink_committee(state, epoch, shard, stateCache)
|
||||||
seed = generate_seed(state, epoch)
|
seed = generate_seed(state, epoch)
|
||||||
|
|
||||||
var
|
var
|
||||||
|
@ -37,7 +37,8 @@ import
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#block-header
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#block-header
|
||||||
proc processBlockHeader(
|
proc processBlockHeader(
|
||||||
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags): bool =
|
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
# Verify that the slots match
|
# Verify that the slots match
|
||||||
if not (blck.slot == state.slot):
|
if not (blck.slot == state.slot):
|
||||||
notice "Block header: slot mismatch",
|
notice "Block header: slot mismatch",
|
||||||
@ -62,7 +63,8 @@ proc processBlockHeader(
|
|||||||
)
|
)
|
||||||
|
|
||||||
# Verify proposer is not slashed
|
# Verify proposer is not slashed
|
||||||
let proposer = state.validator_registry[get_beacon_proposer_index(state)]
|
let proposer =
|
||||||
|
state.validator_registry[get_beacon_proposer_index(state, stateCache)]
|
||||||
if proposer.slashed:
|
if proposer.slashed:
|
||||||
notice "Block header: proposer slashed"
|
notice "Block header: proposer slashed"
|
||||||
return false
|
return false
|
||||||
@ -83,9 +85,10 @@ proc processBlockHeader(
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#randao
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#randao
|
||||||
proc processRandao(
|
proc processRandao(
|
||||||
state: var BeaconState, body: BeaconBlockBody, flags: UpdateFlags): bool =
|
state: var BeaconState, body: BeaconBlockBody, flags: UpdateFlags,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
let
|
let
|
||||||
proposer_index = get_beacon_proposer_index(state)
|
proposer_index = get_beacon_proposer_index(state, stateCache)
|
||||||
proposer = addr state.validator_registry[proposer_index]
|
proposer = addr state.validator_registry[proposer_index]
|
||||||
|
|
||||||
# Verify that the provided randao value is valid
|
# Verify that the provided randao value is valid
|
||||||
@ -128,7 +131,8 @@ func is_slashable_validator(validator: Validator, epoch: Epoch): bool =
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#proposer-slashings
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#proposer-slashings
|
||||||
proc processProposerSlashings(
|
proc processProposerSlashings(
|
||||||
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags): bool =
|
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
if len(blck.body.proposer_slashings) > MAX_PROPOSER_SLASHINGS:
|
if len(blck.body.proposer_slashings) > MAX_PROPOSER_SLASHINGS:
|
||||||
notice "PropSlash: too many!",
|
notice "PropSlash: too many!",
|
||||||
proposer_slashings = len(blck.body.proposer_slashings)
|
proposer_slashings = len(blck.body.proposer_slashings)
|
||||||
@ -166,7 +170,8 @@ proc processProposerSlashings(
|
|||||||
signature_index = i
|
signature_index = i
|
||||||
return false
|
return false
|
||||||
|
|
||||||
slashValidator(state, proposer_slashing.proposer_index.ValidatorIndex)
|
slashValidator(
|
||||||
|
state, proposer_slashing.proposer_index.ValidatorIndex, stateCache)
|
||||||
|
|
||||||
true
|
true
|
||||||
|
|
||||||
@ -183,7 +188,8 @@ func is_slashable_attestation_data(
|
|||||||
data_2.target_epoch < data_1.target_epoch)
|
data_2.target_epoch < data_1.target_epoch)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#attester-slashings
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#attester-slashings
|
||||||
proc processAttesterSlashings(state: var BeaconState, blck: BeaconBlock): bool =
|
proc processAttesterSlashings(state: var BeaconState, blck: BeaconBlock,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
# Process ``AttesterSlashing`` operation.
|
# Process ``AttesterSlashing`` operation.
|
||||||
if len(blck.body.attester_slashings) > MAX_ATTESTER_SLASHINGS:
|
if len(blck.body.attester_slashings) > MAX_ATTESTER_SLASHINGS:
|
||||||
notice "CaspSlash: too many!"
|
notice "CaspSlash: too many!"
|
||||||
@ -221,13 +227,14 @@ proc processAttesterSlashings(state: var BeaconState, blck: BeaconBlock): bool =
|
|||||||
toSet(attesting_indices_2)).items), system.cmp):
|
toSet(attesting_indices_2)).items), system.cmp):
|
||||||
if is_slashable_validator(state.validator_registry[index.int],
|
if is_slashable_validator(state.validator_registry[index.int],
|
||||||
get_current_epoch(state)):
|
get_current_epoch(state)):
|
||||||
slash_validator(state, index.ValidatorIndex)
|
slash_validator(state, index.ValidatorIndex, stateCache)
|
||||||
slashed_any = true
|
slashed_any = true
|
||||||
result = result and slashed_any
|
result = result and slashed_any
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#attestations
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#attestations
|
||||||
proc processAttestations(
|
proc processAttestations(
|
||||||
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags): bool =
|
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
## Each block includes a number of attestations that the proposer chose. Each
|
## Each block includes a number of attestations that the proposer chose. Each
|
||||||
## attestation represents an update to a specific shard and is signed by a
|
## attestation represents an update to a specific shard and is signed by a
|
||||||
## committee of validators.
|
## committee of validators.
|
||||||
@ -238,7 +245,7 @@ proc processAttestations(
|
|||||||
notice "Attestation: too many!", attestations = blck.body.attestations.len
|
notice "Attestation: too many!", attestations = blck.body.attestations.len
|
||||||
return false
|
return false
|
||||||
|
|
||||||
if not blck.body.attestations.allIt(checkAttestation(state, it, flags)):
|
if not blck.body.attestations.allIt(checkAttestation(state, it, flags, stateCache)):
|
||||||
return false
|
return false
|
||||||
|
|
||||||
# All checks passed - update state
|
# All checks passed - update state
|
||||||
@ -262,7 +269,7 @@ proc processAttestations(
|
|||||||
data: attestation.data,
|
data: attestation.data,
|
||||||
aggregation_bitfield: attestation.aggregation_bitfield,
|
aggregation_bitfield: attestation.aggregation_bitfield,
|
||||||
inclusion_delay: state.slot - attestation_slot,
|
inclusion_delay: state.slot - attestation_slot,
|
||||||
proposer_index: get_beacon_proposer_index(state),
|
proposer_index: get_beacon_proposer_index(state, stateCache),
|
||||||
)
|
)
|
||||||
|
|
||||||
if attestation.data.target_epoch == get_current_epoch(state):
|
if attestation.data.target_epoch == get_current_epoch(state):
|
||||||
@ -334,7 +341,7 @@ proc processVoluntaryExits(
|
|||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#transfers
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#transfers
|
||||||
proc processTransfers(state: var BeaconState, blck: BeaconBlock,
|
proc processTransfers(state: var BeaconState, blck: BeaconBlock,
|
||||||
flags: UpdateFlags): bool =
|
flags: UpdateFlags, stateCache: var StateCache): bool =
|
||||||
if not (len(blck.body.transfers) <= MAX_TRANSFERS):
|
if not (len(blck.body.transfers) <= MAX_TRANSFERS):
|
||||||
notice "Transfer: too many transfers"
|
notice "Transfer: too many transfers"
|
||||||
return false
|
return false
|
||||||
@ -387,7 +394,8 @@ proc processTransfers(state: var BeaconState, blck: BeaconBlock,
|
|||||||
state, transfer.sender.ValidatorIndex, transfer.amount + transfer.fee)
|
state, transfer.sender.ValidatorIndex, transfer.amount + transfer.fee)
|
||||||
increase_balance(
|
increase_balance(
|
||||||
state, transfer.recipient.ValidatorIndex, transfer.amount)
|
state, transfer.recipient.ValidatorIndex, transfer.amount)
|
||||||
increase_balance(state, get_beacon_proposer_index(state), transfer.fee)
|
increase_balance(
|
||||||
|
state, get_beacon_proposer_index(state, stateCache), transfer.fee)
|
||||||
|
|
||||||
# Verify balances are not dust
|
# Verify balances are not dust
|
||||||
if not (
|
if not (
|
||||||
@ -429,32 +437,33 @@ func process_slot(state: var BeaconState) =
|
|||||||
signing_root(state.latest_block_header)
|
signing_root(state.latest_block_header)
|
||||||
|
|
||||||
proc processBlock(
|
proc processBlock(
|
||||||
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags): bool =
|
state: var BeaconState, blck: BeaconBlock, flags: UpdateFlags,
|
||||||
|
stateCache: var StateCache): bool =
|
||||||
## When there's a new block, we need to verify that the block is sane and
|
## When there's a new block, we need to verify that the block is sane and
|
||||||
## update the state accordingly
|
## update the state accordingly
|
||||||
|
|
||||||
# TODO when there's a failure, we should reset the state!
|
# TODO when there's a failure, we should reset the state!
|
||||||
# TODO probably better to do all verification first, then apply state changes
|
# TODO probably better to do all verification first, then apply state changes
|
||||||
|
|
||||||
if not processBlockHeader(state, blck, flags):
|
if not processBlockHeader(state, blck, flags, stateCache):
|
||||||
notice "Block header not valid", slot = humaneSlotNum(state.slot)
|
notice "Block header not valid", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
if not processRandao(state, blck.body, flags):
|
if not processRandao(state, blck.body, flags, stateCache):
|
||||||
debug "[Block processing] Randao failure", slot = humaneSlotNum(state.slot)
|
debug "[Block processing] Randao failure", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
processEth1Data(state, blck.body)
|
processEth1Data(state, blck.body)
|
||||||
|
|
||||||
if not processProposerSlashings(state, blck, flags):
|
if not processProposerSlashings(state, blck, flags, stateCache):
|
||||||
debug "[Block processing] Proposer slashing failure", slot = humaneSlotNum(state.slot)
|
debug "[Block processing] Proposer slashing failure", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
if not processAttesterSlashings(state, blck):
|
if not processAttesterSlashings(state, blck, stateCache):
|
||||||
debug "[Block processing] Attester slashing failure", slot = humaneSlotNum(state.slot)
|
debug "[Block processing] Attester slashing failure", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
if not processAttestations(state, blck, flags):
|
if not processAttestations(state, blck, flags, stateCache):
|
||||||
debug "[Block processing] Attestation processing failure", slot = humaneSlotNum(state.slot)
|
debug "[Block processing] Attestation processing failure", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
@ -466,26 +475,12 @@ proc processBlock(
|
|||||||
debug "[Block processing] Exit processing failure", slot = humaneSlotNum(state.slot)
|
debug "[Block processing] Exit processing failure", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
if not processTransfers(state, blck, flags):
|
if not processTransfers(state, blck, flags, stateCache):
|
||||||
debug "[Block processing] Transfer processing failure", slot = humaneSlotNum(state.slot)
|
debug "[Block processing] Transfer processing failure", slot = humaneSlotNum(state.slot)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
true
|
true
|
||||||
|
|
||||||
# TODO this cached version corresponds to the blob/v0.5.1ish get_attesting_indices
|
|
||||||
# rm/make consistent with 0.6 version above
|
|
||||||
func get_attesting_indices_cached(
|
|
||||||
state: BeaconState,
|
|
||||||
attestations: openArray[PendingAttestation], cache: var StateCache):
|
|
||||||
HashSet[ValidatorIndex] =
|
|
||||||
# Union of attesters that participated in some attestations
|
|
||||||
result = initSet[ValidatorIndex]()
|
|
||||||
for attestation in attestations:
|
|
||||||
for validator_index in get_attestation_participants_cached(
|
|
||||||
state, attestation.data, attestation.aggregation_bitfield,
|
|
||||||
cache):
|
|
||||||
result.incl validator_index
|
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#helper-functions-1
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#helper-functions-1
|
||||||
func get_total_active_balance(state: BeaconState): Gwei =
|
func get_total_active_balance(state: BeaconState): Gwei =
|
||||||
return get_total_balance(
|
return get_total_balance(
|
||||||
@ -516,26 +511,22 @@ func get_matching_head_attestations(state: BeaconState, epoch: Epoch):
|
|||||||
)
|
)
|
||||||
|
|
||||||
func get_unslashed_attesting_indices(
|
func get_unslashed_attesting_indices(
|
||||||
state: BeaconState, attestations: seq[PendingAttestation]):
|
state: BeaconState, attestations: seq[PendingAttestation],
|
||||||
HashSet[ValidatorIndex] =
|
stateCache: var StateCache): HashSet[ValidatorIndex] =
|
||||||
result = initSet[ValidatorIndex]()
|
result = initSet[ValidatorIndex]()
|
||||||
for a in attestations:
|
for a in attestations:
|
||||||
result = result.union(get_attesting_indices(
|
result = result.union(get_attesting_indices(
|
||||||
state, a.data, a.aggregation_bitfield))
|
state, a.data, a.aggregation_bitfield, stateCache))
|
||||||
|
|
||||||
for index in result:
|
for index in result:
|
||||||
if state.validator_registry[index].slashed:
|
if state.validator_registry[index].slashed:
|
||||||
result.excl index
|
result.excl index
|
||||||
|
|
||||||
func get_attesting_balance(
|
func get_attesting_balance(
|
||||||
state: BeaconState, attestations: seq[PendingAttestation]): Gwei =
|
|
||||||
get_total_balance(state, get_unslashed_attesting_indices(state, attestations))
|
|
||||||
|
|
||||||
func get_attesting_balance_cached(
|
|
||||||
state: BeaconState, attestations: seq[PendingAttestation],
|
state: BeaconState, attestations: seq[PendingAttestation],
|
||||||
cache: var StateCache): Gwei =
|
stateCache: var StateCache): Gwei =
|
||||||
get_total_balance(state, get_attesting_indices_cached(
|
get_total_balance(state, get_unslashed_attesting_indices(
|
||||||
state, attestations, cache))
|
state, attestations, stateCache))
|
||||||
|
|
||||||
func get_crosslink_from_attestation_data(
|
func get_crosslink_from_attestation_data(
|
||||||
state: BeaconState, data: AttestationData): Crosslink =
|
state: BeaconState, data: AttestationData): Crosslink =
|
||||||
@ -554,9 +545,9 @@ func lowerThan(candidate, current: Eth2Digest): bool =
|
|||||||
if v > candidate.data[i]: return true
|
if v > candidate.data[i]: return true
|
||||||
false
|
false
|
||||||
|
|
||||||
# TODO check/profile if should add cache: var StateCache param
|
|
||||||
func get_winning_crosslink_and_attesting_indices(
|
func get_winning_crosslink_and_attesting_indices(
|
||||||
state: BeaconState, epoch: Epoch, shard: Shard): tuple[a: Crosslink, b: HashSet[ValidatorIndex]] =
|
state: BeaconState, epoch: Epoch, shard: Shard,
|
||||||
|
stateCache: var StateCache): tuple[a: Crosslink, b: HashSet[ValidatorIndex]] =
|
||||||
let
|
let
|
||||||
## TODO Z-F could help here
|
## TODO Z-F could help here
|
||||||
## TODO get_winning_crosslink_and_attesting_indices was profiling hotspot
|
## TODO get_winning_crosslink_and_attesting_indices was profiling hotspot
|
||||||
@ -602,7 +593,8 @@ func get_winning_crosslink_and_attesting_indices(
|
|||||||
## let root_balance = get_attesting_balance_cached(
|
## let root_balance = get_attesting_balance_cached(
|
||||||
## state, attestations_for.getOrDefault(r), cache)
|
## state, attestations_for.getOrDefault(r), cache)
|
||||||
let crosslink_balance =
|
let crosslink_balance =
|
||||||
get_attesting_balance(state, get_attestations_for(candidate_crosslink))
|
get_attesting_balance(
|
||||||
|
state, get_attestations_for(candidate_crosslink), stateCache)
|
||||||
if (crosslink_balance > winning_crosslink_balance or
|
if (crosslink_balance > winning_crosslink_balance or
|
||||||
(winning_crosslink_balance == crosslink_balance and
|
(winning_crosslink_balance == crosslink_balance and
|
||||||
lowerThan(winning_crosslink.crosslink_data_root,
|
lowerThan(winning_crosslink.crosslink_data_root,
|
||||||
@ -612,10 +604,11 @@ func get_winning_crosslink_and_attesting_indices(
|
|||||||
|
|
||||||
(winning_crosslink,
|
(winning_crosslink,
|
||||||
get_unslashed_attesting_indices(state,
|
get_unslashed_attesting_indices(state,
|
||||||
get_attestations_for(winning_crosslink)))
|
get_attestations_for(winning_crosslink), stateCache))
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#justification-and-finalization
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#justification-and-finalization
|
||||||
func process_justification_and_finalization(state: var BeaconState) =
|
func process_justification_and_finalization(
|
||||||
|
state: var BeaconState, stateCache: var StateCache) =
|
||||||
if get_current_epoch(state) <= GENESIS_EPOCH + 1:
|
if get_current_epoch(state) <= GENESIS_EPOCH + 1:
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -631,7 +624,7 @@ func process_justification_and_finalization(state: var BeaconState) =
|
|||||||
state.justification_bitfield = (state.justification_bitfield shl 1)
|
state.justification_bitfield = (state.justification_bitfield shl 1)
|
||||||
let previous_epoch_matching_target_balance =
|
let previous_epoch_matching_target_balance =
|
||||||
get_attesting_balance(state,
|
get_attesting_balance(state,
|
||||||
get_matching_target_attestations(state, previous_epoch))
|
get_matching_target_attestations(state, previous_epoch), stateCache)
|
||||||
if previous_epoch_matching_target_balance * 3 >=
|
if previous_epoch_matching_target_balance * 3 >=
|
||||||
get_total_active_balance(state) * 2:
|
get_total_active_balance(state) * 2:
|
||||||
state.current_justified_epoch = previous_epoch
|
state.current_justified_epoch = previous_epoch
|
||||||
@ -640,7 +633,8 @@ func process_justification_and_finalization(state: var BeaconState) =
|
|||||||
state.justification_bitfield = state.justification_bitfield or (1 shl 1)
|
state.justification_bitfield = state.justification_bitfield or (1 shl 1)
|
||||||
let current_epoch_matching_target_balance =
|
let current_epoch_matching_target_balance =
|
||||||
get_attesting_balance(state,
|
get_attesting_balance(state,
|
||||||
get_matching_target_attestations(state, current_epoch))
|
get_matching_target_attestations(state, current_epoch),
|
||||||
|
stateCache)
|
||||||
if current_epoch_matching_target_balance * 3 >=
|
if current_epoch_matching_target_balance * 3 >=
|
||||||
get_total_active_balance(state) * 2:
|
get_total_active_balance(state) * 2:
|
||||||
state.current_justified_epoch = current_epoch
|
state.current_justified_epoch = current_epoch
|
||||||
@ -680,7 +674,7 @@ func process_justification_and_finalization(state: var BeaconState) =
|
|||||||
state.finalized_root = get_block_root(state, state.finalized_epoch)
|
state.finalized_root = get_block_root(state, state.finalized_epoch)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#crosslinks
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#crosslinks
|
||||||
func process_crosslinks(state: var BeaconState, per_epoch_cache: var StateCache) =
|
func process_crosslinks(state: var BeaconState, stateCache: var StateCache) =
|
||||||
## TODO is there a semantic reason for this, or is this just a way to force
|
## TODO is there a semantic reason for this, or is this just a way to force
|
||||||
## copying? If so, why not just `list(foo)` or similar? This is strange. In
|
## copying? If so, why not just `list(foo)` or similar? This is strange. In
|
||||||
## this case, for type reasons, don't do weird
|
## this case, for type reasons, don't do weird
|
||||||
@ -695,13 +689,15 @@ func process_crosslinks(state: var BeaconState, per_epoch_cache: var StateCache)
|
|||||||
for offset in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
for offset in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
||||||
let
|
let
|
||||||
shard = (get_epoch_start_shard(state, epoch) + offset) mod SHARD_COUNT
|
shard = (get_epoch_start_shard(state, epoch) + offset) mod SHARD_COUNT
|
||||||
crosslink_committee = get_crosslink_committee(state, epoch, shard)
|
crosslink_committee =
|
||||||
|
get_crosslink_committee(state, epoch, shard, stateCache)
|
||||||
# In general, it'll loop over the same shards twice, and
|
# In general, it'll loop over the same shards twice, and
|
||||||
# get_winning_root_and_participants is defined to return
|
# get_winning_root_and_participants is defined to return
|
||||||
# the same results from the previous epoch as current.
|
# the same results from the previous epoch as current.
|
||||||
# TODO cache like before, in 0.5 version of this function
|
# TODO cache like before, in 0.5 version of this function
|
||||||
(winning_crosslink, attesting_indices) =
|
(winning_crosslink, attesting_indices) =
|
||||||
get_winning_crosslink_and_attesting_indices(state, epoch, shard)
|
get_winning_crosslink_and_attesting_indices(
|
||||||
|
state, epoch, shard, stateCache)
|
||||||
if 3'u64 * get_total_balance(state, attesting_indices) >=
|
if 3'u64 * get_total_balance(state, attesting_indices) >=
|
||||||
2'u64 * get_total_balance(state, crosslink_committee):
|
2'u64 * get_total_balance(state, crosslink_committee):
|
||||||
state.current_crosslinks[shard] = winning_crosslink
|
state.current_crosslinks[shard] = winning_crosslink
|
||||||
@ -715,7 +711,7 @@ func get_base_reward(state: BeaconState, index: ValidatorIndex): Gwei =
|
|||||||
integer_squareroot(total_balance) div BASE_REWARDS_PER_EPOCH
|
integer_squareroot(total_balance) div BASE_REWARDS_PER_EPOCH
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#rewards-and-penalties
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#rewards-and-penalties
|
||||||
func get_attestation_deltas(state: BeaconState):
|
func get_attestation_deltas(state: BeaconState, stateCache: var StateCache):
|
||||||
tuple[a: seq[Gwei], b: seq[Gwei]] =
|
tuple[a: seq[Gwei], b: seq[Gwei]] =
|
||||||
let
|
let
|
||||||
previous_epoch = get_previous_epoch(state)
|
previous_epoch = get_previous_epoch(state)
|
||||||
@ -743,8 +739,8 @@ func get_attestation_deltas(state: BeaconState):
|
|||||||
matching_head_attestations]:
|
matching_head_attestations]:
|
||||||
let
|
let
|
||||||
unslashed_attesting_indices =
|
unslashed_attesting_indices =
|
||||||
get_unslashed_attesting_indices(state, attestations)
|
get_unslashed_attesting_indices(state, attestations, stateCache)
|
||||||
attesting_balance = get_attesting_balance(state, attestations)
|
attesting_balance = get_attesting_balance(state, attestations, stateCache)
|
||||||
for index in eligible_validator_indices:
|
for index in eligible_validator_indices:
|
||||||
if index in unslashed_attesting_indices:
|
if index in unslashed_attesting_indices:
|
||||||
rewards[index] +=
|
rewards[index] +=
|
||||||
@ -756,11 +752,13 @@ func get_attestation_deltas(state: BeaconState):
|
|||||||
return (rewards, penalties)
|
return (rewards, penalties)
|
||||||
|
|
||||||
# Proposer and inclusion delay micro-rewards
|
# Proposer and inclusion delay micro-rewards
|
||||||
for index in get_unslashed_attesting_indices(state, matching_source_attestations):
|
for index in get_unslashed_attesting_indices(
|
||||||
|
state, matching_source_attestations, stateCache):
|
||||||
doAssert matching_source_attestations.len > 0
|
doAssert matching_source_attestations.len > 0
|
||||||
var attestation = matching_source_attestations[0]
|
var attestation = matching_source_attestations[0]
|
||||||
for a in matching_source_attestations:
|
for a in matching_source_attestations:
|
||||||
if index notin get_attesting_indices(state, a.data, a.aggregation_bitfield):
|
if index notin get_attesting_indices(
|
||||||
|
state, a.data, a.aggregation_bitfield, stateCache):
|
||||||
continue
|
continue
|
||||||
if a.inclusion_delay < attestation.inclusion_delay:
|
if a.inclusion_delay < attestation.inclusion_delay:
|
||||||
attestation = a
|
attestation = a
|
||||||
@ -774,7 +772,8 @@ func get_attestation_deltas(state: BeaconState):
|
|||||||
let finality_delay = previous_epoch - state.finalized_epoch
|
let finality_delay = previous_epoch - state.finalized_epoch
|
||||||
if finality_delay > MIN_EPOCHS_TO_INACTIVITY_PENALTY:
|
if finality_delay > MIN_EPOCHS_TO_INACTIVITY_PENALTY:
|
||||||
let matching_target_attesting_indices =
|
let matching_target_attesting_indices =
|
||||||
get_unslashed_attesting_indices(state, matching_target_attestations)
|
get_unslashed_attesting_indices(
|
||||||
|
state, matching_target_attestations, stateCache)
|
||||||
for index in eligible_validator_indices:
|
for index in eligible_validator_indices:
|
||||||
penalties[index] +=
|
penalties[index] +=
|
||||||
BASE_REWARDS_PER_EPOCH.uint64 * get_base_reward(state, index)
|
BASE_REWARDS_PER_EPOCH.uint64 * get_base_reward(state, index)
|
||||||
@ -796,9 +795,11 @@ func get_crosslink_deltas(state: BeaconState, cache: var StateCache):
|
|||||||
for offset in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
for offset in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
||||||
let
|
let
|
||||||
shard = (get_epoch_start_shard(state, epoch) + offset) mod SHARD_COUNT
|
shard = (get_epoch_start_shard(state, epoch) + offset) mod SHARD_COUNT
|
||||||
crosslink_committee = get_crosslink_committee(state, epoch, shard)
|
crosslink_committee =
|
||||||
|
get_crosslink_committee(state, epoch, shard, cache)
|
||||||
(winning_crosslink, attesting_indices) =
|
(winning_crosslink, attesting_indices) =
|
||||||
get_winning_crosslink_and_attesting_indices(state, epoch, shard)
|
get_winning_crosslink_and_attesting_indices(
|
||||||
|
state, epoch, shard, cache)
|
||||||
attesting_balance = get_total_balance(state, attesting_indices)
|
attesting_balance = get_total_balance(state, attesting_indices)
|
||||||
committee_balance = get_total_balance(state, crosslink_committee)
|
committee_balance = get_total_balance(state, crosslink_committee)
|
||||||
for index in crosslink_committee:
|
for index in crosslink_committee:
|
||||||
@ -819,7 +820,7 @@ func process_rewards_and_penalties(
|
|||||||
return
|
return
|
||||||
|
|
||||||
let
|
let
|
||||||
(rewards1, penalties1) = get_attestation_deltas(state)
|
(rewards1, penalties1) = get_attestation_deltas(state, cache)
|
||||||
(rewards2, penalties2) = get_crosslink_deltas(state, cache)
|
(rewards2, penalties2) = get_crosslink_deltas(state, cache)
|
||||||
for i in 0 ..< len(state.validator_registry):
|
for i in 0 ..< len(state.validator_registry):
|
||||||
increase_balance(state, i.ValidatorIndex, rewards1[i] + rewards2[i])
|
increase_balance(state, i.ValidatorIndex, rewards1[i] + rewards2[i])
|
||||||
@ -897,22 +898,16 @@ func process_final_updates(state: var BeaconState) =
|
|||||||
state.current_epoch_attestations = @[]
|
state.current_epoch_attestations = @[]
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#per-epoch-processing
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.3/specs/core/0_beacon-chain.md#per-epoch-processing
|
||||||
func get_empty_per_epoch_cache(): StateCache =
|
|
||||||
result.crosslink_committee_cache =
|
|
||||||
initTable[tuple[a: uint64, b: bool], seq[CrosslinkCommittee]]()
|
|
||||||
result.winning_root_participants_cache =
|
|
||||||
initTable[Shard, HashSet[ValidatorIndex]]()
|
|
||||||
|
|
||||||
func processEpoch(state: var BeaconState) =
|
func processEpoch(state: var BeaconState) =
|
||||||
if not (state.slot > GENESIS_SLOT and
|
if not (state.slot > GENESIS_SLOT and
|
||||||
(state.slot + 1) mod SLOTS_PER_EPOCH == 0):
|
(state.slot + 1) mod SLOTS_PER_EPOCH == 0):
|
||||||
return
|
return
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#justification-and-finalization
|
|
||||||
process_justification_and_finalization(state)
|
|
||||||
|
|
||||||
var per_epoch_cache = get_empty_per_epoch_cache()
|
var per_epoch_cache = get_empty_per_epoch_cache()
|
||||||
|
|
||||||
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#justification-and-finalization
|
||||||
|
process_justification_and_finalization(state, per_epoch_cache)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#crosslinks
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#crosslinks
|
||||||
process_crosslinks(state, per_epoch_cache)
|
process_crosslinks(state, per_epoch_cache)
|
||||||
|
|
||||||
@ -920,8 +915,13 @@ func processEpoch(state: var BeaconState) =
|
|||||||
process_rewards_and_penalties(state, per_epoch_cache)
|
process_rewards_and_penalties(state, per_epoch_cache)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#registry-updates
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#registry-updates
|
||||||
|
# Don't rely on caching here.
|
||||||
process_registry_updates(state)
|
process_registry_updates(state)
|
||||||
|
|
||||||
|
## Caching here for get_crosslink_committee(...) can break otherwise, since
|
||||||
|
## get_active_validator_indices(...) usually changes.
|
||||||
|
clear(per_epoch_cache.crosslink_committee_cache)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#slashings
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.7.1/specs/core/0_beacon-chain.md#slashings
|
||||||
process_slashings(state)
|
process_slashings(state)
|
||||||
|
|
||||||
@ -1006,7 +1006,9 @@ proc updateState*(
|
|||||||
# that the block is sane.
|
# that the block is sane.
|
||||||
# TODO what should happen if block processing fails?
|
# TODO what should happen if block processing fails?
|
||||||
# https://github.com/ethereum/eth2.0-specs/issues/293
|
# https://github.com/ethereum/eth2.0-specs/issues/293
|
||||||
if processBlock(state, new_block, flags):
|
var per_epoch_cache = get_empty_per_epoch_cache()
|
||||||
|
|
||||||
|
if processBlock(state, new_block, flags, per_epoch_cache):
|
||||||
# This is a bit awkward - at the end of processing we verify that the
|
# This is a bit awkward - at the end of processing we verify that the
|
||||||
# state we arrive at is what the block producer thought it would be -
|
# state we arrive at is what the block producer thought it would be -
|
||||||
# meaning that potentially, it could fail verification
|
# meaning that potentially, it could fail verification
|
||||||
@ -1059,7 +1061,8 @@ proc updateState*(
|
|||||||
var old_state = state
|
var old_state = state
|
||||||
advanceState(state)
|
advanceState(state)
|
||||||
|
|
||||||
if processBlock(state.data, blck, flags):
|
var per_epoch_cache = get_empty_per_epoch_cache()
|
||||||
|
if processBlock(state.data, blck, flags, per_epoch_cache):
|
||||||
if skipValidation in flags or verifyStateRoot(state.data, blck):
|
if skipValidation in flags or verifyStateRoot(state.data, blck):
|
||||||
# State root is what it should be - we're done!
|
# State root is what it should be - we're done!
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ type Timers = enum
|
|||||||
tBlock = "Process non-epoch slot with block"
|
tBlock = "Process non-epoch slot with block"
|
||||||
tEpoch = "Process epoch slot with block"
|
tEpoch = "Process epoch slot with block"
|
||||||
tHashBlock = "Tree-hash block"
|
tHashBlock = "Tree-hash block"
|
||||||
tShuffle = "Retrieve committee once using get_crosslink_committees_at_slot"
|
tShuffle = "Retrieve committee once using get_crosslink_committee"
|
||||||
tAttest = "Combine committee attestations"
|
tAttest = "Combine committee attestations"
|
||||||
|
|
||||||
template withTimer(stats: var RunningStat, body: untyped) =
|
template withTimer(stats: var RunningStat, body: untyped) =
|
||||||
@ -65,6 +65,7 @@ cli do(slots = 1945,
|
|||||||
attesters: RunningStat
|
attesters: RunningStat
|
||||||
r: Rand
|
r: Rand
|
||||||
blck: BeaconBlock
|
blck: BeaconBlock
|
||||||
|
cache = get_empty_per_epoch_cache()
|
||||||
|
|
||||||
proc maybeWrite() =
|
proc maybeWrite() =
|
||||||
if state.slot mod json_interval.uint64 == 0:
|
if state.slot mod json_interval.uint64 == 0:
|
||||||
@ -96,18 +97,24 @@ cli do(slots = 1945,
|
|||||||
# attesterRatio is the fraction of attesters that actually do their
|
# attesterRatio is the fraction of attesters that actually do their
|
||||||
# work for every slot - we'll randomize it deterministically to give
|
# work for every slot - we'll randomize it deterministically to give
|
||||||
# some variation
|
# some variation
|
||||||
let scass = withTimerRet(timers[tShuffle]):
|
let
|
||||||
get_crosslink_committees_at_slot(state, state.slot)
|
epoch = slot_to_epoch(state.slot)
|
||||||
|
scass = withTimerRet(timers[tShuffle]):
|
||||||
|
mapIt(
|
||||||
|
0'u64 .. (get_epoch_committee_count(state, epoch) - 1),
|
||||||
|
get_crosslink_committee(state, epoch,
|
||||||
|
(it + get_epoch_start_shard(state, epoch)) mod SHARD_COUNT,
|
||||||
|
cache))
|
||||||
|
|
||||||
for scas in scass:
|
for scas in scass:
|
||||||
var
|
var
|
||||||
attestation: Attestation
|
attestation: Attestation
|
||||||
first = true
|
first = true
|
||||||
|
|
||||||
attesters.push scas.committee.len()
|
attesters.push scas.len()
|
||||||
|
|
||||||
withTimer(timers[tAttest]):
|
withTimer(timers[tAttest]):
|
||||||
for v in scas.committee:
|
for v in scas:
|
||||||
if (rand(r, high(int)).float * attesterRatio).int <= high(int):
|
if (rand(r, high(int)).float * attesterRatio).int <= high(int):
|
||||||
if first:
|
if first:
|
||||||
attestation = makeAttestation(state, latest_block_root, v, flags)
|
attestation = makeAttestation(state, latest_block_root, v, flags)
|
||||||
|
@ -35,13 +35,14 @@ suite "Attestation pool processing" & preset():
|
|||||||
genBlock = get_initial_beacon_block(genState)
|
genBlock = get_initial_beacon_block(genState)
|
||||||
|
|
||||||
test "Can add and retrieve simple attestation" & preset():
|
test "Can add and retrieve simple attestation" & preset():
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
withPool:
|
withPool:
|
||||||
let
|
let
|
||||||
# Create an attestation for slot 1!
|
# Create an attestation for slot 1!
|
||||||
crosslink_committees =
|
crosslink_committee = get_crosslink_committee(state.data.data,
|
||||||
get_crosslink_committees_at_slot(state.data.data, state.data.data.slot)
|
slot_to_epoch(state.data.data.slot), 0, cache)
|
||||||
attestation = makeAttestation(
|
attestation = makeAttestation(
|
||||||
state.data.data, state.blck.root, crosslink_committees[0].committee[0])
|
state.data.data, state.blck.root, crosslink_committee[0])
|
||||||
|
|
||||||
pool.add(state.data.data, attestation)
|
pool.add(state.data.data, attestation)
|
||||||
|
|
||||||
@ -55,21 +56,22 @@ suite "Attestation pool processing" & preset():
|
|||||||
attestations.len == 1
|
attestations.len == 1
|
||||||
|
|
||||||
test "Attestations may arrive in any order" & preset():
|
test "Attestations may arrive in any order" & preset():
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
withPool:
|
withPool:
|
||||||
let
|
let
|
||||||
# Create an attestation for slot 1!
|
# Create an attestation for slot 1!
|
||||||
cc0 =
|
cc0 = get_crosslink_committee(state.data.data,
|
||||||
get_crosslink_committees_at_slot(state.data.data, state.data.data.slot)
|
slot_to_epoch(state.data.data.slot), 0, cache)
|
||||||
attestation0 = makeAttestation(
|
attestation0 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[0])
|
state.data.data, state.blck.root, cc0[0])
|
||||||
|
|
||||||
advanceState(state.data)
|
advanceState(state.data)
|
||||||
|
|
||||||
let
|
let
|
||||||
cc1 =
|
cc1 = get_crosslink_committee(state.data.data,
|
||||||
get_crosslink_committees_at_slot(state.data.data, state.data.data.slot)
|
slot_to_epoch(state.data.data.slot), 0, cache)
|
||||||
attestation1 = makeAttestation(
|
attestation1 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc1[0].committee[0])
|
state.data.data, state.blck.root, cc1[0])
|
||||||
|
|
||||||
# test reverse order
|
# test reverse order
|
||||||
pool.add(state.data.data, attestation1)
|
pool.add(state.data.data, attestation1)
|
||||||
@ -84,15 +86,16 @@ suite "Attestation pool processing" & preset():
|
|||||||
attestations.len == 1
|
attestations.len == 1
|
||||||
|
|
||||||
test "Attestations should be combined" & preset():
|
test "Attestations should be combined" & preset():
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
withPool:
|
withPool:
|
||||||
let
|
let
|
||||||
# Create an attestation for slot 1!
|
# Create an attestation for slot 1!
|
||||||
cc0 =
|
cc0 = get_crosslink_committee(state.data.data,
|
||||||
get_crosslink_committees_at_slot(state.data.data, state.data.data.slot)
|
slot_to_epoch(state.data.data.slot), 0, cache)
|
||||||
attestation0 = makeAttestation(
|
attestation0 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[0])
|
state.data.data, state.blck.root, cc0[0])
|
||||||
attestation1 = makeAttestation(
|
attestation1 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[1])
|
state.data.data, state.blck.root, cc0[1])
|
||||||
|
|
||||||
pool.add(state.data.data, attestation0)
|
pool.add(state.data.data, attestation0)
|
||||||
pool.add(state.data.data, attestation1)
|
pool.add(state.data.data, attestation1)
|
||||||
@ -106,16 +109,17 @@ suite "Attestation pool processing" & preset():
|
|||||||
attestations.len == 1
|
attestations.len == 1
|
||||||
|
|
||||||
test "Attestations may overlap, bigger first" & preset():
|
test "Attestations may overlap, bigger first" & preset():
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
withPool:
|
withPool:
|
||||||
|
|
||||||
var
|
var
|
||||||
# Create an attestation for slot 1!
|
# Create an attestation for slot 1!
|
||||||
cc0 =
|
cc0 = get_crosslink_committee(state.data.data,
|
||||||
get_crosslink_committees_at_slot(state.data.data, state.data.data.slot)
|
slot_to_epoch(state.data.data.slot), 0, cache)
|
||||||
attestation0 = makeAttestation(
|
attestation0 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[0])
|
state.data.data, state.blck.root, cc0[0])
|
||||||
attestation1 = makeAttestation(
|
attestation1 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[1])
|
state.data.data, state.blck.root, cc0[1])
|
||||||
|
|
||||||
attestation0.combine(attestation1, {skipValidation})
|
attestation0.combine(attestation1, {skipValidation})
|
||||||
|
|
||||||
@ -131,15 +135,16 @@ suite "Attestation pool processing" & preset():
|
|||||||
attestations.len == 1
|
attestations.len == 1
|
||||||
|
|
||||||
test "Attestations may overlap, smaller first" & preset():
|
test "Attestations may overlap, smaller first" & preset():
|
||||||
|
var cache = get_empty_per_epoch_cache()
|
||||||
withPool:
|
withPool:
|
||||||
var
|
var
|
||||||
# Create an attestation for slot 1!
|
# Create an attestation for slot 1!
|
||||||
cc0 =
|
cc0 = get_crosslink_committee(state.data.data,
|
||||||
get_crosslink_committees_at_slot(state.data.data, state.data.data.slot)
|
slot_to_epoch(state.data.data.slot), 0, cache)
|
||||||
attestation0 = makeAttestation(
|
attestation0 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[0])
|
state.data.data, state.blck.root, cc0[0])
|
||||||
attestation1 = makeAttestation(
|
attestation1 = makeAttestation(
|
||||||
state.data.data, state.blck.root, cc0[0].committee[1])
|
state.data.data, state.blck.root, cc0[1])
|
||||||
|
|
||||||
attestation0.combine(attestation1, {skipValidation})
|
attestation0.combine(attestation1, {skipValidation})
|
||||||
|
|
||||||
|
@ -76,16 +76,17 @@ suite "Block processing" & preset():
|
|||||||
var
|
var
|
||||||
state = genesisState
|
state = genesisState
|
||||||
previous_block_root = genesisRoot
|
previous_block_root = genesisRoot
|
||||||
|
cache = get_empty_per_epoch_cache()
|
||||||
|
|
||||||
# Slot 0 is a finalized slot - won't be making attestations for it..
|
# Slot 0 is a finalized slot - won't be making attestations for it..
|
||||||
advanceState(state)
|
advanceState(state)
|
||||||
|
|
||||||
let
|
let
|
||||||
# Create an attestation for slot 1 signed by the only attester we have!
|
# Create an attestation for slot 1 signed by the only attester we have!
|
||||||
crosslink_committees = get_crosslink_committees_at_slot(state, state.slot)
|
crosslink_committee =
|
||||||
|
get_crosslink_committee(state, slot_to_epoch(state.slot), 0, cache)
|
||||||
attestation = makeAttestation(
|
attestation = makeAttestation(
|
||||||
state, previous_block_root,
|
state, previous_block_root, crosslink_committee[0])
|
||||||
crosslink_committees[0].committee[0])
|
|
||||||
|
|
||||||
# Some time needs to pass before attestations are included - this is
|
# Some time needs to pass before attestations are included - this is
|
||||||
# to let the attestation propagate properly to interested participants
|
# to let the attestation propagate properly to interested participants
|
||||||
|
@ -63,9 +63,12 @@ func getNextBeaconProposerIndex*(state: BeaconState): ValidatorIndex =
|
|||||||
# TODO: This is a special version of get_beacon_proposer_index that takes into
|
# TODO: This is a special version of get_beacon_proposer_index that takes into
|
||||||
# account the partial update done at the start of slot processing -
|
# account the partial update done at the start of slot processing -
|
||||||
# see get_shard_committees_index
|
# see get_shard_committees_index
|
||||||
var next_state = state
|
var
|
||||||
|
next_state = state
|
||||||
|
cache = get_empty_per_epoch_cache()
|
||||||
|
|
||||||
next_state.slot += 1
|
next_state.slot += 1
|
||||||
get_beacon_proposer_index(next_state)
|
get_beacon_proposer_index(next_state, cache)
|
||||||
|
|
||||||
proc addBlock*(
|
proc addBlock*(
|
||||||
state: var BeaconState, previous_block_root: Eth2Digest,
|
state: var BeaconState, previous_block_root: Eth2Digest,
|
||||||
@ -76,7 +79,8 @@ proc addBlock*(
|
|||||||
# but avoids some slow block copies
|
# but avoids some slow block copies
|
||||||
|
|
||||||
state.slot += 1
|
state.slot += 1
|
||||||
let proposer_index = get_beacon_proposer_index(state)
|
var cache = get_empty_per_epoch_cache()
|
||||||
|
let proposer_index = get_beacon_proposer_index(state, cache)
|
||||||
state.slot -= 1
|
state.slot -= 1
|
||||||
|
|
||||||
let
|
let
|
||||||
@ -138,25 +142,29 @@ proc makeBlock*(
|
|||||||
addBlock(next_state, previous_block_root, body)
|
addBlock(next_state, previous_block_root, body)
|
||||||
|
|
||||||
proc find_shard_committee(
|
proc find_shard_committee(
|
||||||
sacs: openArray[CrosslinkCommittee], validator_index: ValidatorIndex): CrosslinkCommittee =
|
state: BeaconState, validator_index: ValidatorIndex): auto =
|
||||||
for sac in sacs:
|
let epoch = slot_to_epoch(state.slot)
|
||||||
if validator_index in sac.committee: return sac
|
var cache = get_empty_per_epoch_cache()
|
||||||
|
for shard in 0'u64 ..< get_epoch_committee_count(state, epoch):
|
||||||
|
let committee = get_crosslink_committee(state, epoch,
|
||||||
|
(shard + get_epoch_start_shard(state, epoch)) mod SHARD_COUNT, cache)
|
||||||
|
if validator_index in committee:
|
||||||
|
return (committee, shard)
|
||||||
doAssert false
|
doAssert false
|
||||||
|
|
||||||
proc makeAttestation*(
|
proc makeAttestation*(
|
||||||
state: BeaconState, beacon_block_root: Eth2Digest,
|
state: BeaconState, beacon_block_root: Eth2Digest,
|
||||||
validator_index: ValidatorIndex, flags: UpdateFlags = {}): Attestation =
|
validator_index: ValidatorIndex, flags: UpdateFlags = {}): Attestation =
|
||||||
let
|
let
|
||||||
sac = find_shard_committee(
|
(committee, shard) = find_shard_committee(state, validator_index)
|
||||||
get_crosslink_committees_at_slot(state, state.slot), validator_index)
|
|
||||||
validator = state.validator_registry[validator_index]
|
validator = state.validator_registry[validator_index]
|
||||||
sac_index = sac.committee.find(validator_index)
|
sac_index = committee.find(validator_index)
|
||||||
data = makeAttestationData(state, sac.shard, beacon_block_root)
|
data = makeAttestationData(state, shard, beacon_block_root)
|
||||||
|
|
||||||
doAssert sac_index != -1, "find_shard_committe should guarantee this"
|
doAssert sac_index != -1, "find_shard_committee should guarantee this"
|
||||||
|
|
||||||
var
|
var
|
||||||
aggregation_bitfield = BitField.init(sac.committee.len)
|
aggregation_bitfield = BitField.init(committee.len)
|
||||||
set_bitfield_bit(aggregation_bitfield, sac_index)
|
set_bitfield_bit(aggregation_bitfield, sac_index)
|
||||||
|
|
||||||
let
|
let
|
||||||
@ -177,7 +185,7 @@ proc makeAttestation*(
|
|||||||
data: data,
|
data: data,
|
||||||
aggregation_bitfield: aggregation_bitfield,
|
aggregation_bitfield: aggregation_bitfield,
|
||||||
signature: sig,
|
signature: sig,
|
||||||
custody_bitfield: BitField.init(sac.committee.len)
|
custody_bitfield: BitField.init(committee.len)
|
||||||
)
|
)
|
||||||
|
|
||||||
proc makeTestDB*(tailState: BeaconState, tailBlock: BeaconBlock): BeaconChainDB =
|
proc makeTestDB*(tailState: BeaconState, tailBlock: BeaconBlock): BeaconChainDB =
|
||||||
|
Loading…
x
Reference in New Issue
Block a user