4331834003
* Move tests -> testing * Directory restructure * Update Cargo.toml during restructure * Update Makefile during restructure * Fix arbitrary path
162 lines
4.9 KiB
Rust
162 lines
4.9 KiB
Rust
//! Library for safe arithmetic on integers, avoiding overflow and division by zero.
|
|
|
|
/// Error representing the failure of an arithmetic operation.
|
|
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
|
pub enum ArithError {
|
|
Overflow,
|
|
DivisionByZero,
|
|
}
|
|
|
|
type Result<T> = std::result::Result<T, ArithError>;
|
|
|
|
macro_rules! assign_method {
|
|
($name:ident, $op:ident, $doc_op:expr) => {
|
|
assign_method!($name, $op, Self, $doc_op);
|
|
};
|
|
($name:ident, $op:ident, $rhs_ty:ty, $doc_op:expr) => {
|
|
#[doc = "Safe variant of `"]
|
|
#[doc = $doc_op]
|
|
#[doc = "`."]
|
|
fn $name(&mut self, other: $rhs_ty) -> Result<()> {
|
|
*self = self.$op(other)?;
|
|
Ok(())
|
|
}
|
|
};
|
|
}
|
|
|
|
/// Trait providing safe arithmetic operations for built-in types.
|
|
pub trait SafeArith: Sized + Copy {
|
|
const ZERO: Self;
|
|
const ONE: Self;
|
|
|
|
/// Safe variant of `+` that guards against overflow.
|
|
fn safe_add(&self, other: Self) -> Result<Self>;
|
|
|
|
/// Safe variant of `-` that guards against overflow.
|
|
fn safe_sub(&self, other: Self) -> Result<Self>;
|
|
|
|
/// Safe variant of `*` that guards against overflow.
|
|
fn safe_mul(&self, other: Self) -> Result<Self>;
|
|
|
|
/// Safe variant of `/` that guards against division by 0.
|
|
fn safe_div(&self, other: Self) -> Result<Self>;
|
|
|
|
/// Safe variant of `%` that guards against division by 0.
|
|
fn safe_rem(&self, other: Self) -> Result<Self>;
|
|
|
|
/// Safe variant of `<<` that guards against overflow.
|
|
fn safe_shl(&self, other: u32) -> Result<Self>;
|
|
|
|
/// Safe variant of `>>` that guards against overflow.
|
|
fn safe_shr(&self, other: u32) -> Result<Self>;
|
|
|
|
assign_method!(safe_add_assign, safe_add, "+=");
|
|
assign_method!(safe_sub_assign, safe_sub, "-=");
|
|
assign_method!(safe_mul_assign, safe_mul, "*=");
|
|
assign_method!(safe_div_assign, safe_div, "/=");
|
|
assign_method!(safe_rem_assign, safe_rem, "%=");
|
|
assign_method!(safe_shl_assign, safe_shl, u32, "<<=");
|
|
assign_method!(safe_shr_assign, safe_shr, u32, ">>=");
|
|
|
|
/// Mutate `self` by adding 1, erroring on overflow.
|
|
fn increment(&mut self) -> Result<()> {
|
|
self.safe_add_assign(Self::ONE)
|
|
}
|
|
}
|
|
|
|
macro_rules! impl_safe_arith {
|
|
($typ:ty) => {
|
|
impl SafeArith for $typ {
|
|
const ZERO: Self = 0;
|
|
const ONE: Self = 1;
|
|
|
|
fn safe_add(&self, other: Self) -> Result<Self> {
|
|
self.checked_add(other).ok_or(ArithError::Overflow)
|
|
}
|
|
|
|
fn safe_sub(&self, other: Self) -> Result<Self> {
|
|
self.checked_sub(other).ok_or(ArithError::Overflow)
|
|
}
|
|
|
|
fn safe_mul(&self, other: Self) -> Result<Self> {
|
|
self.checked_mul(other).ok_or(ArithError::Overflow)
|
|
}
|
|
|
|
fn safe_div(&self, other: Self) -> Result<Self> {
|
|
self.checked_div(other).ok_or(ArithError::DivisionByZero)
|
|
}
|
|
|
|
fn safe_rem(&self, other: Self) -> Result<Self> {
|
|
self.checked_rem(other).ok_or(ArithError::DivisionByZero)
|
|
}
|
|
|
|
fn safe_shl(&self, other: u32) -> Result<Self> {
|
|
self.checked_shl(other).ok_or(ArithError::Overflow)
|
|
}
|
|
|
|
fn safe_shr(&self, other: u32) -> Result<Self> {
|
|
self.checked_shr(other).ok_or(ArithError::Overflow)
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
impl_safe_arith!(u8);
|
|
impl_safe_arith!(u16);
|
|
impl_safe_arith!(u32);
|
|
impl_safe_arith!(u64);
|
|
impl_safe_arith!(usize);
|
|
impl_safe_arith!(i8);
|
|
impl_safe_arith!(i16);
|
|
impl_safe_arith!(i32);
|
|
impl_safe_arith!(i64);
|
|
impl_safe_arith!(isize);
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn basic() {
|
|
let x = 10u32;
|
|
let y = 11;
|
|
assert_eq!(x.safe_add(y), Ok(x + y));
|
|
assert_eq!(y.safe_sub(x), Ok(y - x));
|
|
assert_eq!(x.safe_mul(y), Ok(x * y));
|
|
assert_eq!(x.safe_div(y), Ok(x / y));
|
|
assert_eq!(x.safe_rem(y), Ok(x % y));
|
|
|
|
assert_eq!(x.safe_shl(1), Ok(x << 1));
|
|
assert_eq!(x.safe_shr(1), Ok(x >> 1));
|
|
}
|
|
|
|
#[test]
|
|
fn mutate() {
|
|
let mut x = 0u8;
|
|
x.increment().unwrap();
|
|
x.increment().unwrap();
|
|
assert_eq!(x, 2);
|
|
x.safe_sub_assign(1).unwrap();
|
|
assert_eq!(x, 1);
|
|
x.safe_shl_assign(1).unwrap();
|
|
assert_eq!(x, 2);
|
|
x.safe_mul_assign(3).unwrap();
|
|
assert_eq!(x, 6);
|
|
x.safe_div_assign(4).unwrap();
|
|
assert_eq!(x, 1);
|
|
x.safe_shr_assign(1).unwrap();
|
|
assert_eq!(x, 0);
|
|
}
|
|
|
|
#[test]
|
|
fn errors() {
|
|
assert!(u32::max_value().safe_add(1).is_err());
|
|
assert!(u32::min_value().safe_sub(1).is_err());
|
|
assert!(u32::max_value().safe_mul(2).is_err());
|
|
assert!(u32::max_value().safe_div(0).is_err());
|
|
assert!(u32::max_value().safe_rem(0).is_err());
|
|
assert!(u32::max_value().safe_shl(32).is_err());
|
|
assert!(u32::max_value().safe_shr(32).is_err());
|
|
}
|
|
}
|