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

IF: Implement computing finality digest #2282

Merged
merged 71 commits into from
Mar 14, 2024
Merged
Show file tree
Hide file tree
Changes from 67 commits
Commits
Show all changes
71 commits
Select commit Hold shift + click to select a range
4d02cf5
remove unnecessary proposal_mtree and finality_mtree from block_heade…
linh2931 Mar 4, 2024
f7bc026
add finality_mroot() to block_header_state.hpp and update the comment…
linh2931 Mar 4, 2024
4639791
add valid structure and implement get_finality_mroot
linh2931 Mar 5, 2024
8014a38
implement build_valid_structure and add valid to block_state constructor
linh2931 Mar 5, 2024
90f87f9
add finality_mroot_claim
linh2931 Mar 5, 2024
f0c7501
add validating_finality_mroot_claim as an optional parameter to assem…
linh2931 Mar 5, 2024
14b1173
implement compute_finalizer_digest()
linh2931 Mar 5, 2024
f02f113
verify finality_mroot claim
linh2931 Mar 5, 2024
1f77ef2
remove unnecessary block_num from valid structure
linh2931 Mar 5, 2024
57bf066
rename finality_tree to finality_merkel_tree, validation_tree_roots t…
linh2931 Mar 5, 2024
2260fd5
use a meaning full name for parent_bsp instead of it
linh2931 Mar 5, 2024
f374b90
do not calculate core again if it is already calculated as updated_core
linh2931 Mar 5, 2024
5f83347
fix missing variable definitions in the existing EOS_ASSERT for Block…
linh2931 Mar 5, 2024
1e8d557
do not fetch parent_bsp again if it is available
linh2931 Mar 5, 2024
d3dacfe
add more comments and some minor clean up
linh2931 Mar 5, 2024
472b3e2
update forked_tests/fork_with_bad_block to match corrected exception …
linh2931 Mar 5, 2024
b79e173
remove a line of code added for debugging
linh2931 Mar 5, 2024
af45a31
stop calculating action_mroot in Savanna
linh2931 Mar 6, 2024
ec04b34
construct valid structure for Savanna genesis block properly
linh2931 Mar 6, 2024
c9f9611
Merge branch 'hotstuff_integration' into compute_finality_digest
linh2931 Mar 6, 2024
a9bb0b3
use new version fork_db::get_block for getting block state on root
linh2931 Mar 6, 2024
6824c2b
add missing valid in FC_REFLECT
linh2931 Mar 7, 2024
ef8c82a
rename finality_mroot to action_mroot in finality_leaf_node_t
linh2931 Mar 7, 2024
87fb8b0
do not perform validation again if finality mroot was already validated
linh2931 Mar 7, 2024
32ee804
use action_mroot in leaf node correctly
linh2931 Mar 7, 2024
8918588
Merge branch 'hotstuff_integration' into compute_finality_digest
linh2931 Mar 7, 2024
de0b91a
fix a compile error due to merging latest target branch
linh2931 Mar 7, 2024
700b3ed
use chain_head as the parent of the block being built
linh2931 Mar 7, 2024
35a9227
cache action_mroot in assembled_block to be reused in apply_block; ma…
linh2931 Mar 8, 2024
42fc370
add finality_tree_leaf_version
linh2931 Mar 8, 2024
28ccc0b
make action_mroot parameter in build_valid_structure as a const &
linh2931 Mar 8, 2024
569de69
make building next valid structure as a member function of valid
linh2931 Mar 8, 2024
200764f
In building_block_if, change parent's type from block_header_state to…
linh2931 Mar 8, 2024
367924b
use parent from building block instead of getting from fork_db in new…
linh2931 Mar 8, 2024
3260dfc
do not reuse updated_core
linh2931 Mar 8, 2024
1eec92f
add is_genesis_block_num() to finality_core
linh2931 Mar 9, 2024
bce5309
remove block_num from finality_mroot_claim (only containing finality_…
linh2931 Mar 9, 2024
e630a61
controller.cpp
linh2931 Mar 9, 2024
6cf0d83
Revert "controller.cpp"
linh2931 Mar 9, 2024
ebcdb9e
move creation of block_state valid structure from apply_block to asse…
linh2931 Mar 9, 2024
74a4eff
update validation tree and finality mroot according to a slightly cha…
linh2931 Mar 9, 2024
2e8956b
add light_header_protocol_version_major and light_header_protocol_ver…
linh2931 Mar 9, 2024
b1a81e7
recaculate action_mroot in non-canonical-format for IF genesis block …
linh2931 Mar 11, 2024
5546215
remove unused updated_core
linh2931 Mar 11, 2024
20cc780
make get_qc_data a function, do not use optional for qc_data and fina…
linh2931 Mar 11, 2024
8cd4b8a
Merge branch 'hotstuff_integration' into compute_finality_digest
linh2931 Mar 11, 2024
f15df1d
remove proposal_mtree and finality_mtree from snapshot V7 due to merg…
linh2931 Mar 11, 2024
c86a8a6
change non_canonical_action_mroot to action_mroot_svnn
linh2931 Mar 11, 2024
e4ab6a7
Merge branch 'hotstuff_integration' into compute_finality_digest
linh2931 Mar 12, 2024
fab8b40
refactor to make code more concise
linh2931 Mar 12, 2024
d91dcdd
Merge branch 'hotstuff_integration' into compute_finality_digest
linh2931 Mar 12, 2024
21e23ca
passing parent instead of building_block to get_qc_data
linh2931 Mar 12, 2024
cec7777
Include `block_state::valid` in the snapshot.
greg7mdp Mar 12, 2024
6d5a1fa
Merge pull request #2303 from AntelopeIO/compute_finality_digest_snap…
greg7mdp Mar 12, 2024
2db3cfc
move get_finality_mroot_claim and get_most_ancestor_qc_data from cont…
linh2931 Mar 12, 2024
e194a40
make parameter in get_most_ancestor_qc_data a const&
linh2931 Mar 12, 2024
3f762c4
add is_proper_svnn_block() to block_head and use it for validation
linh2931 Mar 13, 2024
2a2cfc0
make no_finality_tree_associated and empty action_mroot validation as…
linh2931 Mar 13, 2024
229de94
make finality_mroot_claim in block_header_state non-optional
linh2931 Mar 13, 2024
be73881
compute base digest explicitly to deep serializing pointer data
linh2931 Mar 13, 2024
f3daf0c
rename proper_svnn_block_flag to proper_svnn_schedule_version
linh2931 Mar 13, 2024
a4401fd
keep genesis Savanna block's schedule_version unchanged
linh2931 Mar 13, 2024
67c20ef
make get_next_proposer_schedule_version always return the new 2^31
linh2931 Mar 13, 2024
7a57560
do not set header.schedule_version to active_proposer_policy->propose…
linh2931 Mar 13, 2024
3729f9b
move get_qc_data back to controller so that assumption about the bran…
linh2931 Mar 13, 2024
2c63198
validate schedule_version with proper_svnn_schedule_version exists at…
linh2931 Mar 13, 2024
869844c
rename compute_finalizer_digest to compute_finality_digest
linh2931 Mar 13, 2024
6878d5b
Move is_proper_svnn_block from block_header.cpp to block_header.hpp
linh2931 Mar 14, 2024
4a00109
Simplify compute_finality_digest by calculating active_finalizer_poli…
linh2931 Mar 14, 2024
65319ec
Fix indentation
linh2931 Mar 14, 2024
7f3d5e6
Rename canonical_merkle to legacy_merkle, and incremental_canonical_m…
linh2931 Mar 14, 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
8 changes: 8 additions & 0 deletions libraries/chain/block_header.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,14 @@ namespace eosio { namespace chain {
return result;
}

bool block_header::is_proper_svnn_block() const {
// We don't check whether finality extension exists here for performance reason.
// When block header is validated in block_header_state's next(),
// it is already validate if schedule_version == proper_svnn_schedule_version,
// finality extension must exist.
return ( schedule_version == proper_svnn_schedule_version );
heifner marked this conversation as resolved.
Show resolved Hide resolved
}

header_extension_multimap block_header::validate_and_extract_header_extensions()const {
using decompose_t = block_header_extension_types::decompose_t;

Expand Down
121 changes: 102 additions & 19 deletions libraries/chain/block_header_state.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,72 @@

namespace eosio::chain {

// moved this warning out of header so it only uses once
#warning TDDO https://github.com/AntelopeIO/leap/issues/2080
// digest_type compute_finalizer_digest() const { return id; };
// this is a versioning scheme that is separate from protocol features that only
// gets updated if a protocol feature causes a breaking change to light block
// header validation

// data for finality_digest
struct finality_digest_data_v1 {
uint32_t major_version{light_header_protocol_version_major};
uint32_t minor_version{light_header_protocol_version_minor};
uint32_t active_finalizer_policy_generation {0};
digest_type finality_tree_digest;
digest_type base_and_active_finalizer_policy_digest;
};

// data for base_and_active_finalizer_policy_digest
struct base_and_active_finalizer_policy_digest_data_t {
digest_type active_finalizer_policy_digest;
digest_type base_digest;
};

// compute base_digest explicitly because of pointers involved.
digest_type block_header_state::compute_base_digest() const {
digest_type::encoder enc;

fc::raw::pack( enc, header );
fc::raw::pack( enc, core );

for (const auto& fp_pair : finalizer_policies) {
fc::raw::pack( enc, fp_pair.first );
assert(fp_pair.second);
fc::raw::pack( enc, *fp_pair.second );
}
arhag marked this conversation as resolved.
Show resolved Hide resolved

assert(active_proposer_policy);
fc::raw::pack( enc, *active_proposer_policy );

for (const auto& pp_pair : proposer_policies) {
assert(pp_pair.second);
fc::raw::pack( enc, *pp_pair.second );
}
arhag marked this conversation as resolved.
Show resolved Hide resolved

if (activated_protocol_features) {
fc::raw::pack( enc, *activated_protocol_features );
}

return enc.result();
}

digest_type block_header_state::compute_finality_digest() const {
assert(active_finalizer_policy);
auto active_finalizer_policy_digest = fc::sha256::hash(*active_finalizer_policy);
auto base_digest = compute_base_digest();

base_and_active_finalizer_policy_digest_data_t b_afp_digest_data {
.active_finalizer_policy_digest = active_finalizer_policy_digest,
.base_digest = base_digest
};
heifner marked this conversation as resolved.
Show resolved Hide resolved
auto b_afp_digest = fc::sha256::hash(b_afp_digest_data);

finality_digest_data_v1 finality_digest_data {
.active_finalizer_policy_generation = active_finalizer_policy->generation,
.finality_tree_digest = finality_mroot(),
.base_and_active_finalizer_policy_digest = b_afp_digest
};

return fc::sha256::hash(finality_digest_data);
}

const producer_authority& block_header_state::get_scheduled_producer(block_timestamp_type t) const {
return detail::get_scheduled_producer(active_proposer_policy->proposer_schedule.producers, t);
Expand All @@ -33,8 +95,8 @@ block_header_state block_header_state::next(block_header_state_input& input) con
.confirmed = 0,
.previous = input.parent_id,
.transaction_mroot = input.transaction_mroot,
.action_mroot = input.action_mroot,
.schedule_version = header.schedule_version
.action_mroot = input.finality_mroot_claim,
.schedule_version = block_header::proper_svnn_schedule_version
};

// activated protocol features
Expand All @@ -46,10 +108,6 @@ block_header_state block_header_state::next(block_header_state_input& input) con
result.activated_protocol_features = activated_protocol_features;
}

// proposal_mtree and finality_mtree
// ---------------------------------
// [greg todo] ??

// proposer policy
// ---------------
result.active_proposer_policy = active_proposer_policy;
Expand All @@ -59,8 +117,6 @@ block_header_state block_header_state::next(block_header_state_input& input) con
// +1 since this is called after the block is built, this will be the active schedule for the next block
if (it->first.slot <= input.timestamp.slot + 1) {
result.active_proposer_policy = it->second;
result.header.schedule_version = header.schedule_version + 1;
result.active_proposer_policy->proposer_schedule.version = result.header.schedule_version;
Copy link
Member

Choose a reason for hiding this comment

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

I believe this change means it is now possible to skip propser_schedule.version numbers. We should modify the set_proposed_producers logic so we don't skip version numbers. Need a test to cover that case so we don't break it again.

Copy link
Member Author

Choose a reason for hiding this comment

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

Does

return (--parent.proposer_policies.end())->second->proposer_schedule.version + 1;
}
assert(active_proposer_policy);
return active_proposer_policy->proposer_schedule.version + 1;
ensure the version number not being skipped?

Copy link
Member

Choose a reason for hiding this comment

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

No, that doesn't help. We need to make sure that if a proposer policy is scheduled for the same proposer_policy.active_time that it replaces the current one there but doesn't increment the schedule version. I think we will need to change controller::set_proposed_producers to check if there is an existing one that would have the same active_time and replace it and use its version. We should return the correct version to the contract.

Copy link
Member

Choose a reason for hiding this comment

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

Feel free to create a follow-on issue for this.

Copy link
Member Author

Choose a reason for hiding this comment

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

#2313

Thanks.

result.proposer_policies = { ++it, proposer_policies.end() };
} else {
result.proposer_policies = proposer_policies;
Expand All @@ -80,19 +136,19 @@ block_header_state block_header_state::next(block_header_state_input& input) con
// if (input.new_finalizer_policy)
// ++input.new_finalizer_policy->generation;


instant_finality_extension new_if_ext {input.most_recent_ancestor_with_qc,
std::move(input.new_finalizer_policy),
std::move(input.new_proposer_policy)};

// finality_core
// -----------------------
// -------------
block_ref parent_block {
.block_id = input.parent_id,
.timestamp = input.parent_timestamp
};
result.core = core.next(parent_block, input.most_recent_ancestor_with_qc);

// finality extension
// ------------------
instant_finality_extension new_if_ext {input.most_recent_ancestor_with_qc,
std::move(input.new_finalizer_policy),
std::move(input.new_proposer_policy)};
uint16_t if_ext_id = instant_finality_extension::extension_id();
emplace_extension(result.header.header_extensions, if_ext_id, fc::raw::pack(new_if_ext));
result.header_exts.emplace(if_ext_id, std::move(new_if_ext));
Expand Down Expand Up @@ -126,6 +182,9 @@ block_header_state block_header_state::next(const signed_block_header& h, valida
EOS_ASSERT( h.previous == block_id, unlinkable_block_exception, "previous mismatch ${p} != ${id}", ("p", h.previous)("id", block_id) );
EOS_ASSERT( h.producer == producer, wrong_producer, "wrong producer specified" );
EOS_ASSERT( h.confirmed == 0, block_validate_exception, "invalid confirmed ${c}", ("c", h.confirmed) );
EOS_ASSERT( h.schedule_version == block_header::proper_svnn_schedule_version, block_validate_exception,
"invalid schedule_version ${s}, expected: ${e}",
("s", h.schedule_version)("e", block_header::proper_svnn_schedule_version) );
EOS_ASSERT( !h.new_producers, producer_schedule_exception, "Block header contains legacy producer schedule outdated by activation of WTMsig Block Signatures" );

auto exts = h.validate_and_extract_header_extensions();
Expand Down Expand Up @@ -155,11 +214,35 @@ block_header_state block_header_state::next(const signed_block_header& h, valida
.new_protocol_feature_activations = std::move(new_protocol_feature_activations)
};

digest_type action_mroot = {};

if (h.is_proper_svnn_block()) {
// if there is no Finality Tree Root associated with the block,
// then this needs to validate that h.action_mroot is the empty digest
auto next_core_metadata = core.next_metadata(if_ext.qc_claim);
bool no_finality_tree_associated = core.is_genesis_block_num(next_core_metadata.final_on_strong_qc_block_num);

EOS_ASSERT( no_finality_tree_associated == h.action_mroot.empty(),
block_validate_exception,
"No Finality Tree Root associated with the block test does not match with empty action_mroot test: no finality tree associated (${n}), action_mroot empty (${e}), final_on_strong_qc_block_num (${f})",
("n", no_finality_tree_associated)("e", h.action_mroot.empty())("f", next_core_metadata.final_on_strong_qc_block_num));

action_mroot = h.action_mroot;
};

block_header_state_input bhs_input{
bb_input, h.transaction_mroot, h.action_mroot, if_ext.new_proposer_policy, if_ext.new_finalizer_policy,
if_ext.qc_claim };
bb_input,
h.transaction_mroot,
if_ext.new_proposer_policy,
if_ext.new_finalizer_policy,
if_ext.qc_claim,
action_mroot // for finality_mroot_claim
};

return next(bhs_input);
}

} // namespace eosio::chain

FC_REFLECT( eosio::chain::finality_digest_data_v1, (major_version)(minor_version)(active_finalizer_policy_generation)(finality_tree_digest)(base_and_active_finalizer_policy_digest) )
FC_REFLECT( eosio::chain::base_and_active_finalizer_policy_digest_data_t, (active_finalizer_policy_digest)(base_digest) )
96 changes: 85 additions & 11 deletions libraries/chain/block_state.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ block_state::block_state(const block_header_state& prev, signed_block_ptr b, con
const validator_t& validator, bool skip_validate_signee)
: block_header_state(prev.next(*b, validator))
, block(std::move(b))
, strong_digest(compute_finalizer_digest())
, strong_digest(compute_finality_digest())
, weak_digest(create_weak_digest(strong_digest))
, pending_qc(prev.active_finalizer_policy->finalizers.size(), prev.active_finalizer_policy->threshold, prev.active_finalizer_policy->max_weak_sum_before_weak_final())
{
Expand All @@ -25,14 +25,19 @@ block_state::block_state(const block_header_state& prev, signed_block_ptr b, con
}
}

block_state::block_state(const block_header_state& bhs, deque<transaction_metadata_ptr>&& trx_metas,
deque<transaction_receipt>&& trx_receipts, const std::optional<quorum_certificate>& qc,
const signer_callback_type& signer, const block_signing_authority& valid_block_signing_authority)
block_state::block_state(const block_header_state& bhs,
deque<transaction_metadata_ptr>&& trx_metas,
deque<transaction_receipt>&& trx_receipts,
const std::optional<valid_t>& valid,
const std::optional<quorum_certificate>& qc,
const signer_callback_type& signer,
const block_signing_authority& valid_block_signing_authority)
heifner marked this conversation as resolved.
Show resolved Hide resolved
: block_header_state(bhs)
, block(std::make_shared<signed_block>(signed_block_header{bhs.header}))
, strong_digest(compute_finalizer_digest())
, strong_digest(compute_finality_digest())
, weak_digest(create_weak_digest(strong_digest))
, pending_qc(bhs.active_finalizer_policy->finalizers.size(), bhs.active_finalizer_policy->threshold, bhs.active_finalizer_policy->max_weak_sum_before_weak_final())
, valid(valid)
, pub_keys_recovered(true) // called by produce_block so signature recovery of trxs must have been done
, cached_trxs(std::move(trx_metas))
{
Expand All @@ -46,13 +51,28 @@ block_state::block_state(const block_header_state& bhs, deque<transaction_metada
sign(signer, valid_block_signing_authority);
}

// Used for transition from dpos to instant-finality
block_state::block_state(const block_state_legacy& bsp) {
// Used for transition from dpos to Savanna.
block_state::block_state(const block_state_legacy& bsp, const digest_type& action_mroot_svnn) {
greg7mdp marked this conversation as resolved.
Show resolved Hide resolved
block_header_state::block_id = bsp.id();
header = bsp.header;
core = finality_core::create_core_for_genesis_block(bsp.block_num()); // [if todo] instant transition is not acceptable
activated_protocol_features = bsp.activated_protocol_features;

// built leaf_node and validation_tree
valid_t::finality_leaf_node_t leaf_node {
.block_num = bsp.block_num(),
.finality_digest = digest_type{},
.action_mroot = action_mroot_svnn
};
incremental_merkle_tree validation_tree;
validation_tree.append(fc::sha256::hash(leaf_node));

// construct valid structure
valid = valid_t {
.validation_tree = validation_tree,
.validation_mroots = { validation_tree.get_root() }
};

auto if_ext_id = instant_finality_extension::extension_id();
std::optional<block_header_extension> ext = bsp.block->extract_header_extension(if_ext_id);
assert(ext); // required by current transition mechanism
Expand All @@ -78,18 +98,16 @@ block_state::block_state(snapshot_detail::snapshot_block_state_v7&& sbs)
.header = std::move(sbs.header),
.activated_protocol_features = std::move(sbs.activated_protocol_features),
.core = std::move(sbs.core),
.proposal_mtree = std::move(sbs.proposal_mtree),
.finality_mtree = std::move(sbs.finality_mtree),
.active_finalizer_policy = std::move(sbs.active_finalizer_policy),
.active_proposer_policy = std::move(sbs.active_proposer_policy),
.proposer_policies = std::move(sbs.proposer_policies),
.finalizer_policies = std::move(sbs.finalizer_policies)
}
, strong_digest(compute_finalizer_digest())
, strong_digest(compute_finality_digest())
, weak_digest(create_weak_digest(strong_digest))
, pending_qc(active_finalizer_policy->finalizers.size(), active_finalizer_policy->threshold,
active_finalizer_policy->max_weak_sum_before_weak_final()) // just in case we receive votes
// , valid(std::move(sbs.valid) // [snapshot todo]
, valid(std::move(sbs.valid))
{
header_exts = header.validate_and_extract_header_extensions();
}
Expand Down Expand Up @@ -223,6 +241,62 @@ std::optional<quorum_certificate> block_state::get_best_qc() const {
return quorum_certificate{ block_num(), best_qc };
}

valid_t block_state::new_valid(const block_header_state& next_bhs, const digest_type& action_mroot) const {
assert(valid);
assert(next_bhs.core.last_final_block_num() >= core.last_final_block_num());

// Copy parent's validation_tree and validation_mroots.
auto start = next_bhs.core.last_final_block_num() - core.last_final_block_num();
valid_t next_valid {
.validation_tree = valid->validation_tree,
// Trim roots from the front end, up to block number `next_bhs.core.last_final_block_num()`
.validation_mroots = { valid->validation_mroots.cbegin() + start, valid->validation_mroots.cend() }
};

// construct block's finality leaf node.
valid_t::finality_leaf_node_t leaf_node{
.block_num = next_bhs.block_num(),
.finality_digest = next_bhs.compute_finality_digest(),
.action_mroot = action_mroot
};
auto leaf_node_digest = fc::sha256::hash(leaf_node);

// append new finality leaf node digest to validation_tree
next_valid.validation_tree.append(leaf_node_digest);

// append the root of the new Validation Tree to validation_mroots.
next_valid.validation_mroots.emplace_back(next_valid.validation_tree.get_root());

// post condition of validation_mroots
assert(next_valid.validation_mroots.size() == (next_bhs.block_num() - next_bhs.core.last_final_block_num() + 1));

return next_valid;
}

digest_type block_state::get_validation_mroot(block_num_type target_block_num) const {
if (!valid) {
return digest_type{};
}

assert(valid->validation_mroots.size() > 0);
assert(core.last_final_block_num() <= target_block_num &&
target_block_num < core.last_final_block_num() + valid->validation_mroots.size());
assert(target_block_num - core.last_final_block_num() < valid->validation_mroots.size());

return valid->validation_mroots[target_block_num - core.last_final_block_num()];
heifner marked this conversation as resolved.
Show resolved Hide resolved
}

digest_type block_state::get_finality_mroot_claim(const qc_claim_t& qc_claim) const {
auto next_core_metadata = core.next_metadata(qc_claim);

// For proper IF blocks that do not have an associated Finality Tree defined
if (core.is_genesis_block_num(next_core_metadata.final_on_strong_qc_block_num)) {
return digest_type{};
}

return get_validation_mroot(next_core_metadata.final_on_strong_qc_block_num);
}

void inject_additional_signatures( signed_block& b, const std::vector<signature_type>& additional_signatures)
{
if (!additional_signatures.empty()) {
Expand Down
Loading
Loading