34 size_t max_elements) {
35 auto &pview =
peers.get<by_proofid>();
36 auto it = pview.find(proofid);
37 if (it == pview.end()) {
50 size_t max_elements) {
53 const PeerId peerid = it->peerid;
55 auto nit =
nodes.find(nodeid);
56 if (nit ==
nodes.end()) {
57 if (!
nodes.emplace(nodeid, peerid, max_elements).second) {
61 const PeerId oldpeerid = nit->peerid;
62 if (!
nodes.modify(nit, [&](
Node &n) { n.peerid = peerid; })) {
79 const ProofId &proofid = it->getProofId();
101 const uint32_t score = p.
getScore();
103 slots.emplace_back(start, score, it->peerid);
114 auto [begin, end] = remoteProofsView.equal_range(nodeid);
115 remoteProofsView.erase(begin, end);
122 auto it =
nodes.find(nodeid);
123 if (it ==
nodes.end()) {
127 const PeerId peerid = it->peerid;
142 if (it ==
peers.end()) {
146 assert(count <= it->node_count);
152 const uint32_t new_count = it->node_count -
count;
153 if (!
peers.modify(it, [&](
Peer &p) { p.node_count = new_count; })) {
170 const size_t i = it->index;
175 if (i + 1 ==
slots.size()) {
189 auto it =
nodes.find(nodeid);
190 if (it ==
nodes.end()) {
202 auto it =
nodes.find(nodeid);
203 if (it ==
nodes.end()) {
207 if (it->last_round >
response.getRound()) {
212 auto timeout = Now<SteadyMilliseconds>() +
213 std::chrono::milliseconds(
response.getCooldown());
222 auto it =
nodes.find(nodeid);
223 if (it ==
nodes.end()) {
227 return !it->avaproofsSent &&
236 PeerId peerid,
const std::chrono::seconds &nextTime) {
237 auto it =
peers.find(peerid);
238 if (it ==
peers.end()) {
249 return it->nextPossibleConflictTime == nextTime;
253 auto it =
peers.find(peerid);
254 if (it ==
peers.end()) {
264template <
typename ProofContainer>
266 auto &peersView =
peers.get<by_proofid>();
267 for (
const ProofRef &proof : proofs) {
268 auto it = peersView.find(proof->getId());
269 if (it != peersView.end()) {
282 const ProofId &proofid = proof->getId();
285 const std::string &message) {
286 return registrationState.
Invalid(
297 "proof-already-registered");
329 "utxo-missing-or-spent");
336 auto now = GetTime<std::chrono::seconds>();
337 auto nextCooldownTimePoint =
339 "-avalancheconflictingproofcooldown",
344 case ProofPool::AddProofStatus::REJECTED: {
346 auto bestPossibleConflictTime = std::chrono::seconds(0);
347 auto &pview =
peers.get<by_proofid>();
348 for (
auto &conflictingProof : conflictingProofs) {
349 auto it = pview.find(conflictingProof->getId());
350 assert(it != pview.end());
353 bestPossibleConflictTime = std::max(
354 bestPossibleConflictTime, it->nextPossibleConflictTime);
357 nextCooldownTimePoint);
360 if (bestPossibleConflictTime > now) {
364 "cooldown-not-elapsed");
380 ProofPool::AddProofStatus::REJECTED
384 "conflicting-utxos");
394 assert(status == ProofPool::AddProofStatus::SUCCEED);
398 case ProofPool::AddProofStatus::DUPLICATED:
401 "proof-already-registered");
402 case ProofPool::AddProofStatus::SUCCEED:
416 auto inserted =
peers.emplace(peerid, proof, nextCooldownTimePoint);
430 auto &pendingNodesView =
pendingNodes.get<by_proofid>();
431 auto range = pendingNodesView.equal_range(proofid);
436 std::vector<std::pair<NodeId, size_t>> nodeids_and_max_elements;
437 nodeids_and_max_elements.reserve(std::distance(range.first, range.second));
438 std::transform(range.first, range.second,
439 std::back_inserter(nodeids_and_max_elements),
441 return std::make_pair(n.nodeid, n.max_elements);
444 for (
const auto &[nodeid, max_elements] : nodeids_and_max_elements) {
481 auto &pview =
peers.get<by_proofid>();
482 auto it = pview.find(proofid);
483 assert(it != pview.end());
495 if (!conflictingProof) {
511 std::unordered_set<ProofRef, SaltedProofHasher> ®isteredProofs) {
512 registeredProofs.clear();
513 const auto now = GetTime<std::chrono::seconds>();
515 std::vector<ProofRef> newlyDanglingProofs;
520 peer.node_count == 0 &&
527 newlyDanglingProofs.push_back(peer.proof);
534 std::vector<ProofRef> previouslyDanglingProofs;
537 previouslyDanglingProofs.push_back(proof);
540 for (
const ProofRef &proof : previouslyDanglingProofs) {
543 registeredProofs.insert(proof);
547 for (
const ProofRef &proof : newlyDanglingProofs) {
554 "Proof dangling for too long (no connected node): %s\n",
555 proof->getId().GetHex());
578 if (it != nview.end() && it->peerid == p &&
579 it->nextRequestTime <= Now<SteadyMilliseconds>()) {
591 std::vector<ProofId> invalidProofIds;
592 std::vector<ProofRef> newImmatures;
597 for (
const auto &p :
peers) {
601 newImmatures.push_back(p.proof);
603 invalidProofIds.push_back(p.getProofId());
606 "Invalidating proof %s: verification failed (%s)\n",
607 p.proof->getId().GetHex(), state.
ToString());
618 invalidProofIds.push_back(proof->getId());
622 "Invalidating dangling proof %s: verification failed "
624 proof->getId().GetHex(), state.
ToString());
632 for (
const ProofId &invalidProofId : invalidProofIds) {
638 for (
auto &p : newImmatures) {
642 return registeredProofs;
665 auto &pview =
peers.get<by_proofid>();
666 return pview.find(proofid) != pview.end();
694 const bool present) {
709 auto &remoteProofsByLastUpdate =
remoteProofs.get<by_lastUpdate>();
710 auto [begin, end] = remoteProofsByLastUpdate.equal_range(nodeid);
717 while (
size_t(std::distance(begin, end)) >=
720 begin = remoteProofsByLastUpdate.erase(begin);
723 auto it =
remoteProofs.find(boost::make_tuple(proofid, nodeid));
729 .emplace(
RemoteProof{proofid, nodeid, GetTime<std::chrono::seconds>(),
734std::vector<RemoteProof>
736 std::vector<RemoteProof> nodeRemoteProofs;
738 auto &remoteProofsByLastUpdate =
remoteProofs.get<by_lastUpdate>();
739 auto [begin, end] = remoteProofsByLastUpdate.equal_range(nodeid);
741 for (
auto &it = begin; it != end; it++) {
742 nodeRemoteProofs.emplace_back(*it);
745 return nodeRemoteProofs;
750 return view.count(proofid) > 0;
755 auto [begin, end] = view.equal_range(proofid);
756 return std::any_of(begin, end, [](
const auto &remoteProof) {
757 return remoteProof.present;
762 auto it =
peers.find(peerid);
763 if (it ==
peers.end()) {
773 auto range = nview.equal_range(peerid);
774 for (
auto &nit = range.first; nit != range.second; ++nit) {
775 pendingNodes.emplace(it->getProofId(), nit->nodeid, nit->maxElements);
784 boost::make_tuple(peerid, Now<SteadyMilliseconds>())));
825 std::vector<Slot> newslots;
826 newslots.reserve(
peers.size());
828 uint64_t prevStop = 0;
830 for (
auto it =
peers.begin(); it !=
peers.end(); it++) {
831 if (it->node_count == 0) {
835 newslots.emplace_back(prevStop, it->getScore(), it->peerid);
836 prevStop =
slots[i].getStop();
837 if (!
peers.modify(it, [&](
Peer &p) { p.index = i++; })) {
842 slots = std::move(newslots);
844 const uint64_t saved =
slotCount - prevStop;
852 uint64_t prevStop = 0;
853 uint32_t scoreFromSlots = 0;
854 for (
size_t i = 0; i <
slots.size(); i++) {
871 if (it ==
peers.end() || it->index != i) {
876 scoreFromSlots +=
slots[i].getScore();
884 uint32_t scoreFromAllPeers = 0;
885 uint32_t scoreFromPeersWithNodes = 0;
887 std::unordered_set<COutPoint, SaltedOutpointHasher> peersUtxos;
888 for (
const auto &p :
peers) {
890 scoreFromAllPeers += p.getScore();
898 for (
const auto &ss : p.proof->getStakes()) {
899 const COutPoint &outpoint = ss.getStake().getUTXO();
906 if (proof != p.proof) {
911 if (!peersUtxos.emplace(outpoint).second) {
918 const auto count_nodes = [&]() {
921 auto begin = nview.lower_bound(
923 auto end = nview.upper_bound(
926 for (
auto it = begin; it != end; ++it) {
933 if (p.node_count != count_nodes()) {
938 if (p.node_count == 0) {
942 scoreFromPeersWithNodes += p.getScore();
944 if (p.index >=
slots.size() ||
slots[p.index].getPeerId() != p.peerid) {
949 if (
slots[p.index].getScore() != p.getScore()) {
959 if (p.node_count == 0 &&
986 const uint64_t max) {
989 size_t begin = 0, end = slots.size();
990 uint64_t bottom = 0, top = max;
993 while ((end - begin) > 8) {
995 if (slot < bottom || slot >= top) {
1000 size_t i = begin + ((slot - bottom) * (end - begin) / (top - bottom));
1001 assert(begin <= i && i < end);
1004 if (slots[i].contains(slot)) {
1005 return slots[i].getPeerId();
1009 if (slots[i].precedes(slot)) {
1015 bottom = slots[begin].getStart();
1020 if (slots[i].follows(slot)) {
1022 top = slots[end].getStart();
1031 for (
size_t i = begin; i < end; i++) {
1033 if (slots[i].contains(slot)) {
1034 return slots[i].getPeerId();
1055 std::vector<std::pair<ProofId, CScript>> &winners) {
1066 auto registrationDelay = std::chrono::duration_cast<std::chrono::seconds>(
1068 auto maxRegistrationDelay =
1069 std::chrono::duration_cast<std::chrono::seconds>(
1071 auto minRegistrationDelay =
1072 std::chrono::duration_cast<std::chrono::seconds>(
1077 const int64_t targetRegistrationTime = refTime - registrationDelay.count();
1078 const int64_t maxRegistrationTime = refTime - minRegistrationDelay.count();
1079 const int64_t minRegistrationTime = refTime - maxRegistrationDelay.count();
1083 std::vector<ProofRef> selectedProofs;
1085 while (selectedProofs.size() <
peers.size()) {
1086 double bestRewardRank = std::numeric_limits<double>::max();
1088 int64_t selectedProofRegistrationTime{0};
1097 if (!peer.hasFinalized ||
1098 peer.registration_time.count() >= maxRegistrationTime) {
1102 if (std::find_if(selectedProofs.begin(), selectedProofs.end(),
1104 return peer.getProofId() == proof->getId();
1105 }) != selectedProofs.end()) {
1115 "Staking reward hash has a suspicious value of zero for "
1116 "proof %s and blockhash %s, skipping\n",
1117 peer.getProofId().ToString(), prevblockhash.
ToString());
1121 double proofRewardRank =
1127 proofRewardHash, proofRewardRank, peer.getProofId(),
1128 bestRewardHash, bestRewardRank,
1129 selectedProof ? selectedProof->getId()
1131 bestRewardRank = proofRewardRank;
1132 selectedProof = peer.proof;
1133 selectedProofRegistrationTime = peer.registration_time.count();
1134 bestRewardHash = proofRewardHash;
1138 if (!selectedProof) {
1143 if (!firstCompliantProof &&
1144 selectedProofRegistrationTime < targetRegistrationTime) {
1145 firstCompliantProof = selectedProof;
1148 selectedProofs.push_back(selectedProof);
1150 if (selectedProofRegistrationTime < minRegistrationTime &&
1151 !
isFlaky(selectedProof->getId())) {
1158 if (!firstCompliantProof) {
1162 winners.reserve(selectedProofs.size());
1165 for (
const ProofRef &proof : selectedProofs) {
1166 if (proof->getId() == firstCompliantProof->getId()) {
1167 winners.push_back({proof->getId(), proof->getPayoutScript()});
1171 for (
const ProofRef &proof : selectedProofs) {
1172 if (proof->getId() != firstCompliantProof->getId()) {
1173 winners.push_back({proof->getId(), proof->getPayoutScript()});
1203 auto &remoteProofsByNodeId =
remoteProofs.get<by_nodeid>();
1206 std::unordered_map<PeerId, std::unordered_set<ProofId, SaltedProofIdHasher>>
1210 double total_score{0};
1216 auto nodes_range = nview.equal_range(peerid);
1217 for (
auto &nit = nodes_range.first; nit != nodes_range.second; ++nit) {
1218 auto proofs_range = remoteProofsByNodeId.equal_range(nit->nodeid);
1219 for (
auto &proofit = proofs_range.first;
1220 proofit != proofs_range.second; ++proofit) {
1221 if (!proofit->present) {
1222 missing_per_peer[peerid].insert(proofit->proofid);
1228 double missing_score{0};
1231 for (
const auto &[peerid, missingProofs] : missing_per_peer) {
1232 if (missingProofs.size() > 3) {
1237 auto pit =
peers.find(peerid);
1238 if (pit ==
peers.end()) {
1243 if (missingProofs.count(proofid) > 0) {
1244 missing_score += pit->getScore();
1248 return (missing_score / total_score) > 0.3;
1253 auto &remoteProofsView =
remoteProofs.get<by_proofid>();
1254 auto [begin, end] = remoteProofsView.equal_range(proofid);
1258 return std::nullopt;
1261 double total_score{0};
1262 double present_score{0};
1263 double missing_score{0};
1265 for (
auto it = begin; it != end; it++) {
1266 auto nit =
nodes.find(it->nodeid);
1267 if (nit ==
nodes.end()) {
1272 const PeerId peerid = nit->peerid;
1274 auto pit =
peers.find(peerid);
1275 if (pit ==
peers.end()) {
1280 uint32_t node_count = pit->node_count;
1286 if (node_count == 0) {
1291 const double score = double(pit->getScore()) / node_count;
1293 total_score += score;
1295 present_score += score;
1297 missing_score += score;
1302 auto &peersByProofid =
peers.get<by_proofid>();
1305 bool present =
false;
1306 auto pit = peersByProofid.find(proofid);
1307 if (pit != peersByProofid.end()) {
1308 present = pit->node_count > 0;
1311 pit = peersByProofid.find(
localProof->getId());
1312 if (pit != peersByProofid.end()) {
1315 const double score =
1316 double(pit->getScore()) / (1 + pit->node_count);
1318 total_score += score;
1320 present_score += score;
1322 missing_score += score;
1327 if (present_score / total_score > 0.55) {
1328 return std::make_optional(
true);
1331 if (missing_score / total_score > 0.55) {
1332 return std::make_optional(
false);
1335 return std::nullopt;
1340 const fs::path dumpPathTmp = dumpPath +
".new";
1348 file << uint64_t(
peers.size());
1351 file << peer.hasFinalized;
1352 file << int64_t(peer.registration_time.count());
1353 file << int64_t(peer.nextPossibleConflictTime.count());
1357 throw std::runtime_error(
strprintf(
"Failed to commit to file %s",
1363 throw std::runtime_error(
strprintf(
"Rename failed from %s to %s",
1367 }
catch (
const std::exception &e) {
1381 std::unordered_set<ProofRef, SaltedProofHasher> ®isteredProofs) {
1382 registeredProofs.clear();
1386 if (file.IsNull()) {
1388 "Failed to open avalanche peers file from disk.\n");
1398 "Unsupported avalanche peers file version.\n");
1405 auto &peersByProofId =
peers.get<by_proofid>();
1407 for (uint64_t i = 0; i < numPeers; i++) {
1410 int64_t registrationTime;
1411 int64_t nextPossibleConflictTime;
1414 file >> hasFinalized;
1415 file >> registrationTime;
1416 file >> nextPossibleConflictTime;
1419 auto it = peersByProofId.find(proof->getId());
1420 if (it == peersByProofId.end()) {
1429 peersByProofId.modify(it, [&](
Peer &p) {
1432 std::chrono::seconds{registrationTime};
1434 std::chrono::seconds{nextPossibleConflictTime};
1437 registeredProofs.insert(proof);
1440 }
catch (
const std::exception &e) {
1442 "Failed to read the avalanche peers file data on disk: %s.\n",
1459 const ProofId &proofid = proof->getId();
1461 "Cached stake contender with proofid %s, payout %s at block "
1462 "%s (height %d) with id %s\n",
1479 std::vector<std::pair<ProofId, CScript>> &newWinners) {
1508 const std::vector<std::pair<ProofId, CScript>> winners,
size_t maxPollable,
1509 std::vector<StakeContenderId> &pollableContenders) {
1512 for (
const auto &winner : winners) {
1516 "Stake contender set as local winner: proofid %s, payout "
1517 "%s at block %s (height %d) with id %s\n",
1518 winner.first.ToString(),
HexStr(winner.second),
1527 pollableContenders) > 0) {
1532 "Stake contender set as best contender: id %s at block "
1543 const CBlockIndex *pindex,
const std::vector<CScript> &payoutScripts) {
static constexpr PeerId NO_PEER
static constexpr size_t AVALANCHE_DEFAULT_CONFLICTING_PROOF_COOLDOWN
Conflicting proofs cooldown time default value in seconds.
int64_t GetIntArg(const std::string &strArg, int64_t nDefault) const
Return integer argument or default value.
Non-refcounted RAII wrapper for FILE*.
The block chain is a tree shaped structure starting with the genesis block at the root,...
int64_t GetBlockTime() const
BlockHash GetBlockHash() const
int nHeight
height of the entry in the chain. The genesis block has height 0
void insert(Span< const uint8_t > vKey)
bool contains(Span< const uint8_t > vKey) const
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
bool Invalid(Result result, const std::string &reject_reason="", const std::string &debug_message="")
std::string ToString() const
bool selectStakingRewardWinner(const CBlockIndex *pprev, std::vector< std::pair< ProofId, CScript > > &winners)
Deterministically select a list of payout scripts based on the proof set and the previous block hash.
uint32_t connectedPeersScore
std::vector< RemoteProof > getRemoteProofs(const NodeId nodeid) const
bool removeNode(NodeId nodeid)
bool setFinalized(PeerId peerid)
Latch on that this peer has a finalized proof.
bool dumpPeersToFile(const fs::path &dumpPath) const
RemoteProofSet remoteProofs
Remember which node sent which proof so we have an image of the proof set of our peers.
bool updateNextRequestTimeForResponse(NodeId nodeid, const Response &response)
bool isDangling(const ProofId &proofid) const
bool addNode(NodeId nodeid, const ProofId &proofid, size_t max_elements)
Node API.
bool unsetFlaky(const ProofId &proofid)
std::optional< bool > getRemotePresenceStatus(const ProofId &proofid) const
Get the presence remote status of a proof.
bool addNodeToPeer(const PeerSet::iterator &it)
bool exists(const ProofId &proofid) const
Return true if the (valid) proof exists, but only for non-dangling proofs.
PendingNodeSet pendingNodes
bool verify() const
Perform consistency check on internal data structures.
bool hasRemoteProofStatus(const ProofId &proofid) const
bool forPeer(const ProofId &proofid, Callable &&func) const
void finalizeStakeContender(const StakeContenderId &contenderId, BlockHash &prevblockhash, std::vector< std::pair< ProofId, CScript > > &newWinners)
bool latchAvaproofsSent(NodeId nodeid)
Flag that a node did send its compact proofs.
void cleanupStakeContenders(const int requestedMinHeight)
Make some of the contender cache API available.
bool updateNextRequestTimeForPoll(NodeId nodeid, SteadyMilliseconds timeout, uint64_t round)
static constexpr int SELECT_PEER_MAX_RETRY
ProofIdSet m_unbroadcast_proofids
Track proof ids to broadcast.
bool loadPeersFromFile(const fs::path &dumpPath, std::unordered_set< ProofRef, SaltedProofHasher > ®isteredProofs)
RejectionMode
Rejection mode.
void addUnbroadcastProof(const ProofId &proofid)
Proof broadcast API.
std::unordered_set< ProofRef, SaltedProofHasher > updatedBlockTip()
Update the peer set when a new block is connected.
void removeUnbroadcastProof(const ProofId &proofid)
void promoteStakeContendersToBlock(const CBlockIndex *pindex)
bool isBoundToPeer(const ProofId &proofid) const
bool setContenderStatusForLocalWinners(const CBlockIndex *prevblock, const std::vector< std::pair< ProofId, CScript > > winners, size_t maxPollable, std::vector< StakeContenderId > &pollableContenders)
ProofRadixTree shareableProofs
bool saveRemoteProof(const ProofId &proofid, const NodeId nodeid, const bool present)
CRollingBloomFilter invalidProofs
Filter for proofs that are consensus-invalid or were recently invalidated by avalanche (finalized rej...
bool addOrUpdateNode(const PeerSet::iterator &it, NodeId nodeid, size_t max_elements)
uint64_t compact()
Trigger maintenance of internal data structures.
std::vector< Slot > slots
uint32_t totalPeersScore
Quorum management.
ProofPool danglingProofPool
StakeContenderCache stakeContenderCache
void setInvalid(const ProofId &proofid)
int getStakeContenderStatus(const StakeContenderId &contenderId, BlockHash &prevblockhashout) const
bool isFlaky(const ProofId &proofid) const
ChainstateManager & chainman
bool isInvalid(const ProofId &proofid) const
std::unordered_set< ProofId, SaltedProofIdHasher > manualFlakyProofids
bool removePeer(const PeerId peerid)
Remove an existing peer.
bool isImmature(const ProofId &proofid) const
bool rejectProof(const ProofId &proofid, RejectionMode mode=RejectionMode::DEFAULT)
ProofPool immatureProofPool
Amount stakeUtxoDustThreshold
RegistrationMode
Registration mode.
ProofPool conflictingProofPool
bool isStakingPreconsensusActivated() const
static constexpr size_t MAX_REMOTE_PROOFS
bool setFlaky(const ProofId &proofid)
void addStakeContender(const ProofRef &proof)
std::atomic< bool > needMoreNodes
Flag indicating that we failed to select a node and need to expand our node set.
PeerId selectPeer() const
Randomly select a peer to poll.
bool isInConflictingPool(const ProofId &proofid) const
bool isRemotelyPresentProof(const ProofId &proofid) const
static constexpr int SELECT_NODE_MAX_RETRY
void cleanupDanglingProofs(std::unordered_set< ProofRef, SaltedProofHasher > ®isteredProofs)
void acceptStakeContender(const StakeContenderId &contenderId)
ProofRef getProof(const ProofId &proofid) const
bool registerProof(const ProofRef &proof, ProofRegistrationState ®istrationState, RegistrationMode mode=RegistrationMode::DEFAULT)
void rejectStakeContender(const StakeContenderId &contenderId)
bool removeNodeFromPeer(const PeerSet::iterator &it, uint32_t count=1)
bool updateNextPossibleConflictTime(PeerId peerid, const std::chrono::seconds &nextTime)
Proof and Peer related API.
void moveToConflictingPool(const ProofContainer &proofs)
bool setStakeContenderWinners(const CBlockIndex *pindex, const std::vector< CScript > &payoutScripts)
AddProofStatus addProofIfPreferred(const ProofRef &proof, ConflictingProofSet &conflictingProofs)
Attempt to add a proof to the pool.
AddProofStatus addProofIfNoConflict(const ProofRef &proof, ConflictingProofSet &conflictingProofs)
Attempt to add a proof to the pool, and fail if there is a conflict on any UTXO.
size_t countProofs() const
bool removeProof(ProofId proofid)
void forEachProof(Callable &&func) const
ProofRef getProof(const ProofId &proofid) const
std::set< ProofRef, ConflictingProofComparator > ConflictingProofSet
ProofRef getLowestScoreProof() const
std::unordered_set< ProofRef, SaltedProofHasher > rescan(PeerManager &peerManager)
bool getWinners(const BlockHash &prevblockhash, std::vector< std::pair< ProofId, CScript > > &winners) const
bool accept(const StakeContenderId &contenderId)
Helpers to set avalanche state of a contender.
void cleanup(const int requestedMinHeight)
size_t getPollableContenders(const BlockHash &prevblockhash, size_t maxPollable, std::vector< StakeContenderId > &pollableContenders) const
Get the best ranking contenders, accepted contenders ranking first.
bool reject(const StakeContenderId &contenderId)
bool setWinners(const CBlockIndex *pindex, const std::vector< CScript > &payoutScripts)
Set proof(s) that should be treated as winners (already finalized).
bool add(const CBlockIndex *pindex, const ProofRef &proof, uint8_t status=StakeContenderStatus::UNKNOWN)
Add a proof to consider in staking rewards pre-consensus.
void promoteToBlock(const CBlockIndex *activeTip, std::function< bool(const ProofId &proofid)> const &shouldPromote)
Promote cache entries to a the active chain tip.
int getVoteStatus(const StakeContenderId &contenderId, BlockHash &prevblockhashout) const
Get contender acceptance state for avalanche voting.
bool finalize(const StakeContenderId &contenderId)
std::string ToString() const
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
static const uint256 ZERO
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
bool RenameOver(fs::path src, fs::path dest)
Rename src to dest.
bool FileCommit(FILE *file)
Ensure file contents are fully committed to disk, using a platform-specific feature analogous to fsyn...
std::string HexStr(const Span< const uint8_t > s)
Convert a span of bytes to a lower-case hexadecimal string.
#define LogPrint(category,...)
#define LogTrace(category,...)
static constexpr uint32_t AVALANCHE_MAX_IMMATURE_PROOFS
Maximum number of immature proofs the peer manager will accept from the network.
static bool isImmatureState(const ProofValidationState &state)
static constexpr uint64_t PEERS_DUMP_VERSION
PeerId selectPeerImpl(const std::vector< Slot > &slots, const uint64_t slot, const uint64_t max)
Internal methods that are exposed for testing purposes.
RCUPtr< const Proof > ProofRef
static std::string PathToString(const path &path)
Convert path object to byte string.
FILE * fopen(const fs::path &p, const char *mode)
static constexpr NodeId NO_NODE
Special NodeId that represent no node.
static std::string ToString(const CService &ip)
A BlockHash is a unqiue identifier for a block.
RCUPtr< T > remove(const KeyType &key)
Remove an element from the tree.
RCUPtr< T > get(const KeyType &key)
Get the value corresponding to a key.
bool forEachLeaf(Callable &&func) const
bool insert(const RCUPtr< T > &value)
Insert a value into the tree.
Facility for using an uint256 as a radix tree key.
SteadyMilliseconds nextRequestTime
std::chrono::seconds registration_time
std::chrono::seconds nextPossibleConflictTime
static constexpr auto DANGLING_TIMEOUT
Consider dropping the peer if no node is attached after this timeout expired.
uint32_t getScore() const
uint64_t getStart() const
StakeContenderIds are unique for each block to ensure that the peer polling for their acceptance has ...
double ComputeProofRewardRank(uint32_t proofScore) const
To make sure the selection is properly weighted according to the proof score, we normalize the conten...
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define NO_THREAD_SAFETY_ANALYSIS
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
std::chrono::time_point< std::chrono::steady_clock, std::chrono::milliseconds > SteadyMilliseconds