LCOV - code coverage report
Current view: top level - src - pow.cpp (source / functions) Coverage Total Hit
Test: test_bitcoin_coverage.info Lines: 92.3 % 78 72
Test Date: 2025-10-25 04:38:23 Functions: 100.0 % 6 6
Branches: 82.1 % 56 46

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2009-2010 Satoshi Nakamoto
       2                 :             : // Copyright (c) 2009-2022 The Bitcoin Core developers
       3                 :             : // Distributed under the MIT software license, see the accompanying
       4                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       5                 :             : 
       6                 :             : #include <pow.h>
       7                 :             : 
       8                 :             : #include <arith_uint256.h>
       9                 :             : #include <chain.h>
      10                 :             : #include <primitives/block.h>
      11                 :             : #include <uint256.h>
      12                 :             : #include <util/check.h>
      13                 :             : 
      14                 :       28706 : unsigned int GetNextWorkRequired(const CBlockIndex* pindexLast, const CBlockHeader *pblock, const Consensus::Params& params)
      15                 :             : {
      16         [ -  + ]:       28706 :     assert(pindexLast != nullptr);
      17                 :       28706 :     unsigned int nProofOfWorkLimit = UintToArith256(params.powLimit).GetCompact();
      18                 :             : 
      19                 :             :     // Only change once per difficulty adjustment interval
      20         [ +  + ]:       28706 :     if ((pindexLast->nHeight+1) % params.DifficultyAdjustmentInterval() != 0)
      21                 :             :     {
      22         [ +  + ]:       28680 :         if (params.fPowAllowMinDifficultyBlocks)
      23                 :             :         {
      24                 :             :             // Special difficulty rule for testnet:
      25                 :             :             // If the new block's timestamp is more than 2* 10 minutes
      26                 :             :             // then it MUST be a min-difficulty block.
      27         [ +  + ]:       28303 :             if (pblock->GetBlockTime() > pindexLast->GetBlockTime() + params.nPowTargetSpacing*2)
      28                 :             :                 return nProofOfWorkLimit;
      29                 :             :             else
      30                 :             :             {
      31                 :             :                 // Return the last non-special-min-difficulty-rules-block
      32                 :             :                 const CBlockIndex* pindex = pindexLast;
      33   [ +  +  +  +  :     1377197 :                 while (pindex->pprev && pindex->nHeight % params.DifficultyAdjustmentInterval() != 0 && pindex->nBits == nProofOfWorkLimit)
                   +  - ]
      34                 :             :                     pindex = pindex->pprev;
      35                 :       25543 :                 return pindex->nBits;
      36                 :             :             }
      37                 :             :         }
      38                 :         377 :         return pindexLast->nBits;
      39                 :             :     }
      40                 :             : 
      41                 :             :     // Go back by what we want to be 14 days worth of blocks
      42                 :          26 :     int nHeightFirst = pindexLast->nHeight - (params.DifficultyAdjustmentInterval()-1);
      43         [ -  + ]:          26 :     assert(nHeightFirst >= 0);
      44                 :          26 :     const CBlockIndex* pindexFirst = pindexLast->GetAncestor(nHeightFirst);
      45         [ -  + ]:          26 :     assert(pindexFirst);
      46                 :             : 
      47                 :          26 :     return CalculateNextWorkRequired(pindexLast, pindexFirst->GetBlockTime(), params);
      48                 :             : }
      49                 :             : 
      50                 :          30 : unsigned int CalculateNextWorkRequired(const CBlockIndex* pindexLast, int64_t nFirstBlockTime, const Consensus::Params& params)
      51                 :             : {
      52         [ +  + ]:          30 :     if (params.fPowNoRetargeting)
      53                 :          26 :         return pindexLast->nBits;
      54                 :             : 
      55                 :             :     // Limit adjustment step
      56         [ +  + ]:           4 :     int64_t nActualTimespan = pindexLast->GetBlockTime() - nFirstBlockTime;
      57         [ +  + ]:           4 :     if (nActualTimespan < params.nPowTargetTimespan/4)
      58                 :           1 :         nActualTimespan = params.nPowTargetTimespan/4;
      59         [ +  + ]:           4 :     if (nActualTimespan > params.nPowTargetTimespan*4)
      60                 :           1 :         nActualTimespan = params.nPowTargetTimespan*4;
      61                 :             : 
      62                 :             :     // Retarget
      63                 :           4 :     const arith_uint256 bnPowLimit = UintToArith256(params.powLimit);
      64                 :           4 :     arith_uint256 bnNew;
      65                 :             : 
      66                 :             :     // Special difficulty rule for Testnet4
      67         [ -  + ]:           4 :     if (params.enforce_BIP94) {
      68                 :             :         // Here we use the first block of the difficulty period. This way
      69                 :             :         // the real difficulty is always preserved in the first block as
      70                 :             :         // it is not allowed to use the min-difficulty exception.
      71                 :           0 :         int nHeightFirst = pindexLast->nHeight - (params.DifficultyAdjustmentInterval()-1);
      72                 :           0 :         const CBlockIndex* pindexFirst = pindexLast->GetAncestor(nHeightFirst);
      73                 :           0 :         bnNew.SetCompact(pindexFirst->nBits);
      74                 :             :     } else {
      75                 :           4 :         bnNew.SetCompact(pindexLast->nBits);
      76                 :             :     }
      77                 :             : 
      78                 :           4 :     bnNew *= nActualTimespan;
      79                 :           8 :     bnNew /= params.nPowTargetTimespan;
      80                 :             : 
      81         [ +  + ]:           4 :     if (bnNew > bnPowLimit)
      82                 :           1 :         bnNew = bnPowLimit;
      83                 :             : 
      84                 :           4 :     return bnNew.GetCompact();
      85                 :             : }
      86                 :             : 
      87                 :             : // Check that on difficulty adjustments, the new difficulty does not increase
      88                 :             : // or decrease beyond the permitted limits.
      89                 :       90232 : bool PermittedDifficultyTransition(const Consensus::Params& params, int64_t height, uint32_t old_nbits, uint32_t new_nbits)
      90                 :             : {
      91         [ +  + ]:       90232 :     if (params.fPowAllowMinDifficultyBlocks) return true;
      92                 :             : 
      93         [ +  - ]:           6 :     if (height % params.DifficultyAdjustmentInterval() == 0) {
      94                 :           6 :         int64_t smallest_timespan = params.nPowTargetTimespan/4;
      95                 :           6 :         int64_t largest_timespan = params.nPowTargetTimespan*4;
      96                 :             : 
      97                 :           6 :         const arith_uint256 pow_limit = UintToArith256(params.powLimit);
      98                 :           6 :         arith_uint256 observed_new_target;
      99                 :           6 :         observed_new_target.SetCompact(new_nbits);
     100                 :             : 
     101                 :             :         // Calculate the largest difficulty value possible:
     102                 :           6 :         arith_uint256 largest_difficulty_target;
     103                 :           6 :         largest_difficulty_target.SetCompact(old_nbits);
     104                 :           6 :         largest_difficulty_target *= largest_timespan;
     105                 :          12 :         largest_difficulty_target /= params.nPowTargetTimespan;
     106                 :             : 
     107         [ +  + ]:           6 :         if (largest_difficulty_target > pow_limit) {
     108                 :           2 :             largest_difficulty_target = pow_limit;
     109                 :             :         }
     110                 :             : 
     111                 :             :         // Round and then compare this new calculated value to what is
     112                 :             :         // observed.
     113                 :           6 :         arith_uint256 maximum_new_target;
     114                 :           6 :         maximum_new_target.SetCompact(largest_difficulty_target.GetCompact());
     115         [ +  + ]:           6 :         if (maximum_new_target < observed_new_target) return false;
     116                 :             : 
     117                 :             :         // Calculate the smallest difficulty value possible:
     118                 :           5 :         arith_uint256 smallest_difficulty_target;
     119                 :           5 :         smallest_difficulty_target.SetCompact(old_nbits);
     120                 :           5 :         smallest_difficulty_target *= smallest_timespan;
     121                 :          10 :         smallest_difficulty_target /= params.nPowTargetTimespan;
     122                 :             : 
     123         [ -  + ]:           5 :         if (smallest_difficulty_target > pow_limit) {
     124                 :           0 :             smallest_difficulty_target = pow_limit;
     125                 :             :         }
     126                 :             : 
     127                 :             :         // Round and then compare this new calculated value to what is
     128                 :             :         // observed.
     129                 :           5 :         arith_uint256 minimum_new_target;
     130                 :           5 :         minimum_new_target.SetCompact(smallest_difficulty_target.GetCompact());
     131         [ +  + ]:           5 :         if (minimum_new_target > observed_new_target) return false;
     132         [ #  # ]:           0 :     } else if (old_nbits != new_nbits) {
     133                 :           0 :         return false;
     134                 :             :     }
     135                 :             :     return true;
     136                 :             : }
     137                 :             : 
     138                 :             : // Bypasses the actual proof of work check during fuzz testing with a simplified validation checking whether
     139                 :             : // the most significant bit of the last byte of the hash is set.
     140                 :       91946 : bool CheckProofOfWork(uint256 hash, unsigned int nBits, const Consensus::Params& params)
     141                 :             : {
     142                 :       91946 :     if (EnableFuzzDeterminism()) return (hash.data()[31] & 0x80) == 0;
     143                 :       91946 :     return CheckProofOfWorkImpl(hash, nBits, params);
     144                 :             : }
     145                 :             : 
     146                 :       91946 : std::optional<arith_uint256> DeriveTarget(unsigned int nBits, const uint256 pow_limit)
     147                 :             : {
     148                 :       91946 :     bool fNegative;
     149                 :       91946 :     bool fOverflow;
     150                 :       91946 :     arith_uint256 bnTarget;
     151                 :             : 
     152                 :       91946 :     bnTarget.SetCompact(nBits, &fNegative, &fOverflow);
     153                 :             : 
     154                 :             :     // Check range
     155   [ +  +  +  +  :      183891 :     if (fNegative || bnTarget == 0 || fOverflow || bnTarget > UintToArith256(pow_limit))
             +  -  +  + ]
     156                 :           7 :         return {};
     157                 :             : 
     158                 :       91939 :     return bnTarget;
     159                 :             : }
     160                 :             : 
     161                 :       91946 : bool CheckProofOfWorkImpl(uint256 hash, unsigned int nBits, const Consensus::Params& params)
     162                 :             : {
     163                 :       91946 :     auto bnTarget{DeriveTarget(nBits, params.powLimit)};
     164         [ +  + ]:       91946 :     if (!bnTarget) return false;
     165                 :             : 
     166                 :             :     // Check proof of work matches claimed amount
     167         [ +  + ]:       91939 :     if (UintToArith256(hash) > bnTarget)
     168                 :       36718 :         return false;
     169                 :             : 
     170                 :             :     return true;
     171                 :             : }
        

Generated by: LCOV version 2.0-1