Line data Source code
1 : // Copyright (c) 2020-2022 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 <test/fuzz/FuzzedDataProvider.h> 6 : #include <test/fuzz/fuzz.h> 7 : #include <test/fuzz/util.h> 8 : #include <util/overflow.h> 9 : 10 : #include <cstdint> 11 : #include <string> 12 : #include <vector> 13 : 14 : #if defined(__has_builtin) 15 : #if __has_builtin(__builtin_add_overflow) 16 : #define HAVE_BUILTIN_ADD_OVERFLOW 17 : #endif 18 : #elif defined(__GNUC__) 19 : #define HAVE_BUILTIN_ADD_OVERFLOW 20 : #endif 21 : 22 : namespace { 23 : template <typename T> 24 0 : void TestAdditionOverflow(FuzzedDataProvider& fuzzed_data_provider) 25 2 : { 26 0 : const T i = fuzzed_data_provider.ConsumeIntegral<T>(); 27 0 : const T j = fuzzed_data_provider.ConsumeIntegral<T>(); 28 0 : const bool is_addition_overflow_custom = AdditionOverflow(i, j); 29 0 : const auto maybe_add{CheckedAdd(i, j)}; 30 0 : const auto sat_add{SaturatingAdd(i, j)}; 31 0 : assert(is_addition_overflow_custom == !maybe_add.has_value()); 32 0 : assert(is_addition_overflow_custom == AdditionOverflow(j, i)); 33 0 : assert(maybe_add == CheckedAdd(j, i)); 34 0 : assert(sat_add == SaturatingAdd(j, i)); 35 : #if defined(HAVE_BUILTIN_ADD_OVERFLOW) 36 : T result_builtin; 37 0 : const bool is_addition_overflow_builtin = __builtin_add_overflow(i, j, &result_builtin); 38 0 : assert(is_addition_overflow_custom == is_addition_overflow_builtin); 39 0 : if (!is_addition_overflow_custom) { 40 0 : assert(i + j == result_builtin); 41 0 : } 42 : #endif 43 0 : if (is_addition_overflow_custom) { 44 0 : assert(sat_add == std::numeric_limits<T>::min() || sat_add == std::numeric_limits<T>::max()); 45 0 : } else { 46 0 : const auto add{i + j}; 47 0 : assert(add == maybe_add.value()); 48 0 : assert(add == sat_add); 49 : } 50 0 : } 51 : } // namespace 52 : 53 4 : FUZZ_TARGET(addition_overflow) 54 : { 55 0 : FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size()); 56 0 : TestAdditionOverflow<int64_t>(fuzzed_data_provider); 57 0 : TestAdditionOverflow<uint64_t>(fuzzed_data_provider); 58 0 : TestAdditionOverflow<int32_t>(fuzzed_data_provider); 59 0 : TestAdditionOverflow<uint32_t>(fuzzed_data_provider); 60 0 : TestAdditionOverflow<int16_t>(fuzzed_data_provider); 61 0 : TestAdditionOverflow<uint16_t>(fuzzed_data_provider); 62 0 : TestAdditionOverflow<char>(fuzzed_data_provider); 63 0 : TestAdditionOverflow<unsigned char>(fuzzed_data_provider); 64 0 : TestAdditionOverflow<signed char>(fuzzed_data_provider); 65 0 : }