blob: 96c2f517b4b205b041325bd4fd57ac8ffeacf1bf [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 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) {
#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<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<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