Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

prospective-parachains rework: take II #4937

Merged
merged 64 commits into from
Aug 12, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
64 commits
Select commit Hold shift + click to select a range
8362d1e
first prototype
alindima Jun 26, 2024
d634867
first prototype part 2
alindima Jun 28, 2024
ded4ee9
first sane version
alindima Jul 1, 2024
53f9327
some more code
alindima Jul 2, 2024
67d6887
bugfixes
alindima Jul 3, 2024
7a59fcf
add doc comments to HypotheticalOrConcreteCandidate
alindima Jul 3, 2024
33f239e
fix compilation
alindima Jul 4, 2024
ac9285e
some minor refactors
alindima Jul 4, 2024
2b58595
mostly cosmetics
alindima Jul 4, 2024
ce00024
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Jul 4, 2024
dfba945
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Jul 5, 2024
e34de0e
start updating comments
alindima Jul 5, 2024
2993cd7
more comments
alindima Jul 5, 2024
a8ee808
update inclusion emulator docs
alindima Jul 9, 2024
49634ea
metrics
alindima Jul 9, 2024
2ab16a8
rename chain to best_chain and group it in a struct for better readab…
alindima Jul 9, 2024
fdd8e54
nits
alindima Jul 9, 2024
297781b
start modifying tests
alindima Jul 11, 2024
386273e
some unit tests
alindima Jul 12, 2024
3c0d7d8
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Jul 24, 2024
a56f371
more unit tests
alindima Jul 24, 2024
cb91a20
typos
alindima Jul 24, 2024
1f1fde5
unit test for ForkWithCandidatePendingAvailability
alindima Jul 24, 2024
1a07e56
update lock
alindima Jul 24, 2024
eae9345
fmt
alindima Jul 24, 2024
27dd91d
don't keep candidates from previous leaf if they used to be pending a…
alindima Jul 25, 2024
b691b86
more tests
alindima Jul 25, 2024
b13c536
unit tests for backable chain
alindima Jul 25, 2024
45b4133
move wild import
alindima Jul 25, 2024
2cbc770
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Jul 25, 2024
6b10a11
add rand dependency
alindima Jul 25, 2024
a81f149
small refactor
alindima Jul 26, 2024
dedfd68
more unit testing
alindima Jul 26, 2024
a83d78f
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Jul 29, 2024
d758d14
unit testing
alindima Jul 29, 2024
8858f36
remove unused deps
alindima Jul 30, 2024
b2612e9
finish up tests
alindima Jul 30, 2024
3443af1
add prdoc
alindima Jul 31, 2024
4b975cb
rollback CI yaml changes
alindima Jul 31, 2024
e4dd399
correct log
alindima Aug 1, 2024
d43f024
fix another log
alindima Aug 1, 2024
10e3425
remove CandidateAlreadyPendingAvailability error variant
alindima Aug 1, 2024
e9da176
fix verbose log
alindima Aug 2, 2024
0c5f6c3
fix bug with relay chain forks not getting the candidates from the pr…
alindima Aug 2, 2024
90b6983
log active leaf updates
alindima Aug 2, 2024
797e9b6
add test for bugfix
alindima Aug 2, 2024
b5eaf54
use backing implicit view in prospective-parachains
alindima Aug 5, 2024
e54566f
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Aug 7, 2024
5166663
update some comments
alindima Aug 7, 2024
9a4b88d
address some comments
alindima Aug 7, 2024
4ba7b0f
a bit of refactoring
alindima Aug 7, 2024
021515e
optimise candidate_backed
alindima Aug 7, 2024
a9c5799
move public items to the top of the impl blocks
alindima Aug 8, 2024
243466f
remove comment
alindima Aug 8, 2024
78cf31a
refactor fragment chain constructor and acitve leave update handling
alindima Aug 8, 2024
0243915
restrict the visibility of some items
alindima Aug 8, 2024
034706e
dedup into a From impl
alindima Aug 8, 2024
09eb5ee
update prdoc
alindima Aug 8, 2024
80e52cb
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Aug 8, 2024
64546fe
update some comments
alindima Aug 9, 2024
1c31af0
address a todo
alindima Aug 9, 2024
142432d
keep tracking candidates for deactivated leaves in implicit view
alindima Aug 9, 2024
a9d7131
add test for bounded implicit view
alindima Aug 9, 2024
32b2ebf
Merge remote-tracking branch 'origin/master' into alindima/prospectiv…
alindima Aug 9, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
278 changes: 142 additions & 136 deletions polkadot/node/core/prospective-parachains/src/fragment_chain/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -216,7 +216,7 @@ impl CandidateStorage {
}

