// Copyright (c) 2020-2021 The Bitcoin Core developers // Distributed under the MIT software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include #include #include #include #include #include #include #include #include #include #include #include void initialize_pow() { SelectParams(CBaseChainParams::MAIN); } FUZZ_TARGET_INIT(pow, initialize_pow) { FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size()); const Consensus::Params& consensus_params = Params().GetConsensus(); std::vector blocks; const uint32_t fixed_time = fuzzed_data_provider.ConsumeIntegral(); const uint32_t fixed_bits = fuzzed_data_provider.ConsumeIntegral(); LIMITED_WHILE(fuzzed_data_provider.remaining_bytes() > 0, 10000) { const std::optional block_header = ConsumeDeserializable(fuzzed_data_provider); if (!block_header) { continue; } CBlockIndex current_block{*block_header}; { CBlockIndex* previous_block = blocks.empty() ? nullptr : &PickValue(fuzzed_data_provider, blocks); const int current_height = (previous_block != nullptr && previous_block->nHeight != std::numeric_limits::max()) ? previous_block->nHeight + 1 : 0; if (fuzzed_data_provider.ConsumeBool()) { current_block.pprev = previous_block; } if (fuzzed_data_provider.ConsumeBool()) { current_block.nHeight = current_height; } if (fuzzed_data_provider.ConsumeBool()) { const uint32_t seconds = current_height * consensus_params.nPowTargetSpacing; if (!AdditionOverflow(fixed_time, seconds)) { current_block.nTime = fixed_time + seconds; } } if (fuzzed_data_provider.ConsumeBool()) { current_block.nBits = fixed_bits; } if (fuzzed_data_provider.ConsumeBool()) { current_block.nChainWork = previous_block != nullptr ? previous_block->nChainWork + GetBlockProof(*previous_block) : arith_uint256{0}; } else { current_block.nChainWork = ConsumeArithUInt256(fuzzed_data_provider); } blocks.push_back(current_block); } { (void)GetBlockProof(current_block); (void)CalculateNextWorkRequired(¤t_block, fuzzed_data_provider.ConsumeIntegralInRange(0, std::numeric_limits::max()), consensus_params); if (current_block.nHeight != std::numeric_limits::max() && current_block.nHeight - (consensus_params.DifficultyAdjustmentInterval() - 1) >= 0) { (void)GetNextWorkRequired(¤t_block, &(*block_header), consensus_params); } } { const CBlockIndex* to = &PickValue(fuzzed_data_provider, blocks); const CBlockIndex* from = &PickValue(fuzzed_data_provider, blocks); const CBlockIndex* tip = &PickValue(fuzzed_data_provider, blocks); try { (void)GetBlockProofEquivalentTime(*to, *from, *tip, consensus_params); } catch (const uint_error&) { } } { const std::optional hash = ConsumeDeserializable(fuzzed_data_provider); if (hash) { (void)CheckProofOfWork(*hash, fuzzed_data_provider.ConsumeIntegral(), consensus_params); } } } } FUZZ_TARGET_INIT(pow_transition, initialize_pow) { FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size()); const Consensus::Params& consensus_params{Params().GetConsensus()}; std::vector> blocks; const uint32_t old_time{fuzzed_data_provider.ConsumeIntegral()}; const uint32_t new_time{fuzzed_data_provider.ConsumeIntegral()}; const int32_t version{fuzzed_data_provider.ConsumeIntegral()}; uint32_t nbits{fuzzed_data_provider.ConsumeIntegral()}; const arith_uint256 pow_limit = UintToArith256(consensus_params.powLimit); arith_uint256 old_target; old_target.SetCompact(nbits); if (old_target > pow_limit) { nbits = pow_limit.GetCompact(); } // Create one difficulty adjustment period worth of headers for (int height = 0; height < consensus_params.DifficultyAdjustmentInterval(); ++height) { CBlockHeader header; header.nVersion = version; header.nTime = old_time; header.nBits = nbits; if (height == consensus_params.DifficultyAdjustmentInterval() - 1) { header.nTime = new_time; } auto current_block{std::make_unique(header)}; current_block->pprev = blocks.empty() ? nullptr : blocks.back().get(); current_block->nHeight = height; blocks.emplace_back(std::move(current_block)); } auto last_block{blocks.back().get()}; unsigned int new_nbits{GetNextWorkRequired(last_block, nullptr, consensus_params)}; Assert(PermittedDifficultyTransition(consensus_params, last_block->nHeight + 1, last_block->nBits, new_nbits)); }