blob: eb3c8037fd34002357e2b85298852a491356762e [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef UI_GFX_GEOMETRY_SAFE_INTEGER_CONVERSIONS_H_
#define UI_GFX_GEOMETRY_SAFE_INTEGER_CONVERSIONS_H_
#include <cmath>
#include <limits>
#include "base/numerics/safe_conversions.h"
#include "ui/gfx/gfx_export.h"
namespace gfx {
inline int ToFlooredInt(float value) {
return base::saturated_cast<int>(std::floor(value));
}
inline int ToCeiledInt(float value) {
return base::saturated_cast<int>(std::ceil(value));
}
inline int ToFlooredInt(double value) {
return base::saturated_cast<int>(std::floor(value));
}
inline int ToCeiledInt(double value) {
return base::saturated_cast<int>(std::ceil(value));
}
inline int ToRoundedInt(float value) {
float rounded;
if (value >= 0.0f)
rounded = std::floor(value + 0.5f);
else
rounded = std::ceil(value - 0.5f);
return base::saturated_cast<int>(rounded);
}
inline int ToRoundedInt(double value) {
double rounded;
if (value >= 0.0)
rounded = std::floor(value + 0.5);
else
rounded = std::ceil(value - 0.5);
return base::saturated_cast<int>(rounded);
}
inline bool IsExpressibleAsInt(float value) {
if (value != value)
return false; // no int NaN.
if (value > std::numeric_limits<int>::max())
return false;
if (value < std::numeric_limits<int>::min())
return false;
return true;
}
// Returns true iff a+b would overflow max int.
constexpr bool AddWouldOverflow(int a, int b) {
// In this function, GCC tries to make optimizations that would only
// work if max - a wouldn't overflow but it isn't smart enough to notice that
// a > 0. So cast everything to unsigned to avoid this. As it is guaranteed
// that max - a and b are both already positive, the cast is a noop.
//
// This is intended to be: a > 0 && max - a < b
return a > 0 && b > 0 &&
static_cast<unsigned>(std::numeric_limits<int>::max() - a) <
static_cast<unsigned>(b);
}
// Returns true iff a+b would underflow min int.
constexpr bool AddWouldUnderflow(int a, int b) {
return a < 0 && b < std::numeric_limits<int>::min() - a;
}
// Returns true iff a-b would overflow max int.
constexpr bool SubtractWouldOverflow(int a, int b) {
return b < 0 && a > std::numeric_limits<int>::max() + b;
}
// Returns true iff a-b would underflow min int.
constexpr bool SubtractWouldUnderflow(int a, int b) {
return b > 0 && a < std::numeric_limits<int>::min() + b;
}
// Safely adds a+b without integer overflow/underflow. Exceeding these
// bounds will clamp to the max or min int limit.
constexpr int SafeAdd(int a, int b) {
return AddWouldOverflow(a, b)
? std::numeric_limits<int>::max()
: AddWouldUnderflow(a, b) ? std::numeric_limits<int>::min()
: a + b;
}
// Safely subtracts a-b without integer overflow/underflow. Exceeding these
// bounds will clamp to the max or min int limit.
constexpr int SafeSubtract(int a, int b) {
return SubtractWouldOverflow(a, b)
? std::numeric_limits<int>::max()
: SubtractWouldUnderflow(a, b) ? std::numeric_limits<int>::min()
: a - b;
}
} // namespace gfx
#endif // UI_GFX_GEOMETRY_SAFE_INTEGER_CONVERSIONS_H_