lib.rs 14.0 KB
Newer Older
1
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
2 3 4 5 6 7 8 9 10
// 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.

A
Alex Crichton 已提交
11 12 13 14 15 16 17
//! Interface to random number generators in Rust.
//!
//! This is an experimental library which lives underneath the standard library
//! in its dependency chain. This library is intended to define the interface
//! for random number generation and also provide utilities around doing so. It
//! is not recommended to use this library directly, but rather the official
//! interface through `std::rand`.
18

19 20
// Do not remove on snapshot creation. Needed for bootstrap. (Issue #22364)
#![cfg_attr(stage0, feature(custom_attribute))]
21
#![crate_name = "rand"]
22
#![crate_type = "rlib"]
23
#![doc(html_logo_url = "https://www.rust-lang.org/logos/rust-logo-128x128-blk.png",
A
Alex Crichton 已提交
24
       html_favicon_url = "https://doc.rust-lang.org/favicon.ico",
25
       html_root_url = "https://doc.rust-lang.org/nightly/",
K
Kevin Butler 已提交
26 27
       html_playground_url = "https://play.rust-lang.org/",
       test(attr(deny(warnings))))]
A
Alex Crichton 已提交
28
#![no_std]
B
Brian Anderson 已提交
29
#![staged_api]
30
#![unstable(feature = "rand",
31 32
            reason = "use `rand` from crates.io",
            issue = "27703")]
33
#![feature(core_float)]
34
#![feature(core_intrinsics)]
35
#![feature(core_slice_ext)]
36
#![feature(no_std)]
37
#![feature(num_bits_bytes)]
38
#![feature(staged_api)]
A
Aaron Turon 已提交
39
#![feature(step_by)]
40 41
#![feature(custom_attribute)]
#![allow(unused_attributes)]
42

43
#![cfg_attr(test, feature(test, rand, rustc_private, iter_order_deprecated))]
A
Alexis 已提交
44

45
#![allow(deprecated)]
A
Alex Crichton 已提交
46

M
Marcello Seri 已提交
47 48 49 50 51 52
#[cfg(test)]
#[macro_use]
extern crate std;
#[cfg(test)]
#[macro_use]
extern crate log;
A
Alex Crichton 已提交
53

54 55
use core::f64;
use core::intrinsics;
56
use core::marker::PhantomData;
57

H
Huon Wilson 已提交
58
pub use isaac::{IsaacRng, Isaac64Rng};
59
pub use chacha::ChaChaRng;
H
Huon Wilson 已提交
60

N
fallout  
Nick Cameron 已提交
61
use distributions::{Range, IndependentSample};
H
Huon Wilson 已提交
62
use distributions::range::SampleRange;
63

A
Alex Crichton 已提交
64
#[cfg(test)]
65
const RAND_BENCH_N: u64 = 100;
A
Alex Crichton 已提交
66

67
pub mod distributions;
68
pub mod isaac;
69
pub mod chacha;
70
pub mod reseeding;
71
mod rand_impls;
72

73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
// Temporary trait to implement a few floating-point routines
// needed by librand; this is necessary because librand doesn't
// depend on libstd.  This will go away when librand is integrated
// into libstd.
trait FloatMath : Sized {
    fn exp(self) -> Self;
    fn ln(self) -> Self;
    fn sqrt(self) -> Self;
    fn powf(self, n: Self) -> Self;
}

impl FloatMath for f64 {
    #[inline]
    fn exp(self) -> f64 {
        unsafe { intrinsics::expf64(self) }
    }

    #[inline]
    fn ln(self) -> f64 {
        unsafe { intrinsics::logf64(self) }
    }

    #[inline]
    fn powf(self, n: f64) -> f64 {
        unsafe { intrinsics::powf64(self, n) }
    }

    #[inline]
    fn sqrt(self) -> f64 {
        if self < 0.0 {
            f64::NAN
        } else {
            unsafe { intrinsics::sqrtf64(self) }
        }
    }
}

110
/// A type that can be randomly generated using an `Rng`.
S
Steve Klabnik 已提交
111
#[doc(hidden)]
112
pub trait Rand : Sized {
113
    /// Generates a random instance of this type using the specified source of
114
    /// randomness.
115
    fn rand<R: Rng>(rng: &mut R) -> Self;
116 117
}

