#[cfg(feature = "proptest")]
use proptest_derive::Arbitrary;
use serde::{Deserialize, Serialize};
use std::ops::{Add, AddAssign, Div, Mul, Neg, Rem, Sub, SubAssign};
#[derive(Debug, Default, Ord, PartialOrd, Eq, PartialEq, Copy, Clone, Serialize, Deserialize)]
#[cfg_attr(feature = "proptest", derive(Arbitrary))]
pub struct Overflowing<T>(T);
#[derive(Debug)]
pub enum OverflowingBehavior {
Panic,
SoftPanic,
Ignore,
}
impl std::str::FromStr for OverflowingBehavior {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
_ if s.eq_ignore_ascii_case("panic") => Ok(OverflowingBehavior::Panic),
_ if s.eq_ignore_ascii_case("soft_panic") => Ok(OverflowingBehavior::SoftPanic),
_ if s.eq_ignore_ascii_case("ignore") => Ok(OverflowingBehavior::Ignore),
_ => Err(format!("Invalid OverflowingBehavior: {s}")),
}
}
}
pub fn set_behavior(behavior: OverflowingBehavior) {
overflowing_support::set_overflowing_mode(behavior);
}
impl<T> Overflowing<T> {
pub fn into_inner(self) -> T {
self.0
}
}
impl<T: std::fmt::Display> std::fmt::Display for Overflowing<T> {
#[inline(always)]
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.0.fmt(f)
}
}
#[cfg(feature = "columnar")]
mod columnar {
use crate::overflowing::Overflowing;
use columnar::common::index::CopyAs;
use columnar::{AsBytes, Clear, Columnar, Container, FromBytes, Index, IndexAs, Len, Push};
use serde::{Deserialize, Serialize};
impl<T: Columnar + Copy + Send> Columnar for Overflowing<T>
where
Vec<T>: Container<T>,
Overflowing<T>: From<T>,
for<'a> <T as Columnar>::Ref<'a>: CopyAs<T>,
{
type Ref<'a> = Overflowing<T>;
#[inline(always)]
fn into_owned<'a>(other: Self::Ref<'a>) -> Self {
other
}
type Container = Overflows<T, Vec<T>>;
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
pub struct Overflows<T, TC>(TC, std::marker::PhantomData<T>);
impl<T, TC: Default> Default for Overflows<T, TC> {
#[inline(always)]
fn default() -> Self {
Self(TC::default(), std::marker::PhantomData)
}
}
impl<T: Columnar + Copy + Send, TC: Container<T>> Container<Overflowing<T>> for Overflows<T, TC>
where
Vec<T>: Container<T>,
Overflowing<T>: From<T>,
for<'a> <T as Columnar>::Ref<'a>: CopyAs<T>,
{
type Borrowed<'a>
= Overflows<T, TC::Borrowed<'a>>
where
Self: 'a;
#[inline(always)]
fn borrow<'a>(&'a self) -> Self::Borrowed<'a> {
Overflows(self.0.borrow(), std::marker::PhantomData)
}
}
impl<'a, T: Copy, TC: AsBytes<'a>> AsBytes<'a> for Overflows<T, TC> {
#[inline(always)]
fn as_bytes(&self) -> impl Iterator<Item = (u64, &'a [u8])> {
self.0.as_bytes()
}
}
impl<'a, T: Copy, TC: FromBytes<'a>> FromBytes<'a> for Overflows<T, TC> {
#[inline(always)]
fn from_bytes(bytes: &mut impl Iterator<Item = &'a [u8]>) -> Self {
Self(TC::from_bytes(bytes), std::marker::PhantomData)
}
}
impl<T: Copy, TC: Len> Len for Overflows<T, TC> {
#[inline(always)]
fn len(&self) -> usize {
self.0.len()
}
}
impl<T: Copy, TC: Clear> Clear for Overflows<T, TC> {
#[inline(always)]
fn clear(&mut self) {
self.0.clear();
}
}
impl<T: Copy, TC: IndexAs<T>> Index for Overflows<T, TC>
where
Overflowing<T>: From<T>,
{
type Ref = Overflowing<T>;
#[inline(always)]
fn get(&self, index: usize) -> Self::Ref {
self.0.index_as(index).into()
}
}
impl<T: Copy, TC: Push<T>> Push<Overflowing<T>> for Overflows<T, TC> {
#[inline(always)]
fn push(&mut self, item: Overflowing<T>) {
self.0.push(item.0);
}
}
impl<T: Copy, TC: Push<T>> Push<&Overflowing<T>> for Overflows<T, TC> {
#[inline(always)]
fn push(&mut self, item: &Overflowing<T>) {
self.0.push(item.0);
}
}
}
macro_rules! impl_overflowing {
($t:ty) => {
impl Overflowing<$t> {
pub const ZERO: Self = Self(0);
pub const ONE: Self = Self(1);
pub const MIN: Self = Self(<$t>::MIN);
pub const MAX: Self = Self(<$t>::MAX);
#[inline(always)]
pub fn checked_add(self, rhs: Self) -> Option<Self> {
self.0.checked_add(rhs.0).map(Self)
}
#[inline(always)]
pub fn wrapping_add(self, rhs: Self) -> Self {
Self(self.0.wrapping_add(rhs.0))
}
#[inline(always)]
pub fn checked_mul(self, rhs: Self) -> Option<Self> {
self.0.checked_mul(rhs.0).map(Self)
}
#[inline(always)]
pub fn wrapping_mul(self, rhs: Self) -> Self {
Self(self.0.wrapping_mul(rhs.0))
}
pub fn is_zero(self) -> bool {
self == Self::ZERO
}
}
impl Add<Self> for Overflowing<$t> {
type Output = Self;
#[inline(always)]
fn add(self, rhs: Self) -> Self::Output {
match self.0.overflowing_add(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} + {rhs}"))
}
(result, false) => Self(result),
}
}
}
impl<'a> Add<&'a Self> for Overflowing<$t> {
type Output = Self;
#[inline(always)]
fn add(self, rhs: &'a Self) -> Self::Output {
match self.0.overflowing_add(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} + {rhs}"))
}
(result, false) => Self(result),
}
}
}
impl AddAssign<Self> for Overflowing<$t> {
#[inline(always)]
fn add_assign(&mut self, rhs: Self) {
*self = *self + rhs;
}
}
impl AddAssign<&Self> for Overflowing<$t> {
#[inline(always)]
fn add_assign(&mut self, rhs: &Self) {
*self = *self + *rhs;
}
}
impl Div<Self> for Overflowing<$t> {
type Output = Overflowing<<$t as Div>::Output>;
#[inline(always)]
fn div(self, rhs: Self) -> Self::Output {
match self.0.overflowing_div(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} / {rhs}"))
}
(result, false) => Self(result),
}
}
}
impl Rem<Self> for Overflowing<$t> {
type Output = Overflowing<<$t as Rem>::Output>;
#[inline(always)]
fn rem(self, rhs: Self) -> Self::Output {
match self.0.overflowing_rem(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} % {rhs}"))
}
(result, false) => Self(result),
}
}
}
impl Sub<Self> for Overflowing<$t> {
type Output = Self;
#[inline(always)]
fn sub(self, rhs: Self) -> Self::Output {
match self.0.overflowing_sub(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} - {rhs}"))
}
(result, false) => Self(result),
}
}
}
impl<'a> Sub<&'a Self> for Overflowing<$t> {
type Output = Self;
#[inline(always)]
fn sub(self, rhs: &'a Self) -> Self::Output {
match self.0.overflowing_sub(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} - {rhs}"))
}
(result, false) => Self(result),
}
}
}
impl SubAssign<Self> for Overflowing<$t> {
#[inline(always)]
fn sub_assign(&mut self, rhs: Self) {
*self = *self - rhs;
}
}
impl SubAssign<&Self> for Overflowing<$t> {
#[inline(always)]
fn sub_assign(&mut self, rhs: &Self) {
*self = *self - *rhs;
}
}
impl std::iter::Sum<Overflowing<$t>> for Overflowing<$t> {
#[inline(always)]
fn sum<I: Iterator<Item = Self>>(iter: I) -> Self {
iter.fold(Self::ZERO, |a, b| a + b)
}
}
impl<'a> std::iter::Sum<&'a Overflowing<$t>> for Overflowing<$t> {
#[inline(always)]
fn sum<I: Iterator<Item = &'a Self>>(iter: I) -> Self {
iter.fold(Self::ZERO, |a, b| a + b)
}
}
impl Mul for Overflowing<$t> {
type Output = Self;
#[inline(always)]
fn mul(self, rhs: Self) -> Self::Output {
match self.0.overflowing_mul(rhs.0) {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("{self} * {rhs}"))
}
(result, false) => Self(result),
}
}
}
#[cfg(feature = "differential-dataflow")]
impl differential_dataflow::difference::IsZero for Overflowing<$t> {
#[inline(always)]
fn is_zero(&self) -> bool {
self.0.is_zero()
}
}
#[cfg(feature = "differential-dataflow")]
impl differential_dataflow::difference::Semigroup for Overflowing<$t> {
#[inline(always)]
fn plus_equals(&mut self, rhs: &Self) {
*self += *rhs
}
}
#[cfg(feature = "differential-dataflow")]
impl differential_dataflow::difference::Monoid for Overflowing<$t> {
#[inline(always)]
fn zero() -> Self {
Self::ZERO
}
}
#[cfg(feature = "differential-dataflow")]
impl differential_dataflow::difference::Multiply<Self> for Overflowing<$t> {
type Output = Self;
#[inline(always)]
fn multiply(self, rhs: &Self) -> Self::Output {
self * *rhs
}
}
#[cfg(feature = "columnation")]
impl columnation::Columnation for Overflowing<$t> {
type InnerRegion = columnation::CopyRegion<Self>;
}
impl std::str::FromStr for Overflowing<$t> {
type Err = <$t as std::str::FromStr>::Err;
#[inline(always)]
fn from_str(s: &str) -> Result<Self, Self::Err> {
<$t>::from_str(s).map(Self)
}
}
impl std::hash::Hash for Overflowing<$t> {
#[inline(always)]
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.0.hash(state);
}
}
impl<T> crate::cast::CastFrom<T> for Overflowing<$t>
where
$t: crate::cast::CastFrom<T>,
{
#[inline(always)]
fn cast_from(value: T) -> Self {
Self(<$t>::cast_from(value))
}
}
#[cfg(feature = "num-traits")]
impl num_traits::identities::Zero for Overflowing<$t> {
#[inline(always)]
fn zero() -> Self {
Self::ZERO
}
#[inline(always)]
fn is_zero(&self) -> bool {
self.0.is_zero()
}
}
#[cfg(feature = "num-traits")]
impl num_traits::identities::One for Overflowing<$t> {
#[inline(always)]
fn one() -> Self {
Self::ONE
}
}
#[cfg(feature = "num-traits")]
impl num_traits::Num for Overflowing<$t> {
type FromStrRadixErr = <$t as num_traits::Num>::FromStrRadixErr;
#[inline(always)]
fn from_str_radix(str: &str, radix: u32) -> Result<Self, Self::FromStrRadixErr> {
<$t>::from_str_radix(str, radix).map(Self)
}
}
};
}
macro_rules! impl_overflowing_from {
($t:ty, $($f:ty)+) => {
$(
impl From<$f> for Overflowing<$t> {
#[inline(always)]
fn from(value: $f) -> Self {
Self(value.into())
}
}
)+
};
}
macro_rules! impl_overflowing_from_overflowing {
($t:ty, $($f:ty)+) => {
$(
impl From<Overflowing<$f>> for Overflowing<$t> {
#[inline(always)]
fn from(value: Overflowing<$f>) -> Self {
Self(value.0.into())
}
}
)+
};
}
macro_rules! impl_overflowing_try_from {
($t:ty, $($f:ty)+) => {
$(
impl TryFrom<$f> for Overflowing<$t> {
type Error = <$t as TryFrom<$f>>::Error;
#[inline(always)]
fn try_from(value: $f) -> Result<Self, Self::Error> {
<$t>::try_from(value).map(Self)
}
}
impl TryFrom<Overflowing<$f>> for Overflowing<$t> {
type Error = <$t as TryFrom<$f>>::Error;
#[inline(always)]
fn try_from(value: Overflowing<$f>) -> Result<Self, Self::Error> {
<$t>::try_from(value.0).map(Self)
}
}
)+
};
}
macro_rules! impl_overflowing_signed {
($t:ty, $u:ty) => {
impl Overflowing<$t> {
pub const MINUS_ONE: Self = Self(-1);
pub fn abs(self) -> Self {
Self(self.0.abs())
}
#[inline(always)]
pub fn unsigned_abs(self) -> $u {
self.0.unsigned_abs()
}
pub fn is_positive(self) -> bool {
self > Self::ZERO
}
pub fn is_negative(self) -> bool {
self < Self::ZERO
}
}
impl Neg for Overflowing<$t> {
type Output = Overflowing<<$t as Neg>::Output>;
#[inline(always)]
fn neg(self) -> Self::Output {
match self.0.overflowing_neg() {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("-{self}"))
}
(result, false) => Self(result),
}
}
}
impl Neg for &Overflowing<$t> {
type Output = Overflowing<<$t as Neg>::Output>;
#[inline(always)]
fn neg(self) -> Self::Output {
match self.0.overflowing_neg() {
(result, true) => {
overflowing_support::handle_overflow(result, format_args!("-{self}"))
}
(result, false) => Overflowing(result),
}
}
}
#[cfg(feature = "differential-dataflow")]
impl differential_dataflow::difference::Abelian for Overflowing<$t> {
#[inline(always)]
fn negate(&mut self) {
*self = -*self
}
}
#[cfg(feature = "num-traits")]
impl num_traits::sign::Signed for Overflowing<$t> {
#[inline(always)]
fn abs(&self) -> Self {
Self(self.0.abs())
}
#[inline(always)]
fn abs_sub(&self, other: &Self) -> Self {
Self(self.0.abs_sub(&other.0))
}
#[inline(always)]
fn signum(&self) -> Self {
Self(self.0.signum())
}
#[inline(always)]
fn is_positive(&self) -> bool {
self.0.is_positive()
}
#[inline(always)]
fn is_negative(&self) -> bool {
self.0.is_negative()
}
}
};
}
macro_rules! overflowing {
($t:ty, $($fit:ty)+, $($may_fit:ty)+ $(, $unsigned:ty)?) => {
impl_overflowing!($t);
impl_overflowing_from!($t, $($fit)+ $t);
impl_overflowing_from_overflowing!($t, $($fit)+);
impl_overflowing_try_from!($t, $($may_fit)+);
$( impl_overflowing_signed!($t, $unsigned); )?
};
}
overflowing!(u8, bool, u16 u32 u64 u128 i8 i16 i32 i64 i128 isize usize);
overflowing!(u16, bool u8, u32 u64 u128 i8 i16 i32 i64 i128 isize usize);
overflowing!(u32, bool u8 u16, u64 u128 i8 i16 i32 i64 i128 isize usize);
overflowing!(u64, bool u8 u16 u32, u128 i8 i16 i32 i64 i128 isize usize);
overflowing!(u128, bool u8 u16 u32 u64, i8 i16 i32 i64 i128 isize usize);
overflowing!(i8, bool, u8 i16 u16 i32 u32 i64 u64 i128 u128 isize usize, u8);
overflowing!(i16, bool i8 u8, u16 i32 u32 i64 u64 i128 u128 isize usize, u16);
overflowing!(i32, bool i8 u8 i16 u16, u32 i64 u64 i128 u128 isize usize, u32);
overflowing!(i64, bool i8 u8 i16 u16 i32 u32, u64 i128 u128 isize usize, u64);
overflowing!(i128, bool i8 u8 i16 u16 i32 u32 i64 u64, u128 isize usize, u128);
mod overflowing_support {
use std::sync::atomic::AtomicUsize;
use crate::overflowing::OverflowingBehavior;
const MODE_IGNORE: usize = 0;
const MODE_SOFT_PANIC: usize = 1;
const MODE_PANIC: usize = 2;
static OVERFLOWING_MODE: AtomicUsize = AtomicUsize::new(MODE_IGNORE);
#[track_caller]
#[cold]
pub(super) fn handle_overflow<T: Into<O>, O>(result: T, description: std::fmt::Arguments) -> O {
let mode = OVERFLOWING_MODE.load(std::sync::atomic::Ordering::Relaxed);
match mode {
#[cfg(not(target_arch = "wasm32"))]
MODE_SOFT_PANIC => crate::soft_panic_or_log!("Overflow: {description}"),
#[cfg(target_arch = "wasm32")]
MODE_SOFT_PANIC => panic!("Overflow: {description}"),
MODE_PANIC => panic!("Overflow: {description}"),
_ => {}
}
result.into()
}
pub(crate) fn set_overflowing_mode(behavior: OverflowingBehavior) {
let value = match behavior {
OverflowingBehavior::Panic => MODE_PANIC,
OverflowingBehavior::SoftPanic => MODE_SOFT_PANIC,
OverflowingBehavior::Ignore => MODE_IGNORE,
};
OVERFLOWING_MODE.store(value, std::sync::atomic::Ordering::Relaxed);
}
}
#[cfg(test)]
mod test {
use super::*;
#[cfg(debug_assertions)]
#[crate::test]
#[should_panic]
fn test_panicking_add() {
set_behavior(OverflowingBehavior::Panic);
let _ = Overflowing::<i8>::MAX + Overflowing::<i8>::ONE;
}
#[crate::test]
fn test_wrapping_add() {
let result = Overflowing::<i8>::MAX.wrapping_add(Overflowing::<i8>::ONE);
assert_eq!(result, Overflowing::<i8>::MIN);
}
#[crate::test]
fn test_checked_add() {
let result = Overflowing::<i8>::MAX.checked_add(Overflowing::<i8>::ONE);
assert_eq!(result, None);
}
}