2017cc3ce4
Like next_power_of_two, but returns None on overflow.
250 lines
7.0 KiB
Rust
250 lines
7.0 KiB
Rust
// 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 `uint`
|
|
|
|
use num;
|
|
use num::{CheckedAdd, CheckedSub, CheckedMul};
|
|
use option::{Option, Some, None};
|
|
use unstable::intrinsics;
|
|
use sys;
|
|
|
|
pub use self::generated::*;
|
|
|
|
uint_module!(uint, int, ::int::bits)
|
|
|
|
///
|
|
/// Divide two numbers, return the result, rounded up.
|
|
///
|
|
/// # Arguments
|
|
///
|
|
/// * x - an integer
|
|
/// * y - an integer distinct from 0u
|
|
///
|
|
/// # Return value
|
|
///
|
|
/// The smallest integer `q` such that `x/y <= q`.
|
|
///
|
|
pub fn div_ceil(x: uint, y: uint) -> uint {
|
|
let div = x / y;
|
|
if x % y == 0u { div }
|
|
else { div + 1u }
|
|
}
|
|
|
|
///
|
|
/// Divide two numbers, return the result, rounded to the closest integer.
|
|
///
|
|
/// # Arguments
|
|
///
|
|
/// * x - an integer
|
|
/// * y - an integer distinct from 0u
|
|
///
|
|
/// # Return value
|
|
///
|
|
/// The integer `q` closest to `x/y`.
|
|
///
|
|
pub fn div_round(x: uint, y: uint) -> uint {
|
|
let div = x / y;
|
|
if x % y * 2u < y { div }
|
|
else { div + 1u }
|
|
}
|
|
|
|
///
|
|
/// Divide two numbers, return the result, rounded down.
|
|
///
|
|
/// Note: This is the same function as `div`.
|
|
///
|
|
/// # Arguments
|
|
///
|
|
/// * x - an integer
|
|
/// * y - an integer distinct from 0u
|
|
///
|
|
/// # Return value
|
|
///
|
|
/// The smallest integer `q` such that `x/y <= q`. This
|
|
/// is either `x/y` or `x/y + 1`.
|
|
///
|
|
pub fn div_floor(x: uint, y: uint) -> uint { return x / y; }
|
|
|
|
impl num::Times for uint {
|
|
#[inline]
|
|
///
|
|
/// A convenience form for basic repetition. Given a uint `x`,
|
|
/// `do x.times { ... }` executes the given block x times.
|
|
///
|
|
/// Equivalent to `for uint::range(0, x) |_| { ... }`.
|
|
///
|
|
/// Not defined on all integer types to permit unambiguous
|
|
/// use with integer literals of inferred integer-type as
|
|
/// the self-value (eg. `do 100.times { ... }`).
|
|
///
|
|
fn times(&self, it: &fn()) {
|
|
let mut i = *self;
|
|
while i > 0 {
|
|
it();
|
|
i -= 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Returns the smallest power of 2 greater than or equal to `n`
|
|
#[inline]
|
|
pub fn next_power_of_two(n: uint) -> uint {
|
|
let halfbits: uint = sys::size_of::<uint>() * 4u;
|
|
let mut tmp: uint = n - 1u;
|
|
let mut shift: uint = 1u;
|
|
while shift <= halfbits { tmp |= tmp >> shift; shift <<= 1u; }
|
|
tmp + 1u
|
|
}
|
|
|
|
/// Returns the smallest power of 2 greater than or equal to `n`
|
|
#[inline]
|
|
pub fn next_power_of_two_opt(n: uint) -> Option<uint> {
|
|
let halfbits: uint = sys::size_of::<uint>() * 4u;
|
|
let mut tmp: uint = n - 1u;
|
|
let mut shift: uint = 1u;
|
|
while shift <= halfbits { tmp |= tmp >> shift; shift <<= 1u; }
|
|
tmp.checked_add(&1)
|
|
}
|
|
|
|
#[cfg(target_word_size = "32")]
|
|
impl CheckedAdd for uint {
|
|
#[inline]
|
|
fn checked_add(&self, v: &uint) -> Option<uint> {
|
|
unsafe {
|
|
let (x, y) = intrinsics::u32_add_with_overflow(*self as u32, *v as u32);
|
|
if y { None } else { Some(x as uint) }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(target_word_size = "64")]
|
|
impl CheckedAdd for uint {
|
|
#[inline]
|
|
fn checked_add(&self, v: &uint) -> Option<uint> {
|
|
unsafe {
|
|
let (x, y) = intrinsics::u64_add_with_overflow(*self as u64, *v as u64);
|
|
if y { None } else { Some(x as uint) }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(target_word_size = "32")]
|
|
impl CheckedSub for uint {
|
|
#[inline]
|
|
fn checked_sub(&self, v: &uint) -> Option<uint> {
|
|
unsafe {
|
|
let (x, y) = intrinsics::u32_sub_with_overflow(*self as u32, *v as u32);
|
|
if y { None } else { Some(x as uint) }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(target_word_size = "64")]
|
|
impl CheckedSub for uint {
|
|
#[inline]
|
|
fn checked_sub(&self, v: &uint) -> Option<uint> {
|
|
unsafe {
|
|
let (x, y) = intrinsics::u64_sub_with_overflow(*self as u64, *v as u64);
|
|
if y { None } else { Some(x as uint) }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(target_word_size = "32")]
|
|
impl CheckedMul for uint {
|
|
#[inline]
|
|
fn checked_mul(&self, v: &uint) -> Option<uint> {
|
|
unsafe {
|
|
let (x, y) = intrinsics::u32_mul_with_overflow(*self as u32, *v as u32);
|
|
if y { None } else { Some(x as uint) }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(target_word_size = "64")]
|
|
impl CheckedMul for uint {
|
|
#[inline]
|
|
fn checked_mul(&self, v: &uint) -> Option<uint> {
|
|
unsafe {
|
|
let (x, y) = intrinsics::u64_mul_with_overflow(*self as u64, *v as u64);
|
|
if y { None } else { Some(x as uint) }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_next_power_of_two() {
|
|
assert!((next_power_of_two(0u) == 0u));
|
|
assert!((next_power_of_two(1u) == 1u));
|
|
assert!((next_power_of_two(2u) == 2u));
|
|
assert!((next_power_of_two(3u) == 4u));
|
|
assert!((next_power_of_two(4u) == 4u));
|
|
assert!((next_power_of_two(5u) == 8u));
|
|
assert!((next_power_of_two(6u) == 8u));
|
|
assert!((next_power_of_two(7u) == 8u));
|
|
assert!((next_power_of_two(8u) == 8u));
|
|
assert!((next_power_of_two(9u) == 16u));
|
|
assert!((next_power_of_two(10u) == 16u));
|
|
assert!((next_power_of_two(11u) == 16u));
|
|
assert!((next_power_of_two(12u) == 16u));
|
|
assert!((next_power_of_two(13u) == 16u));
|
|
assert!((next_power_of_two(14u) == 16u));
|
|
assert!((next_power_of_two(15u) == 16u));
|
|
assert!((next_power_of_two(16u) == 16u));
|
|
assert!((next_power_of_two(17u) == 32u));
|
|
assert!((next_power_of_two(18u) == 32u));
|
|
assert!((next_power_of_two(19u) == 32u));
|
|
assert!((next_power_of_two(20u) == 32u));
|
|
assert!((next_power_of_two(21u) == 32u));
|
|
assert!((next_power_of_two(22u) == 32u));
|
|
assert!((next_power_of_two(23u) == 32u));
|
|
assert!((next_power_of_two(24u) == 32u));
|
|
assert!((next_power_of_two(25u) == 32u));
|
|
assert!((next_power_of_two(26u) == 32u));
|
|
assert!((next_power_of_two(27u) == 32u));
|
|
assert!((next_power_of_two(28u) == 32u));
|
|
assert!((next_power_of_two(29u) == 32u));
|
|
assert!((next_power_of_two(30u) == 32u));
|
|
assert!((next_power_of_two(31u) == 32u));
|
|
assert!((next_power_of_two(32u) == 32u));
|
|
assert!((next_power_of_two(33u) == 64u));
|
|
assert!((next_power_of_two(34u) == 64u));
|
|
assert!((next_power_of_two(35u) == 64u));
|
|
assert!((next_power_of_two(36u) == 64u));
|
|
assert!((next_power_of_two(37u) == 64u));
|
|
assert!((next_power_of_two(38u) == 64u));
|
|
assert!((next_power_of_two(39u) == 64u));
|
|
}
|
|
|
|
#[test]
|
|
fn test_overflows() {
|
|
use uint;
|
|
assert!((uint::max_value > 0u));
|
|
assert!((uint::min_value <= 0u));
|
|
assert!((uint::min_value + uint::max_value + 1u == 0u));
|
|
}
|
|
|
|
#[test]
|
|
fn test_div() {
|
|
assert!((div_floor(3u, 4u) == 0u));
|
|
assert!((div_ceil(3u, 4u) == 1u));
|
|
assert!((div_round(3u, 4u) == 1u));
|
|
}
|
|
|
|
#[test]
|
|
pub fn test_times() {
|
|
use num::Times;
|
|
let ten = 10 as uint;
|
|
let mut accum = 0;
|
|
do ten.times { accum += 1; }
|
|
assert!((accum == 10));
|
|
}
|