aboutsummaryrefslogtreecommitdiff
path: root/src/util
diff options
context:
space:
mode:
authorMarcoFalke <falke.marco@gmail.com>2021-11-02 09:48:10 +0100
committerMarcoFalke <falke.marco@gmail.com>2021-12-17 10:46:39 +0100
commitfac01888d17423d6c23a9ce15d98fc88fb34e3cc (patch)
tree9a53718485b5521f4fb2ee6458e5cbaacf8d5033 /src/util
parentfa526d8fb6ab8f2678a30d4536aa9c45218f5269 (diff)
downloadbitcoin-fac01888d17423d6c23a9ce15d98fc88fb34e3cc.tar.xz
Move AdditionOverflow to util, Add CheckedAdd with unit tests
Diffstat (limited to 'src/util')
-rw-r--r--src/util/overflow.h31
1 files changed, 31 insertions, 0 deletions
diff --git a/src/util/overflow.h b/src/util/overflow.h
new file mode 100644
index 0000000000..5982af8d04
--- /dev/null
+++ b/src/util/overflow.h
@@ -0,0 +1,31 @@
+// Copyright (c) 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.
+
+#ifndef BITCOIN_UTIL_OVERFLOW_H
+#define BITCOIN_UTIL_OVERFLOW_H
+
+#include <limits>
+#include <type_traits>
+
+template <class T>
+[[nodiscard]] bool AdditionOverflow(const T i, const T j) noexcept
+{
+ static_assert(std::is_integral<T>::value, "Integral required.");
+ if (std::numeric_limits<T>::is_signed) {
+ return (i > 0 && j > std::numeric_limits<T>::max() - i) ||
+ (i < 0 && j < std::numeric_limits<T>::min() - i);
+ }
+ return std::numeric_limits<T>::max() - i < j;
+}
+
+template <class T>
+[[nodiscard]] std::optional<T> CheckedAdd(const T i, const T j) noexcept
+{
+ if (AdditionOverflow(i, j)) {
+ return std::nullopt;
+ }
+ return i + j;
+}
+
+#endif // BITCOIN_UTIL_OVERFLOW_H