This repository has been archived by the owner on Nov 15, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 379
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
refactor unincluded segment length into a ConsensusHook (#2501)
* refactor unincluded segment length into a ConsensusHook * add docs * refactor bandwidth_out calculation Co-authored-by: Chris Sosnin <[email protected]> * test for limits from impl * fmt * make tests compile * update comment * uncomment test * fix collator test by adding parent to state proof * patch HRMP watermark rules for unincluded segment * get consensus-common tests to pass, using unincluded segment * fix unincluded segment tests * get all tests passing * fmt * rustdoc CI * aura-ext: limit the number of authored blocks per slot (#2551) * aura_ext consensus hook * reverse dependency * include weight into hook * fix tests * remove stray println Co-authored-by: Chris Sosnin <[email protected]> * fix test warning * fix doc link --------- Co-authored-by: Chris Sosnin <[email protected]> Co-authored-by: Chris Sosnin <[email protected]>
- Loading branch information
1 parent
86b5d3d
commit dcf62e5
Showing
30 changed files
with
627 additions
and
157 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,56 @@ | ||
// Copyright 2023 Parity Technologies (UK) Ltd. | ||
// This file is part of Cumulus. | ||
|
||
// Cumulus is free software: you can redistribute it and/or modify | ||
// it under the terms of the GNU General Public License as published by | ||
// the Free Software Foundation, either version 3 of the License, or | ||
// (at your option) any later version. | ||
|
||
// Cumulus is distributed in the hope that it will be useful, | ||
// but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
// GNU General Public License for more details. | ||
|
||
// You should have received a copy of the GNU General Public License | ||
// along with Cumulus. If not, see <http://www.gnu.org/licenses/>. | ||
|
||
//! The definition of a [`FixedVelocityConsensusHook`] for consensus logic to manage | ||
//! block velocity. | ||
//! | ||
//! The velocity `V` refers to the rate of block processing by the relay chain. | ||
use super::pallet; | ||
use cumulus_pallet_parachain_system::{ | ||
consensus_hook::{ConsensusHook, UnincludedSegmentCapacity}, | ||
relay_state_snapshot::RelayChainStateProof, | ||
}; | ||
use frame_support::pallet_prelude::*; | ||
use sp_std::{marker::PhantomData, num::NonZeroU32}; | ||
|
||
/// A consensus hook for a fixed block processing velocity and unincluded segment capacity. | ||
pub struct FixedVelocityConsensusHook<T, const V: u32, const C: u32>(PhantomData<T>); | ||
|
||
impl<T: pallet::Config, const V: u32, const C: u32> ConsensusHook | ||
for FixedVelocityConsensusHook<T, V, C> | ||
{ | ||
// Validates the number of authored blocks within the slot with respect to the `V + 1` limit. | ||
fn on_state_proof(_state_proof: &RelayChainStateProof) -> (Weight, UnincludedSegmentCapacity) { | ||
// Ensure velocity is non-zero. | ||
let velocity = V.max(1); | ||
|
||
let authored = pallet::Pallet::<T>::slot_info() | ||
.map(|(_slot, authored)| authored) | ||
.expect("slot info is inserted on block initialization"); | ||
if authored > velocity + 1 { | ||
panic!("authored blocks limit is reached for the slot") | ||
} | ||
let weight = T::DbWeight::get().reads(1); | ||
|
||
( | ||
weight, | ||
NonZeroU32::new(sp_std::cmp::max(C, 1)) | ||
.expect("1 is the minimum value and non-zero; qed") | ||
.into(), | ||
) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,109 @@ | ||
// Copyright 2023 Parity Technologies (UK) Ltd. | ||
// This file is part of Cumulus. | ||
|
||
// Cumulus is free software: you can redistribute it and/or modify | ||
// it under the terms of the GNU General Public License as published by | ||
// the Free Software Foundation, either version 3 of the License, or | ||
// (at your option) any later version. | ||
|
||
// Cumulus is distributed in the hope that it will be useful, | ||
// but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
// GNU General Public License for more details. | ||
|
||
// You should have received a copy of the GNU General Public License | ||
// along with Cumulus. If not, see <http://www.gnu.org/licenses/>. | ||
|
||
//! The definition of a [`ConsensusHook`] trait for consensus logic to manage the backlog | ||
//! of parachain blocks ready to submit to the relay chain, as well as some basic implementations. | ||
use super::relay_state_snapshot::RelayChainStateProof; | ||
use frame_support::weights::Weight; | ||
use sp_std::num::NonZeroU32; | ||
|
||
/// The possible capacity of the unincluded segment. | ||
#[derive(Clone)] | ||
pub struct UnincludedSegmentCapacity(UnincludedSegmentCapacityInner); | ||
|
||
impl UnincludedSegmentCapacity { | ||
pub(crate) fn get(&self) -> u32 { | ||
match self.0 { | ||
UnincludedSegmentCapacityInner::ExpectParentIncluded => 1, | ||
UnincludedSegmentCapacityInner::Value(v) => v.get(), | ||
} | ||
} | ||
|
||
pub(crate) fn is_expecting_included_parent(&self) -> bool { | ||
match self.0 { | ||
UnincludedSegmentCapacityInner::ExpectParentIncluded => true, | ||
UnincludedSegmentCapacityInner::Value(_) => false, | ||
} | ||
} | ||
} | ||
|
||
#[derive(Clone)] | ||
pub(crate) enum UnincludedSegmentCapacityInner { | ||
ExpectParentIncluded, | ||
Value(NonZeroU32), | ||
} | ||
|
||
impl From<NonZeroU32> for UnincludedSegmentCapacity { | ||
fn from(value: NonZeroU32) -> Self { | ||
UnincludedSegmentCapacity(UnincludedSegmentCapacityInner::Value(value)) | ||
} | ||
} | ||
|
||
/// The consensus hook for dealing with the unincluded segment. | ||
/// | ||
/// Higher-level and user-configurable consensus logic is more informed about the | ||
/// desired unincluded segment length, as well as any rules for adapting it dynamically | ||
/// according to the relay-chain state. | ||
pub trait ConsensusHook { | ||
/// This hook is called partway through the `set_validation_data` inherent in parachain-system. | ||
/// | ||
/// The hook is allowed to panic if customized consensus rules aren't met and is required | ||
/// to return a maximum capacity for the unincluded segment with weight consumed. | ||
fn on_state_proof(state_proof: &RelayChainStateProof) -> (Weight, UnincludedSegmentCapacity); | ||
} | ||
|
||
/// A special consensus hook for handling the migration to asynchronous backing gracefully, | ||
/// even if collators haven't been updated to provide the last included parent in the state | ||
/// proof yet. | ||
/// | ||
/// This behaves as though the parent is included, even if the relay chain state proof doesn't contain | ||
/// the included para head. If the para head is present in the state proof, this does ensure the | ||
/// parent is included. | ||
pub struct ExpectParentIncluded; | ||
|
||
impl ConsensusHook for ExpectParentIncluded { | ||
fn on_state_proof(_state_proof: &RelayChainStateProof) -> (Weight, UnincludedSegmentCapacity) { | ||
( | ||
Weight::zero(), | ||
UnincludedSegmentCapacity(UnincludedSegmentCapacityInner::ExpectParentIncluded), | ||
) | ||
} | ||
} | ||
|
||
/// A consensus hook for a fixed unincluded segment length. This hook does nothing but | ||
/// set the capacity of the unincluded segment to the constant N. | ||
/// | ||
/// Since it is illegal to provide an unincluded segment length of 0, this sets a minimum of | ||
/// 1. | ||
pub struct FixedCapacityUnincludedSegment<const N: u32>; | ||
|
||
impl<const N: u32> ConsensusHook for FixedCapacityUnincludedSegment<N> { | ||
fn on_state_proof(_state_proof: &RelayChainStateProof) -> (Weight, UnincludedSegmentCapacity) { | ||
( | ||
Weight::zero(), | ||
NonZeroU32::new(sp_std::cmp::max(N, 1)) | ||
.expect("1 is the minimum value and non-zero; qed") | ||
.into(), | ||
) | ||
} | ||
} | ||
|
||
/// A fixed-capacity unincluded segment hook, which requires that the parent block is | ||
/// included prior to the current block being authored. | ||
/// | ||
/// This is a simple type alias around a fixed-capacity unincluded segment with a size of 1. | ||
pub type RequireParentIncluded = FixedCapacityUnincludedSegment<1>; |
Oops, something went wrong.