malachite-base 0.3.2

A collection of utilities, including new arithmetic traits and iterators that generate all values of a type
Documentation
use malachite_base::num::arithmetic::traits::{ArithmeticCheckedShl, ArithmeticCheckedShr};
use malachite_base::num::basic::integers::PrimitiveInt;
use malachite_base::num::basic::signeds::PrimitiveSigned;
use malachite_base::num::basic::unsigneds::PrimitiveUnsigned;
use malachite_base::num::conversion::traits::ExactFrom;
use malachite_base::test_util::generators::{
    signed_gen, signed_gen_var_5, signed_pair_gen_var_2, unsigned_gen,
    unsigned_signed_pair_gen_var_1,
};

#[test]
fn test_arithmetic_checked_shr() {
    fn test<T: ArithmeticCheckedShr<U, Output = T> + PrimitiveInt, U: PrimitiveInt>(
        t: T,
        u: U,
        out: Option<T>,
    ) {
        assert_eq!(t.arithmetic_checked_shr(u), out);
    }
    test::<u32, i8>(100, 3, Some(12));
    test::<u32, i16>(100, 100, Some(0));

    test::<i8, i8>(3, -5, Some(96));
    test::<i8, i16>(3, -6, None);
    test::<i8, i32>(-3, -5, Some(-96));
    test::<i8, i64>(-3, -6, None);
    test::<i16, i128>(3, -100, None);
    test::<i16, isize>(-3, -100, None);
    test::<i32, i8>(0, -100, Some(0));
    test::<i32, i16>(100, 3, Some(12));
    test::<i32, i32>(-100, 3, Some(-13));
    test::<i64, i64>(100, 100, Some(0));
    test::<i64, i128>(-100, 100, Some(-1));
}

// Type repetition to avoid long line
#[allow(clippy::type_repetition_in_bounds)]
fn arithmetic_checked_shr_properties_helper_unsigned_signed<
    T: ArithmeticCheckedShl<U, Output = T> + ArithmeticCheckedShr<U, Output = T>,
    U: PrimitiveSigned,
>()
where
    u64: ExactFrom<U>,
    T: PrimitiveUnsigned,
{
    unsigned_signed_pair_gen_var_1::<T, U>().test_properties(|(n, i)| {
        let shifted = n.arithmetic_checked_shr(i);
        if shifted.is_none() {
            assert_ne!(n, T::ZERO);
        }
        if i != U::MIN {
            assert_eq!(n.arithmetic_checked_shl(-i), shifted);
        }
    });

    unsigned_gen::<T>().test_properties(|n| {
        assert_eq!(n.arithmetic_checked_shr(U::ZERO), Some(n));
    });

    signed_gen_var_5::<U>().test_properties(|i| {
        assert_eq!(T::ZERO.arithmetic_checked_shr(i), Some(T::ZERO));
    });
}

fn arithmetic_checked_shr_properties_helper_signed_signed<
    T: ArithmeticCheckedShl<U, Output = T> + ArithmeticCheckedShr<U, Output = T> + PrimitiveSigned,
    U: PrimitiveSigned,
>()
where
    u64: ExactFrom<U>,
{
    signed_pair_gen_var_2::<T, U>().test_properties(|(n, i)| {
        let shifted = n.arithmetic_checked_shr(i);
        if shifted.is_none() {
            assert_ne!(n, T::ZERO);
        }
        if i != U::MIN {
            assert_eq!(n.arithmetic_checked_shl(-i), shifted);
        }
    });

    signed_gen::<T>().test_properties(|n| {
        assert_eq!(n.arithmetic_checked_shr(U::ZERO), Some(n));
    });

    signed_gen_var_5::<U>().test_properties(|i| {
        assert_eq!(T::ZERO.arithmetic_checked_shr(i), Some(T::ZERO));
    });
}

#[test]
fn arithmetic_checked_shr_properties() {
    apply_fn_to_unsigneds_and_signeds!(arithmetic_checked_shr_properties_helper_unsigned_signed);
    apply_fn_to_signeds_and_signeds!(arithmetic_checked_shr_properties_helper_signed_signed);
}