/root/bitcoin/src/test/fuzz/util/descriptor.h
Line | Count | Source |
1 | | // Copyright (c) 2023-present 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 | | #ifndef BITCOIN_TEST_FUZZ_UTIL_DESCRIPTOR_H |
6 | | #define BITCOIN_TEST_FUZZ_UTIL_DESCRIPTOR_H |
7 | | |
8 | | #include <array> |
9 | | #include <cinttypes> |
10 | | #include <cstddef> |
11 | | #include <limits> |
12 | | #include <optional> |
13 | | #include <span> |
14 | | #include <string> |
15 | | #include <string_view> |
16 | | |
17 | | /** |
18 | | * Converts a mocked descriptor string to a valid one. Every key in a mocked descriptor is |
19 | | * represented by 2 hex characters preceded by the '%' character. We parse the two hex characters |
20 | | * as an index in a list of pre-generated keys. This list contains keys of the various types |
21 | | * accepted in descriptor key expressions. |
22 | | */ |
23 | | class MockedDescriptorConverter { |
24 | | private: |
25 | | //! Types are raw (un)compressed pubkeys, raw xonly pubkeys, raw privkeys (WIF), xpubs, xprvs. |
26 | | static constexpr uint8_t KEY_TYPES_COUNT{6}; |
27 | | //! How many keys we'll generate in total. |
28 | | static constexpr size_t TOTAL_KEYS_GENERATED{std::numeric_limits<uint8_t>::max() + 1}; |
29 | | //! 256 keys of various types. |
30 | | std::array<std::string, TOTAL_KEYS_GENERATED> keys_str; |
31 | | |
32 | | public: |
33 | | // We derive the type of key to generate from the 1-byte id parsed from hex. |
34 | 0 | bool IdIsCompPubKey(uint8_t idx) const { return idx % KEY_TYPES_COUNT == 0; } |
35 | 0 | bool IdIsUnCompPubKey(uint8_t idx) const { return idx % KEY_TYPES_COUNT == 1; } |
36 | 0 | bool IdIsXOnlyPubKey(uint8_t idx) const { return idx % KEY_TYPES_COUNT == 2; } |
37 | 0 | bool IdIsConstPrivKey(uint8_t idx) const { return idx % KEY_TYPES_COUNT == 3; } |
38 | 0 | bool IdIsXpub(uint8_t idx) const { return idx % KEY_TYPES_COUNT == 4; } |
39 | 0 | bool IdIsXprv(uint8_t idx) const { return idx % KEY_TYPES_COUNT == 5; } |
40 | | |
41 | | //! When initializing the target, populate the list of keys. |
42 | | void Init(); |
43 | | |
44 | | //! Parse an id in the keys vectors from a 2-characters hex string. |
45 | | std::optional<uint8_t> IdxFromHex(std::string_view hex_characters) const; |
46 | | |
47 | | //! Get an actual descriptor string from a descriptor string whose keys were mocked. |
48 | | std::optional<std::string> GetDescriptor(std::string_view mocked_desc) const; |
49 | | }; |
50 | | |
51 | | //! Default maximum number of derivation indexes in a single derivation path when limiting its depth. |
52 | | constexpr int MAX_DEPTH{2}; |
53 | | |
54 | | /** |
55 | | * Whether the buffer, if it represents a valid descriptor, contains a derivation path deeper than |
56 | | * a given maximum depth. Note this may also be hit for deriv paths in origins. |
57 | | */ |
58 | | bool HasDeepDerivPath(std::span<const uint8_t> buff, int max_depth = MAX_DEPTH); |
59 | | |
60 | | //! Default maximum number of sub-fragments. |
61 | | constexpr int MAX_SUBS{1'000}; |
62 | | //! Maximum number of nested sub-fragments we'll allow in a descriptor. |
63 | | constexpr size_t MAX_NESTED_SUBS{10'000}; |
64 | | |
65 | | /** |
66 | | * Whether the buffer, if it represents a valid descriptor, contains a fragment with more |
67 | | * sub-fragments than the given maximum. |
68 | | */ |
69 | | bool HasTooManySubFrag(std::span<const uint8_t> buff, int max_subs = MAX_SUBS, |
70 | | size_t max_nested_subs = MAX_NESTED_SUBS); |
71 | | |
72 | | //! Default maximum number of wrappers per fragment. |
73 | | constexpr int MAX_WRAPPERS{100}; |
74 | | |
75 | | /** |
76 | | * Whether the buffer, if it represents a valid descriptor, contains a fragment with more |
77 | | * wrappers than the given maximum. |
78 | | */ |
79 | | bool HasTooManyWrappers(std::span<const uint8_t> buff, int max_wrappers = MAX_WRAPPERS); |
80 | | |
81 | | /// Default maximum leaf size. This should be large enough to cover an extended |
82 | | /// key, including paths "/", inside and outside of "[]". |
83 | | constexpr uint32_t MAX_LEAF_SIZE{200}; |
84 | | |
85 | | /// Whether the expanded buffer (after calling GetDescriptor() in |
86 | | /// MockedDescriptorConverter) has a leaf size too large. |
87 | | bool HasTooLargeLeafSize(std::span<const uint8_t> buff, uint32_t max_leaf_size = MAX_LEAF_SIZE); |
88 | | |
89 | | /// Deriving "expensive" descriptors will consume useful fuzz compute. The |
90 | | /// compute is better spent on a smaller subset of descriptors, which still |
91 | | /// covers all real end-user settings. |
92 | | /// |
93 | | /// Use this function after MockedDescriptorConverter::GetDescriptor() |
94 | | inline bool IsTooExpensive(std::span<const uint8_t> buffer) |
95 | 0 | { |
96 | | // Key derivation is expensive. Deriving deep derivation paths takes a lot of compute and we'd |
97 | | // rather spend time elsewhere in this target, like on the actual descriptor syntax. So rule |
98 | | // out strings which could correspond to a descriptor containing a too large derivation path. |
99 | 0 | if (HasDeepDerivPath(buffer)) return true; Branch (99:9): [True: 0, False: 0]
|
100 | | |
101 | | // Some fragments can take a virtually unlimited number of sub-fragments (thresh, multi_a) but |
102 | | // may perform quadratic operations on them. Limit the number of sub-fragments per fragment. |
103 | 0 | if (HasTooManySubFrag(buffer)) return true; Branch (103:9): [True: 0, False: 0]
|
104 | | |
105 | | // The script building logic performs quadratic copies in the number of nested wrappers. Limit |
106 | | // the number of nested wrappers per fragment. |
107 | 0 | if (HasTooManyWrappers(buffer)) return true; Branch (107:9): [True: 0, False: 0]
|
108 | | |
109 | | // If any suspected leaf is too large, it will likely not represent a valid |
110 | | // use-case. Also, possible base58 parsing in the leaf is quadratic. So |
111 | | // limit the leaf size. |
112 | 0 | if (HasTooLargeLeafSize(buffer)) return true; Branch (112:9): [True: 0, False: 0]
|
113 | | |
114 | 0 | return false; |
115 | 0 | } |
116 | | #endif // BITCOIN_TEST_FUZZ_UTIL_DESCRIPTOR_H |