/// Introduce a new candidate entry.
pub fn add_candidate_entry(&mut self, candidate: CandidateEntry) -> Result<(), Error> {
fn add_candidate_entry(&mut self, candidate: CandidateEntry) -> Result<(), Error> {
let candidate_hash = candidate.candidate_hash;
if self.by_candidate_hash.contains_key(&candidate_hash) {
return Err(Error::CandidateAlreadyKnown)
Expand Down Expand Up @@ -274,11 +274,6 @@ impl CandidateStorage {
self.by_candidate_hash.values()
}

/// Consume self into an iterator over the stored candidates, in arbitrary order.
pub fn into_candidates(self) -> impl Iterator<Item = CandidateEntry> {
self.by_candidate_hash.into_values()
}

/// Try getting head-data by hash.
fn head_data_by_hash(&self, hash: &Hash) -> Option<&HeadData> {
// First, search for candidates outputting this head data and extract the head data
Expand Down Expand Up @@ -363,6 +358,10 @@ impl CandidateEntry {
Self::new(candidate_hash, candidate, persisted_validation_data, CandidateState::Seconded)
}

pub fn hash(&self) -> CandidateHash {
self.candidate_hash
}

fn new(
candidate_hash: CandidateHash,
candidate: CommittedCandidateReceipt,
Expand Down Expand Up @@ -394,10 +393,6 @@ impl CandidateEntry {
}),
})
}

pub fn hash(&self) -> CandidateHash {
self.candidate_hash
}
}

