2014-03-10 08:46:53 -07:00
|
|
|
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
|
|
|
// Copyright (c) 2009-2014 The Bitcoin developers
|
|
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#include "pow.h"
|
|
|
|
|
|
|
|
#include "chainparams.h"
|
|
|
|
#include "core.h"
|
|
|
|
#include "main.h"
|
2014-06-28 14:03:06 +02:00
|
|
|
#include "timedata.h"
|
2014-03-10 08:46:53 -07:00
|
|
|
#include "uint256.h"
|
Split up util.cpp/h
Split up util.cpp/h into:
- string utilities (hex, base32, base64): no internal dependencies, no dependency on boost (apart from foreach)
- money utilities (parsesmoney, formatmoney)
- time utilities (gettime*, sleep, format date):
- and the rest (logging, argument parsing, config file parsing)
The latter is basically the environment and OS handling,
and is stripped of all utility functions, so we may want to
rename it to something else than util.cpp/h for clarity (Matt suggested
osinterface).
Breaks dependency of sha256.cpp on all the things pulled in by util.
2014-08-21 16:11:09 +02:00
|
|
|
#include "util.h"
|
2014-03-10 08:46:53 -07:00
|
|
|
|
|
|
|
unsigned int GetNextWorkRequired(const CBlockIndex* pindexLast, const CBlockHeader *pblock)
|
|
|
|
{
|
|
|
|
unsigned int nProofOfWorkLimit = Params().ProofOfWorkLimit().GetCompact();
|
|
|
|
|
|
|
|
// Genesis block
|
|
|
|
if (pindexLast == NULL)
|
|
|
|
return nProofOfWorkLimit;
|
|
|
|
|
|
|
|
// Only change once per interval
|
2014-03-22 18:29:34 +01:00
|
|
|
if ((pindexLast->nHeight+1) % Params().Interval() != 0)
|
2014-03-10 08:46:53 -07:00
|
|
|
{
|
|
|
|
if (Params().AllowMinDifficultyBlocks())
|
|
|
|
{
|
|
|
|
// Special difficulty rule for testnet:
|
|
|
|
// If the new block's timestamp is more than 2* 10 minutes
|
|
|
|
// then allow mining of a min-difficulty block.
|
2014-06-28 23:36:06 +02:00
|
|
|
if (pblock->GetBlockTime() > pindexLast->GetBlockTime() + Params().TargetSpacing()*2)
|
2014-03-10 08:46:53 -07:00
|
|
|
return nProofOfWorkLimit;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Return the last non-special-min-difficulty-rules-block
|
|
|
|
const CBlockIndex* pindex = pindexLast;
|
2014-03-22 18:29:34 +01:00
|
|
|
while (pindex->pprev && pindex->nHeight % Params().Interval() != 0 && pindex->nBits == nProofOfWorkLimit)
|
2014-03-10 08:46:53 -07:00
|
|
|
pindex = pindex->pprev;
|
|
|
|
return pindex->nBits;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return pindexLast->nBits;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go back by what we want to be 14 days worth of blocks
|
|
|
|
const CBlockIndex* pindexFirst = pindexLast;
|
2014-03-22 18:29:34 +01:00
|
|
|
for (int i = 0; pindexFirst && i < Params().Interval()-1; i++)
|
2014-03-10 08:46:53 -07:00
|
|
|
pindexFirst = pindexFirst->pprev;
|
|
|
|
assert(pindexFirst);
|
|
|
|
|
|
|
|
// Limit adjustment step
|
|
|
|
int64_t nActualTimespan = pindexLast->GetBlockTime() - pindexFirst->GetBlockTime();
|
|
|
|
LogPrintf(" nActualTimespan = %d before bounds\n", nActualTimespan);
|
2014-03-22 18:29:34 +01:00
|
|
|
if (nActualTimespan < Params().TargetTimespan()/4)
|
|
|
|
nActualTimespan = Params().TargetTimespan()/4;
|
|
|
|
if (nActualTimespan > Params().TargetTimespan()*4)
|
|
|
|
nActualTimespan = Params().TargetTimespan()*4;
|
2014-03-10 08:46:53 -07:00
|
|
|
|
|
|
|
// Retarget
|
|
|
|
uint256 bnNew;
|
|
|
|
uint256 bnOld;
|
|
|
|
bnNew.SetCompact(pindexLast->nBits);
|
|
|
|
bnOld = bnNew;
|
|
|
|
bnNew *= nActualTimespan;
|
2014-03-22 18:29:34 +01:00
|
|
|
bnNew /= Params().TargetTimespan();
|
2014-03-10 08:46:53 -07:00
|
|
|
|
|
|
|
if (bnNew > Params().ProofOfWorkLimit())
|
|
|
|
bnNew = Params().ProofOfWorkLimit();
|
|
|
|
|
|
|
|
/// debug print
|
|
|
|
LogPrintf("GetNextWorkRequired RETARGET\n");
|
2014-03-22 18:29:34 +01:00
|
|
|
LogPrintf("Params().TargetTimespan() = %d nActualTimespan = %d\n", Params().TargetTimespan(), nActualTimespan);
|
2014-03-10 08:46:53 -07:00
|
|
|
LogPrintf("Before: %08x %s\n", pindexLast->nBits, bnOld.ToString());
|
|
|
|
LogPrintf("After: %08x %s\n", bnNew.GetCompact(), bnNew.ToString());
|
|
|
|
|
|
|
|
return bnNew.GetCompact();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CheckProofOfWork(uint256 hash, unsigned int nBits)
|
|
|
|
{
|
|
|
|
bool fNegative;
|
|
|
|
bool fOverflow;
|
|
|
|
uint256 bnTarget;
|
2014-09-04 16:23:42 -03:00
|
|
|
|
|
|
|
if (Params().SkipProofOfWorkCheck())
|
|
|
|
return true;
|
|
|
|
|
2014-03-10 08:46:53 -07:00
|
|
|
bnTarget.SetCompact(nBits, &fNegative, &fOverflow);
|
|
|
|
|
|
|
|
// Check range
|
|
|
|
if (fNegative || bnTarget == 0 || fOverflow || bnTarget > Params().ProofOfWorkLimit())
|
|
|
|
return error("CheckProofOfWork() : nBits below minimum work");
|
|
|
|
|
|
|
|
// Check proof of work matches claimed amount
|
|
|
|
if (hash > bnTarget)
|
|
|
|
return error("CheckProofOfWork() : hash doesn't match nBits");
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2014-06-20 20:55:42 +02:00
|
|
|
// true if nBits is greater than the minimum amount of work that could
|
|
|
|
// possibly be required deltaTime after minimum work required was nBase
|
2014-03-10 08:46:53 -07:00
|
|
|
//
|
2014-06-20 20:55:42 +02:00
|
|
|
bool CheckMinWork(unsigned int nBits, unsigned int nBase, int64_t deltaTime)
|
2014-03-10 08:46:53 -07:00
|
|
|
{
|
2014-06-20 20:55:42 +02:00
|
|
|
bool fOverflow = false;
|
|
|
|
uint256 bnNewBlock;
|
|
|
|
bnNewBlock.SetCompact(nBits, NULL, &fOverflow);
|
|
|
|
if (fOverflow)
|
|
|
|
return false;
|
|
|
|
|
2014-03-10 08:46:53 -07:00
|
|
|
const uint256 &bnLimit = Params().ProofOfWorkLimit();
|
|
|
|
// Testnet has min-difficulty blocks
|
2014-03-22 18:29:34 +01:00
|
|
|
// after Params().TargetSpacing()*2 time between blocks:
|
2014-06-20 20:55:42 +02:00
|
|
|
if (Params().AllowMinDifficultyBlocks() && deltaTime > Params().TargetSpacing()*2)
|
|
|
|
return bnNewBlock <= bnLimit;
|
2014-03-10 08:46:53 -07:00
|
|
|
|
|
|
|
uint256 bnResult;
|
|
|
|
bnResult.SetCompact(nBase);
|
2014-06-20 20:55:42 +02:00
|
|
|
while (deltaTime > 0 && bnResult < bnLimit)
|
2014-03-10 08:46:53 -07:00
|
|
|
{
|
|
|
|
// Maximum 400% adjustment...
|
|
|
|
bnResult *= 4;
|
|
|
|
// ... in best-case exactly 4-times-normal target time
|
2014-06-20 20:55:42 +02:00
|
|
|
deltaTime -= Params().TargetTimespan()*4;
|
2014-03-10 08:46:53 -07:00
|
|
|
}
|
|
|
|
if (bnResult > bnLimit)
|
|
|
|
bnResult = bnLimit;
|
2014-06-20 20:55:42 +02:00
|
|
|
|
|
|
|
return bnNewBlock <= bnResult;
|
2014-03-10 08:46:53 -07:00
|
|
|
}
|
2014-06-28 14:03:06 +02:00
|
|
|
|
|
|
|
void UpdateTime(CBlockHeader* pblock, const CBlockIndex* pindexPrev)
|
|
|
|
{
|
|
|
|
pblock->nTime = std::max(pindexPrev->GetMedianTimePast()+1, GetAdjustedTime());
|
|
|
|
|
|
|
|
// Updating time can change work required on testnet:
|
|
|
|
if (Params().AllowMinDifficultyBlocks())
|
|
|
|
pblock->nBits = GetNextWorkRequired(pindexPrev, pblock);
|
|
|
|
}
|
2014-07-05 12:05:33 +02:00
|
|
|
|
|
|
|
uint256 GetProofIncrement(unsigned int nBits)
|
|
|
|
{
|
|
|
|
uint256 bnTarget;
|
|
|
|
bool fNegative;
|
|
|
|
bool fOverflow;
|
|
|
|
bnTarget.SetCompact(nBits, &fNegative, &fOverflow);
|
|
|
|
if (fNegative || fOverflow || bnTarget == 0)
|
|
|
|
return 0;
|
|
|
|
// We need to compute 2**256 / (bnTarget+1), but we can't represent 2**256
|
|
|
|
// as it's too large for a uint256. However, as 2**256 is at least as large
|
|
|
|
// as bnTarget+1, it is equal to ((2**256 - bnTarget - 1) / (bnTarget+1)) + 1,
|
|
|
|
// or ~bnTarget / (nTarget+1) + 1.
|
|
|
|
return (~bnTarget / (bnTarget + 1)) + 1;
|
2014-03-10 08:46:53 -07:00
|
|
|
}
|