Branch data Line data Source code
1 : : // Copyright (c) 2020-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 <script/script.h>
6 : : #include <test/fuzz/FuzzedDataProvider.h>
7 : : #include <test/fuzz/fuzz.h>
8 : : #include <test/fuzz/util.h>
9 : :
10 : : #include <cassert>
11 : : #include <cstdint>
12 : : #include <limits>
13 : : #include <vector>
14 : :
15 : : namespace {
16 : 417678 : bool IsValidAddition(const CScriptNum& lhs, const CScriptNum& rhs)
17 : : {
18 [ + + + + : 417678 : return rhs == 0 || (rhs > 0 && lhs <= CScriptNum{std::numeric_limits<int64_t>::max()} - rhs) || (rhs < 0 && lhs >= CScriptNum{std::numeric_limits<int64_t>::min()} - rhs);
+ + + + +
+ ]
19 : : }
20 : :
21 : 435282 : bool IsValidSubtraction(const CScriptNum& lhs, const CScriptNum& rhs)
22 : : {
23 [ + + + + : 435282 : return rhs == 0 || (rhs > 0 && lhs >= CScriptNum{std::numeric_limits<int64_t>::min()} + rhs) || (rhs < 0 && lhs <= CScriptNum{std::numeric_limits<int64_t>::max()} + rhs);
+ + + + +
+ ]
24 : : }
25 : : } // namespace
26 : :
27 [ + - ]: 621 : FUZZ_TARGET(scriptnum_ops)
28 : : {
29 : 207 : FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size());
30 : 207 : CScriptNum script_num = ConsumeScriptNum(fuzzed_data_provider);
31 [ + + + - ]: 631649 : LIMITED_WHILE(fuzzed_data_provider.remaining_bytes() > 0, 1000000) {
32 : 631442 : CallOneOf(
33 : : fuzzed_data_provider,
34 : 332525 : [&] {
35 : 332525 : const int64_t i = fuzzed_data_provider.ConsumeIntegral<int64_t>();
36 [ - + ]: 332525 : assert((script_num == i) != (script_num != i));
37 [ - + ]: 332525 : assert((script_num <= i) != (script_num > i));
38 [ - + ]: 332525 : assert((script_num >= i) != (script_num < i));
39 : : // Avoid signed integer overflow:
40 : : // script/script.h:264:93: runtime error: signed integer overflow: -2261405121394637306 + -9223372036854775802 cannot be represented in type 'long'
41 [ + + ]: 332525 : if (IsValidAddition(script_num, CScriptNum{i})) {
42 : 155006 : assert((script_num + i) - i == script_num);
43 : : }
44 : : // Avoid signed integer overflow:
45 : : // script/script.h:265:93: runtime error: signed integer overflow: 9223371895120855039 - -9223372036854710486 cannot be represented in type 'long'
46 [ + + ]: 332525 : if (IsValidSubtraction(script_num, CScriptNum{i})) {
47 : 193517 : assert((script_num - i) + i == script_num);
48 : : }
49 : 332525 : },
50 : 18322 : [&] {
51 : 18322 : const CScriptNum random_script_num = ConsumeScriptNum(fuzzed_data_provider);
52 [ - + ]: 18322 : assert((script_num == random_script_num) != (script_num != random_script_num));
53 [ - + ]: 18322 : assert((script_num <= random_script_num) != (script_num > random_script_num));
54 [ - + ]: 18322 : assert((script_num >= random_script_num) != (script_num < random_script_num));
55 : : // Avoid signed integer overflow:
56 : : // script/script.h:264:93: runtime error: signed integer overflow: -9223126527765971126 + -9223372036854756825 cannot be represented in type 'long'
57 [ + + ]: 18322 : if (IsValidAddition(script_num, random_script_num)) {
58 : 11639 : assert((script_num + random_script_num) - random_script_num == script_num);
59 : : }
60 : : // Avoid signed integer overflow:
61 : : // script/script.h:265:93: runtime error: signed integer overflow: 6052837899185946624 - -9223372036854775808 cannot be represented in type 'long'
62 [ + + ]: 18322 : if (IsValidSubtraction(script_num, random_script_num)) {
63 : 12685 : assert((script_num - random_script_num) + random_script_num == script_num);
64 : : }
65 : 18322 : },
66 : 41432 : [&] {
67 : 41432 : const CScriptNum random_script_num = ConsumeScriptNum(fuzzed_data_provider);
68 [ + + ]: 41432 : if (!IsValidAddition(script_num, random_script_num)) {
69 : : // Avoid assertion failure:
70 : : // ./script/script.h:292: CScriptNum &CScriptNum::operator+=(const int64_t &): Assertion `rhs == 0 || (rhs > 0 && m_value <= std::numeric_limits<int64_t>::max() - rhs) || (rhs < 0 && m_value >= std::numeric_limits<int64_t>::min() - rhs)' failed.
71 : : return;
72 : : }
73 : 30795 : script_num += random_script_num;
74 : : },
75 : 60970 : [&] {
76 : 60970 : const CScriptNum random_script_num = ConsumeScriptNum(fuzzed_data_provider);
77 [ + + ]: 60970 : if (!IsValidSubtraction(script_num, random_script_num)) {
78 : : // Avoid assertion failure:
79 : : // ./script/script.h:300: CScriptNum &CScriptNum::operator-=(const int64_t &): Assertion `rhs == 0 || (rhs > 0 && m_value >= std::numeric_limits<int64_t>::min() + rhs) || (rhs < 0 && m_value <= std::numeric_limits<int64_t>::max() + rhs)' failed.
80 : : return;
81 : : }
82 : 29917 : script_num -= random_script_num;
83 : : },
84 : 14460 : [&] {
85 : 14460 : script_num = script_num & fuzzed_data_provider.ConsumeIntegral<int64_t>();
86 : 14460 : },
87 : 13151 : [&] {
88 : 13151 : script_num = script_num & ConsumeScriptNum(fuzzed_data_provider);
89 : 13151 : },
90 : 11530 : [&] {
91 : 11530 : script_num &= ConsumeScriptNum(fuzzed_data_provider);
92 : 11530 : },
93 : 64433 : [&] {
94 [ + + ]: 64433 : if (script_num == CScriptNum{std::numeric_limits<int64_t>::min()}) {
95 : : // Avoid assertion failure:
96 : : // ./script/script.h:279: CScriptNum CScriptNum::operator-() const: Assertion `m_value != std::numeric_limits<int64_t>::min()' failed.
97 : : return;
98 : : }
99 : 61872 : script_num = -script_num;
100 : : },
101 : 17685 : [&] {
102 : 17685 : script_num = fuzzed_data_provider.ConsumeIntegral<int64_t>();
103 : 17685 : },
104 : 25399 : [&] {
105 : 25399 : const int64_t random_integer = fuzzed_data_provider.ConsumeIntegral<int64_t>();
106 [ + + ]: 25399 : if (!IsValidAddition(script_num, CScriptNum{random_integer})) {
107 : : // Avoid assertion failure:
108 : : // ./script/script.h:292: CScriptNum &CScriptNum::operator+=(const int64_t &): Assertion `rhs == 0 || (rhs > 0 && m_value <= std::numeric_limits<int64_t>::max() - rhs) || (rhs < 0 && m_value >= std::numeric_limits<int64_t>::min() - rhs)' failed.
109 : : return;
110 : : }
111 : 18425 : script_num += random_integer;
112 : : },
113 : 23465 : [&] {
114 : 23465 : const int64_t random_integer = fuzzed_data_provider.ConsumeIntegral<int64_t>();
115 [ + + ]: 23465 : if (!IsValidSubtraction(script_num, CScriptNum{random_integer})) {
116 : : // Avoid assertion failure:
117 : : // ./script/script.h:300: CScriptNum &CScriptNum::operator-=(const int64_t &): Assertion `rhs == 0 || (rhs > 0 && m_value >= std::numeric_limits<int64_t>::min() + rhs) || (rhs < 0 && m_value <= std::numeric_limits<int64_t>::max() + rhs)' failed.
118 : : return;
119 : : }
120 : 13582 : script_num -= random_integer;
121 : : },
122 : 8070 : [&] {
123 : 8070 : script_num &= fuzzed_data_provider.ConsumeIntegral<int64_t>();
124 : 8070 : });
125 : 631442 : (void)script_num.getint();
126 : 631442 : (void)script_num.getvch();
127 : : }
128 : 207 : }
|