118
/// A random number generator.
119
pub trait Rng : Sized {
120
    /// Return the next random u32.
121
    ///
122 123
    /// This rarely needs to be called directly, prefer `r.gen()` to
    /// `r.next_u32()`.
124 125
    // FIXME #7771: Should be implemented in terms of next_u64
    fn next_u32(&mut self) -> u32;
126

127
    /// Return the next random u64.
128 129 130
    ///
    /// By default this is implemented in terms of `next_u32`. An
    /// implementation of this trait must provide at least one of
131 132
    /// these two methods. Similarly to `next_u32`, this rarely needs
    /// to be called directly, prefer `r.gen()` to `r.next_u64()`.
133
    fn next_u64(&mut self) -> u64 {
134
        ((self.next_u32() as u64) << 32) | (self.next_u32() as u64)
135
    }
136

H
Huon Wilson 已提交
137 138 139 140 141 142 143 144 145 146 147
    /// Return the next random f32 selected from the half-open
    /// interval `[0, 1)`.
    ///
    /// By default this is implemented in terms of `next_u32`, but a
    /// random number generator which can generate numbers satisfying
    /// the requirements directly can overload this for performance.
    /// It is required that the return value lies in `[0, 1)`.
    ///
    /// See `Closed01` for the closed interval `[0,1]`, and
    /// `Open01` for the open interval `(0,1)`.
    fn next_f32(&mut self) -> f32 {
148 149
        const MANTISSA_BITS: usize = 24;
        const IGNORED_BITS: usize = 8;
H
Huon Wilson 已提交
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
        const SCALE: f32 = (1u64 << MANTISSA_BITS) as f32;

        // using any more than `MANTISSA_BITS` bits will
        // cause (e.g.) 0xffff_ffff to correspond to 1
        // exactly, so we need to drop some (8 for f32, 11
        // for f64) to guarantee the open end.
        (self.next_u32() >> IGNORED_BITS) as f32 / SCALE
    }

    /// Return the next random f64 selected from the half-open
    /// interval `[0, 1)`.
    ///
    /// By default this is implemented in terms of `next_u64`, but a
    /// random number generator which can generate numbers satisfying
    /// the requirements directly can overload this for performance.
    /// It is required that the return value lies in `[0, 1)`.
    ///
    /// See `Closed01` for the closed interval `[0,1]`, and
    /// `Open01` for the open interval `(0,1)`.
    fn next_f64(&mut self) -> f64 {
170 171
        const MANTISSA_BITS: usize = 53;
        const IGNORED_BITS: usize = 11;
H
Huon Wilson 已提交
172 173 174 175 176
        const SCALE: f64 = (1u64 << MANTISSA_BITS) as f64;

        (self.next_u64() >> IGNORED_BITS) as f64 / SCALE
    }

177 178 179
    /// Fill `dest` with random data.
    ///
    /// This has a default implementation in terms of `next_u64` and
H
Huon Wilson 已提交
180
    /// `next_u32`, but should be overridden by implementations that
181 182 183 184 185 186 187 188 189 190
    /// offer a more efficient solution than just calling those
    /// methods repeatedly.
    ///
    /// This method does *not* have a requirement to bear any fixed
    /// relationship to the other methods, for example, it does *not*
    /// have to result in the same output as progressively filling
    /// `dest` with `self.gen::<u8>()`, and any such behaviour should
    /// not be relied upon.
    ///
    /// This method should guarantee that `dest` is entirely filled
S
Steve Klabnik 已提交
191
    /// with new data, and may panic if this is impossible
192 193
    /// (e.g. reading past the end of a file that is being used as the
    /// source of randomness).
194
    fn fill_bytes(&mut self, dest: &mut [u8]) {
195 196 197 198 199 200
        // this could, in theory, be done by transmuting dest to a
        // [u64], but this is (1) likely to be undefined behaviour for
        // LLVM, (2) has to be very careful about alignment concerns,
        // (3) adds more `unsafe` that needs to be checked, (4)
        // probably doesn't give much performance gain if
        // optimisations are on.
T
Tobias Bucher 已提交
201
        let mut count = 0;
202
        let mut num = 0;
203
        for byte in dest {
204 205 206 207 208 209
            if count == 0 {
                // we could micro-optimise here by generating a u32 if
                // we only need a few more bytes to fill the vector
                // (i.e. at most 4).
                num = self.next_u64();
                count = 8;
210
            }
211 212 213 214

            *byte = (num & 0xff) as u8;
            num >>= 8;
            count -= 1;
215 216
        }
    }
217

218
    /// Return a random value of a `Rand` type.
219
    #[inline(always)]
220
    fn gen<T: Rand>(&mut self) -> T {
221
        Rand::rand(self)
222
    }
B
Brian Anderson 已提交
223

224
    /// Return an iterator that will yield an infinite number of randomly
A
Alex Crichton 已提交
225 226
    /// generated items.
    fn gen_iter<'a, T: Rand>(&'a mut self) -> Generator<'a, T, Self> {
M
Marcello Seri 已提交
227 228 229 230
        Generator {
            rng: self,
            _marker: PhantomData,
        }
231 232
    }

233
    /// Generate a random value in the range [`low`, `high`).
234
    ///
