OpenTTD Source 20250205-master-gfd85ab1e2c
overflowsafe_type.hpp
Go to the documentation of this file.
1/*
2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
6 */
7
10#ifndef OVERFLOWSAFE_TYPE_HPP
11#define OVERFLOWSAFE_TYPE_HPP
12
13#include "math_func.hpp"
14
15
16#ifdef __has_builtin
17# if __has_builtin(__builtin_add_overflow) && __has_builtin(__builtin_sub_overflow) && __has_builtin(__builtin_mul_overflow)
18# define HAS_OVERFLOW_BUILTINS
19# endif
20#endif
21
28template <class T>
30{
31private:
32 static constexpr T T_MAX = std::numeric_limits<T>::max();
33 static constexpr T T_MIN = std::numeric_limits<T>::min();
34
37public:
38 using BaseType = T;
39
40 constexpr OverflowSafeInt() : m_value(0) { }
41
42 constexpr OverflowSafeInt(const OverflowSafeInt &other) : m_value(other.m_value) { }
43 constexpr OverflowSafeInt(const T int_) : m_value(int_) { }
44
45 inline constexpr OverflowSafeInt& operator = (const OverflowSafeInt& other) { this->m_value = other.m_value; return *this; }
46 inline constexpr OverflowSafeInt& operator = (T other) { this->m_value = other; return *this; }
47
48 inline constexpr OverflowSafeInt operator - () const { return OverflowSafeInt(this->m_value == T_MIN ? T_MAX : -this->m_value); }
49
55 inline constexpr OverflowSafeInt& operator += (const OverflowSafeInt& other)
56 {
57#ifdef HAS_OVERFLOW_BUILTINS
58 if (__builtin_add_overflow(this->m_value, other.m_value, &this->m_value)) [[unlikely]] {
59 this->m_value = (other.m_value < 0) ? T_MIN : T_MAX;
60 }
61#else
62 if (this->m_value > 0 && other.m_value > 0 && (T_MAX - other.m_value) < this->m_value) {
63 this->m_value = T_MAX;
64 } else if (this->m_value < 0 && other.m_value < 0 && (this->m_value == T_MIN || other.m_value == T_MIN || ((T_MAX + this->m_value) + other.m_value < (T_MIN + T_MAX)))) {
65 this->m_value = T_MIN;
66 } else {
67 this->m_value += other.m_value;
68 }
69#endif
70 return *this;
71 }
72
78 inline constexpr OverflowSafeInt& operator -= (const OverflowSafeInt& other)
79 {
80#ifdef HAS_OVERFLOW_BUILTINS
81 if (__builtin_sub_overflow(this->m_value, other.m_value, &this->m_value)) [[unlikely]] {
82 this->m_value = (other.m_value < 0) ? T_MAX : T_MIN;
83 }
84#else
85 if (this->m_value > 0 && other.m_value < 0 && (T_MAX + other.m_value) < this->m_value) {
86 this->m_value = T_MAX;
87 } else if (this->m_value < 0 && other.m_value > 0 && (T_MAX + this->m_value) < (T_MIN + T_MAX) + other.m_value) {
88 this->m_value = T_MIN;
89 } else {
90 this->m_value -= other.m_value;
91 }
92#endif
93 return *this;
94 }
95
96 /* Operators for addition and subtraction. */
97 inline constexpr OverflowSafeInt operator + (const OverflowSafeInt& other) const { OverflowSafeInt result = *this; result += other; return result; }
98 inline constexpr OverflowSafeInt operator + (const int other) const { OverflowSafeInt result = *this; result += (int64_t)other; return result; }
99 inline constexpr OverflowSafeInt operator + (const uint other) const { OverflowSafeInt result = *this; result += (int64_t)other; return result; }
100 inline constexpr OverflowSafeInt operator - (const OverflowSafeInt& other) const { OverflowSafeInt result = *this; result -= other; return result; }
101 inline constexpr OverflowSafeInt operator - (const int other) const { OverflowSafeInt result = *this; result -= (int64_t)other; return result; }
102 inline constexpr OverflowSafeInt operator - (const uint other) const { OverflowSafeInt result = *this; result -= (int64_t)other; return result; }
103
104 inline constexpr OverflowSafeInt& operator ++ () { return *this += 1; }
105 inline constexpr OverflowSafeInt& operator -- () { return *this += -1; }
106 inline constexpr OverflowSafeInt operator ++ (int) { OverflowSafeInt org = *this; *this += 1; return org; }
107 inline constexpr OverflowSafeInt operator -- (int) { OverflowSafeInt org = *this; *this += -1; return org; }
108
115 inline constexpr OverflowSafeInt& operator *= (const int factor)
116 {
117#ifdef HAS_OVERFLOW_BUILTINS
118 const bool is_result_positive = (this->m_value < 0) == (factor < 0); // -ve * -ve == +ve
119 if (__builtin_mul_overflow(this->m_value, factor, &this->m_value)) [[unlikely]] {
120 this->m_value = is_result_positive ? T_MAX : T_MIN;
121 }
122#else
123 if (factor == -1) {
124 this->m_value = (this->m_value == T_MIN) ? T_MAX : -this->m_value;
125 } else if (factor > 0 && this->m_value > 0 && (T_MAX / factor) < this->m_value) {
126 this->m_value = T_MAX;
127 } else if (factor > 0 && this->m_value < 0 && (T_MIN / factor) > this->m_value) {
128 this->m_value = T_MIN;
129 } else if (factor < 0 && this->m_value > 0 && (T_MIN / factor) < this->m_value) {
130 this->m_value = T_MIN;
131 } else if (factor < 0 && this->m_value < 0 && (T_MAX / factor) > this->m_value) {
132 this->m_value = T_MAX;
133 } else {
134 this->m_value *= factor;
135 }
136#endif
137 return *this;
138 }
139
140 /* Operators for multiplication. */
141 inline constexpr OverflowSafeInt operator * (const int64_t factor) const { OverflowSafeInt result = *this; result *= factor; return result; }
142 inline constexpr OverflowSafeInt operator * (const int factor) const { OverflowSafeInt result = *this; result *= (int64_t)factor; return result; }
143 inline constexpr OverflowSafeInt operator * (const uint factor) const { OverflowSafeInt result = *this; result *= (int64_t)factor; return result; }
144 inline constexpr OverflowSafeInt operator * (const uint16_t factor) const { OverflowSafeInt result = *this; result *= (int64_t)factor; return result; }
145 inline constexpr OverflowSafeInt operator * (const uint8_t factor) const { OverflowSafeInt result = *this; result *= (int64_t)factor; return result; }
146
147 /* Operators for division. */
148 inline constexpr OverflowSafeInt& operator /= (const int64_t divisor) { this->m_value /= divisor; return *this; }
149 inline constexpr OverflowSafeInt operator / (const OverflowSafeInt& divisor) const { OverflowSafeInt result = *this; result /= divisor.m_value; return result; }
150 inline constexpr OverflowSafeInt operator / (const int divisor) const { OverflowSafeInt result = *this; result /= divisor; return result; }
151 inline constexpr OverflowSafeInt operator / (const uint divisor) const { OverflowSafeInt result = *this; result /= (int)divisor; return result; }
152
153 /* Operators for modulo */
154 inline constexpr OverflowSafeInt& operator %= (const int divisor) { this->m_value %= divisor; return *this; }
155 inline constexpr OverflowSafeInt operator % (const int divisor) const { OverflowSafeInt result = *this; result %= divisor; return result; }
156
157 /* Operators for shifting. */
158 inline constexpr OverflowSafeInt& operator <<= (const int shift) { this->m_value <<= shift; return *this; }
159 inline constexpr OverflowSafeInt operator << (const int shift) const { OverflowSafeInt result = *this; result <<= shift; return result; }
160 inline constexpr OverflowSafeInt& operator >>= (const int shift) { this->m_value >>= shift; return *this; }
161 inline constexpr OverflowSafeInt operator >> (const int shift) const { OverflowSafeInt result = *this; result >>= shift; return result; }
162
163 /* Operators for (in)equality when comparing overflow safe ints. */
164 inline constexpr bool operator == (const OverflowSafeInt& other) const { return this->m_value == other.m_value; }
165 inline constexpr bool operator != (const OverflowSafeInt& other) const { return !(*this == other); }
166 inline constexpr bool operator > (const OverflowSafeInt& other) const { return this->m_value > other.m_value; }
167 inline constexpr bool operator >= (const OverflowSafeInt& other) const { return this->m_value >= other.m_value; }
168 inline constexpr bool operator < (const OverflowSafeInt& other) const { return !(*this >= other); }
169 inline constexpr bool operator <= (const OverflowSafeInt& other) const { return !(*this > other); }
170
171 /* Operators for (in)equality when comparing non-overflow safe ints. */
172 inline constexpr bool operator == (const int other) const { return this->m_value == other; }
173 inline constexpr bool operator != (const int other) const { return !(*this == other); }
174 inline constexpr bool operator > (const int other) const { return this->m_value > other; }
175 inline constexpr bool operator >= (const int other) const { return this->m_value >= other; }
176 inline constexpr bool operator < (const int other) const { return !(*this >= other); }
177 inline constexpr bool operator <= (const int other) const { return !(*this > other); }
178
179 inline constexpr operator T () const { return this->m_value; }
180
181 static inline constexpr OverflowSafeInt<T> max() { return T_MAX; }
182 static inline constexpr OverflowSafeInt<T> min() { return T_MIN; }
183
184 BaseType base() const noexcept { return this->m_value; }
185};
186
187
188/* Sometimes we got int64_t operator OverflowSafeInt instead of vice versa. Handle that properly. */
189template <class T> inline constexpr OverflowSafeInt<T> operator + (const int64_t a, const OverflowSafeInt<T> b) { return b + a; }
190template <class T> inline constexpr OverflowSafeInt<T> operator - (const int64_t a, const OverflowSafeInt<T> b) { return -b + a; }
191template <class T> inline constexpr OverflowSafeInt<T> operator * (const int64_t a, const OverflowSafeInt<T> b) { return b * a; }
192template <class T> inline constexpr OverflowSafeInt<T> operator / (const int64_t a, const OverflowSafeInt<T> b) { return (OverflowSafeInt<T>)a / (int)b; }
193
194/* Sometimes we got int operator OverflowSafeInt instead of vice versa. Handle that properly. */
195template <class T> inline constexpr OverflowSafeInt<T> operator + (const int a, const OverflowSafeInt<T> b) { return b + a; }
196template <class T> inline constexpr OverflowSafeInt<T> operator - (const int a, const OverflowSafeInt<T> b) { return -b + a; }
197template <class T> inline constexpr OverflowSafeInt<T> operator * (const int a, const OverflowSafeInt<T> b) { return b * a; }
198template <class T> inline constexpr OverflowSafeInt<T> operator / (const int a, const OverflowSafeInt<T> b) { return (OverflowSafeInt<T>)a / (int)b; }
199
200/* Sometimes we got uint operator OverflowSafeInt instead of vice versa. Handle that properly. */
201template <class T> inline constexpr OverflowSafeInt<T> operator + (const uint a, const OverflowSafeInt<T> b) { return b + a; }
202template <class T> inline constexpr OverflowSafeInt<T> operator - (const uint a, const OverflowSafeInt<T> b) { return -b + a; }
203template <class T> inline constexpr OverflowSafeInt<T> operator * (const uint a, const OverflowSafeInt<T> b) { return b * a; }
204template <class T> inline constexpr OverflowSafeInt<T> operator / (const uint a, const OverflowSafeInt<T> b) { return (OverflowSafeInt<T>)a / (int)b; }
205
206/* Sometimes we got byte operator OverflowSafeInt instead of vice versa. Handle that properly. */
207template <class T> inline constexpr OverflowSafeInt<T> operator + (const uint8_t a, const OverflowSafeInt<T> b) { return b + (uint)a; }
208template <class T> inline constexpr OverflowSafeInt<T> operator - (const uint8_t a, const OverflowSafeInt<T> b) { return -b + (uint)a; }
209template <class T> inline constexpr OverflowSafeInt<T> operator * (const uint8_t a, const OverflowSafeInt<T> b) { return b * (uint)a; }
210template <class T> inline constexpr OverflowSafeInt<T> operator / (const uint8_t a, const OverflowSafeInt<T> b) { return (OverflowSafeInt<T>)a / (int)b; }
211
214
215/* Some basic "unit tests". Also has the bonus of confirming that constexpr is working. */
216static_assert(OverflowSafeInt32(INT32_MIN) - 1 == OverflowSafeInt32(INT32_MIN));
217static_assert(OverflowSafeInt32(INT32_MAX) + 1 == OverflowSafeInt32(INT32_MAX));
218static_assert(OverflowSafeInt32(INT32_MAX) * 2 == OverflowSafeInt32(INT32_MAX));
219static_assert(OverflowSafeInt32(INT32_MIN) * 2 == OverflowSafeInt32(INT32_MIN));
220
221/* Specialisation of the generic ClampTo function for overflow safe integers to normal integers. */
222template <typename To, typename From>
223constexpr To ClampTo(OverflowSafeInt<From> value) { return ClampTo<To>(From(value)); }
224
225#undef HAS_OVERFLOW_BUILTINS
226
227#endif /* OVERFLOWSAFE_TYPE_HPP */
Overflow safe template for integers, i.e.
constexpr OverflowSafeInt & operator+=(const OverflowSafeInt &other)
Safe implementation of addition.
constexpr OverflowSafeInt & operator*=(const int factor)
Safe implementation of multiplication.
T m_value
The non-overflow safe backend to store the value in.
constexpr OverflowSafeInt & operator-=(const OverflowSafeInt &other)
Safe implementation of subtraction.
Integer math functions.
constexpr To ClampTo(From value)
Clamp the given value down to lie within the requested type.