mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-24 07:44:10 +00:00
Merge pull request #156 from yvt/fix-int-ops
Implement `saturating_{add, sub}` for non-native integer types
This commit is contained in:
commit
d69ada6bf3
@ -967,34 +967,55 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
|
||||
}
|
||||
|
||||
fn saturating_add(&mut self, lhs: RValue<'gcc>, rhs: RValue<'gcc>, signed: bool, width: u64) -> RValue<'gcc> {
|
||||
let func = self.current_func.borrow().expect("func");
|
||||
|
||||
let result_type = lhs.get_type();
|
||||
if signed {
|
||||
// Algorithm from: https://stackoverflow.com/a/56531252/389119
|
||||
let after_block = func.new_block("after");
|
||||
let func_name =
|
||||
match width {
|
||||
8 => "__builtin_add_overflow",
|
||||
16 => "__builtin_add_overflow",
|
||||
32 => "__builtin_sadd_overflow",
|
||||
64 => "__builtin_saddll_overflow",
|
||||
128 => "__builtin_add_overflow",
|
||||
_ => unreachable!(),
|
||||
};
|
||||
let overflow_func = self.context.get_builtin_function(func_name);
|
||||
let result_type = lhs.get_type();
|
||||
// Based on algorithm from: https://stackoverflow.com/a/56531252/389119
|
||||
let func = self.current_func.borrow().expect("func");
|
||||
let res = func.new_local(None, result_type, "saturating_sum");
|
||||
let overflow = self.overflow_call(overflow_func, &[lhs, rhs, res.get_address(None)], None);
|
||||
let supports_native_type = self.is_native_int_type(result_type);
|
||||
let overflow =
|
||||
if supports_native_type {
|
||||
let func_name =
|
||||
match width {
|
||||
8 => "__builtin_add_overflow",
|
||||
16 => "__builtin_add_overflow",
|
||||
32 => "__builtin_sadd_overflow",
|
||||
64 => "__builtin_saddll_overflow",
|
||||
128 => "__builtin_add_overflow",
|
||||
_ => unreachable!(),
|
||||
};
|
||||
let overflow_func = self.context.get_builtin_function(func_name);
|
||||
self.overflow_call(overflow_func, &[lhs, rhs, res.get_address(None)], None)
|
||||
}
|
||||
else {
|
||||
let func_name =
|
||||
match width {
|
||||
128 => "__rust_i128_addo",
|
||||
_ => unreachable!(),
|
||||
};
|
||||
let param_a = self.context.new_parameter(None, result_type, "a");
|
||||
let param_b = self.context.new_parameter(None, result_type, "b");
|
||||
let result_field = self.context.new_field(None, result_type, "result");
|
||||
let overflow_field = self.context.new_field(None, self.bool_type, "overflow");
|
||||
let return_type = self.context.new_struct_type(None, "result_overflow", &[result_field, overflow_field]);
|
||||
let func = self.context.new_function(None, FunctionType::Extern, return_type.as_type(), &[param_a, param_b], func_name, false);
|
||||
let result = self.context.new_call(None, func, &[lhs, rhs]);
|
||||
let overflow = result.access_field(None, overflow_field);
|
||||
let int_result = result.access_field(None, result_field);
|
||||
self.llbb().add_assignment(None, res, int_result);
|
||||
overflow
|
||||
};
|
||||
|
||||
let then_block = func.new_block("then");
|
||||
let after_block = func.new_block("after");
|
||||
|
||||
let unsigned_type = self.context.new_int_type(width as i32 / 8, false);
|
||||
let shifted = self.context.new_cast(None, lhs, unsigned_type) >> self.context.new_rvalue_from_int(unsigned_type, width as i32 - 1);
|
||||
let uint_max = self.context.new_unary_op(None, UnaryOp::BitwiseNegate, unsigned_type,
|
||||
self.context.new_rvalue_from_int(unsigned_type, 0)
|
||||
);
|
||||
let int_max = uint_max >> self.context.new_rvalue_one(unsigned_type);
|
||||
then_block.add_assignment(None, res, self.context.new_cast(None, shifted + int_max, result_type));
|
||||
// Return `result_type`'s maximum or minimum value on overflow
|
||||
// NOTE: convert the type to unsigned to have an unsigned shift.
|
||||
let unsigned_type = result_type.to_unsigned(&self.cx);
|
||||
let shifted = self.gcc_lshr(self.gcc_int_cast(lhs, unsigned_type), self.gcc_int(unsigned_type, width as i64 - 1));
|
||||
let uint_max = self.gcc_not(self.gcc_int(unsigned_type, 0));
|
||||
let int_max = self.gcc_lshr(uint_max, self.gcc_int(unsigned_type, 1));
|
||||
then_block.add_assignment(None, res, self.gcc_int_cast(self.gcc_add(shifted, int_max), result_type));
|
||||
then_block.end_with_jump(None, after_block);
|
||||
|
||||
self.llbb().end_with_conditional(None, overflow, then_block, after_block);
|
||||
@ -1007,19 +1028,18 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
|
||||
}
|
||||
else {
|
||||
// Algorithm from: http://locklessinc.com/articles/sat_arithmetic/
|
||||
let res = lhs + rhs;
|
||||
let res_type = res.get_type();
|
||||
let cond = self.context.new_comparison(None, ComparisonOp::LessThan, res, lhs);
|
||||
let value = self.context.new_unary_op(None, UnaryOp::Minus, res_type, self.context.new_cast(None, cond, res_type));
|
||||
res | value
|
||||
let res = self.gcc_add(lhs, rhs);
|
||||
let cond = self.gcc_icmp(IntPredicate::IntULT, res, lhs);
|
||||
let value = self.gcc_neg(self.gcc_int_cast(cond, result_type));
|
||||
self.gcc_or(res, value)
|
||||
}
|
||||
}
|
||||
|
||||
// Algorithm from: https://locklessinc.com/articles/sat_arithmetic/
|
||||
fn saturating_sub(&mut self, lhs: RValue<'gcc>, rhs: RValue<'gcc>, signed: bool, width: u64) -> RValue<'gcc> {
|
||||
let result_type = lhs.get_type();
|
||||
if signed {
|
||||
// Also based on algorithm from: https://stackoverflow.com/a/56531252/389119
|
||||
let result_type = lhs.get_type();
|
||||
// Based on algorithm from: https://stackoverflow.com/a/56531252/389119
|
||||
let func = self.current_func.borrow().expect("func");
|
||||
let res = func.new_local(None, result_type, "saturating_diff");
|
||||
let supports_native_type = self.is_native_int_type(result_type);
|
||||
@ -1059,6 +1079,7 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
|
||||
let then_block = func.new_block("then");
|
||||
let after_block = func.new_block("after");
|
||||
|
||||
// Return `result_type`'s maximum or minimum value on overflow
|
||||
// NOTE: convert the type to unsigned to have an unsigned shift.
|
||||
let unsigned_type = result_type.to_unsigned(&self.cx);
|
||||
let shifted = self.gcc_lshr(self.gcc_int_cast(lhs, unsigned_type), self.gcc_int(unsigned_type, width as i64 - 1));
|
||||
@ -1076,11 +1097,10 @@ impl<'a, 'gcc, 'tcx> Builder<'a, 'gcc, 'tcx> {
|
||||
res.to_rvalue()
|
||||
}
|
||||
else {
|
||||
let res = lhs - rhs;
|
||||
let comparison = self.context.new_comparison(None, ComparisonOp::LessThanEquals, res, lhs);
|
||||
let comparison = self.context.new_cast(None, comparison, lhs.get_type());
|
||||
let unary_op = self.context.new_unary_op(None, UnaryOp::Minus, comparison.get_type(), comparison);
|
||||
self.and(res, unary_op)
|
||||
let res = self.gcc_sub(lhs, rhs);
|
||||
let comparison = self.gcc_icmp(IntPredicate::IntULE, res, lhs);
|
||||
let value = self.gcc_neg(self.gcc_int_cast(comparison, result_type));
|
||||
self.gcc_and(res, value)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
405
tests/run/int.rs
405
tests/run/int.rs
@ -3,32 +3,13 @@
|
||||
// Run-time:
|
||||
// status: 0
|
||||
|
||||
#![feature(arbitrary_self_types, auto_traits, core_intrinsics, lang_items, start, intrinsics)]
|
||||
#![feature(bench_black_box, const_black_box, core_intrinsics, start)]
|
||||
|
||||
#![no_std]
|
||||
|
||||
mod intrinsics {
|
||||
extern "rust-intrinsic" {
|
||||
pub fn abort() -> !;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Core
|
||||
*/
|
||||
|
||||
mod libc {
|
||||
#[link(name = "c")]
|
||||
extern "C" {
|
||||
pub fn puts(s: *const u8) -> i32;
|
||||
}
|
||||
}
|
||||
|
||||
#[panic_handler]
|
||||
fn panic_handler(_: &core::panic::PanicInfo) -> ! {
|
||||
unsafe {
|
||||
core::intrinsics::abort();
|
||||
}
|
||||
core::intrinsics::abort();
|
||||
}
|
||||
|
||||
/*
|
||||
@ -36,118 +17,324 @@ fn panic_handler(_: &core::panic::PanicInfo) -> ! {
|
||||
*/
|
||||
|
||||
#[start]
|
||||
fn main(argc: isize, _argv: *const *const u8) -> isize {
|
||||
let var = 134217856_u128;
|
||||
let var2 = 10475372733397991552_u128;
|
||||
let var3 = 193236519889708027473620326106273939584_u128;
|
||||
let var4 = 123236519889708027473620326106273939584_u128;
|
||||
let var5 = 153236519889708027473620326106273939584_u128;
|
||||
let var6 = 18446744073709551616_i128;
|
||||
let var7 = 170141183460469231731687303715884105728_u128;
|
||||
fn main(_argc: isize, _argv: *const *const u8) -> isize {
|
||||
use core::hint::black_box;
|
||||
|
||||
// Shifts.
|
||||
assert_eq!(var << (argc as u128 - 1), var);
|
||||
assert_eq!(var << argc as u128, 268435712);
|
||||
assert_eq!(var << (argc + 32) as u128, 1152922604118474752);
|
||||
assert_eq!(var << (argc + 48) as u128, 75557935783508361347072);
|
||||
assert_eq!(var << (argc + 60) as u128, 309485304969250248077606912);
|
||||
assert_eq!(var << (argc + 62) as u128, 1237941219877000992310427648);
|
||||
assert_eq!(var << (argc + 63) as u128, 2475882439754001984620855296);
|
||||
assert_eq!(var << (argc + 80) as u128, 324518863143436548128224745357312);
|
||||
macro_rules! check {
|
||||
($ty:ty, $expr:expr) => {
|
||||
{
|
||||
const EXPECTED: $ty = $expr;
|
||||
assert_eq!($expr, EXPECTED);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
assert_eq!(var2 << argc as u128, 20950745466795983104);
|
||||
assert_eq!(var2 << (argc as u128 - 1), var2);
|
||||
assert_eq!(var2 << (argc + 32) as u128, 89982766606709001335848566784);
|
||||
assert_eq!(var2 << (argc + 48) as u128, 5897110592337281111546171672756224);
|
||||
assert_eq!(var2 << (argc + 60) as u128, 24154564986213503432893119171609493504);
|
||||
assert_eq!(var2 << (argc + 62) as u128, 96618259944854013731572476686437974016);
|
||||
assert_eq!(var2 << (argc + 63) as u128, 193236519889708027463144953372875948032);
|
||||
check!(u32, (2220326408_u32 + black_box(1)) >> (32 - 6));
|
||||
|
||||
assert_eq!(var3 << argc as u128, 46190672858477591483866044780779667712);
|
||||
assert_eq!(var3 << (argc as u128 - 1), var3);
|
||||
assert_eq!(var3 << (argc + 32) as u128, 21267668304951024224840338247585366016);
|
||||
assert_eq!(var3 << (argc + 48) as u128, 1335125106377253154015353231953100800);
|
||||
assert_eq!(var3 << (argc + 60) as u128, 24154564986213503432893119171609493504);
|
||||
assert_eq!(var3 << (argc + 62) as u128, 96618259944854013731572476686437974016);
|
||||
assert_eq!(var3 << (argc + 63) as u128, 193236519889708027463144953372875948032);
|
||||
/// Generate `check!` tests for integer types at least as wide as 128 bits.
|
||||
macro_rules! check_ops128 {
|
||||
() => {
|
||||
check_ops64!();
|
||||
|
||||
assert_eq!((2220326408_u32 + argc as u32) >> (32 - 6), 33);
|
||||
// Shifts.
|
||||
check!(T, VAL1 << black_box(64));
|
||||
check!(T, VAL1 << black_box(81));
|
||||
check!(T, VAL3 << black_box(63));
|
||||
check!(T, VAL3 << black_box(64));
|
||||
|
||||
assert_eq!(var >> (argc as u128 - 1), var);
|
||||
assert_eq!(var >> argc as u128, 67108928);
|
||||
assert_eq!(var >> (argc + 32) as u128, 0);
|
||||
assert_eq!(var >> (argc + 48) as u128, 0);
|
||||
assert_eq!(var >> (argc + 60) as u128, 0);
|
||||
assert_eq!(var >> (argc + 62) as u128, 0);
|
||||
assert_eq!(var >> (argc + 63) as u128, 0);
|
||||
check!(T, VAL1 >> black_box(64));
|
||||
check!(T, VAL2 >> black_box(64));
|
||||
check!(T, VAL3 >> black_box(64));
|
||||
check!(T, VAL3 >> black_box(81));
|
||||
};
|
||||
}
|
||||
|
||||
assert_eq!(var2 >> argc as u128, 5237686366698995776);
|
||||
assert_eq!(var2 >> (argc as u128 - 1), var2);
|
||||
assert_eq!(var2 >> (argc + 32) as u128, 1219493888);
|
||||
assert_eq!(var2 >> (argc + 48) as u128, 18608);
|
||||
assert_eq!(var2 >> (argc + 60) as u128, 4);
|
||||
assert_eq!(var2 >> (argc + 62) as u128, 1);
|
||||
assert_eq!(var2 >> (argc + 63) as u128, 0);
|
||||
/// Generate `check!` tests for integer types at least as wide as 64 bits.
|
||||
macro_rules! check_ops64 {
|
||||
() => {
|
||||
check_ops32!();
|
||||
|
||||
assert_eq!(var3 >> (argc as u128 - 1), var3);
|
||||
assert_eq!(var3 >> argc as u128, 96618259944854013736810163053136969792);
|
||||
assert_eq!(var3 >> (argc + 32) as u128, 22495691651677250335181635584);
|
||||
assert_eq!(var3 >> (argc + 48) as u128, 343257013727985387194544);
|
||||
assert_eq!(var3 >> (argc + 60) as u128, 83802981867183932420);
|
||||
assert_eq!(var3 >> (argc + 62) as u128, 20950745466795983105);
|
||||
assert_eq!(var3 >> (argc + 63) as u128, 10475372733397991552);
|
||||
assert_eq!(var3 >> (argc + 80) as u128, 79920751444992);
|
||||
// Shifts.
|
||||
check!(T, VAL2 << black_box(33));
|
||||
check!(T, VAL2 << black_box(49));
|
||||
check!(T, VAL2 << black_box(61));
|
||||
check!(T, VAL2 << black_box(63));
|
||||
|
||||
assert_eq!(var6 >> argc as u128, 9223372036854775808);
|
||||
assert_eq!((var6 - 1) >> argc as u128, 9223372036854775807);
|
||||
assert_eq!(var7 >> argc as u128, 85070591730234615865843651857942052864);
|
||||
check!(T, VAL3 << black_box(33));
|
||||
check!(T, VAL3 << black_box(49));
|
||||
check!(T, VAL3 << black_box(61));
|
||||
|
||||
// Casts
|
||||
assert_eq!((var >> (argc + 32) as u128) as u64, 0);
|
||||
assert_eq!((var >> argc as u128) as u64, 67108928);
|
||||
check!(T, VAL1 >> black_box(33));
|
||||
check!(T, VAL1 >> black_box(49));
|
||||
check!(T, VAL1 >> black_box(61));
|
||||
check!(T, VAL1 >> black_box(63));
|
||||
|
||||
// Addition.
|
||||
assert_eq!(var + argc as u128, 134217857);
|
||||
check!(T, VAL2 >> black_box(33));
|
||||
check!(T, VAL2 >> black_box(49));
|
||||
check!(T, VAL2 >> black_box(61));
|
||||
check!(T, VAL2 >> black_box(63));
|
||||
|
||||
assert_eq!(var2 + argc as u128, 10475372733397991553);
|
||||
assert_eq!(var2 + (var2 + argc as u128) as u128, 20950745466795983105);
|
||||
check!(T, VAL3 >> black_box(33));
|
||||
check!(T, VAL3 >> black_box(49));
|
||||
check!(T, VAL3 >> black_box(61));
|
||||
check!(T, VAL3 >> black_box(63));
|
||||
};
|
||||
}
|
||||
|
||||
assert_eq!(var3 + argc as u128, 193236519889708027473620326106273939585);
|
||||
/// Generate `check!` tests for integer types at least as wide as 32 bits.
|
||||
macro_rules! check_ops32 {
|
||||
() => {
|
||||
// Shifts.
|
||||
check!(T, VAL2 << black_box(1));
|
||||
check!(T, VAL2 << black_box(0));
|
||||
|
||||
// Subtraction
|
||||
assert_eq!(var - argc as u128, 134217855);
|
||||
check!(T, VAL3 << black_box(1));
|
||||
check!(T, VAL3 << black_box(0));
|
||||
|
||||
assert_eq!(var2 - argc as u128, 10475372733397991551);
|
||||
check!(T, VAL1.wrapping_shl(black_box(0)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(1)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(33)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(49)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(61)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(63)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(64)));
|
||||
check!(T, VAL1.wrapping_shl(black_box(81)));
|
||||
|
||||
assert_eq!(var3 - argc as u128, 193236519889708027473620326106273939583);
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(0)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(1)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(33)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(49)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(61)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(63)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(64)));
|
||||
check!(Option<T>, VAL1.checked_shl(black_box(81)));
|
||||
|
||||
// Multiplication
|
||||
assert_eq!(var * (argc + 1) as u128, 268435712);
|
||||
assert_eq!(var * (argc as u128 + var2), 1405982069077538020949770368);
|
||||
check!(T, VAL1 >> black_box(0));
|
||||
check!(T, VAL1 >> black_box(1));
|
||||
|
||||
assert_eq!(var2 * (argc + 1) as u128, 20950745466795983104);
|
||||
assert_eq!(var2 * (argc as u128 + var2), 109733433903618109003204073240861360256);
|
||||
check!(T, VAL2 >> black_box(1));
|
||||
check!(T, VAL2 >> black_box(0));
|
||||
|
||||
assert_eq!(var3 * argc as u128, 193236519889708027473620326106273939584);
|
||||
check!(T, VAL3 >> black_box(0));
|
||||
check!(T, VAL3 >> black_box(1));
|
||||
|
||||
assert_eq!(var4 * (argc + 1) as u128, 246473039779416054947240652212547879168);
|
||||
check!(T, VAL1.wrapping_shr(black_box(0)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(1)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(33)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(49)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(61)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(63)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(64)));
|
||||
check!(T, VAL1.wrapping_shr(black_box(81)));
|
||||
|
||||
assert_eq!(var5 * (argc + 1) as u128, 306473039779416054947240652212547879168);
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(0)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(1)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(33)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(49)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(61)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(63)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(64)));
|
||||
check!(Option<T>, VAL1.checked_shr(black_box(81)));
|
||||
|
||||
// Division.
|
||||
assert_eq!(var / (argc + 1) as u128, 67108928);
|
||||
assert_eq!(var / (argc + 2) as u128, 44739285);
|
||||
// Casts
|
||||
check!(u64, (VAL1 >> black_box(1)) as u64);
|
||||
|
||||
assert_eq!(var2 / (argc + 1) as u128, 5237686366698995776);
|
||||
assert_eq!(var2 / (argc + 2) as u128, 3491790911132663850);
|
||||
// Addition.
|
||||
check!(T, VAL1 + black_box(1));
|
||||
check!(T, VAL2 + black_box(1));
|
||||
check!(T, VAL2 + (VAL2 + black_box(1)));
|
||||
check!(T, VAL3 + black_box(1));
|
||||
|
||||
assert_eq!(var3 / (argc + 1) as u128, 96618259944854013736810163053136969792);
|
||||
assert_eq!(var3 / (argc + 2) as u128, 64412173296569342491206775368757979861);
|
||||
assert_eq!(var3 / (argc as u128 + var4), 1);
|
||||
assert_eq!(var3 / (argc as u128 + var2), 18446744073709551615);
|
||||
check!(Option<T>, VAL1.checked_add(black_box(1)));
|
||||
check!(Option<T>, VAL2.checked_add(black_box(1)));
|
||||
check!(Option<T>, VAL2.checked_add(VAL2 + black_box(1)));
|
||||
check!(Option<T>, VAL3.checked_add(T::MAX));
|
||||
check!(Option<T>, VAL3.checked_add(T::MIN));
|
||||
|
||||
assert_eq!(var4 / (argc + 1) as u128, 61618259944854013736810163053136969792);
|
||||
assert_eq!(var4 / (argc + 2) as u128, 41078839963236009157873442035424646528);
|
||||
check!(T, VAL1.wrapping_add(black_box(1)));
|
||||
check!(T, VAL2.wrapping_add(black_box(1)));
|
||||
check!(T, VAL2.wrapping_add(VAL2 + black_box(1)));
|
||||
check!(T, VAL3.wrapping_add(T::MAX));
|
||||
check!(T, VAL3.wrapping_add(T::MIN));
|
||||
|
||||
check!((T, bool), VAL1.overflowing_add(black_box(1)));
|
||||
check!((T, bool), VAL2.overflowing_add(black_box(1)));
|
||||
check!((T, bool), VAL2.overflowing_add(VAL2 + black_box(1)));
|
||||
check!((T, bool), VAL3.overflowing_add(T::MAX));
|
||||
check!((T, bool), VAL3.overflowing_add(T::MIN));
|
||||
|
||||
check!(T, VAL1.saturating_add(black_box(1)));
|
||||
check!(T, VAL2.saturating_add(black_box(1)));
|
||||
check!(T, VAL2.saturating_add(VAL2 + black_box(1)));
|
||||
check!(T, VAL3.saturating_add(T::MAX));
|
||||
check!(T, VAL3.saturating_add(T::MIN));
|
||||
|
||||
// Subtraction
|
||||
check!(T, VAL1 - black_box(1));
|
||||
check!(T, VAL2 - black_box(1));
|
||||
check!(T, VAL3 - black_box(1));
|
||||
|
||||
check!(Option<T>, VAL1.checked_sub(black_box(1)));
|
||||
check!(Option<T>, VAL2.checked_sub(black_box(1)));
|
||||
check!(Option<T>, VAL2.checked_sub(VAL2 + black_box(1)));
|
||||
check!(Option<T>, VAL3.checked_sub(T::MAX));
|
||||
check!(Option<T>, VAL3.checked_sub(T::MIN));
|
||||
|
||||
check!(T, VAL1.wrapping_sub(black_box(1)));
|
||||
check!(T, VAL2.wrapping_sub(black_box(1)));
|
||||
check!(T, VAL2.wrapping_sub(VAL2 + black_box(1)));
|
||||
check!(T, VAL3.wrapping_sub(T::MAX));
|
||||
check!(T, VAL3.wrapping_sub(T::MIN));
|
||||
|
||||
check!((T, bool), VAL1.overflowing_sub(black_box(1)));
|
||||
check!((T, bool), VAL2.overflowing_sub(black_box(1)));
|
||||
check!((T, bool), VAL2.overflowing_sub(VAL2 + black_box(1)));
|
||||
check!((T, bool), VAL3.overflowing_sub(T::MAX));
|
||||
check!((T, bool), VAL3.overflowing_sub(T::MIN));
|
||||
|
||||
check!(T, VAL1.saturating_sub(black_box(1)));
|
||||
check!(T, VAL2.saturating_sub(black_box(1)));
|
||||
check!(T, VAL2.saturating_sub(VAL2 + black_box(1)));
|
||||
check!(T, VAL3.saturating_sub(T::MAX));
|
||||
check!(T, VAL3.saturating_sub(T::MIN));
|
||||
|
||||
// Multiplication
|
||||
check!(T, VAL1 * black_box(2));
|
||||
check!(T, VAL1 * (black_box(1) + VAL2));
|
||||
check!(T, VAL2 * black_box(2));
|
||||
check!(T, VAL2 * (black_box(1) + VAL2));
|
||||
check!(T, VAL3 * black_box(1));
|
||||
check!(T, VAL4 * black_box(2));
|
||||
check!(T, VAL5 * black_box(2));
|
||||
|
||||
check!(Option<T>, VAL1.checked_mul(black_box(2)));
|
||||
check!(Option<T>, VAL1.checked_mul(black_box(1) + VAL2));
|
||||
check!(Option<T>, VAL3.checked_mul(VAL3));
|
||||
check!(Option<T>, VAL4.checked_mul(black_box(2)));
|
||||
check!(Option<T>, VAL5.checked_mul(black_box(2)));
|
||||
|
||||
check!(T, VAL1.wrapping_mul(black_box(2)));
|
||||
check!(T, VAL1.wrapping_mul((black_box(1) + VAL2)));
|
||||
check!(T, VAL3.wrapping_mul(VAL3));
|
||||
check!(T, VAL4.wrapping_mul(black_box(2)));
|
||||
check!(T, VAL5.wrapping_mul(black_box(2)));
|
||||
|
||||
check!((T, bool), VAL1.overflowing_mul(black_box(2)));
|
||||
check!((T, bool), VAL1.overflowing_mul(black_box(1) + VAL2));
|
||||
check!((T, bool), VAL3.overflowing_mul(VAL3));
|
||||
check!((T, bool), VAL4.overflowing_mul(black_box(2)));
|
||||
check!((T, bool), VAL5.overflowing_mul(black_box(2)));
|
||||
|
||||
check!(T, VAL1.saturating_mul(black_box(2)));
|
||||
check!(T, VAL1.saturating_mul(black_box(1) + VAL2));
|
||||
check!(T, VAL3.saturating_mul(VAL3));
|
||||
check!(T, VAL4.saturating_mul(black_box(2)));
|
||||
check!(T, VAL5.saturating_mul(black_box(2)));
|
||||
|
||||
// Division.
|
||||
check!(T, VAL1 / black_box(2));
|
||||
check!(T, VAL1 / black_box(3));
|
||||
|
||||
check!(T, VAL2 / black_box(2));
|
||||
check!(T, VAL2 / black_box(3));
|
||||
|
||||
check!(T, VAL3 / black_box(2));
|
||||
check!(T, VAL3 / black_box(3));
|
||||
check!(T, VAL3 / (black_box(1) + VAL4));
|
||||
check!(T, VAL3 / (black_box(1) + VAL2));
|
||||
|
||||
check!(T, VAL4 / black_box(2));
|
||||
check!(T, VAL4 / black_box(3));
|
||||
|
||||
check!(Option<T>, VAL1.checked_div(black_box(2)));
|
||||
check!(Option<T>, VAL1.checked_div(black_box(1) + VAL2));
|
||||
check!(Option<T>, VAL3.checked_div(VAL3));
|
||||
check!(Option<T>, VAL4.checked_div(black_box(2)));
|
||||
check!(Option<T>, VAL5.checked_div(black_box(2)));
|
||||
check!(Option<T>, (T::MIN).checked_div(black_box(0 as T).wrapping_sub(1)));
|
||||
check!(Option<T>, VAL5.checked_div(black_box(0))); // var5 / 0
|
||||
|
||||
check!(T, VAL1.wrapping_div(black_box(2)));
|
||||
check!(T, VAL1.wrapping_div(black_box(1) + VAL2));
|
||||
check!(T, VAL3.wrapping_div(VAL3));
|
||||
check!(T, VAL4.wrapping_div(black_box(2)));
|
||||
check!(T, VAL5.wrapping_div(black_box(2)));
|
||||
check!(T, (T::MIN).wrapping_div(black_box(0 as T).wrapping_sub(1)));
|
||||
|
||||
check!((T, bool), VAL1.overflowing_div(black_box(2)));
|
||||
check!((T, bool), VAL1.overflowing_div(black_box(1) + VAL2));
|
||||
check!((T, bool), VAL3.overflowing_div(VAL3));
|
||||
check!((T, bool), VAL4.overflowing_div(black_box(2)));
|
||||
check!((T, bool), VAL5.overflowing_div(black_box(2)));
|
||||
check!((T, bool), (T::MIN).overflowing_div(black_box(0 as T).wrapping_sub(1)));
|
||||
|
||||
check!(T, VAL1.saturating_div(black_box(2)));
|
||||
check!(T, VAL1.saturating_div((black_box(1) + VAL2)));
|
||||
check!(T, VAL3.saturating_div(VAL3));
|
||||
check!(T, VAL4.saturating_div(black_box(2)));
|
||||
check!(T, VAL5.saturating_div(black_box(2)));
|
||||
check!(T, (T::MIN).saturating_div((0 as T).wrapping_sub(black_box(1))));
|
||||
};
|
||||
}
|
||||
|
||||
{
|
||||
type T = u32;
|
||||
const VAL1: T = 14162_u32;
|
||||
const VAL2: T = 14556_u32;
|
||||
const VAL3: T = 323656954_u32;
|
||||
const VAL4: T = 2023651954_u32;
|
||||
const VAL5: T = 1323651954_u32;
|
||||
check_ops32!();
|
||||
}
|
||||
|
||||
{
|
||||
type T = i32;
|
||||
const VAL1: T = 13456_i32;
|
||||
const VAL2: T = 10475_i32;
|
||||
const VAL3: T = 923653954_i32;
|
||||
const VAL4: T = 993198738_i32;
|
||||
const VAL5: T = 1023653954_i32;
|
||||
check_ops32!();
|
||||
}
|
||||
|
||||
{
|
||||
type T = u64;
|
||||
const VAL1: T = 134217856_u64;
|
||||
const VAL2: T = 104753732_u64;
|
||||
const VAL3: T = 12323651988970863954_u64;
|
||||
const VAL4: T = 7323651988970863954_u64;
|
||||
const VAL5: T = 8323651988970863954_u64;
|
||||
check_ops64!();
|
||||
}
|
||||
|
||||
{
|
||||
type T = i64;
|
||||
const VAL1: T = 134217856_i64;
|
||||
const VAL2: T = 104753732_i64;
|
||||
const VAL3: T = 6323651988970863954_i64;
|
||||
const VAL4: T = 2323651988970863954_i64;
|
||||
const VAL5: T = 3323651988970863954_i64;
|
||||
check_ops64!();
|
||||
}
|
||||
|
||||
{
|
||||
type T = u128;
|
||||
const VAL1: T = 134217856_u128;
|
||||
const VAL2: T = 10475372733397991552_u128;
|
||||
const VAL3: T = 193236519889708027473620326106273939584_u128;
|
||||
const VAL4: T = 123236519889708027473620326106273939584_u128;
|
||||
const VAL5: T = 153236519889708027473620326106273939584_u128;
|
||||
check_ops128!();
|
||||
}
|
||||
{
|
||||
type T = i128;
|
||||
const VAL1: T = 134217856_i128;
|
||||
const VAL2: T = 10475372733397991552_i128;
|
||||
const VAL3: T = 83236519889708027473620326106273939584_i128;
|
||||
const VAL4: T = 63236519889708027473620326106273939584_i128;
|
||||
const VAL5: T = 73236519889708027473620326106273939584_i128;
|
||||
check_ops128!();
|
||||
}
|
||||
|
||||
0
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user