#![doc = include_str!("../README.md")]
#![cfg_attr(not(feature = "std"), no_std)]
#![feature(array_chunks, assert_matches, let_chains, portable_simd)]
#![warn(unused_must_use, unsafe_code, unused_variables)]
#[cfg(not(feature = "std"))]
extern crate alloc;
#[cfg(test)]
mod mock;
#[cfg(test)]
mod tests;
#[cfg(feature = "runtime-benchmarks")]
mod benchmarking;
pub mod weights;
#[cfg(not(feature = "std"))]
use alloc::string::String;
use codec::{Decode, Encode, MaxEncodedLen};
use core::num::NonZeroU64;
use frame_support::dispatch::DispatchResult;
use frame_support::traits::Get;
use frame_system::offchain::{SendTransactionTypes, SubmitTransaction};
use frame_system::pallet_prelude::*;
use log::{debug, error, warn};
pub use pallet::*;
use scale_info::TypeInfo;
use schnorrkel::SignatureError;
use sp_consensus_slots::Slot;
use sp_consensus_subspace::consensus::{is_proof_of_time_valid, verify_solution};
use sp_consensus_subspace::digests::CompatibleDigestItem;
use sp_consensus_subspace::{
PotParameters, PotParametersChange, SignedVote, Vote, WrappedPotOutput,
};
use sp_runtime::generic::DigestItem;
use sp_runtime::traits::{BlockNumberProvider, CheckedSub, Hash, One, Zero};
use sp_runtime::transaction_validity::{
InvalidTransaction, TransactionPriority, TransactionSource, TransactionValidity,
TransactionValidityError, ValidTransaction,
};
use sp_std::collections::btree_map::BTreeMap;
use sp_std::prelude::*;
use subspace_core_primitives::pieces::PieceOffset;
use subspace_core_primitives::sectors::{SectorId, SectorIndex};
use subspace_core_primitives::segments::{
ArchivedHistorySegment, HistorySize, SegmentHeader, SegmentIndex,
};
use subspace_core_primitives::solutions::{RewardSignature, SolutionRange};
use subspace_core_primitives::{
BlockHash, PublicKey, ScalarBytes, SlotNumber, REWARD_SIGNING_CONTEXT,
};
use subspace_verification::{
check_reward_signature, derive_next_solution_range, derive_pot_entropy, PieceCheckParams,
VerifySolutionParams,
};
pub trait EraChangeTrigger {
fn trigger<T: Config>(block_number: BlockNumberFor<T>);
}
pub struct NormalEraChange;
impl EraChangeTrigger for NormalEraChange {
fn trigger<T: Config>(block_number: BlockNumberFor<T>) {
if <Pallet<T>>::should_era_change(block_number) {
<Pallet<T>>::enact_era_change();
}
}
}
#[derive(Copy, Clone, Eq, PartialEq, Encode, Decode, MaxEncodedLen, TypeInfo)]
struct VoteVerificationData {
solution_range: SolutionRange,
vote_solution_range: SolutionRange,
current_slot: Slot,
parent_slot: Slot,
}
#[frame_support::pallet]
pub mod pallet {
use super::{EraChangeTrigger, VoteVerificationData};
use crate::weights::WeightInfo;
use frame_support::pallet_prelude::*;
use frame_system::pallet_prelude::*;
use sp_consensus_slots::Slot;
use sp_consensus_subspace::digests::CompatibleDigestItem;
use sp_consensus_subspace::inherents::{InherentError, InherentType, INHERENT_IDENTIFIER};
use sp_consensus_subspace::SignedVote;
use sp_runtime::traits::One;
use sp_runtime::DigestItem;
use sp_std::collections::btree_map::BTreeMap;
use sp_std::num::NonZeroU32;
use sp_std::prelude::*;
use subspace_core_primitives::hashes::Blake3Hash;
use subspace_core_primitives::pieces::PieceOffset;
use subspace_core_primitives::pot::PotCheckpoints;
use subspace_core_primitives::sectors::SectorIndex;
use subspace_core_primitives::segments::{HistorySize, SegmentHeader, SegmentIndex};
use subspace_core_primitives::solutions::{RewardSignature, SolutionRange};
use subspace_core_primitives::{PublicKey, Randomness, ScalarBytes};
pub(super) struct InitialSolutionRanges<T: Config> {
_config: T,
}
impl<T: Config> Get<sp_consensus_subspace::SolutionRanges> for InitialSolutionRanges<T> {
fn get() -> sp_consensus_subspace::SolutionRanges {
sp_consensus_subspace::SolutionRanges {
current: T::InitialSolutionRange::get(),
next: None,
voting_current: if T::ShouldAdjustSolutionRange::get() {
T::InitialSolutionRange::get()
.saturating_mul(u64::from(T::ExpectedVotesPerBlock::get()) + 1)
} else {
T::InitialSolutionRange::get()
},
voting_next: None,
}
}
}
#[derive(Debug, Encode, Decode, TypeInfo)]
pub(super) struct SolutionRangeOverride {
pub(super) solution_range: SolutionRange,
pub(super) voting_solution_range: SolutionRange,
}
#[pallet::pallet]
#[pallet::without_storage_info]
pub struct Pallet<T>(_);
#[pallet::config]
#[pallet::disable_frame_system_supertrait_check]
pub trait Config: frame_system::Config {
type RuntimeEvent: From<Event<Self>> + IsType<<Self as frame_system::Config>::RuntimeEvent>;
#[pallet::constant]
type BlockAuthoringDelay: Get<Slot>;
#[pallet::constant]
type PotEntropyInjectionInterval: Get<BlockNumberFor<Self>>;
#[pallet::constant]
type PotEntropyInjectionLookbackDepth: Get<u8>;
#[pallet::constant]
type PotEntropyInjectionDelay: Get<Slot>;
#[pallet::constant]
type EraDuration: Get<BlockNumberFor<Self>>;
#[pallet::constant]
type InitialSolutionRange: Get<SolutionRange>;
#[pallet::constant]
type SlotProbability: Get<(u64, u64)>;
#[pallet::constant]
type ConfirmationDepthK: Get<BlockNumberFor<Self>>;
#[pallet::constant]
type RecentSegments: Get<HistorySize>;
#[pallet::constant]
type RecentHistoryFraction: Get<(HistorySize, HistorySize)>;
#[pallet::constant]
type MinSectorLifetime: Get<HistorySize>;
#[pallet::constant]
type ExpectedVotesPerBlock: Get<u32>;
#[pallet::constant]
type MaxPiecesInSector: Get<u16>;
type ShouldAdjustSolutionRange: Get<bool>;
type EraChangeTrigger: EraChangeTrigger;
type WeightInfo: WeightInfo;
#[pallet::constant]
type BlockSlotCount: Get<u32>;
}
#[derive(Debug, Default, Encode, Decode, TypeInfo)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub enum AllowAuthoringBy {
#[default]
Anyone,
FirstFarmer,
RootFarmer(PublicKey),
}
#[derive(Debug, Copy, Clone, Encode, Decode, TypeInfo)]
pub(super) struct PotEntropyValue {
pub(super) target_slot: Option<Slot>,
pub(super) entropy: Blake3Hash,
}
#[derive(Debug, Copy, Clone, Encode, Decode, TypeInfo, PartialEq)]
pub(super) struct PotSlotIterationsValue {
pub(super) slot_iterations: NonZeroU32,
pub(super) update: Option<PotSlotIterationsUpdate>,
}
#[derive(Debug, Copy, Clone, Encode, Decode, TypeInfo, PartialEq)]
pub(super) struct PotSlotIterationsUpdate {
pub(super) target_slot: Option<Slot>,
pub(super) slot_iterations: NonZeroU32,
}
#[derive(Debug, Copy, Clone, Eq, PartialEq, Encode, Decode, TypeInfo)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub enum EnableRewardsAt<BlockNumber> {
Height(BlockNumber),
SolutionRange(u64),
Manually,
}
#[pallet::genesis_config]
pub struct GenesisConfig<T>
where
T: Config,
{
pub enable_rewards_at: EnableRewardsAt<BlockNumberFor<T>>,
pub allow_authoring_by: AllowAuthoringBy,
pub pot_slot_iterations: NonZeroU32,
#[serde(skip)]
pub phantom: PhantomData<T>,
}
impl<T> Default for GenesisConfig<T>
where
T: Config,
{
#[inline]
fn default() -> Self {
Self {
enable_rewards_at: EnableRewardsAt::Height(BlockNumberFor::<T>::one()),
allow_authoring_by: AllowAuthoringBy::Anyone,
pot_slot_iterations: NonZeroU32::MIN,
phantom: PhantomData,
}
}
}
#[pallet::genesis_build]
impl<T> BuildGenesisConfig for GenesisConfig<T>
where
T: Config,
{
fn build(&self) {
match self.enable_rewards_at {
EnableRewardsAt::Height(block_number) => {
EnableRewards::<T>::put(block_number);
}
EnableRewardsAt::SolutionRange(solution_range) => {
EnableRewardsBelowSolutionRange::<T>::put(solution_range);
}
EnableRewardsAt::Manually => {
}
}
match &self.allow_authoring_by {
AllowAuthoringBy::Anyone => {
AllowAuthoringByAnyone::<T>::put(true);
}
AllowAuthoringBy::FirstFarmer => {
AllowAuthoringByAnyone::<T>::put(false);
}
AllowAuthoringBy::RootFarmer(root_farmer) => {
AllowAuthoringByAnyone::<T>::put(false);
RootPlotPublicKey::<T>::put(root_farmer);
}
}
PotSlotIterations::<T>::put(PotSlotIterationsValue {
slot_iterations: self.pot_slot_iterations,
update: None,
});
}
}
#[pallet::event]
#[pallet::generate_deposit(pub (super) fn deposit_event)]
pub enum Event<T: Config> {
SegmentHeaderStored { segment_header: SegmentHeader },
FarmerVote {
public_key: PublicKey,
reward_address: T::AccountId,
height: BlockNumberFor<T>,
parent_hash: T::Hash,
},
}
#[pallet::error]
pub enum Error<T> {
SolutionRangeAdjustmentAlreadyEnabled,
NotMultipleOfCheckpoints,
PotSlotIterationsMustIncrease,
PotSlotIterationsUpdateAlreadyScheduled,
}
#[pallet::storage]
#[pallet::getter(fn block_slots)]
pub(super) type BlockSlots<T: Config> =
StorageValue<_, BoundedBTreeMap<BlockNumberFor<T>, Slot, T::BlockSlotCount>, ValueQuery>;
#[pallet::storage]
#[pallet::getter(fn solution_ranges)]
pub(super) type SolutionRanges<T: Config> = StorageValue<
_,
sp_consensus_subspace::SolutionRanges,
ValueQuery,
InitialSolutionRanges<T>,
>;
#[pallet::storage]
#[pallet::getter(fn should_adjust_solution_range)]
pub(super) type ShouldAdjustSolutionRange<T: Config> =
StorageValue<_, bool, ValueQuery, T::ShouldAdjustSolutionRange>;
#[pallet::storage]
pub(super) type NextSolutionRangeOverride<T> = StorageValue<_, SolutionRangeOverride>;
#[pallet::storage]
pub(super) type EraStartSlot<T> = StorageValue<_, Slot>;
#[pallet::storage]
#[pallet::getter(fn segment_commitment)]
pub(super) type SegmentCommitment<T> = CountedStorageMap<
_,
Twox64Concat,
SegmentIndex,
subspace_core_primitives::segments::SegmentCommitment,
>;
#[pallet::storage]
pub(super) type DidProcessSegmentHeaders<T: Config> = StorageValue<_, bool, ValueQuery>;
#[pallet::storage]
pub(super) type ParentVoteVerificationData<T> = StorageValue<_, VoteVerificationData>;
#[pallet::storage]
pub(super) type ParentBlockAuthorInfo<T> =
StorageValue<_, (PublicKey, SectorIndex, PieceOffset, ScalarBytes, Slot)>;
#[pallet::storage]
pub(super) type EnableRewards<T: Config> = StorageValue<_, BlockNumberFor<T>>;
#[pallet::storage]
pub(super) type EnableRewardsBelowSolutionRange<T: Config> = StorageValue<_, u64>;
#[pallet::storage]
pub(super) type CurrentBlockAuthorInfo<T: Config> = StorageValue<
_,
(
PublicKey,
SectorIndex,
PieceOffset,
ScalarBytes,
Slot,
Option<T::AccountId>,
),
>;
#[pallet::storage]
pub(super) type ParentBlockVoters<T: Config> = StorageValue<
_,
BTreeMap<
(PublicKey, SectorIndex, PieceOffset, ScalarBytes, Slot),
(Option<T::AccountId>, RewardSignature),
>,
ValueQuery,
>;
#[pallet::storage]
pub(super) type CurrentBlockVoters<T: Config> = StorageValue<
_,
BTreeMap<
(PublicKey, SectorIndex, PieceOffset, ScalarBytes, Slot),
(Option<T::AccountId>, RewardSignature),
>,
>;
#[pallet::storage]
pub(super) type PotSlotIterations<T> = StorageValue<_, PotSlotIterationsValue>;
#[pallet::storage]
pub(super) type PotEntropy<T: Config> =
StorageValue<_, BTreeMap<BlockNumberFor<T>, PotEntropyValue>, ValueQuery>;
#[pallet::storage]
pub type BlockRandomness<T> = StorageValue<_, Randomness>;
#[pallet::storage]
pub(super) type AllowAuthoringByAnyone<T> = StorageValue<_, bool, ValueQuery>;
#[pallet::storage]
#[pallet::getter(fn root_plot_public_key)]
pub(super) type RootPlotPublicKey<T> = StorageValue<_, PublicKey>;
#[pallet::hooks]
impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
fn on_initialize(block_number: BlockNumberFor<T>) -> Weight {
Self::do_initialize(block_number);
Weight::zero()
}
fn on_finalize(block_number: BlockNumberFor<T>) {
Self::do_finalize(block_number)
}
}
#[pallet::call]
impl<T: Config> Pallet<T> {
#[pallet::call_index(0)]
#[pallet::weight((< T as Config >::WeightInfo::store_segment_headers(segment_headers.len() as u32), DispatchClass::Mandatory))]
pub fn store_segment_headers(
origin: OriginFor<T>,
segment_headers: Vec<SegmentHeader>,
) -> DispatchResult {
ensure_none(origin)?;
Self::do_store_segment_headers(segment_headers)
}
#[pallet::call_index(1)]
#[pallet::weight(< T as Config >::WeightInfo::enable_solution_range_adjustment())]
pub fn enable_solution_range_adjustment(
origin: OriginFor<T>,
solution_range_override: Option<u64>,
voting_solution_range_override: Option<u64>,
) -> DispatchResult {
ensure_root(origin)?;
Self::do_enable_solution_range_adjustment(
solution_range_override,
voting_solution_range_override,
)?;
frame_system::Pallet::<T>::deposit_log(
DigestItem::enable_solution_range_adjustment_and_override(solution_range_override),
);
Ok(())
}
#[pallet::call_index(2)]
#[pallet::weight((< T as Config >::WeightInfo::vote(), DispatchClass::Operational))]
#[allow(clippy::boxed_local)]
pub fn vote(
origin: OriginFor<T>,
signed_vote: Box<SignedVote<BlockNumberFor<T>, T::Hash, T::AccountId>>,
) -> DispatchResult {
ensure_none(origin)?;
Self::do_vote(*signed_vote)
}
#[pallet::call_index(3)]
#[pallet::weight(< T as Config >::WeightInfo::enable_rewards_at())]
pub fn enable_rewards_at(
origin: OriginFor<T>,
enable_rewards_at: EnableRewardsAt<BlockNumberFor<T>>,
) -> DispatchResult {
ensure_root(origin)?;
Self::do_enable_rewards_at(enable_rewards_at)
}
#[pallet::call_index(4)]
#[pallet::weight(< T as Config >::WeightInfo::enable_authoring_by_anyone())]
pub fn enable_authoring_by_anyone(origin: OriginFor<T>) -> DispatchResult {
ensure_root(origin)?;
AllowAuthoringByAnyone::<T>::put(true);
RootPlotPublicKey::<T>::take();
frame_system::Pallet::<T>::deposit_log(DigestItem::root_plot_public_key_update(None));
Ok(())
}
#[pallet::call_index(5)]
#[pallet::weight(< T as Config >::WeightInfo::set_pot_slot_iterations())]
pub fn set_pot_slot_iterations(
origin: OriginFor<T>,
slot_iterations: NonZeroU32,
) -> DispatchResult {
ensure_root(origin)?;
if slot_iterations.get() % u32::from(PotCheckpoints::NUM_CHECKPOINTS.get() * 2) != 0 {
return Err(Error::<T>::NotMultipleOfCheckpoints.into());
}
let mut pot_slot_iterations =
PotSlotIterations::<T>::get().expect("Always initialized during genesis; qed");
if pot_slot_iterations.slot_iterations >= slot_iterations {
return Err(Error::<T>::PotSlotIterationsMustIncrease.into());
}
if let Some(pot_slot_iterations_update_value) = pot_slot_iterations.update
&& pot_slot_iterations_update_value.target_slot.is_some()
{
return Err(Error::<T>::PotSlotIterationsUpdateAlreadyScheduled.into());
}
pot_slot_iterations.update.replace(PotSlotIterationsUpdate {
target_slot: None,
slot_iterations,
});
PotSlotIterations::<T>::put(pot_slot_iterations);
Ok(())
}
}
#[pallet::inherent]
impl<T: Config> ProvideInherent for Pallet<T> {
type Call = Call<T>;
type Error = InherentError;
const INHERENT_IDENTIFIER: InherentIdentifier = INHERENT_IDENTIFIER;
fn create_inherent(data: &InherentData) -> Option<Self::Call> {
let inherent_data = data
.get_data::<InherentType>(&INHERENT_IDENTIFIER)
.expect("Subspace inherent data not correctly encoded")
.expect("Subspace inherent data must be provided");
let segment_headers = inherent_data.segment_headers;
if segment_headers.is_empty() {
None
} else {
Some(Call::store_segment_headers { segment_headers })
}
}
fn is_inherent_required(data: &InherentData) -> Result<Option<Self::Error>, Self::Error> {
let inherent_data = data
.get_data::<InherentType>(&INHERENT_IDENTIFIER)
.expect("Subspace inherent data not correctly encoded")
.expect("Subspace inherent data must be provided");
Ok(if inherent_data.segment_headers.is_empty() {
None
} else {
Some(InherentError::MissingSegmentHeadersList)
})
}
fn check_inherent(call: &Self::Call, data: &InherentData) -> Result<(), Self::Error> {
if let Call::store_segment_headers { segment_headers } = call {
let inherent_data = data
.get_data::<InherentType>(&INHERENT_IDENTIFIER)
.expect("Subspace inherent data not correctly encoded")
.expect("Subspace inherent data must be provided");
if segment_headers != &inherent_data.segment_headers {
return Err(InherentError::IncorrectSegmentHeadersList {
expected: inherent_data.segment_headers,
actual: segment_headers.clone(),
});
}
}
Ok(())
}
fn is_inherent(call: &Self::Call) -> bool {
matches!(call, Call::store_segment_headers { .. })
}
}
#[pallet::validate_unsigned]
impl<T: Config> ValidateUnsigned for Pallet<T> {
type Call = Call<T>;
fn validate_unsigned(source: TransactionSource, call: &Self::Call) -> TransactionValidity {
match call {
Call::store_segment_headers { segment_headers } => {
Self::validate_segment_header(source, segment_headers)
}
Call::vote { signed_vote } => Self::validate_vote(signed_vote),
_ => InvalidTransaction::Call.into(),
}
}
fn pre_dispatch(call: &Self::Call) -> Result<(), TransactionValidityError> {
match call {
Call::store_segment_headers { segment_headers } => {
Self::pre_dispatch_segment_header(segment_headers)
}
Call::vote { signed_vote } => Self::pre_dispatch_vote(signed_vote),
_ => Err(InvalidTransaction::Call.into()),
}
}
}
}
impl<T: Config> Pallet<T> {
pub fn history_size() -> HistorySize {
let number_of_segments = u64::from(SegmentCommitment::<T>::count()).max(1);
HistorySize::from(NonZeroU64::new(number_of_segments).expect("Not zero; qed"))
}
fn should_era_change(block_number: BlockNumberFor<T>) -> bool {
block_number % T::EraDuration::get() == Zero::zero()
}
fn enact_era_change() {
let slot_probability = T::SlotProbability::get();
let current_slot = Self::current_slot();
SolutionRanges::<T>::mutate(|solution_ranges| {
let next_solution_range;
let next_voting_solution_range;
if !ShouldAdjustSolutionRange::<T>::get() {
next_solution_range = solution_ranges.current;
next_voting_solution_range = solution_ranges.current;
} else if let Some(solution_range_override) = NextSolutionRangeOverride::<T>::take() {
next_solution_range = solution_range_override.solution_range;
next_voting_solution_range = solution_range_override.voting_solution_range;
} else {
next_solution_range = derive_next_solution_range(
u64::from(EraStartSlot::<T>::get().unwrap_or_default()),
u64::from(current_slot),
slot_probability,
solution_ranges.current,
T::EraDuration::get()
.try_into()
.unwrap_or_else(|_| panic!("Era duration is always within u64; qed")),
);
next_voting_solution_range = next_solution_range
.saturating_mul(u64::from(T::ExpectedVotesPerBlock::get()) + 1);
};
solution_ranges.next.replace(next_solution_range);
solution_ranges
.voting_next
.replace(next_voting_solution_range);
if let Some(solution_range_for_rewards) = EnableRewardsBelowSolutionRange::<T>::get() {
if next_solution_range <= solution_range_for_rewards {
EnableRewardsBelowSolutionRange::<T>::take();
let next_block_number =
frame_system::Pallet::<T>::current_block_number() + One::one();
EnableRewards::<T>::put(next_block_number);
}
}
});
EraStartSlot::<T>::put(current_slot);
}
fn do_initialize(block_number: BlockNumberFor<T>) {
let pre_digest = frame_system::Pallet::<T>::digest()
.logs
.iter()
.find_map(|s| s.as_subspace_pre_digest::<T::AccountId>())
.expect("Block must always have pre-digest");
let current_slot = pre_digest.slot();
BlockSlots::<T>::mutate(|block_slots| {
if let Some(to_remove) = block_number.checked_sub(&T::BlockSlotCount::get().into()) {
block_slots.remove(&to_remove);
}
block_slots
.try_insert(block_number, current_slot)
.expect("one entry just removed before inserting; qed");
});
{
CurrentBlockAuthorInfo::<T>::take();
let farmer_public_key = pre_digest.solution().public_key;
if !AllowAuthoringByAnyone::<T>::get() {
RootPlotPublicKey::<T>::mutate(|maybe_root_plot_public_key| {
if let Some(root_plot_public_key) = maybe_root_plot_public_key {
if root_plot_public_key != &farmer_public_key {
panic!("Client bug, authoring must be only done by the root user");
}
} else {
maybe_root_plot_public_key.replace(farmer_public_key);
frame_system::Pallet::<T>::deposit_log(
DigestItem::root_plot_public_key_update(Some(farmer_public_key)),
);
}
});
}
let key = (
farmer_public_key,
pre_digest.solution().sector_index,
pre_digest.solution().piece_offset,
pre_digest.solution().chunk,
current_slot,
);
if !ParentBlockVoters::<T>::get().contains_key(&key) {
let (public_key, sector_index, piece_offset, chunk, slot) = key;
CurrentBlockAuthorInfo::<T>::put((
public_key,
sector_index,
piece_offset,
chunk,
slot,
Some(pre_digest.solution().reward_address.clone()),
));
}
}
CurrentBlockVoters::<T>::put(BTreeMap::<
(PublicKey, SectorIndex, PieceOffset, ScalarBytes, Slot),
(Option<T::AccountId>, RewardSignature),
>::default());
if let sp_consensus_subspace::SolutionRanges {
next: Some(next),
voting_next: Some(voting_next),
..
} = SolutionRanges::<T>::get()
{
SolutionRanges::<T>::put(sp_consensus_subspace::SolutionRanges {
current: next,
next: None,
voting_current: voting_next,
voting_next: None,
});
}
let block_randomness = pre_digest
.pot_info()
.proof_of_time()
.derive_global_randomness();
BlockRandomness::<T>::put(block_randomness);
frame_system::Pallet::<T>::deposit_log(DigestItem::solution_range(
SolutionRanges::<T>::get().current,
));
T::EraChangeTrigger::trigger::<T>(block_number);
{
let mut pot_slot_iterations =
PotSlotIterations::<T>::get().expect("Always initialized during genesis; qed");
frame_system::Pallet::<T>::deposit_log(DigestItem::pot_slot_iterations(
pot_slot_iterations.slot_iterations,
));
if let Some(update) = pot_slot_iterations.update
&& let Some(target_slot) = update.target_slot
&& target_slot <= current_slot
{
debug!(
target: "runtime::subspace",
"Applying PoT slots update, changing to {} at block #{:?}",
update.slot_iterations,
block_number
);
pot_slot_iterations = PotSlotIterationsValue {
slot_iterations: update.slot_iterations,
update: None,
};
PotSlotIterations::<T>::put(pot_slot_iterations);
}
let pot_entropy_injection_interval = T::PotEntropyInjectionInterval::get();
let pot_entropy_injection_delay = T::PotEntropyInjectionDelay::get();
let mut entropy = PotEntropy::<T>::get();
let lookback_in_blocks = pot_entropy_injection_interval
* BlockNumberFor::<T>::from(T::PotEntropyInjectionLookbackDepth::get());
let last_entropy_injection_block =
block_number / pot_entropy_injection_interval * pot_entropy_injection_interval;
let maybe_entropy_source_block_number =
last_entropy_injection_block.checked_sub(&lookback_in_blocks);
if (block_number % pot_entropy_injection_interval).is_zero() {
let current_block_entropy = derive_pot_entropy(
&pre_digest.solution().chunk,
pre_digest.pot_info().proof_of_time(),
);
entropy.insert(
block_number,
PotEntropyValue {
target_slot: None,
entropy: current_block_entropy,
},
);
if let Some(entropy_source_block_number) = maybe_entropy_source_block_number {
if let Some(entropy_value) = entropy.get_mut(&entropy_source_block_number) {
let target_slot = pre_digest
.slot()
.saturating_add(pot_entropy_injection_delay);
debug!(
target: "runtime::subspace",
"Pot entropy injection will happen at slot {target_slot:?}",
);
entropy_value.target_slot.replace(target_slot);
if let Some(update) = &mut pot_slot_iterations.update
&& update.target_slot.is_none()
{
debug!(
target: "runtime::subspace",
"Scheduling PoT slots update to happen at slot {target_slot:?}"
);
update.target_slot.replace(target_slot);
PotSlotIterations::<T>::put(pot_slot_iterations);
}
}
}
PotEntropy::<T>::put(entropy.clone());
}
if let Some(entropy_source_block_number) = maybe_entropy_source_block_number {
let maybe_entropy_value = entropy.get(&entropy_source_block_number).copied();
if let Some(PotEntropyValue {
target_slot,
entropy,
}) = maybe_entropy_value
{
let target_slot = target_slot
.expect("Target slot is guaranteed to be present due to logic above; qed");
let slot_iterations = if let Some(update) = pot_slot_iterations.update
&& let Some(update_target_slot) = update.target_slot
&& update_target_slot == target_slot
{
debug!(
target: "runtime::subspace",
"Applying PoT slots update to the next PoT parameters change"
);
update.slot_iterations
} else {
pot_slot_iterations.slot_iterations
};
frame_system::Pallet::<T>::deposit_log(DigestItem::pot_parameters_change(
PotParametersChange {
slot: target_slot,
slot_iterations,
entropy,
},
));
}
}
if let Some(entry) = entropy.first_entry() {
if let Some(target_slot) = entry.get().target_slot
&& target_slot < current_slot
{
entry.remove();
PotEntropy::<T>::put(entropy);
}
}
}
}
fn do_finalize(_block_number: BlockNumberFor<T>) {
if let Some(next_solution_range) = SolutionRanges::<T>::get().next {
frame_system::Pallet::<T>::deposit_log(DigestItem::next_solution_range(
next_solution_range,
));
}
if let Some((public_key, sector_index, piece_offset, scalar, slot, _reward_address)) =
CurrentBlockAuthorInfo::<T>::get()
{
ParentBlockAuthorInfo::<T>::put((public_key, sector_index, piece_offset, scalar, slot));
} else {
ParentBlockAuthorInfo::<T>::take();
}
ParentVoteVerificationData::<T>::put(current_vote_verification_data::<T>(true));
ParentBlockVoters::<T>::put(CurrentBlockVoters::<T>::get().unwrap_or_default());
DidProcessSegmentHeaders::<T>::take();
}
fn do_store_segment_headers(segment_headers: Vec<SegmentHeader>) -> DispatchResult {
assert!(
!DidProcessSegmentHeaders::<T>::exists(),
"Segment headers must be updated only once in the block"
);
for segment_header in segment_headers {
SegmentCommitment::<T>::insert(
segment_header.segment_index(),
segment_header.segment_commitment(),
);
frame_system::Pallet::<T>::deposit_log(DigestItem::segment_commitment(
segment_header.segment_index(),
segment_header.segment_commitment(),
));
Self::deposit_event(Event::SegmentHeaderStored { segment_header });
}
DidProcessSegmentHeaders::<T>::put(true);
Ok(())
}
fn do_enable_solution_range_adjustment(
solution_range_override: Option<u64>,
voting_solution_range_override: Option<u64>,
) -> DispatchResult {
if ShouldAdjustSolutionRange::<T>::get() {
return Err(Error::<T>::SolutionRangeAdjustmentAlreadyEnabled.into());
}
ShouldAdjustSolutionRange::<T>::put(true);
if let Some(solution_range) = solution_range_override {
let voting_solution_range = voting_solution_range_override.unwrap_or_else(|| {
solution_range.saturating_mul(u64::from(T::ExpectedVotesPerBlock::get()) + 1)
});
SolutionRanges::<T>::mutate(|solution_ranges| {
if solution_ranges.next.is_some() {
solution_ranges.next.replace(solution_range);
solution_ranges.voting_next.replace(voting_solution_range);
} else {
solution_ranges.current = solution_range;
solution_ranges.voting_current = voting_solution_range;
NextSolutionRangeOverride::<T>::put(SolutionRangeOverride {
solution_range,
voting_solution_range,
});
frame_system::Pallet::<T>::deposit_log(DigestItem::next_solution_range(
solution_range,
));
}
});
}
Ok(())
}
fn do_vote(
signed_vote: SignedVote<BlockNumberFor<T>, T::Hash, T::AccountId>,
) -> DispatchResult {
let Vote::V0 {
height,
parent_hash,
solution,
..
} = signed_vote.vote;
Self::deposit_event(Event::FarmerVote {
public_key: solution.public_key,
reward_address: solution.reward_address,
height,
parent_hash,
});
Ok(())
}
fn do_enable_rewards_at(
enable_rewards_at: EnableRewardsAt<BlockNumberFor<T>>,
) -> DispatchResult {
match enable_rewards_at {
EnableRewardsAt::Height(block_number) => {
let next_block_number =
frame_system::Pallet::<T>::current_block_number() + One::one();
EnableRewards::<T>::put(block_number.max(next_block_number));
}
EnableRewardsAt::SolutionRange(solution_range) => {
EnableRewardsBelowSolutionRange::<T>::put(solution_range);
}
EnableRewardsAt::Manually => {
}
}
Ok(())
}
pub fn pot_parameters() -> PotParameters {
let block_number = frame_system::Pallet::<T>::block_number();
let pot_slot_iterations =
PotSlotIterations::<T>::get().expect("Always initialized during genesis; qed");
let pot_entropy_injection_interval = T::PotEntropyInjectionInterval::get();
let entropy = PotEntropy::<T>::get();
let lookback_in_blocks = pot_entropy_injection_interval
* BlockNumberFor::<T>::from(T::PotEntropyInjectionLookbackDepth::get());
let last_entropy_injection_block =
block_number / pot_entropy_injection_interval * pot_entropy_injection_interval;
let maybe_entropy_source_block_number =
last_entropy_injection_block.checked_sub(&lookback_in_blocks);
let mut next_change = None;
if let Some(entropy_source_block_number) = maybe_entropy_source_block_number {
let maybe_entropy_value = entropy.get(&entropy_source_block_number).copied();
if let Some(PotEntropyValue {
target_slot,
entropy,
}) = maybe_entropy_value
{
let target_slot = target_slot.expect(
"Always present due to identical check present in block initialization; qed",
);
let slot_iterations = if let Some(update) = pot_slot_iterations.update
&& let Some(update_target_slot) = update.target_slot
&& update_target_slot == target_slot
{
update.slot_iterations
} else {
pot_slot_iterations.slot_iterations
};
next_change.replace(PotParametersChange {
slot: target_slot,
slot_iterations,
entropy,
});
}
}
PotParameters::V0 {
slot_iterations: pot_slot_iterations.slot_iterations,
next_change,
}
}
pub fn current_slot() -> Slot {
BlockSlots::<T>::get()
.last_key_value()
.map(|(_block, slot)| *slot)
.unwrap_or_default()
}
pub fn archived_history_size() -> u64 {
let archived_segments = SegmentCommitment::<T>::count();
u64::from(archived_segments) * ArchivedHistorySegment::SIZE as u64
}
}
impl<T> Pallet<T>
where
T: Config + SendTransactionTypes<Call<T>>,
{
pub fn submit_vote(signed_vote: SignedVote<BlockNumberFor<T>, T::Hash, T::AccountId>) {
let call = Call::vote {
signed_vote: Box::new(signed_vote),
};
match SubmitTransaction::<T, Call<T>>::submit_unsigned_transaction(call.into()) {
Ok(()) => {
debug!(target: "runtime::subspace", "Submitted Subspace vote");
}
Err(()) => {
error!(target: "runtime::subspace", "Error submitting Subspace vote");
}
}
}
}
impl<T: Config> Pallet<T> {
fn validate_segment_header(
source: TransactionSource,
segment_headers: &[SegmentHeader],
) -> TransactionValidity {
if !matches!(
source,
TransactionSource::Local | TransactionSource::InBlock,
) {
warn!(
target: "runtime::subspace",
"Rejecting segment header extrinsic because it is not local/in-block.",
);
return InvalidTransaction::Call.into();
}
check_segment_headers::<T>(segment_headers)?;
ValidTransaction::with_tag_prefix("SubspaceSegmentHeader")
.priority(TransactionPriority::MAX)
.longevity(0)
.propagate(false)
.build()
}
fn pre_dispatch_segment_header(
segment_headers: &[SegmentHeader],
) -> Result<(), TransactionValidityError> {
check_segment_headers::<T>(segment_headers)
}
fn validate_vote(
signed_vote: &SignedVote<BlockNumberFor<T>, T::Hash, T::AccountId>,
) -> TransactionValidity {
check_vote::<T>(signed_vote, false)?;
ValidTransaction::with_tag_prefix("SubspaceVote")
.priority(TransactionPriority::MAX)
.longevity(2)
.and_provides(signed_vote.signature)
.build()
}
fn pre_dispatch_vote(
signed_vote: &SignedVote<BlockNumberFor<T>, T::Hash, T::AccountId>,
) -> Result<(), TransactionValidityError> {
match check_vote::<T>(signed_vote, true) {
Ok(()) => Ok(()),
Err(CheckVoteError::Equivocated { .. }) => {
Ok(())
}
Err(error) => Err(error.into()),
}
}
}
fn current_vote_verification_data<T: Config>(is_block_initialized: bool) -> VoteVerificationData {
let solution_ranges = SolutionRanges::<T>::get();
VoteVerificationData {
solution_range: if is_block_initialized {
solution_ranges.current
} else {
solution_ranges.next.unwrap_or(solution_ranges.current)
},
vote_solution_range: if is_block_initialized {
solution_ranges.voting_current
} else {
solution_ranges
.voting_next
.unwrap_or(solution_ranges.voting_current)
},
current_slot: Pallet::<T>::current_slot(),
parent_slot: ParentVoteVerificationData::<T>::get()
.map(|parent_vote_verification_data| {
if is_block_initialized {
parent_vote_verification_data.current_slot
} else {
parent_vote_verification_data.parent_slot
}
})
.unwrap_or_else(Pallet::<T>::current_slot),
}
}
#[derive(Debug, Eq, PartialEq)]
enum CheckVoteError {
UnexpectedBeforeHeightTwo,
HeightInTheFuture,
HeightInThePast,
IncorrectParentHash,
SlotInTheFuture,
SlotInThePast,
BadRewardSignature(SignatureError),
UnknownSegmentCommitment,
InvalidHistorySize,
InvalidSolution(String),
QualityTooHigh,
InvalidProofOfTime,
InvalidFutureProofOfTime,
DuplicateVote,
Equivocated { slot: Slot, offender: PublicKey },
}
impl From<CheckVoteError> for TransactionValidityError {
#[inline]
fn from(error: CheckVoteError) -> Self {
TransactionValidityError::Invalid(match error {
CheckVoteError::UnexpectedBeforeHeightTwo => InvalidTransaction::Call,
CheckVoteError::HeightInTheFuture => InvalidTransaction::Future,
CheckVoteError::HeightInThePast => InvalidTransaction::Stale,
CheckVoteError::IncorrectParentHash => InvalidTransaction::Call,
CheckVoteError::SlotInTheFuture => InvalidTransaction::Future,
CheckVoteError::SlotInThePast => InvalidTransaction::Stale,
CheckVoteError::BadRewardSignature(_) => InvalidTransaction::BadProof,
CheckVoteError::UnknownSegmentCommitment => InvalidTransaction::Call,
CheckVoteError::InvalidHistorySize => InvalidTransaction::Call,
CheckVoteError::InvalidSolution(_) => InvalidTransaction::Call,
CheckVoteError::QualityTooHigh => InvalidTransaction::Call,
CheckVoteError::InvalidProofOfTime => InvalidTransaction::Future,
CheckVoteError::InvalidFutureProofOfTime => InvalidTransaction::Call,
CheckVoteError::DuplicateVote => InvalidTransaction::Call,
CheckVoteError::Equivocated { .. } => InvalidTransaction::BadSigner,
})
}
}
fn check_vote<T: Config>(
signed_vote: &SignedVote<BlockNumberFor<T>, T::Hash, T::AccountId>,
pre_dispatch: bool,
) -> Result<(), CheckVoteError> {
let Vote::V0 {
height,
parent_hash,
slot,
solution,
proof_of_time,
future_proof_of_time,
} = &signed_vote.vote;
let height = *height;
let slot = *slot;
let current_block_number = frame_system::Pallet::<T>::current_block_number();
if current_block_number <= One::one() || height <= One::one() {
debug!(
target: "runtime::subspace",
"Votes are not expected at height below 2"
);
return Err(CheckVoteError::UnexpectedBeforeHeightTwo);
}
if !(height == current_block_number || height == current_block_number - One::one()) {
debug!(
target: "runtime::subspace",
"Vote verification error: bad height {height:?}, current block number is \
{current_block_number:?}"
);
return Err(if height > current_block_number {
CheckVoteError::HeightInTheFuture
} else {
CheckVoteError::HeightInThePast
});
}
if *parent_hash != frame_system::Pallet::<T>::block_hash(height - One::one()) {
debug!(
target: "runtime::subspace",
"Vote verification error: parent hash {parent_hash:?}",
);
return Err(CheckVoteError::IncorrectParentHash);
}
let current_vote_verification_data = current_vote_verification_data::<T>(pre_dispatch);
let parent_vote_verification_data = ParentVoteVerificationData::<T>::get()
.expect("Above check for block number ensures that this value is always present; qed");
if pre_dispatch {
let current_slot = current_vote_verification_data.current_slot;
if slot > current_slot || (slot == current_slot && height != current_block_number) {
debug!(
target: "runtime::subspace",
"Vote slot {slot:?} must be before current slot {current_slot:?}",
);
return Err(CheckVoteError::SlotInTheFuture);
}
}
let parent_slot = if pre_dispatch {
if height == current_block_number {
parent_vote_verification_data.current_slot
} else {
parent_vote_verification_data.parent_slot
}
} else {
if height == current_block_number {
current_vote_verification_data.current_slot
} else {
current_vote_verification_data.parent_slot
}
};
if slot <= parent_slot {
debug!(
target: "runtime::subspace",
"Vote slot {slot:?} must be after parent slot {parent_slot:?}",
);
return Err(CheckVoteError::SlotInThePast);
}
if let Err(error) = check_reward_signature(
signed_vote.vote.hash().as_bytes(),
&signed_vote.signature,
&solution.public_key,
&schnorrkel::signing_context(REWARD_SIGNING_CONTEXT),
) {
debug!(
target: "runtime::subspace",
"Vote verification error: {error:?}"
);
return Err(CheckVoteError::BadRewardSignature(error));
}
let vote_verification_data = if height == current_block_number {
current_vote_verification_data
} else {
parent_vote_verification_data
};
let sector_id = SectorId::new(
solution.public_key.hash(),
solution.sector_index,
solution.history_size,
);
let recent_segments = T::RecentSegments::get();
let recent_history_fraction = (
T::RecentHistoryFraction::get().0,
T::RecentHistoryFraction::get().1,
);
let segment_index = sector_id
.derive_piece_index(
solution.piece_offset,
solution.history_size,
T::MaxPiecesInSector::get(),
recent_segments,
recent_history_fraction,
)
.segment_index();
let segment_commitment =
if let Some(segment_commitment) = Pallet::<T>::segment_commitment(segment_index) {
segment_commitment
} else {
debug!(
target: "runtime::subspace",
"Vote verification error: no segment commitment for segment index {segment_index}"
);
return Err(CheckVoteError::UnknownSegmentCommitment);
};
let sector_expiration_check_segment_commitment = Pallet::<T>::segment_commitment(
solution
.history_size
.sector_expiration_check(T::MinSectorLifetime::get())
.ok_or(CheckVoteError::InvalidHistorySize)?
.segment_index(),
);
match verify_solution(
solution.into(),
slot.into(),
(&VerifySolutionParams {
proof_of_time: *proof_of_time,
solution_range: vote_verification_data.vote_solution_range,
piece_check_params: Some(PieceCheckParams {
max_pieces_in_sector: T::MaxPiecesInSector::get(),
segment_commitment,
recent_segments,
recent_history_fraction,
min_sector_lifetime: T::MinSectorLifetime::get(),
current_history_size: Pallet::<T>::history_size(),
sector_expiration_check_segment_commitment,
}),
})
.into(),
) {
Ok(solution_distance) => {
if solution_distance <= vote_verification_data.solution_range / 2 {
debug!(
target: "runtime::subspace",
"Vote quality is too high"
);
return Err(CheckVoteError::QualityTooHigh);
}
}
Err(error) => {
debug!(
target: "runtime::subspace",
"Vote verification error: {error:?}"
);
return Err(CheckVoteError::InvalidSolution(error));
}
}
if !is_proof_of_time_valid(
BlockHash::try_from(parent_hash.as_ref())
.expect("Must be able to convert to block hash type"),
SlotNumber::from(slot),
WrappedPotOutput::from(*proof_of_time),
!pre_dispatch,
) {
debug!(target: "runtime::subspace", "Invalid proof of time");
return Err(CheckVoteError::InvalidProofOfTime);
}
if pre_dispatch
&& !is_proof_of_time_valid(
BlockHash::try_from(parent_hash.as_ref())
.expect("Must be able to convert to block hash type"),
SlotNumber::from(slot + T::BlockAuthoringDelay::get()),
WrappedPotOutput::from(*future_proof_of_time),
false,
)
{
debug!(target: "runtime::subspace", "Invalid future proof of time");
return Err(CheckVoteError::InvalidFutureProofOfTime);
}
let key = (
solution.public_key,
solution.sector_index,
solution.piece_offset,
solution.chunk,
slot,
);
let mut is_equivocating = ParentBlockAuthorInfo::<T>::get().as_ref() == Some(&key)
|| CurrentBlockAuthorInfo::<T>::get()
.map(
|(public_key, sector_index, piece_offset, chunk, slot, _reward_address)| {
(public_key, sector_index, piece_offset, chunk, slot)
},
)
.as_ref()
== Some(&key);
if !is_equivocating {
if let Some((_reward_address, signature)) = ParentBlockVoters::<T>::get().get(&key) {
if signature != &signed_vote.signature {
is_equivocating = true;
} else {
return Err(CheckVoteError::DuplicateVote);
}
}
}
if !is_equivocating {
if let Some((_reward_address, signature)) =
CurrentBlockVoters::<T>::get().unwrap_or_default().get(&key)
{
if signature != &signed_vote.signature {
is_equivocating = true;
} else {
return Err(CheckVoteError::DuplicateVote);
}
}
}
if pre_dispatch {
CurrentBlockVoters::<T>::mutate(|current_reward_receivers| {
current_reward_receivers
.as_mut()
.expect("Always set during block initialization")
.insert(
key,
(
if is_equivocating {
None
} else {
Some(solution.reward_address.clone())
},
signed_vote.signature,
),
);
});
}
if is_equivocating {
let offender = solution.public_key;
CurrentBlockAuthorInfo::<T>::mutate(|maybe_info| {
if let Some((public_key, _sector_index, _piece_offset, _chunk, _slot, reward_address)) =
maybe_info
{
if public_key == &offender {
reward_address.take();
}
}
});
CurrentBlockVoters::<T>::mutate(|current_reward_receivers| {
if let Some(current_reward_receivers) = current_reward_receivers {
for (
(public_key, _sector_index, _piece_offset, _chunk, _slot),
(reward_address, _signature),
) in current_reward_receivers.iter_mut()
{
if public_key == &offender {
reward_address.take();
}
}
}
});
return Err(CheckVoteError::Equivocated { slot, offender });
}
Ok(())
}
fn check_segment_headers<T: Config>(
segment_headers: &[SegmentHeader],
) -> Result<(), TransactionValidityError> {
let mut segment_headers_iter = segment_headers.iter();
let first_segment_header = match segment_headers_iter.next() {
Some(first_segment_header) => first_segment_header,
None => {
return Err(InvalidTransaction::BadMandatory.into());
}
};
if first_segment_header.segment_index() > SegmentIndex::ZERO
&& !SegmentCommitment::<T>::contains_key(
first_segment_header.segment_index() - SegmentIndex::ONE,
)
{
return Err(InvalidTransaction::BadMandatory.into());
}
if SegmentCommitment::<T>::contains_key(first_segment_header.segment_index()) {
return Err(InvalidTransaction::BadMandatory.into());
}
let mut last_segment_index = first_segment_header.segment_index();
for segment_header in segment_headers_iter {
let segment_index = segment_header.segment_index();
if segment_index != last_segment_index + SegmentIndex::ONE {
return Err(InvalidTransaction::BadMandatory.into());
}
if SegmentCommitment::<T>::contains_key(segment_index) {
return Err(InvalidTransaction::BadMandatory.into());
}
last_segment_index = segment_index;
}
Ok(())
}
impl<T: Config> subspace_runtime_primitives::RewardsEnabled for Pallet<T> {
fn rewards_enabled() -> bool {
if let Some(height) = EnableRewards::<T>::get() {
frame_system::Pallet::<T>::current_block_number() >= height
} else {
false
}
}
}
impl<T: Config> subspace_runtime_primitives::FindBlockRewardAddress<T::AccountId> for Pallet<T> {
fn find_block_reward_address() -> Option<T::AccountId> {
CurrentBlockAuthorInfo::<T>::get().and_then(
|(_public_key, _sector_index, _piece_offset, _chunk, _slot, reward_address)| {
if let Some(height) = EnableRewards::<T>::get() {
if frame_system::Pallet::<T>::current_block_number() >= height {
return reward_address;
}
}
None
},
)
}
}
impl<T: Config> subspace_runtime_primitives::FindVotingRewardAddresses<T::AccountId> for Pallet<T> {
fn find_voting_reward_addresses() -> Vec<T::AccountId> {
if let Some(height) = EnableRewards::<T>::get() {
if frame_system::Pallet::<T>::current_block_number() >= height {
return CurrentBlockVoters::<T>::get()
.unwrap_or_else(ParentBlockVoters::<T>::get)
.into_values()
.filter_map(|(reward_address, _signature)| reward_address)
.collect();
}
}
Vec::new()
}
}
impl<T: Config> frame_support::traits::Randomness<T::Hash, BlockNumberFor<T>> for Pallet<T> {
fn random(subject: &[u8]) -> (T::Hash, BlockNumberFor<T>) {
let mut subject = subject.to_vec();
subject.extend_from_slice(
BlockRandomness::<T>::get()
.expect("Block randomness is always set in block initialization; qed")
.as_ref(),
);
(
T::Hashing::hash(&subject),
frame_system::Pallet::<T>::current_block_number(),
)
}
fn random_seed() -> (T::Hash, BlockNumberFor<T>) {
(
T::Hashing::hash(
BlockRandomness::<T>::get()
.expect("Block randomness is always set in block initialization; qed")
.as_ref(),
),
frame_system::Pallet::<T>::current_block_number(),
)
}
}