123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132 |
- use int::{DInt, HInt, Int};
- trait Ashl: DInt {
- /// Returns `a << b`, requires `b < Self::BITS`
- fn ashl(self, shl: u32) -> Self {
- let n_h = Self::H::BITS;
- if shl & n_h != 0 {
- // we only need `self.lo()` because `self.hi()` will be shifted out entirely
- (self.lo() << (shl - n_h)).widen_hi()
- } else if shl == 0 {
- self
- } else {
- Self::from_lo_hi(
- self.lo() << shl,
- self.lo().logical_shr(n_h - shl) | (self.hi() << shl),
- )
- }
- }
- }
- impl Ashl for u32 {}
- impl Ashl for u64 {}
- impl Ashl for u128 {}
- trait Ashr: DInt {
- /// Returns arithmetic `a >> b`, requires `b < Self::BITS`
- fn ashr(self, shr: u32) -> Self {
- let n_h = Self::H::BITS;
- if shr & n_h != 0 {
- Self::from_lo_hi(
- self.hi() >> (shr - n_h),
- // smear the sign bit
- self.hi() >> (n_h - 1),
- )
- } else if shr == 0 {
- self
- } else {
- Self::from_lo_hi(
- self.lo().logical_shr(shr) | (self.hi() << (n_h - shr)),
- self.hi() >> shr,
- )
- }
- }
- }
- impl Ashr for i32 {}
- impl Ashr for i64 {}
- impl Ashr for i128 {}
- trait Lshr: DInt {
- /// Returns logical `a >> b`, requires `b < Self::BITS`
- fn lshr(self, shr: u32) -> Self {
- let n_h = Self::H::BITS;
- if shr & n_h != 0 {
- self.hi().logical_shr(shr - n_h).zero_widen()
- } else if shr == 0 {
- self
- } else {
- Self::from_lo_hi(
- self.lo().logical_shr(shr) | (self.hi() << (n_h - shr)),
- self.hi().logical_shr(shr),
- )
- }
- }
- }
- impl Lshr for u32 {}
- impl Lshr for u64 {}
- impl Lshr for u128 {}
- intrinsics! {
- #[maybe_use_optimized_c_shim]
- pub extern "C" fn __ashlsi3(a: u32, b: u32) -> u32 {
- a.ashl(b)
- }
- #[maybe_use_optimized_c_shim]
- #[arm_aeabi_alias = __aeabi_llsl]
- pub extern "C" fn __ashldi3(a: u64, b: u32) -> u64 {
- a.ashl(b)
- }
- pub extern "C" fn __ashlti3(a: u128, b: u32) -> u128 {
- a.ashl(b)
- }
- #[maybe_use_optimized_c_shim]
- pub extern "C" fn __ashrsi3(a: i32, b: u32) -> i32 {
- a.ashr(b)
- }
- #[maybe_use_optimized_c_shim]
- #[arm_aeabi_alias = __aeabi_lasr]
- pub extern "C" fn __ashrdi3(a: i64, b: u32) -> i64 {
- a.ashr(b)
- }
- pub extern "C" fn __ashrti3(a: i128, b: u32) -> i128 {
- a.ashr(b)
- }
- #[maybe_use_optimized_c_shim]
- pub extern "C" fn __lshrsi3(a: u32, b: u32) -> u32 {
- a.lshr(b)
- }
- #[maybe_use_optimized_c_shim]
- #[arm_aeabi_alias = __aeabi_llsr]
- pub extern "C" fn __lshrdi3(a: u64, b: u32) -> u64 {
- a.lshr(b)
- }
- pub extern "C" fn __lshrti3(a: u128, b: u32) -> u128 {
- a.lshr(b)
- }
- pub extern "C" fn __rust_i128_shlo(a: i128, b: u128) -> (i128, bool) {
- (__ashlti3(a as _, b as _) as _, b >= 128)
- }
- pub extern "C" fn __rust_u128_shlo(a: u128, b: u128) -> (u128, bool) {
- (__ashlti3(a, b as _), b >= 128)
- }
- pub extern "C" fn __rust_i128_shro(a: i128, b: u128) -> (i128, bool) {
- (__ashrti3(a, b as _), b >= 128)
- }
- pub extern "C" fn __rust_u128_shro(a: u128, b: u128) -> (u128, bool) {
- (__lshrti3(a, b as _), b >= 128)
- }
- }
|