blob: 0cb743372eb7b4e42d8a9597ffd5f3891b9e2e00 [file] [log] [blame]
//========================================================================
//
// GooCheckedOps.h
//
// This file is licensed under the GPLv2 or later
//
// Copyright (C) 2018 Adam Reichold <adam.reichold@t-online.de>
// Copyright (C) 2019 LE GARREC Vincent <legarrec.vincent@gmail.com>
// Copyright (C) 2019-2021 Albert Astals Cid <aacid@kde.org>
//
//========================================================================
#ifndef GOO_CHECKED_OPS_H
#define GOO_CHECKED_OPS_H
#include <limits>
#include <type_traits>
template<typename T>
inline bool checkedAssign(long long lz, T *z)
{
static_assert((std::numeric_limits<long long>::max)() > (std::numeric_limits<T>::max)(), "The max of long long type must be larger to perform overflow checks.");
static_assert((std::numeric_limits<long long>::min)() < (std::numeric_limits<T>::min)(), "The min of long long type must be smaller to perform overflow checks.");
if (lz > (std::numeric_limits<T>::max)() || lz < (std::numeric_limits<T>::min)()) {
return true;
}
*z = static_cast<T>(lz);
return false;
}
#ifndef __has_builtin
# define __has_builtin(x) 0
#endif
template<typename T>
inline bool checkedAdd(T x, T y, T *z)
{
// The __GNUC__ checks can not be removed until we depend on GCC >= 10.1
// which is the first version that returns true for __has_builtin(__builtin_add_overflow)
#if __GNUC__ >= 5 || __has_builtin(__builtin_add_overflow)
return __builtin_add_overflow(x, y, z);
#else
const auto lz = static_cast<long long>(x) + static_cast<long long>(y);
return checkedAssign(lz, z);
#endif
}
template<>
inline bool checkedAdd<long long>(long long x, long long y, long long *z)
{
#if __GNUC__ >= 5 || __has_builtin(__builtin_add_overflow)
return __builtin_add_overflow(x, y, z);
#else
if (x > 0 && y > 0) {
if (x > (std::numeric_limits<long long>::max)() - y) {
return true;
}
} else if (x < 0 && y < 0) {
if (x < (std::numeric_limits<long long>::min)() - y) {
return true;
}
}
*z = x + y;
return false;
#endif
}
template<typename T>
inline bool checkedSubtraction(T x, T y, T *z)
{
#if __GNUC__ >= 5 || __has_builtin(__builtin_sub_overflow)
return __builtin_sub_overflow(x, y, z);
#else
const auto lz = static_cast<long long>(x) - static_cast<long long>(y);
return checkedAssign(lz, z);
#endif
}
template<typename T>
inline bool checkedMultiply(T x, T y, T *z)
{
#if __GNUC__ >= 5 || __has_builtin(__builtin_mul_overflow)
return __builtin_mul_overflow(x, y, z);
#else
const auto lz = static_cast<long long>(x) * static_cast<long long>(y);
return checkedAssign(lz, z);
#endif
}
template<>
inline bool checkedMultiply<long long>(long long x, long long y, long long *z)
{
#if __GNUC__ >= 5 || __has_builtin(__builtin_mul_overflow)
return __builtin_mul_overflow(x, y, z);
#else
if (x != 0 && (std::numeric_limits<long long>::max)() / x < y) {
return true;
}
*z = x * y;
return false;
#endif
}
template<typename T>
inline T safeAverage(T a, T b)
{
static_assert((std::numeric_limits<long long>::max)() > (std::numeric_limits<T>::max)(), "The max of long long type must be larger to perform overflow checks.");
static_assert((std::numeric_limits<long long>::min)() < (std::numeric_limits<T>::min)(), "The min of long long type must be smaller to perform overflow checks.");
return static_cast<T>((static_cast<long long>(a) + static_cast<long long>(b)) / 2);
}
#endif // GOO_CHECKED_OPS_H