| // Copyright 2012 The Rust Project Developers. See the COPYRIGHT |
| // file at the top-level directory of this distribution and at |
| // http://rust-lang.org/COPYRIGHT. |
| // |
| // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or |
| // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license |
| // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your |
| // option. This file may not be copied, modified, or distributed |
| // except according to those terms. |
| |
| //! Operations and constants for `int` |
| |
| #[allow(non_uppercase_statics)]; |
| |
| use num::{BitCount, CheckedAdd, CheckedSub, CheckedMul}; |
| use option::{Option, Some, None}; |
| use unstable::intrinsics; |
| |
| pub use self::generated::*; |
| |
| #[cfg(target_word_size = "32")] pub static bits: uint = 32; |
| #[cfg(target_word_size = "64")] pub static bits: uint = 64; |
| |
| int_module!(int, super::bits) |
| |
| #[cfg(target_word_size = "32")] |
| impl BitCount for int { |
| /// Counts the number of bits set. Wraps LLVM's `ctpop` intrinsic. |
| #[inline] |
| fn population_count(&self) -> int { (*self as i32).population_count() as int } |
| |
| /// Counts the number of leading zeros. Wraps LLVM's `ctlz` intrinsic. |
| #[inline] |
| fn leading_zeros(&self) -> int { (*self as i32).leading_zeros() as int } |
| |
| /// Counts the number of trailing zeros. Wraps LLVM's `cttz` intrinsic. |
| #[inline] |
| fn trailing_zeros(&self) -> int { (*self as i32).trailing_zeros() as int } |
| } |
| |
| #[cfg(target_word_size = "64")] |
| impl BitCount for int { |
| /// Counts the number of bits set. Wraps LLVM's `ctpop` intrinsic. |
| #[inline] |
| fn population_count(&self) -> int { (*self as i64).population_count() as int } |
| |
| /// Counts the number of leading zeros. Wraps LLVM's `ctlz` intrinsic. |
| #[inline] |
| fn leading_zeros(&self) -> int { (*self as i64).leading_zeros() as int } |
| |
| /// Counts the number of trailing zeros. Wraps LLVM's `cttz` intrinsic. |
| #[inline] |
| fn trailing_zeros(&self) -> int { (*self as i64).trailing_zeros() as int } |
| } |
| |
| #[cfg(target_word_size = "32")] |
| impl CheckedAdd for int { |
| #[inline] |
| fn checked_add(&self, v: &int) -> Option<int> { |
| unsafe { |
| let (x, y) = intrinsics::i32_add_with_overflow(*self as i32, *v as i32); |
| if y { None } else { Some(x as int) } |
| } |
| } |
| } |
| |
| #[cfg(target_word_size = "64")] |
| impl CheckedAdd for int { |
| #[inline] |
| fn checked_add(&self, v: &int) -> Option<int> { |
| unsafe { |
| let (x, y) = intrinsics::i64_add_with_overflow(*self as i64, *v as i64); |
| if y { None } else { Some(x as int) } |
| } |
| } |
| } |
| |
| #[cfg(target_word_size = "32")] |
| impl CheckedSub for int { |
| #[inline] |
| fn checked_sub(&self, v: &int) -> Option<int> { |
| unsafe { |
| let (x, y) = intrinsics::i32_sub_with_overflow(*self as i32, *v as i32); |
| if y { None } else { Some(x as int) } |
| } |
| } |
| } |
| |
| #[cfg(target_word_size = "64")] |
| impl CheckedSub for int { |
| #[inline] |
| fn checked_sub(&self, v: &int) -> Option<int> { |
| unsafe { |
| let (x, y) = intrinsics::i64_sub_with_overflow(*self as i64, *v as i64); |
| if y { None } else { Some(x as int) } |
| } |
| } |
| } |
| |
| #[cfg(target_word_size = "32")] |
| impl CheckedMul for int { |
| #[inline] |
| fn checked_mul(&self, v: &int) -> Option<int> { |
| unsafe { |
| let (x, y) = intrinsics::i32_mul_with_overflow(*self as i32, *v as i32); |
| if y { None } else { Some(x as int) } |
| } |
| } |
| } |
| |
| #[cfg(target_word_size = "64")] |
| impl CheckedMul for int { |
| #[inline] |
| fn checked_mul(&self, v: &int) -> Option<int> { |
| unsafe { |
| let (x, y) = intrinsics::i64_mul_with_overflow(*self as i64, *v as i64); |
| if y { None } else { Some(x as int) } |
| } |
| } |
| } |
| |
| /// Returns `base` raised to the power of `exponent` |
| pub fn pow(base: int, exponent: uint) -> int { |
| if exponent == 0u { |
| //Not mathemtically true if ~[base == 0] |
| return 1; |
| } |
| if base == 0 { return 0; } |
| let mut my_pow = exponent; |
| let mut acc = 1; |
| let mut multiplier = base; |
| while(my_pow > 0u) { |
| if my_pow % 2u == 1u { |
| acc *= multiplier; |
| } |
| my_pow /= 2u; |
| multiplier *= multiplier; |
| } |
| return acc; |
| } |
| |
| #[test] |
| fn test_pow() { |
| assert!((pow(0, 0u) == 1)); |
| assert!((pow(0, 1u) == 0)); |
| assert!((pow(0, 2u) == 0)); |
| assert!((pow(-1, 0u) == 1)); |
| assert!((pow(1, 0u) == 1)); |
| assert!((pow(-3, 2u) == 9)); |
| assert!((pow(-3, 3u) == -27)); |
| assert!((pow(4, 9u) == 262144)); |
| } |
| |
| #[test] |
| fn test_overflows() { |
| assert!((::int::max_value > 0)); |
| assert!((::int::min_value <= 0)); |
| assert!((::int::min_value + ::int::max_value + 1 == 0)); |
| } |