225 lines
8.7 KiB
C++
225 lines
8.7 KiB
C++
// Copyright (c) 2014-2015 The Dash developers
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include "main.h"
|
|
#include "masternode-sync.h"
|
|
#include "masternode-payments.h"
|
|
#include "masternode.h"
|
|
#include "masternodeman.h"
|
|
#include "util.h"
|
|
#include "addrman.h"
|
|
|
|
class CMasternodeSync;
|
|
CMasternodeSync masternodeSync;
|
|
|
|
CMasternodeSync::CMasternodeSync()
|
|
{
|
|
lastMasternodeList = 0;
|
|
lastMasternodeWinner = 0;
|
|
lastBudgetItem = 0;
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_INITIAL;
|
|
RequestedMasternodeAttempt = 0;
|
|
}
|
|
|
|
bool CMasternodeSync::IsSynced()
|
|
{
|
|
return (RequestedMasternodeAssets == MASTERNODE_SYNC_FINISHED);
|
|
}
|
|
|
|
void CMasternodeSync::AddedMasternodeList()
|
|
{
|
|
lastMasternodeList = GetTime();
|
|
}
|
|
|
|
void CMasternodeSync::AddedMasternodeWinner()
|
|
{
|
|
lastMasternodeWinner = GetTime();
|
|
}
|
|
|
|
void CMasternodeSync::AddedBudgetItem()
|
|
{
|
|
lastBudgetItem = GetTime();
|
|
}
|
|
|
|
void CMasternodeSync::GetNextAsset()
|
|
{
|
|
switch(RequestedMasternodeAssets)
|
|
{
|
|
case(MASTERNODE_SYNC_INITIAL):
|
|
lastMasternodeList = 0;
|
|
lastMasternodeWinner = 0;
|
|
lastBudgetItem = 0;
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_SPORKS;
|
|
break;
|
|
case(MASTERNODE_SYNC_SPORKS):
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_LIST;
|
|
break;
|
|
case(MASTERNODE_SYNC_LIST):
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_MNW;
|
|
break;
|
|
case(MASTERNODE_SYNC_MNW):
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_BUDGET;
|
|
break;
|
|
case(MASTERNODE_SYNC_BUDGET):
|
|
LogPrintf("CMasternodeSync::GetNextAsset - Sync has finished\n");
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_FINISHED;
|
|
break;
|
|
}
|
|
RequestedMasternodeAttempt = 0;
|
|
}
|
|
|
|
void CMasternodeSync::Process()
|
|
{
|
|
static int tick = 0;
|
|
|
|
if(tick++ % MASTERNODE_SYNC_TIMEOUT != 0) return;
|
|
|
|
CBlockIndex* pindexPrev = chainActive.Tip();
|
|
if(pindexPrev == NULL) return;
|
|
|
|
if(IsSynced()) {
|
|
/*
|
|
Resync if we lose all masternodes from sleep/wake or failure to sync originally
|
|
*/
|
|
if(mnodeman.CountEnabled() == 0) {
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_INITIAL;
|
|
} else
|
|
return;
|
|
}
|
|
|
|
if(fDebug) LogPrintf("CMasternodeSync::Process() - tick %d RequestedMasternodeAssets %d\n", tick, RequestedMasternodeAssets);
|
|
|
|
if(RequestedMasternodeAssets == MASTERNODE_SYNC_INITIAL) GetNextAsset();
|
|
|
|
LOCK(cs_vNodes);
|
|
BOOST_FOREACH(CNode* pnode, vNodes)
|
|
{
|
|
|
|
//set to synced
|
|
if(Params().NetworkID() == CBaseChainParams::REGTEST && tick >= 10) {
|
|
LogPrintf("CMasternodeSync::Process - Sync has finished\n");
|
|
RequestedMasternodeAssets = MASTERNODE_SYNC_FINISHED;
|
|
RequestedMasternodeAttempt = 0;
|
|
} else if(Params().NetworkID() != CBaseChainParams::REGTEST) {
|
|
if(RequestedMasternodeAssets == MASTERNODE_SYNC_SPORKS){
|
|
if(pnode->HasFulfilledRequest("getspork")) continue;
|
|
pnode->FulfilledRequest("getspork");
|
|
|
|
pnode->PushMessage("getsporks"); //get current network sporks
|
|
if(RequestedMasternodeAttempt >= 2) GetNextAsset();
|
|
RequestedMasternodeAttempt++;
|
|
|
|
return;
|
|
}
|
|
} else if (RequestedMasternodeAssets == MASTERNODE_SYNC_SPORKS) {
|
|
GetNextAsset();
|
|
return;
|
|
}
|
|
|
|
|
|
//don't begin syncing until we're almost at a recent block
|
|
if(pindexPrev->nHeight + 4 < pindexBestHeader->nHeight && pindexPrev->nTime + 600 < GetTime()) return;
|
|
|
|
if (pnode->nVersion >= masternodePayments.GetMinMasternodePaymentsProto()) {
|
|
|
|
if(RequestedMasternodeAssets == MASTERNODE_SYNC_LIST) {
|
|
if(fDebug) LogPrintf("CMasternodeSync::Process() - lastMasternodeList %lld (GetTime() - MASTERNODE_SYNC_TIMEOUT) %lld\n", lastMasternodeList, GetTime() - MASTERNODE_SYNC_TIMEOUT);
|
|
if(lastMasternodeList > 0 && lastMasternodeList < GetTime() - MASTERNODE_SYNC_TIMEOUT){ //hasn't received a new item in the last five seconds, so we'll move to the
|
|
GetNextAsset();
|
|
return;
|
|
}
|
|
|
|
// If it's already more then MASTERNODE_SYNC_TIMEOUT seconds passed since we asked
|
|
// and we still have nothing, assume there is nothing to sync but give it another
|
|
// MASTERNODE_SYNC_TIMEOUT seconds until we move further
|
|
static int64_t lastTimeAsked = 0;
|
|
if(lastMasternodeList == 0 && lastTimeAsked > 0 && lastTimeAsked < GetTime() - MASTERNODE_SYNC_TIMEOUT) {
|
|
lastMasternodeList = GetTime();
|
|
}
|
|
|
|
if(pnode->HasFulfilledRequest("mnsync")) continue;
|
|
pnode->FulfilledRequest("mnsync");
|
|
|
|
if((lastMasternodeList == 0 || lastMasternodeList > GetTime() - MASTERNODE_SYNC_TIMEOUT)
|
|
&& RequestedMasternodeAttempt <= 2){
|
|
mnodeman.DsegUpdate(pnode);
|
|
RequestedMasternodeAttempt++;
|
|
lastTimeAsked = GetTime();
|
|
}
|
|
return;
|
|
}
|
|
|
|
if(RequestedMasternodeAssets == MASTERNODE_SYNC_MNW) {
|
|
if(lastMasternodeWinner > 0 && lastMasternodeWinner < GetTime() - MASTERNODE_SYNC_TIMEOUT){ //hasn't received a new item in the last five seconds, so we'll move to the
|
|
GetNextAsset();
|
|
return;
|
|
}
|
|
|
|
// If it's already more then MASTERNODE_SYNC_TIMEOUT seconds passed since we asked
|
|
// and we still have nothing, assume there is nothing to sync but give it another
|
|
// MASTERNODE_SYNC_TIMEOUT seconds until we move further
|
|
static int64_t lastTimeAsked = 0;
|
|
if(lastMasternodeWinner == 0 && lastTimeAsked > 0 && lastTimeAsked < GetTime() - MASTERNODE_SYNC_TIMEOUT) {
|
|
lastMasternodeWinner = GetTime();
|
|
}
|
|
|
|
if(pnode->HasFulfilledRequest("mnwsync")) continue;
|
|
pnode->FulfilledRequest("mnwsync");
|
|
|
|
if((lastMasternodeWinner == 0 || lastMasternodeWinner > GetTime() - MASTERNODE_SYNC_TIMEOUT)
|
|
&& RequestedMasternodeAttempt <= 2){
|
|
|
|
CBlockIndex* pindexPrev = chainActive.Tip();
|
|
if(pindexPrev == NULL) return;
|
|
|
|
int nMnCount = mnodeman.CountEnabled()*2;
|
|
int nCountNeeded = (pindexPrev->nHeight - masternodePayments.GetNewestBlock());
|
|
int nHaveBlocks = (pindexPrev->nHeight - masternodePayments.GetOldestBlock());
|
|
if(nHaveBlocks < nMnCount || nCountNeeded > nMnCount) {
|
|
//We have less blocks than there are masternodes, we need more history
|
|
// - or our cache is old
|
|
nCountNeeded = nMnCount;
|
|
}
|
|
|
|
pnode->PushMessage("mnget", nCountNeeded); //sync payees
|
|
RequestedMasternodeAttempt++;
|
|
lastTimeAsked = GetTime();
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (pnode->nVersion >= MIN_BUDGET_PEER_PROTO_VERSION) {
|
|
|
|
if(RequestedMasternodeAssets == MASTERNODE_SYNC_BUDGET){
|
|
if(lastBudgetItem > 0 && lastBudgetItem < GetTime() - MASTERNODE_SYNC_TIMEOUT){ //hasn't received a new item in the last five seconds, so we'll move to the
|
|
GetNextAsset();
|
|
return;
|
|
}
|
|
|
|
// If it's already more then MASTERNODE_SYNC_TIMEOUT seconds passed since we asked
|
|
// and we still have nothing, assume there is nothing to sync but give it another
|
|
// MASTERNODE_SYNC_TIMEOUT seconds until we move further
|
|
static int64_t lastTimeAsked = 0;
|
|
if(lastBudgetItem == 0 && lastTimeAsked > 0 && lastTimeAsked < GetTime() - MASTERNODE_SYNC_TIMEOUT) {
|
|
lastBudgetItem = GetTime();
|
|
}
|
|
|
|
if(pnode->HasFulfilledRequest("busync")) continue;
|
|
pnode->FulfilledRequest("busync");
|
|
|
|
if((lastBudgetItem == 0 || lastBudgetItem > GetTime() - MASTERNODE_SYNC_TIMEOUT)
|
|
&& RequestedMasternodeAttempt <= 2){
|
|
uint256 n = 0;
|
|
pnode->PushMessage("mnvs", n); //sync masternode votes
|
|
RequestedMasternodeAttempt++;
|
|
lastTimeAsked = GetTime();
|
|
}
|
|
return;
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|