2020-01-27 12:36:17 +01:00
|
|
|
# Copyright (c) 2018-2020 Status Research & Development GmbH
|
2018-11-23 16:44:43 -06:00
|
|
|
# Licensed and distributed under either of
|
2019-11-25 15:30:02 +00:00
|
|
|
# * 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).
|
2018-11-23 16:44:43 -06:00
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
|
|
|
# Helpers and functions pertaining to managing the validator set
|
|
|
|
|
2020-04-22 07:53:02 +02:00
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2018-11-23 13:42:47 -06:00
|
|
|
import
|
speed up shuffling
Replace shuffling function with zrnt version - `get_shuffled_seq` in
particular puts more strain on the GC by allocating superfluous seq's
which turns out to have a significant impact on block processing (when
replaying blocks for example) - 4x improvement on non-epoch, 1.5x on
epoch blocks (replay is done without signature checking)
Medalla, first 10k slots - pre:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
76855.848, 0.000, 76855.848, 76855.848, 1,
Initialize DB
1.073, 0.914, 0.071, 12.454, 7831,
Load block from database
31.382, 0.000, 31.382, 31.382, 1,
Load state from database
85.644, 30.350, 3.056, 466.136, 7519,
Apply block
506.569, 91.129, 130.654, 874.786, 312,
Apply epoch block
```
post:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
72457.303, 0.000, 72457.303, 72457.303, 1,
Initialize DB
1.015, 0.858, 0.070, 11.231, 7831,
Load block from database
28.983, 0.000, 28.983, 28.983, 1,
Load state from database
21.725, 17.461, 2.659, 393.217, 7519,
Apply block
324.012, 33.954, 45.452, 440.532, 312,
Apply epoch block
```
2020-08-21 12:06:26 +02:00
|
|
|
options, math, tables,
|
2020-07-27 18:04:44 +02:00
|
|
|
./datatypes, ./digest, ./helpers
|
2018-11-23 13:42:47 -06:00
|
|
|
|
speed up shuffling
Replace shuffling function with zrnt version - `get_shuffled_seq` in
particular puts more strain on the GC by allocating superfluous seq's
which turns out to have a significant impact on block processing (when
replaying blocks for example) - 4x improvement on non-epoch, 1.5x on
epoch blocks (replay is done without signature checking)
Medalla, first 10k slots - pre:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
76855.848, 0.000, 76855.848, 76855.848, 1,
Initialize DB
1.073, 0.914, 0.071, 12.454, 7831,
Load block from database
31.382, 0.000, 31.382, 31.382, 1,
Load state from database
85.644, 30.350, 3.056, 466.136, 7519,
Apply block
506.569, 91.129, 130.654, 874.786, 312,
Apply epoch block
```
post:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
72457.303, 0.000, 72457.303, 72457.303, 1,
Initialize DB
1.015, 0.858, 0.070, 11.231, 7831,
Load block from database
28.983, 0.000, 28.983, 28.983, 1,
Load state from database
21.725, 17.461, 2.659, 393.217, 7519,
Apply block
324.012, 33.954, 45.452, 440.532, 312,
Apply epoch block
```
2020-08-21 12:06:26 +02:00
|
|
|
const
|
|
|
|
SEED_SIZE = sizeof(Eth2Digest)
|
|
|
|
ROUND_SIZE = 1
|
|
|
|
POSITION_WINDOW_SIZE = 4
|
|
|
|
PIVOT_VIEW_SIZE = SEED_SIZE + ROUND_SIZE
|
|
|
|
TOTAL_SIZE = PIVOT_VIEW_SIZE + POSITION_WINDOW_SIZE
|
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#compute_shuffled_index
|
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#compute_committee
|
speed up shuffling
Replace shuffling function with zrnt version - `get_shuffled_seq` in
particular puts more strain on the GC by allocating superfluous seq's
which turns out to have a significant impact on block processing (when
replaying blocks for example) - 4x improvement on non-epoch, 1.5x on
epoch blocks (replay is done without signature checking)
Medalla, first 10k slots - pre:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
76855.848, 0.000, 76855.848, 76855.848, 1,
Initialize DB
1.073, 0.914, 0.071, 12.454, 7831,
Load block from database
31.382, 0.000, 31.382, 31.382, 1,
Load state from database
85.644, 30.350, 3.056, 466.136, 7519,
Apply block
506.569, 91.129, 130.654, 874.786, 312,
Apply epoch block
```
post:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
72457.303, 0.000, 72457.303, 72457.303, 1,
Initialize DB
1.015, 0.858, 0.070, 11.231, 7831,
Load block from database
28.983, 0.000, 28.983, 28.983, 1,
Load state from database
21.725, 17.461, 2.659, 393.217, 7519,
Apply block
324.012, 33.954, 45.452, 440.532, 312,
Apply epoch block
```
2020-08-21 12:06:26 +02:00
|
|
|
# Port of https://github.com/protolambda/zrnt/blob/master/eth2/beacon/shuffle.go
|
|
|
|
# Shuffles or unshuffles, depending on the `dir` (true for shuffling, false for unshuffling
|
|
|
|
func shuffle_list*(input: var seq[ValidatorIndex], seed: Eth2Digest) =
|
|
|
|
let list_size = input.lenu64
|
|
|
|
|
|
|
|
if list_size <= 1: return
|
|
|
|
|
|
|
|
var buf {.noinit.}: array[TOTAL_SIZE, byte]
|
|
|
|
|
|
|
|
# Seed is always the first 32 bytes of the hash input, we never have to change
|
|
|
|
# this part of the buffer.
|
|
|
|
buf[0..<32] = seed.data
|
|
|
|
|
|
|
|
# The original code includes a direction flag, but only the reverse direction
|
|
|
|
# is used in eth2, so we simplify it here
|
|
|
|
for r in 0'u8..<SHUFFLE_ROUND_COUNT.uint8:
|
|
|
|
# spec: pivot = bytes_to_int(hash(seed + int_to_bytes1(round))[0:8]) % list_size
|
|
|
|
# This is the "int_to_bytes1(round)", appended to the seed.
|
|
|
|
buf[SEED_SIZE] = (SHUFFLE_ROUND_COUNT.uint8 - r - 1)
|
|
|
|
|
|
|
|
# Seed is already in place, now just hash the correct part of the buffer,
|
|
|
|
# and take a uint64 from it, and modulo it to get a pivot within range.
|
|
|
|
let
|
|
|
|
pivotDigest = eth2digest(buf.toOpenArray(0, PIVOT_VIEW_SIZE - 1))
|
|
|
|
pivot = bytes_to_uint64(pivotDigest.data.toOpenArray(0, 7)) mod listSize
|
|
|
|
|
|
|
|
# Split up the for-loop in two:
|
|
|
|
# 1. Handle the part from 0 (incl) to pivot (incl). This is mirrored around
|
|
|
|
# (pivot / 2)
|
|
|
|
# 2. Handle the part from pivot (excl) to N (excl). This is mirrored around
|
|
|
|
# ((pivot / 2) + (size/2))
|
|
|
|
# The pivot defines a split in the array, with each of the splits mirroring
|
|
|
|
# their data within the split.
|
|
|
|
# Print out some example even/odd sized index lists, with some even/odd pivots,
|
|
|
|
# and you can deduce how the mirroring works exactly.
|
|
|
|
# Note that the mirror is strict enough to not consider swapping the index
|
|
|
|
# @mirror with itself.
|
|
|
|
# Since we are iterating through the "positions" in order, we can just
|
|
|
|
# repeat the hash every 256th position.
|
|
|
|
# No need to pre-compute every possible hash for efficiency like in the
|
|
|
|
# example code.
|
|
|
|
# We only need it consecutively (we are going through each in reverse order
|
|
|
|
# however, but same thing)
|
|
|
|
|
|
|
|
# spec: source = hash(seed + int_to_bytes1(round) + int_to_bytes4(position // 256))
|
|
|
|
# - seed is still in 0:32 (excl., 32 bytes)
|
|
|
|
# - round number is still in 32
|
|
|
|
# - mix in the position for randomness, except the last byte of it,
|
|
|
|
# which will be used later to select a bit from the resulting hash.
|
|
|
|
# We start from the pivot position, and work back to the mirror position
|
|
|
|
# (of the part left to the pivot).
|
|
|
|
# This makes us process each pear exactly once (instead of unnecessarily
|
|
|
|
# twice, like in the spec)
|
|
|
|
buf[33..<37] = uint_to_bytes4(pivot shr 8)
|
|
|
|
|
|
|
|
var
|
|
|
|
mirror = (pivot + 1) shr 1
|
|
|
|
source = eth2digest(buf)
|
|
|
|
byteV = source.data[(pivot and 0xff) shr 3]
|
|
|
|
i = 0'u64
|
|
|
|
j = pivot
|
|
|
|
|
|
|
|
template shuffle =
|
|
|
|
while i < mirror:
|
|
|
|
# The pair is i,j. With j being the bigger of the two, hence the "position" identifier of the pair.
|
|
|
|
# Every 256th bit (aligned to j).
|
|
|
|
if (j and 0xff) == 0xff:
|
|
|
|
# just overwrite the last part of the buffer, reuse the start (seed, round)
|
|
|
|
buf[33..<37] = uint_to_bytes4(j shr 8)
|
|
|
|
source = eth2digest(buf)
|
|
|
|
|
|
|
|
# Same trick with byte retrieval. Only every 8th.
|
|
|
|
if (j and 0x07) == 0x7:
|
|
|
|
byteV = source.data[(j and 0xff'u64) shr 3]
|
|
|
|
|
|
|
|
let
|
|
|
|
bitV = (byteV shr (j and 0x7)) and 0x1
|
|
|
|
|
|
|
|
if bitV == 1:
|
|
|
|
swap(input[i], input[j])
|
|
|
|
|
|
|
|
i.inc
|
|
|
|
j.dec
|
|
|
|
|
|
|
|
shuffle
|
|
|
|
|
|
|
|
# Now repeat, but for the part after the pivot.
|
|
|
|
mirror = (pivot + list_size + 1) shr 1
|
|
|
|
let lend = list_size - 1
|
|
|
|
# Again, seed and round input is in place, just update the position.
|
|
|
|
# We start at the end, and work back to the mirror point.
|
|
|
|
# This makes us process each pear exactly once (instead of unnecessarily twice, like in the spec)
|
|
|
|
buf[33..<37] = uint_to_bytes4(lend shr 8)
|
|
|
|
|
|
|
|
source = eth2digest(buf)
|
|
|
|
byteV = source.data[(lend and 0xff) shr 3]
|
|
|
|
i = pivot + 1'u64
|
|
|
|
j = lend
|
|
|
|
|
|
|
|
shuffle
|
2019-03-01 23:50:01 +00:00
|
|
|
|
2020-05-29 06:10:20 +00:00
|
|
|
func get_shuffled_active_validator_indices*(state: BeaconState, epoch: Epoch):
|
|
|
|
seq[ValidatorIndex] =
|
|
|
|
# Non-spec function, to cache a data structure from which one can cheaply
|
|
|
|
# compute both get_active_validator_indexes() and get_beacon_committee().
|
speed up shuffling
Replace shuffling function with zrnt version - `get_shuffled_seq` in
particular puts more strain on the GC by allocating superfluous seq's
which turns out to have a significant impact on block processing (when
replaying blocks for example) - 4x improvement on non-epoch, 1.5x on
epoch blocks (replay is done without signature checking)
Medalla, first 10k slots - pre:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
76855.848, 0.000, 76855.848, 76855.848, 1,
Initialize DB
1.073, 0.914, 0.071, 12.454, 7831,
Load block from database
31.382, 0.000, 31.382, 31.382, 1,
Load state from database
85.644, 30.350, 3.056, 466.136, 7519,
Apply block
506.569, 91.129, 130.654, 874.786, 312,
Apply epoch block
```
post:
```
Loaded 68973 blocks, head slot 117077
All time are ms
Average, StdDev, Min, Max, Samples,
Test
Validation is turned off meaning that no BLS operations are performed
72457.303, 0.000, 72457.303, 72457.303, 1,
Initialize DB
1.015, 0.858, 0.070, 11.231, 7831,
Load block from database
28.983, 0.000, 28.983, 28.983, 1,
Load state from database
21.725, 17.461, 2.659, 393.217, 7519,
Apply block
324.012, 33.954, 45.452, 440.532, 312,
Apply epoch block
```
2020-08-21 12:06:26 +02:00
|
|
|
var active_validator_indices = get_active_validator_indices(state, epoch)
|
|
|
|
|
|
|
|
shuffle_list(
|
|
|
|
active_validator_indices, get_seed(state, epoch, DOMAIN_BEACON_ATTESTER))
|
|
|
|
|
|
|
|
active_validator_indices
|
2020-05-29 06:10:20 +00:00
|
|
|
|
2020-07-15 10:44:18 +00:00
|
|
|
func get_shuffled_active_validator_indices*(
|
2020-07-27 18:04:44 +02:00
|
|
|
cache: var StateCache, state: BeaconState, epoch: Epoch):
|
|
|
|
var seq[ValidatorIndex] =
|
|
|
|
# `cache` comes first because of nim's borrowing rules for the `var` return -
|
|
|
|
# the `var` returns avoids copying the validator set.
|
|
|
|
cache.shuffled_active_validator_indices.withValue(epoch, validator_indices) do:
|
|
|
|
return validator_indices[]
|
|
|
|
do:
|
|
|
|
let indices = get_shuffled_active_validator_indices(state, epoch)
|
|
|
|
return cache.shuffled_active_validator_indices.mgetOrPut(epoch, indices)
|
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_active_validator_indices
|
2020-07-27 18:04:44 +02:00
|
|
|
func count_active_validators*(state: BeaconState,
|
|
|
|
epoch: Epoch,
|
|
|
|
cache: var StateCache): uint64 =
|
|
|
|
cache.get_shuffled_active_validator_indices(state, epoch).lenu64
|
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_committee_count_per_slot
|
2020-07-27 18:04:44 +02:00
|
|
|
func get_committee_count_per_slot*(num_active_validators: uint64): uint64 =
|
|
|
|
clamp(
|
|
|
|
num_active_validators div SLOTS_PER_EPOCH div TARGET_COMMITTEE_SIZE,
|
|
|
|
1'u64, MAX_COMMITTEES_PER_SLOT)
|
|
|
|
|
|
|
|
func get_committee_count_per_slot*(state: BeaconState,
|
|
|
|
epoch: Epoch,
|
|
|
|
cache: var StateCache): uint64 =
|
|
|
|
# Return the number of committees at ``slot``.
|
|
|
|
|
|
|
|
# TODO this is mostly used in for loops which have indexes which then need to
|
|
|
|
# be converted to CommitteeIndex types for get_beacon_committee(...); replace
|
|
|
|
# with better and more type-safe use pattern, probably beginning with using a
|
|
|
|
# CommitteeIndex return type here.
|
|
|
|
let
|
|
|
|
active_validator_count = count_active_validators(state, epoch, cache)
|
|
|
|
result = get_committee_count_per_slot(active_validator_count)
|
|
|
|
|
|
|
|
# Otherwise, get_beacon_committee(...) cannot access some committees.
|
|
|
|
doAssert (SLOTS_PER_EPOCH * MAX_COMMITTEES_PER_SLOT) >= uint64(result)
|
|
|
|
|
|
|
|
func get_committee_count_per_slot*(state: BeaconState,
|
|
|
|
slot: Slot,
|
|
|
|
cache: var StateCache): uint64 =
|
|
|
|
get_committee_count_per_slot(state, slot.compute_epoch_at_slot, cache)
|
2020-07-15 10:44:18 +00:00
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_previous_epoch
|
2020-07-22 07:51:45 +00:00
|
|
|
func get_previous_epoch*(current_epoch: Epoch): Epoch =
|
2019-07-01 11:05:22 +02:00
|
|
|
# Return the previous epoch (unless the current epoch is ``GENESIS_EPOCH``).
|
2019-06-14 13:50:47 +00:00
|
|
|
if current_epoch == GENESIS_EPOCH:
|
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
|
|
|
current_epoch
|
2019-06-14 13:50:47 +00:00
|
|
|
else:
|
More 0.8.0 updates (#311)
* replace BeaconState.finalized_{epoch,root} with BeaconState.finalized_checkpoint; rename get_delayed_activation_exit_epoch(...) to compute_activation_exit_epoch(...) and mark as 0.8.0; update get_churn_limit(...)/get_validator_churn_limit(...) to 0.8.0; update process_registry_updates(...) to 0.8.0
* update process_crosslinks(...) to 0.8.0; mark compute_start_slot_of_epoch(...) and get_committee_count(...) as 0.8.0
* mark Fork, is_slashable_validator(...), and get_beacon_proposer_index(...) as 0.8.0
* rename LATEST_SLASHED_EXIT_LENGTH to EPOCHS_PER_SLASHINGS_VECTOR; update process_slashings(...) to 0.8.0; remove pointless type conversion warning in get_previous_epoch(...)
* convert remaining references to finalized_epoch to finalized_checkpoint.epoch
* update slash_validator(...) to 0.8.0; mark inital value, Gwei, and time constants as 0.8.0; mark hash(...) and processBlockHeader(...) as 0.8.0
* rename WHISTLEBLOWING_REWARD_QUOTIENT to WHISTLEBLOWER_REWARD_QUOTIENT; rename LATEST_ACTIVE_INDEX_ROOTS_LENGTH to EPOCHS_PER_HISTORICAL_VECTOR (randao will also get merged into this); remove get_active_index_root(...); mark time parameter, signature domain types, and max operations per block constants as 0.8.0; update rewards and penalties constants to 0.8.0
* update is_valid_indexed_attestation(...) to 0.8.0; mark process_slot(...) as 0.8.0
* replace BeaconState.{current,previous}_justified_{epoch,root} with BeaconState.{current,previous}_justified_checkpoint
2019-07-05 08:30:05 +00:00
|
|
|
current_epoch - 1
|
2019-02-11 16:10:46 +01:00
|
|
|
|
2020-07-22 07:51:45 +00:00
|
|
|
func get_previous_epoch*(state: BeaconState): Epoch =
|
|
|
|
# Return the previous epoch (unless the current epoch is ``GENESIS_EPOCH``).
|
|
|
|
get_previous_epoch(get_current_epoch(state))
|
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#compute_committee
|
2020-08-27 09:34:12 +02:00
|
|
|
func compute_committee_slice*(
|
|
|
|
active_validators, index, count: uint64): Slice[int] =
|
|
|
|
doAssert active_validators <= ValidatorIndex.high.uint64
|
|
|
|
|
|
|
|
let
|
|
|
|
start = (active_validators * index) div count
|
|
|
|
endIdx = (active_validators * (index + 1)) div count
|
|
|
|
|
|
|
|
start.int..(endIdx.int - 1)
|
|
|
|
|
2020-07-28 15:54:32 +02:00
|
|
|
func compute_committee*(shuffled_indices: seq[ValidatorIndex],
|
2020-06-01 07:44:50 +00:00
|
|
|
index: uint64, count: uint64): seq[ValidatorIndex] =
|
2019-07-01 11:05:22 +02:00
|
|
|
## Return the committee corresponding to ``indices``, ``seed``, ``index``,
|
|
|
|
## and committee ``count``.
|
2020-07-27 18:04:44 +02:00
|
|
|
## In this version, we pass in the shuffled indices meaning we no longer need
|
|
|
|
## the seed.
|
2020-06-01 20:27:57 +02:00
|
|
|
let
|
2020-08-27 09:34:12 +02:00
|
|
|
slice = compute_committee_slice(shuffled_indices.lenu64, index, count)
|
2020-04-22 07:53:02 +02:00
|
|
|
|
2020-06-01 20:27:57 +02:00
|
|
|
# In spec, this calls get_shuffled_index() every time, but that's wasteful
|
|
|
|
# Here, get_beacon_committee() gets the shuffled version.
|
2020-08-27 09:34:12 +02:00
|
|
|
shuffled_indices[slice]
|
2019-05-23 11:13:02 +00:00
|
|
|
|
2020-08-27 09:34:12 +02:00
|
|
|
func compute_committee_len*(
|
|
|
|
active_validators, index, count: uint64): uint64 =
|
2020-07-27 18:04:44 +02:00
|
|
|
## Return the committee corresponding to ``indices``, ``seed``, ``index``,
|
|
|
|
## and committee ``count``.
|
|
|
|
|
|
|
|
let
|
2020-08-27 09:34:12 +02:00
|
|
|
slice = compute_committee_slice(active_validators, index, count)
|
2020-07-27 18:04:44 +02:00
|
|
|
|
2020-08-27 09:34:12 +02:00
|
|
|
(slice.b - slice.a + 1).uint64
|
2020-07-27 18:04:44 +02:00
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_beacon_committee
|
2020-04-15 11:01:36 +02:00
|
|
|
func get_beacon_committee*(
|
|
|
|
state: BeaconState, slot: Slot, index: CommitteeIndex,
|
|
|
|
cache: var StateCache): seq[ValidatorIndex] =
|
2020-09-08 08:54:55 +00:00
|
|
|
## Return the beacon committee at ``slot`` for ``index``.
|
2019-11-07 22:13:27 +01:00
|
|
|
let
|
|
|
|
epoch = compute_epoch_at_slot(slot)
|
2020-07-27 18:04:44 +02:00
|
|
|
committees_per_slot = get_committee_count_per_slot(state, epoch, cache)
|
|
|
|
compute_committee(
|
|
|
|
cache.get_shuffled_active_validator_indices(state, epoch),
|
|
|
|
(slot mod SLOTS_PER_EPOCH) * committees_per_slot +
|
|
|
|
index.uint64,
|
|
|
|
committees_per_slot * SLOTS_PER_EPOCH
|
|
|
|
)
|
|
|
|
|
2020-08-06 21:48:47 +02:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_beacon_committee
|
2020-07-27 18:04:44 +02:00
|
|
|
func get_beacon_committee_len*(
|
|
|
|
state: BeaconState, slot: Slot, index: CommitteeIndex,
|
|
|
|
cache: var StateCache): uint64 =
|
|
|
|
# Return the number of members in the beacon committee at ``slot`` for ``index``.
|
|
|
|
let
|
|
|
|
epoch = compute_epoch_at_slot(slot)
|
|
|
|
committees_per_slot = get_committee_count_per_slot(state, epoch, cache)
|
2019-11-07 22:13:27 +01:00
|
|
|
|
2020-07-27 18:04:44 +02:00
|
|
|
compute_committee_len(
|
|
|
|
count_active_validators(state, epoch, cache),
|
|
|
|
(slot mod SLOTS_PER_EPOCH) * committees_per_slot +
|
|
|
|
index.uint64,
|
|
|
|
committees_per_slot * SLOTS_PER_EPOCH
|
|
|
|
)
|
2019-11-07 22:13:27 +01:00
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#compute_shuffled_index
|
2020-07-15 10:44:18 +00:00
|
|
|
func compute_shuffled_index(
|
|
|
|
index: uint64, index_count: uint64, seed: Eth2Digest): uint64 =
|
2020-09-08 08:54:55 +00:00
|
|
|
## Return the shuffled index corresponding to ``seed`` (and ``index_count``).
|
2020-07-15 10:44:18 +00:00
|
|
|
doAssert index < index_count
|
|
|
|
|
|
|
|
var
|
|
|
|
pivot_buffer: array[(32+1), byte]
|
|
|
|
source_buffer: array[(32+1+4), byte]
|
|
|
|
cur_idx_permuted = index
|
|
|
|
|
|
|
|
pivot_buffer[0..31] = seed.data
|
|
|
|
source_buffer[0..31] = seed.data
|
|
|
|
|
|
|
|
# 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
|
2020-07-27 18:04:44 +02:00
|
|
|
for current_round in 0'u8 ..< SHUFFLE_ROUND_COUNT.uint8:
|
|
|
|
pivot_buffer[32] = current_round
|
|
|
|
source_buffer[32] = current_round
|
2020-07-15 10:44:18 +00:00
|
|
|
|
|
|
|
let
|
|
|
|
# If using multiple indices, can amortize this
|
|
|
|
pivot =
|
2020-07-27 10:59:57 +00:00
|
|
|
bytes_to_uint64(eth2digest(pivot_buffer).data.toOpenArray(0, 7)) mod
|
2020-07-15 10:44:18 +00:00
|
|
|
index_count
|
|
|
|
|
|
|
|
flip = ((index_count + pivot) - cur_idx_permuted) mod index_count
|
|
|
|
position = max(cur_idx_permuted.int, flip.int)
|
2020-07-27 10:59:57 +00:00
|
|
|
source_buffer[33..36] = uint_to_bytes4((position div 256).uint64)
|
2020-07-15 10:44:18 +00:00
|
|
|
let
|
|
|
|
source = eth2digest(source_buffer).data
|
|
|
|
byte_value = source[(position mod 256) div 8]
|
|
|
|
bit = (byte_value shr (position mod 8)) mod 2
|
|
|
|
|
|
|
|
cur_idx_permuted = if bit != 0: flip else: cur_idx_permuted
|
|
|
|
|
|
|
|
cur_idx_permuted
|
0.6.2 updates (#275)
* update process_justification_and_finalization to 0.6.2; mark AttesterSlashing as 0.6.2
* replace get_effective_balance(...) with state.validator_registry[idx].effective_balance; rm get_effective_balance, process_ejections, should_update_validator_registry, update_validator_registry, and update_registry_and_shuffling_data; update get_total_balance to 0.6.2; implement process_registry_updates
* rm exit_validator; implement is_slashable_attestation_data; partly update processAttesterSlashings
* mark HistoricalBatch and Eth1Data as 0.6.2; implement get_shard_delta(...); replace 0.5 finish_epoch_update with 0.6 process_final_updates
* mark increase_balance, decrease_balance, get_delayed_activation_exit_epoch, bls_aggregate_pubkeys, bls_verify_multiple, Attestation, Transfer, slot_to_epoch, Crosslink, get_current_epoch, int_to_bytes*, various constants, processEth1Data, processTransfers, and verifyStateRoot as 0.6.2; rm is_double_vote and is_surround_vote
* mark get_bitfield_bit, verify_bitfield, ProposerSlashing, DepositData, VoluntaryExit, PendingAttestation, Fork, integer_squareroot, get_epoch_start_slot, is_active_validator, generate_seed, some constants to 0.6.2; rename MIN_PENALTY_QUOTIENT to MIN_SLASHING_PENALTY_QUOTIENT
* rm get_previous_total_balance, get_current_epoch_boundary_attestations, get_previous_epoch_boundary_attestations, and get_previous_epoch_matching_head_attestations
* update BeaconState to 0.6.2; simplify legacy get_crosslink_committees_at_slot infrastructure a bit by noting that registry_change is always false; reimplment 0.5 get_crosslink_committees_at_slot in terms of 0.6 get_crosslink_committee
* mark process_deposit(...), get_block_root_at_slot(...), get_block_root(...), Deposit, BeaconBlockHeader, BeaconBlockBody, hash(...), get_active_index_root(...), various constants, get_shard_delta(...), get_epoch_start_shard(...), get_crosslink_committee(...), processRandao(...), processVoluntaryExits(...), cacheState(...) as 0.6.2
* rm removed-since-0.5 split(...), is_power_of_2(...), get_shuffling(...); rm 0.5 versions of get_active_validator_indices and get_epoch_committee_count; add a few tests for integer_squareroot
* mark bytes_to_int(...) and advanceState(...) as 0.6.2
* rm 0.5 get_attesting_indices; update get_attesting_balance to 0.6.2
* another tiny commit to poke AppVeyor to maybe not timeout at connecting to GitHub partway through CI: mark get_churn_limit(...), initiate_validator_exit(...), and Validator as 0.6.2
* mark get_attestation_slot(...), AttestationDataAndCustodyBit, and BeaconBlock as 0.6.2
2019-06-03 10:31:04 +00:00
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#compute_proposer_index
|
2019-11-25 07:31:37 +01:00
|
|
|
func compute_proposer_index(state: BeaconState, indices: seq[ValidatorIndex],
|
2020-06-03 05:42:08 +00:00
|
|
|
seed: Eth2Digest): Option[ValidatorIndex] =
|
2020-09-08 08:54:55 +00:00
|
|
|
## Return from ``indices`` a random index sampled by effective balance.
|
2019-11-10 00:03:41 +00:00
|
|
|
const MAX_RANDOM_BYTE = 255
|
|
|
|
|
2019-12-04 10:49:59 +00:00
|
|
|
if len(indices) == 0:
|
|
|
|
return none(ValidatorIndex)
|
2019-05-23 11:13:02 +00:00
|
|
|
|
2020-07-26 18:55:48 +00:00
|
|
|
let seq_len = indices.lenu64
|
2019-11-06 15:50:12 +00:00
|
|
|
|
2019-05-23 11:13:02 +00:00
|
|
|
var
|
2020-07-15 10:44:18 +00:00
|
|
|
i = 0'u64
|
2019-11-10 00:03:41 +00:00
|
|
|
buffer: array[32+8, byte]
|
2019-05-23 11:13:02 +00:00
|
|
|
buffer[0..31] = seed.data
|
|
|
|
while true:
|
2020-07-27 10:59:57 +00:00
|
|
|
buffer[32..39] = uint_to_bytes8(i div 32)
|
2019-05-23 11:13:02 +00:00
|
|
|
let
|
2020-07-15 10:44:18 +00:00
|
|
|
candidate_index =
|
|
|
|
indices[compute_shuffled_index(i mod seq_len, seq_len, seed)]
|
2020-06-16 14:16:43 +02:00
|
|
|
random_byte = (eth2digest(buffer).data)[i mod 32]
|
2020-07-15 10:44:18 +00:00
|
|
|
effective_balance = state.validators[candidate_index].effective_balance
|
0.7.0 updates which don't change semantics; mostly comment changes (#281)
* 0.7.0 updates which don't change semantics; mostly comment changes
* mark get_attesting_indices, bls_verify, ProposerSlashing, BeaconBlockBody, deposit contract constants, state list length constants, compute_committee, get_crosslink_committee, is_slashable_attestation_data, processAttesterSlashings as 0.7.0; rename BASE_REWARD_QUOTIENT to BASE_REWARD_FACTOR
* complete marking unchanged-in-0.7.0 datatypes as such; a few more have trivial field renamings, etc
2019-06-13 07:40:58 +00:00
|
|
|
if effective_balance * MAX_RANDOM_BYTE >=
|
|
|
|
MAX_EFFECTIVE_BALANCE * random_byte:
|
2019-12-04 10:49:59 +00:00
|
|
|
return some(candidate_index)
|
2019-05-23 11:13:02 +00:00
|
|
|
i += 1
|
2019-11-10 00:03:41 +00:00
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_beacon_proposer_index
|
2020-06-03 05:42:08 +00:00
|
|
|
func get_beacon_proposer_index*(state: BeaconState, cache: var StateCache, slot: Slot):
|
2019-12-04 10:49:59 +00:00
|
|
|
Option[ValidatorIndex] =
|
2020-07-27 18:04:44 +02:00
|
|
|
cache.beacon_proposer_indices.withValue(slot, proposer) do:
|
|
|
|
return proposer[]
|
|
|
|
do:
|
2020-06-04 14:03:16 +02:00
|
|
|
|
2020-07-27 18:04:44 +02:00
|
|
|
# Return the beacon proposer index at the current slot.
|
|
|
|
let epoch = get_current_epoch(state)
|
2019-11-10 00:03:41 +00:00
|
|
|
|
2020-07-27 18:04:44 +02:00
|
|
|
var buffer: array[32 + 8, byte]
|
|
|
|
buffer[0..31] = get_seed(state, epoch, DOMAIN_BEACON_PROPOSER).data
|
2019-11-10 00:03:41 +00:00
|
|
|
|
2020-07-27 18:04:44 +02:00
|
|
|
# There's exactly one beacon proposer per slot.
|
2020-06-03 05:42:08 +00:00
|
|
|
|
|
|
|
let
|
2020-08-19 10:03:50 +02:00
|
|
|
# active validator indices are kept in cache but sorting them takes
|
|
|
|
# quite a while
|
|
|
|
indices = get_active_validator_indices(state, epoch)
|
2020-08-20 18:30:47 +02:00
|
|
|
start = epoch.compute_start_slot_at_epoch()
|
2020-08-19 10:03:50 +02:00
|
|
|
|
|
|
|
var res: Option[ValidatorIndex]
|
|
|
|
for i in 0..<SLOTS_PER_EPOCH:
|
|
|
|
buffer[32..39] = uint_to_bytes8((start + i).uint64)
|
|
|
|
let seed = eth2digest(buffer)
|
|
|
|
let pi = compute_proposer_index(state, indices, seed)
|
|
|
|
if start + i == slot:
|
|
|
|
res = pi
|
|
|
|
cache.beacon_proposer_indices[start + i] = pi
|
|
|
|
|
|
|
|
return res
|
2020-02-20 23:22:59 +01:00
|
|
|
|
2020-07-27 16:49:46 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/beacon-chain.md#get_beacon_proposer_index
|
2020-06-04 14:03:16 +02:00
|
|
|
func get_beacon_proposer_index*(state: BeaconState, cache: var StateCache):
|
2020-05-22 20:04:52 +03:00
|
|
|
Option[ValidatorIndex] =
|
2020-06-04 14:03:16 +02:00
|
|
|
get_beacon_proposer_index(state, cache, state.slot)
|
2020-05-22 20:04:52 +03:00
|
|
|
|
2020-07-29 12:47:03 +00:00
|
|
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.12.2/specs/phase0/validator.md#validator-assignments
|
2020-05-27 20:06:28 +03:00
|
|
|
func get_committee_assignment*(
|
2020-07-10 09:24:04 +00:00
|
|
|
state: BeaconState, epoch: Epoch,
|
2020-07-22 08:04:21 +00:00
|
|
|
validator_index: ValidatorIndex):
|
|
|
|
Option[tuple[a: seq[ValidatorIndex], b: CommitteeIndex, c: Slot]] =
|
2020-09-08 08:54:55 +00:00
|
|
|
## Return the committee assignment in the ``epoch`` for ``validator_index``.
|
|
|
|
## ``assignment`` returned is a tuple of the following form:
|
|
|
|
## * ``assignment[0]`` is the list of validators in the committee
|
|
|
|
## * ``assignment[1]`` is the index to which the committee is assigned
|
|
|
|
## * ``assignment[2]`` is the slot at which the committee is assigned
|
|
|
|
## Return None if no assignment.
|
2020-02-20 23:22:59 +01:00
|
|
|
let next_epoch = get_current_epoch(state) + 1
|
|
|
|
doAssert epoch <= next_epoch
|
|
|
|
|
2020-07-15 10:44:18 +00:00
|
|
|
var cache = StateCache()
|
2020-02-20 23:22:59 +01:00
|
|
|
|
2020-07-27 10:59:57 +00:00
|
|
|
let
|
|
|
|
start_slot = compute_start_slot_at_epoch(epoch)
|
|
|
|
committee_count_per_slot =
|
|
|
|
get_committee_count_per_slot(state, epoch, cache)
|
2020-02-20 23:22:59 +01:00
|
|
|
for slot in start_slot ..< start_slot + SLOTS_PER_EPOCH:
|
2020-07-27 10:59:57 +00:00
|
|
|
for index in 0'u64 ..< committee_count_per_slot:
|
2020-04-15 11:01:36 +02:00
|
|
|
let idx = index.CommitteeIndex
|
2020-07-10 09:24:04 +00:00
|
|
|
let committee = get_beacon_committee(state, slot, idx, cache)
|
2020-07-22 08:04:21 +00:00
|
|
|
if validator_index in committee:
|
2020-04-15 11:01:36 +02:00
|
|
|
return some((committee, idx, slot))
|
|
|
|
none(tuple[a: seq[ValidatorIndex], b: CommitteeIndex, c: Slot])
|