235 236 237 238 239
    /// This is a convenience wrapper around
    /// `distributions::Range`. If this function will be called
    /// repeatedly with the same arguments, one should use `Range`, as
    /// that will amortize the computations that allow for perfect
    /// uniformity, as they only happen on initialization.
240
    ///
241 242 243
    /// # Panics
    ///
    /// Panics if `low >= high`.
244
    fn gen_range<T: PartialOrd + SampleRange>(&mut self, low: T, high: T) -> T {
245
        assert!(low < high, "Rng.gen_range called with low >= high");
N
fallout  
Nick Cameron 已提交
246
        Range::new(low, high).ind_sample(self)
247 248
    }

249
    /// Return a bool with a 1 in n chance of true
250
    fn gen_weighted_bool(&mut self, n: usize) -> bool {
251
        n <= 1 || self.gen_range(0, n) == 0
252 253
    }

A
Alex Crichton 已提交
254 255 256
    /// Return an iterator of random characters from the set A-Z,a-z,0-9.
    fn gen_ascii_chars<'a>(&'a mut self) -> AsciiGenerator<'a, Self> {
        AsciiGenerator { rng: self }
257
    }
258

259 260 261 262
    /// Return a random element from `values`.
    ///
    /// Return `None` if `values` is empty.
    fn choose<'a, T>(&mut self, values: &'a [T]) -> Option<&'a T> {
263
        if values.is_empty() {
B
Brian Anderson 已提交
264
            None
265
        } else {
A
Alfie John 已提交
266
            Some(&values[self.gen_range(0, values.len())])
267 268 269
        }
    }

270 271
    /// Shuffle a mutable slice in place.
    fn shuffle<T>(&mut self, values: &mut [T]) {
272
        let mut i = values.len();
A
Alfie John 已提交
273
        while i >= 2 {
274
            // invariant: elements with index >= i have been locked in place.
A
Alfie John 已提交
275
            i -= 1;
276
            // lock element i in place.
A
Alfie John 已提交
277
            values.swap(i, self.gen_range(0, i + 1));
278 279
        }
    }
A
Alex Crichton 已提交
280
}
281

A
Alex Crichton 已提交
282 283 284
/// Iterator which will generate a stream of random items.
///
/// This iterator is created via the `gen_iter` method on `Rng`.
M
Marcello Seri 已提交
285
pub struct Generator<'a, T, R: 'a> {
A
Alex Crichton 已提交
286
    rng: &'a mut R,
M
Marcello Seri 已提交
287
    _marker: PhantomData<T>,
A
Alex Crichton 已提交
288
}
289

J
Jorge Aparicio 已提交
290 291 292
impl<'a, T: Rand, R: Rng> Iterator for Generator<'a, T, R> {
    type Item = T;

A
Alex Crichton 已提交
293 294 295 296 297 298 299 300
    fn next(&mut self) -> Option<T> {
        Some(self.rng.gen())
    }
}

/// Iterator which will continuously generate random ascii characters.
///
/// This iterator is created via the `gen_ascii_chars` method on `Rng`.
M
Marcello Seri 已提交
301
pub struct AsciiGenerator<'a, R: 'a> {
A
Alex Crichton 已提交
302 303 304
    rng: &'a mut R,
}

J
Jorge Aparicio 已提交
305 306 307
impl<'a, R: Rng> Iterator for AsciiGenerator<'a, R> {
    type Item = char;

A
Alex Crichton 已提交
308
    fn next(&mut self) -> Option<char> {
309
        const GEN_ASCII_STR_CHARSET: &'static [u8] =
S
Simon Sapin 已提交
310 311 312
            b"ABCDEFGHIJKLMNOPQRSTUVWXYZ\
              abcdefghijklmnopqrstuvwxyz\
              0123456789";
A
Alex Crichton 已提交
313
        Some(*self.rng.choose(GEN_ASCII_STR_CHARSET).unwrap() as char)
314
    }
315
}
316

317 318 319 320 321 322 323 324 325 326
/// A random number generator that can be explicitly seeded to produce
/// the same stream of randomness multiple times.
pub trait SeedableRng<Seed>: Rng {
    /// Reseed an RNG with the given seed.
    fn reseed(&mut self, Seed);

    /// Create a new RNG with the given seed.
    fn from_seed(seed: Seed) -> Self;
}

327 328
/// An Xorshift[1] random number
/// generator.
329 330 331
///
/// The Xorshift algorithm is not suitable for cryptographic purposes
/// but is very fast. If you do not know for sure that it fits your
P
Piotr Jawniak 已提交
332
/// requirements, use a more secure one such as `IsaacRng` or `OsRng`.
333 334 335 336
///
/// [1]: Marsaglia, George (July 2003). ["Xorshift
/// RNGs"](http://www.jstatsoft.org/v08/i14/paper). *Journal of
/// Statistical Software*. Vol. 8 (Issue 14).
A
Alex Crichton 已提交
337
#[derive(Clone)]
338
pub struct XorShiftRng {
339 340 341 342
    x: u32,
    y: u32,
    z: u32,
    w: u32,
343
}
344

