LCOV - code coverage report
Current view: top level - src/test/fuzz - descriptor_parse.cpp (source / functions) Coverage Total Hit
Test: fuzz_coverage.info Lines: 100.0 % 69 69
Test Date: 2025-01-22 04:09:46 Functions: 100.0 % 7 7
Branches: 68.8 % 96 66

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2009-2021 The Bitcoin Core developers
       2                 :             : // Distributed under the MIT software license, see the accompanying
       3                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       4                 :             : 
       5                 :             : #include <chainparams.h>
       6                 :             : #include <key_io.h>
       7                 :             : #include <pubkey.h>
       8                 :             : #include <script/descriptor.h>
       9                 :             : #include <test/fuzz/fuzz.h>
      10                 :             : #include <test/fuzz/util/descriptor.h>
      11                 :             : #include <util/chaintype.h>
      12                 :             : #include <util/strencodings.h>
      13                 :             : 
      14                 :             : //! The converter of mocked descriptors, needs to be initialized when the target is.
      15                 :             : MockedDescriptorConverter MOCKED_DESC_CONVERTER;
      16                 :             : 
      17                 :             : /** Test a successfully parsed descriptor. */
      18                 :       15517 : static void TestDescriptor(const Descriptor& desc, FlatSigningProvider& sig_provider, std::string& dummy, std::optional<bool>& is_ranged, std::optional<bool>& is_solvable)
      19                 :             : {
      20                 :             :     // Trivial helpers.
      21                 :       15517 :     (void)desc.IsRange();
      22                 :       15517 :     (void)desc.IsSingleType();
      23                 :       15517 :     (void)desc.GetOutputType();
      24                 :             : 
      25         [ +  + ]:       15517 :     if (is_ranged.has_value()) {
      26         [ -  + ]:       11450 :         assert(desc.IsRange() == *is_ranged);
      27                 :             :     } else {
      28                 :        4067 :         is_ranged = desc.IsRange();
      29                 :             :     }
      30         [ +  + ]:       15517 :     if (is_solvable.has_value()) {
      31         [ -  + ]:       11450 :         assert(desc.IsSolvable() == *is_solvable);
      32                 :             :     } else {
      33                 :        4067 :         is_solvable = desc.IsSolvable();
      34                 :             :     }
      35                 :             : 
      36                 :             :     // Serialization to string representation.
      37                 :       15517 :     (void)desc.ToString();
      38                 :       15517 :     (void)desc.ToPrivateString(sig_provider, dummy);
      39                 :       15517 :     (void)desc.ToNormalizedString(sig_provider, dummy);
      40                 :             : 
      41                 :             :     // Serialization to Script.
      42                 :       15517 :     DescriptorCache cache;
      43                 :       15517 :     std::vector<CScript> out_scripts;
      44         [ +  - ]:       15517 :     (void)desc.Expand(0, sig_provider, out_scripts, sig_provider, &cache);
      45         [ +  - ]:       15517 :     (void)desc.ExpandPrivate(0, sig_provider, sig_provider);
      46         [ +  - ]:       15517 :     (void)desc.ExpandFromCache(0, cache, out_scripts, sig_provider);
      47                 :             : 
      48                 :             :     // If we could serialize to script we must be able to infer using the same provider.
      49         [ +  + ]:       15517 :     if (!out_scripts.empty()) {
      50   [ +  -  -  + ]:       14650 :         assert(InferDescriptor(out_scripts.back(), sig_provider));
      51                 :             : 
      52                 :             :         // The ScriptSize() must match the size of the serialized Script. (ScriptSize() is set for all descs but 'combo()'.)
      53         [ +  - ]:       14650 :         const bool is_combo{!desc.IsSingleType()};
      54   [ +  +  +  +  :       26279 :         assert(is_combo || desc.ScriptSize() == out_scripts.back().size());
                   +  - ]
      55                 :             :     }
      56                 :             : 
      57         [ +  - ]:       15517 :     const auto max_sat_maxsig{desc.MaxSatisfactionWeight(true)};
      58         [ +  - ]:       15517 :     const auto max_sat_nonmaxsig{desc.MaxSatisfactionWeight(true)};
      59         [ +  - ]:       15517 :     const auto max_elems{desc.MaxSatisfactionElems()};
      60                 :             :     // We must be able to estimate the max satisfaction size for any solvable descriptor (but combo).
      61   [ +  +  +  -  :       15517 :     const bool is_nontop_or_nonsolvable{!*is_solvable || !desc.GetOutputType()};
                   +  + ]
      62   [ +  +  +  -  :       15517 :     const bool is_input_size_info_set{max_sat_maxsig && max_sat_nonmaxsig && max_elems};
                   -  + ]
      63         [ -  + ]:         749 :     assert(is_input_size_info_set || is_nontop_or_nonsolvable);
      64                 :       15517 : }
      65                 :             : 
      66                 :           2 : void initialize_descriptor_parse()
      67                 :             : {
      68   [ +  -  +  -  :           2 :     static ECC_Context ecc_context{};
                   +  - ]
      69                 :           2 :     SelectParams(ChainType::MAIN);
      70                 :           2 : }
      71                 :             : 
      72                 :           1 : void initialize_mocked_descriptor_parse()
      73                 :             : {
      74                 :           1 :     initialize_descriptor_parse();
      75                 :           1 :     MOCKED_DESC_CONVERTER.Init();
      76                 :           1 : }
      77                 :             : 
      78         [ +  - ]:        4381 : FUZZ_TARGET(mocked_descriptor_parse, .init = initialize_mocked_descriptor_parse)
      79                 :             : {
      80                 :             :     // Key derivation is expensive. Deriving deep derivation paths take a lot of compute and we'd
      81                 :             :     // rather spend time elsewhere in this target, like on the actual descriptor syntax. So rule
      82                 :             :     // out strings which could correspond to a descriptor containing a too large derivation path.
      83         [ +  + ]:        3967 :     if (HasDeepDerivPath(buffer)) return;
      84                 :             : 
      85                 :             :     // Some fragments can take a virtually unlimited number of sub-fragments (thresh, multi_a) but
      86                 :             :     // may perform quadratic operations on them. Limit the number of sub-fragments per fragment.
      87         [ +  + ]:        3966 :     if (HasTooManySubFrag(buffer)) return;
      88                 :             : 
      89                 :             :     // The script building logic performs quadratic copies in the number of nested wrappers. Limit
      90                 :             :     // the number of nested wrappers per fragment.
      91         [ +  + ]:        3964 :     if (HasTooManyWrappers(buffer)) return;
      92                 :             : 
      93         [ +  - ]:        3962 :     const std::string mocked_descriptor{buffer.begin(), buffer.end()};
      94   [ +  -  +  + ]:        3962 :     if (const auto descriptor = MOCKED_DESC_CONVERTER.GetDescriptor(mocked_descriptor)) {
      95                 :        3937 :         FlatSigningProvider signing_provider;
      96         [ +  - ]:        3937 :         std::string error;
      97         [ +  - ]:        3937 :         const auto desc = Parse(*descriptor, signing_provider, error);
      98                 :        3937 :         std::optional<bool> is_ranged;
      99                 :        3937 :         std::optional<bool> is_solvable;
     100         [ +  + ]:       17768 :         for (const auto& d : desc) {
     101         [ -  + ]:       13831 :             assert(d);
     102         [ +  - ]:       13831 :             TestDescriptor(*d, signing_provider, error, is_ranged, is_solvable);
     103                 :             :         }
     104                 :        7899 :     }
     105                 :        3962 : }
     106                 :             : 
     107         [ +  - ]:        3724 : FUZZ_TARGET(descriptor_parse, .init = initialize_descriptor_parse)
     108                 :             : {
     109                 :             :     // See comments above for rationales.
     110         [ +  + ]:        3310 :     if (HasDeepDerivPath(buffer)) return;
     111         [ +  + ]:        3309 :     if (HasTooManySubFrag(buffer)) return;
     112         [ +  + ]:        3306 :     if (HasTooManyWrappers(buffer)) return;
     113                 :             : 
     114                 :        3302 :     const std::string descriptor(buffer.begin(), buffer.end());
     115                 :        3302 :     FlatSigningProvider signing_provider;
     116                 :        3302 :     std::string error;
     117         [ +  + ]:        9906 :     for (const bool require_checksum : {true, false}) {
     118         [ +  - ]:        6604 :         const auto desc = Parse(descriptor, signing_provider, error, require_checksum);
     119                 :        6604 :         std::optional<bool> is_ranged;
     120                 :        6604 :         std::optional<bool> is_solvable;
     121         [ +  + ]:        8290 :         for (const auto& d : desc) {
     122         [ -  + ]:        1686 :             assert(d);
     123         [ +  - ]:        1686 :             TestDescriptor(*d, signing_provider, error, is_ranged, is_solvable);
     124                 :             :         }
     125                 :        6604 :     }
     126                 :        3302 : }
        

Generated by: LCOV version 2.0-1