impl HypotheticalOrConcreteCandidate for CandidateEntry {
Expand Down Expand Up @@ -542,22 +537,24 @@ impl Scope {
self.ancestors_by_hash.get(hash).map(|info| info.clone())
}

/// Get the base constraints of the scope
pub fn base_constraints(&self) -> &Constraints {
&self.base_constraints
}

/// Whether the candidate in question is one pending availability in this scope.
pub fn get_pending_availability(
fn get_pending_availability(
&self,
candidate_hash: &CandidateHash,
) -> Option<&PendingAvailability> {
self.pending_availability.iter().find(|c| &c.candidate_hash == candidate_hash)
}

/// Get the base constraints of the scope
pub fn base_constraints(&self) -> &Constraints {
&self.base_constraints
}
}

#[cfg_attr(test, derive(Clone))]
pub struct FragmentNode {
/// A node that is part of a `BackedChain`. It holds constraints based on the ancestors in the
/// chain.
struct FragmentNode {
fragment: Fragment,
candidate_hash: CandidateHash,
cumulative_modifications: ConstraintModifications,
Expand All @@ -571,6 +568,22 @@ impl FragmentNode {
}
}

impl From<&FragmentNode> for CandidateEntry {
fn from(node: &FragmentNode) -> Self {
// We don't need to perform the checks done in `CandidateEntry::new()`, since a
// `FragmentNode` always comes from a `CandidateEntry`
Self {
candidate_hash: node.candidate_hash,
parent_head_data_hash: node.parent_head_data_hash,
output_head_data_hash: node.output_head_data_hash,
candidate: node.fragment.candidate_clone(),
relay_parent: node.relay_parent(),
// A fragment node is always backed.
state: CandidateState::Backed,
}
}
}

/// A candidate chain of backed/backable candidates.
/// Includes the candidates pending availability and candidates which may be backed on-chain.
#[derive(Default)]
Expand Down Expand Up @@ -659,28 +672,56 @@ pub(crate) struct FragmentChain {
}

impl FragmentChain {
/// Create a new [`FragmentChain`] with given scope and populated from the given storage.
/// The `prev_storage` should contain the candidates of the `FragmentChain` at the previous
/// relay parent, as well as the candidates pending availability at this relay parent.
pub fn populate(scope: Scope, mut prev_storage: CandidateStorage) -> Self {
// Initialize as empty
/// Create a new [`FragmentChain`] with the given scope and populate it with the candidates
/// pending availability.
pub fn init(scope: Scope, mut candidates_pending_availability: CandidateStorage) -> Self {
let mut fragment_chain = Self {
scope,
best_chain: BackedChain::default(),
unconnected: CandidateStorage::default(),
};

// We only need to populate the best backable chain. Candidates pending availability must
// form a chain with the latest included head.
fragment_chain.populate_chain(&mut candidates_pending_availability);

// TODO: return error if not all candidates were introduced successfully.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reminder.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. I chose to log a message instead


fragment_chain
}

/// Populate the [`FragmentChain`] given the new candidates pending availability and the
/// optional previous fragment chain (of the previous relay parent).
pub fn populate_from_previous(&mut self, prev_fragment_chain: &FragmentChain) {
let mut prev_storage = prev_fragment_chain.unconnected.clone();

for candidate in prev_fragment_chain.best_chain.chain.iter() {
// If they used to be pending availability, don't add them. This is fine
// because:
// - if they still are pending availability, they have already been added to the new
// storage.
// - if they were included, no point in keeping them.
//
// This cannot happen for the candidates in the unconnected storage. The pending
// availability candidates will always be part of the best chain.
if prev_fragment_chain
.scope
.get_pending_availability(&candidate.candidate_hash)
.is_none()
{
let _ = prev_storage.add_candidate_entry(candidate.into());
}
}

// First populate the best backable chain.
fragment_chain.populate_chain(&mut prev_storage);
self.populate_chain(&mut prev_storage);

// Now that we picked the best backable chain, trim the forks generated by candidates which
// are not present in the best chain.
fragment_chain.trim_uneligible_forks(&mut prev_storage, None);
self.trim_uneligible_forks(&mut prev_storage, None);

// Finally, keep any candidates which haven't been trimmed but still have potential.
fragment_chain.populate_unconnected_potential_candidates(prev_storage);

fragment_chain
self.populate_unconnected_potential_candidates(prev_storage);
}

/// Get the scope of the [`FragmentChain`].
Expand Down Expand Up @@ -722,34 +763,78 @@ impl FragmentChain {
)
}

/// Return a new [`CandidateStorage`] containing all the candidates from this `FragmentChain`,
/// as well as the unconnected ones. This does not contain the candidates that used to be
/// pending availability.
pub fn advance_scope(&self) -> CandidateStorage {
let mut storage = self.unconnected.clone();
/// Mark a candidate as backed. This can trigger a recreation of the best backable chain.
pub fn candidate_backed(&mut self, newly_backed_candidate: &CandidateHash) {
// Already backed.
if self.best_chain.candidates.contains(newly_backed_candidate) {
return
}
let Some(parent_head_hash) = self
.unconnected
.by_candidate_hash
.get(newly_backed_candidate)
.map(|entry| entry.parent_head_data_hash)
else {
// Candidate is not in unconnected storage.
return
};

for candidate in self.best_chain.chain.iter() {
// If they used to be pending availability, don't add them. This is fine
// because:
// - if they still are pending availability, they have already been added to the new
// storage.
// - if they were included, no point in keeping them.
//
// This cannot happen for the candidates in the unconnected storage. The pending
// availability candidates will always be part of the best chain.
if self.scope.get_pending_availability(&candidate.candidate_hash).is_none() {
let _ = storage.add_candidate_entry(CandidateEntry {
candidate_hash: candidate.candidate_hash,
parent_head_data_hash: candidate.parent_head_data_hash,
output_head_data_hash: candidate.output_head_data_hash,
relay_parent: candidate.relay_parent(),
candidate: candidate.fragment.candidate_clone(), // This clone is very cheap.
state: CandidateState::Backed,
});
}
// Mark the candidate hash.
self.unconnected.mark_backed(newly_backed_candidate);

// Revert to parent_head_hash
if !self.revert_to(&parent_head_hash) {
// If nothing was reverted, there is nothing we can do for now.
return
}

let mut prev_storage = std::mem::take(&mut self.unconnected);

// Populate the chain.
self.populate_chain(&mut prev_storage);

// Now that we picked the best backable chain, trim the forks generated by candidates
// which are not present in the best chain. We can start trimming from this candidate
// onwards.
self.trim_uneligible_forks(&mut prev_storage, Some(parent_head_hash));

// Finally, keep any candidates which haven't been trimmed but still have potential.
self.populate_unconnected_potential_candidates(prev_storage);
}

/// Checks if this candidate could be added in the future to this chain.
/// This will return `Error::CandidateAlreadyKnown` if the candidate is already in the chain or
/// the unconnected candidate storage.
pub fn can_add_candidate_as_potential(
&self,
candidate: &impl HypotheticalOrConcreteCandidate,
) -> Result<(), Error> {
let candidate_hash = candidate.candidate_hash();

if self.best_chain.contains(&candidate_hash) || self.unconnected.contains(&candidate_hash) {
return Err(Error::CandidateAlreadyKnown)
}

self.check_potential(candidate)
}

/// Try adding a seconded candidate, if the candidate has potential. It will never be added to
/// the chain directly in the seconded state, it will only be part of the unconnected storage.
pub fn try_adding_seconded_candidate(
&mut self,
candidate: &CandidateEntry,
) -> Result<(), Error> {
if candidate.state == CandidateState::Backed {
return Err(Error::IntroduceBackedCandidate);
}

storage
self.can_add_candidate_as_potential(candidate)?;

// This clone is cheap, as it uses an Arc for the expensive stuff.
// We can't consume the candidate because other fragment chains may use it also.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Best to leave that decision to the caller. If we need a CandidateEntry and not a &CandidateEntry, we should just state it so and leave it to the caller whether any cloning is required or not.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the idea was to not clone unless we have to. There's no need to clone if the self.can_add_candidate_as_potential(candidate)?; returns an Error

self.unconnected.add_candidate_entry(candidate.clone())?;

Ok(())
}

/// Try getting the full head data associated with this hash.
Expand Down Expand Up @@ -877,44 +962,9 @@ impl FragmentChain {
.unwrap_or_else(|| self.scope.earliest_relay_parent())
}

/// Checks if this candidate could be added in the future to this chain.
/// This will return `Error::CandidateAlreadyKnown` if the candidate is already in the chain or
/// the unconnected candidate storage.
pub fn can_add_candidate_as_potential(
&self,
candidate: &impl HypotheticalOrConcreteCandidate,
) -> Result<(), Error> {
let candidate_hash = candidate.candidate_hash();

if self.best_chain.contains(&candidate_hash) || self.unconnected.contains(&candidate_hash) {
return Err(Error::CandidateAlreadyKnown)
}

self.check_potential(candidate)
}

/// Try adding a seconded candidate, if the candidate has potential. It will never be added to
/// the chain directly in the seconded state, it will only be part of the unconnected storage.
pub fn try_adding_seconded_candidate(
&mut self,
candidate: &CandidateEntry,
) -> Result<(), Error> {
if candidate.state == CandidateState::Backed {
return Err(Error::IntroduceBackedCandidate);
}

self.can_add_candidate_as_potential(candidate)?;

// This clone is cheap, as it uses an Arc for the expensive stuff.
// We can't consume the candidate because other fragment chains may use it also.
self.unconnected.add_candidate_entry(candidate.clone())?;

Ok(())
}

// Populate the unconnected potential candidate storage starting from a previous storage.
fn populate_unconnected_potential_candidates(&mut self, old_storage: CandidateStorage) {
for candidate in old_storage.into_candidates() {
for candidate in old_storage.by_candidate_hash.into_values() {
// Sanity check, all pending availability candidates should be already present in the
// chain.
if self.scope.get_pending_availability(&candidate.candidate_hash).is_some() {
Expand Down Expand Up @@ -1060,6 +1110,8 @@ impl FragmentChain {

// Once the backable chain was populated, trim the forks generated by candidates which
// are not present in the best chain. Fan this out into a full breadth-first search.
// If `starting_point` is `Some()`, start the search from the candidates having this parent head
// hash.
fn trim_uneligible_forks(&self, storage: &mut CandidateStorage, starting_point: Option<Hash>) {
alindima marked this conversation as resolved.
Show resolved Hide resolved
// Start out with the candidates in the chain. They are all valid candidates.
let mut queue: VecDeque<_> = if let Some(starting_point) = starting_point {
Expand Down Expand Up @@ -1275,45 +1327,6 @@ impl FragmentChain {
}
}

/// Mark a candidate as backed. This can trigger a recreation of the best backable chain.
pub fn candidate_backed(&mut self, newly_backed_candidate: &CandidateHash) {
// Already backed.
if self.best_chain.candidates.contains(newly_backed_candidate) {
return
}
let Some(parent_head_hash) = self
.unconnected
.by_candidate_hash
.get(newly_backed_candidate)
.map(|entry| entry.parent_head_data_hash)
else {
// Candidate is not in unconnected storage.
return
};

// Mark the candidate hash.
self.unconnected.mark_backed(newly_backed_candidate);

// Revert to parent_head_hash
if !self.revert_to(&parent_head_hash) {
// If nothing was reverted, there is nothing we can do for now.
return
}

let mut prev_storage = std::mem::take(&mut self.unconnected);

// Populate the chain.
self.populate_chain(&mut prev_storage);

// Now that we picked the best backable chain, trim the forks generated by candidates
// which are not present in the best chain. We can start trimming from this candidate
// onwards.
self.trim_uneligible_forks(&mut prev_storage, Some(parent_head_hash));

// Finally, keep any candidates which haven't been trimmed but still have potential.
self.populate_unconnected_potential_candidates(prev_storage);
}

// Revert the best backable chain so that the last candidate will be one outputting the given
// `parent_head_hash`. If the `parent_head_hash` is exactly the required parent of the base
// constraints (builds on the latest included candidate), revert the entire chain.
Expand All @@ -1333,15 +1346,8 @@ impl FragmentChain {

// Even if it's empty, we need to return true, because we'll be able to add a new candidate
// to the chain.
for node in removed_items {
let _ = self.unconnected.add_candidate_entry(CandidateEntry {
candidate_hash: node.candidate_hash,
parent_head_data_hash: node.parent_head_data_hash,
output_head_data_hash: node.output_head_data_hash,
candidate: node.fragment.candidate_clone(),
relay_parent: node.relay_parent(),
state: CandidateState::Backed,
});
for node in &removed_items {
let _ = self.unconnected.add_candidate_entry(node.into());
}
true
}
Expand Down
Loading
Loading