A
Alex Crichton 已提交
345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361
impl XorShiftRng {
    /// Creates a new XorShiftRng instance which is not seeded.
    ///
    /// The initial values of this RNG are constants, so all generators created
    /// by this function will yield the same stream of random numbers. It is
    /// highly recommended that this is created through `SeedableRng` instead of
    /// this function
    pub fn new_unseeded() -> XorShiftRng {
        XorShiftRng {
            x: 0x193a6754,
            y: 0xa8a7d469,
            z: 0x97830e05,
            w: 0x113ba7bb,
        }
    }
}

362
impl Rng for XorShiftRng {
363
    #[inline]
364
    fn next_u32(&mut self) -> u32 {
365
        let x = self.x;
366
        let t = x ^ (x << 11);
367 368 369 370 371 372 373 374 375
        self.x = self.y;
        self.y = self.z;
        self.z = self.w;
        let w = self.w;
        self.w = w ^ (w >> 19) ^ (t ^ (t >> 8));
        self.w
    }
}

N
Nick Cameron 已提交
376
impl SeedableRng<[u32; 4]> for XorShiftRng {
S
Steve Klabnik 已提交
377
    /// Reseed an XorShiftRng. This will panic if `seed` is entirely 0.
N
Nick Cameron 已提交
378
    fn reseed(&mut self, seed: [u32; 4]) {
379 380 381 382 383 384 385 386 387
        assert!(!seed.iter().all(|&x| x == 0),
                "XorShiftRng.reseed called with an all zero seed.");

        self.x = seed[0];
        self.y = seed[1];
        self.z = seed[2];
        self.w = seed[3];
    }

S
Steve Klabnik 已提交
388
    /// Create a new XorShiftRng. This will panic if `seed` is entirely 0.
N
Nick Cameron 已提交
389
    fn from_seed(seed: [u32; 4]) -> XorShiftRng {
390 391 392 393 394 395 396
        assert!(!seed.iter().all(|&x| x == 0),
                "XorShiftRng::from_seed called with an all zero seed.");

        XorShiftRng {
            x: seed[0],
            y: seed[1],
            z: seed[2],
M
Marcello Seri 已提交
397
            w: seed[3],
398 399 400 401
        }
    }
}

A
Alex Crichton 已提交
402 403 404 405 406
impl Rand for XorShiftRng {
    fn rand<R: Rng>(rng: &mut R) -> XorShiftRng {
        let mut tuple: (u32, u32, u32, u32) = rng.gen();
        while tuple == (0, 0, 0, 0) {
            tuple = rng.gen();
407
        }
A
Alex Crichton 已提交
408
        let (x, y, z, w) = tuple;
M
Marcello Seri 已提交
409 410 411 412 413 414
        XorShiftRng {
            x: x,
            y: y,
            z: z,
            w: w,
        }
415 416
    }
}
417

418 419 420 421 422 423
/// A wrapper for generating floating point numbers uniformly in the
/// open interval `(0,1)` (not including either endpoint).
///
/// Use `Closed01` for the closed interval `[0,1]`, and the default
/// `Rand` implementation for `f32` and `f64` for the half-open
/// `[0,1)`.
424
pub struct Open01<F>(pub F);
425 426 427 428 429 430 431

/// A wrapper for generating floating point numbers uniformly in the
/// closed interval `[0,1]` (including both endpoints).
///
/// Use `Open01` for the closed interval `(0,1)`, and the default
/// `Rand` implementation of `f32` and `f64` for the half-open
/// `[0,1)`.
432
pub struct Closed01<F>(pub F);
433

K
Kiet Tran 已提交
434
#[cfg(test)]
A
Alex Crichton 已提交
435
mod test {
436
    use std::__rand as rand;
437

M
Marcello Seri 已提交
438 439 440
    pub struct MyRng<R> {
        inner: R,
    }
441

A
Alex Crichton 已提交
442 443
    impl<R: rand::Rng> ::Rng for MyRng<R> {
        fn next_u32(&mut self) -> u32 {
444
            rand::Rng::next_u32(&mut self.inner)
A
Alex Crichton 已提交
445
        }
446 447
    }

S
Simonas Kazlauskas 已提交
448
    pub fn rng() -> MyRng<rand::ThreadRng> {
A
Alex Crichton 已提交
449
        MyRng { inner: rand::thread_rng() }
450 451
    }

452
    pub fn weak_rng() -> MyRng<rand::ThreadRng> {
A
Alex Crichton 已提交
453
        MyRng { inner: rand::thread_rng() }
454
    }
455
}