rust/src/libextra/par.rs

143 lines
4.3 KiB
Rust
Raw Normal View History

2013-07-02 14:47:32 -05:00
// Cloneright 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.
use std::cast;
use std::num;
use std::ptr;
use std::sys;
use std::vec;
2012-09-04 13:23:53 -05:00
use future_spawn = future::spawn;
2012-05-30 12:36:29 -05:00
/**
* The maximum number of tasks this module will spawn for a single
* operation.
*/
static MAX_TASKS : uint = 32u;
2012-05-30 12:36:29 -05:00
/// The minimum number of elements each task will process.
static MIN_GRANULARITY : uint = 1024u;
2012-05-30 12:36:29 -05:00
/**
* An internal helper to map a function over a large vector and
* return the intermediate results.
*
* This is used to build most of the other parallel vector functions,
* like map or alli.
*/
2013-07-02 14:47:32 -05:00
fn map_slices<A:Clone + Send,B:Clone + Send>(
xs: &[A],
f: &fn() -> ~fn(uint, v: &[A]) -> B)
-> ~[B] {
2012-05-30 12:36:29 -05:00
let len = xs.len();
if len < MIN_GRANULARITY {
2013-03-08 14:39:42 -06:00
info!("small slice");
2012-05-30 12:36:29 -05:00
// This is a small vector, fall back on the normal map.
~[f()(0u, xs)]
2013-07-02 14:47:32 -05:00
} else {
let num_tasks = num::min(MAX_TASKS, len / MIN_GRANULARITY);
2012-05-30 12:36:29 -05:00
let items_per_task = len / num_tasks;
let mut futures = ~[];
2012-05-30 12:36:29 -05:00
let mut base = 0u;
2013-03-08 14:39:42 -06:00
info!("spawning tasks");
2012-05-30 12:36:29 -05:00
while base < len {
let end = num::min(len, base + items_per_task);
do xs.as_imm_buf |p, _len| {
let f = f();
let base = base;
2013-02-15 01:30:30 -06:00
let f = do future_spawn() || {
2012-05-30 12:36:29 -05:00
unsafe {
let len = end - base;
let slice = (ptr::offset(p, base),
len * sys::size_of::<A>());
2013-03-08 14:39:42 -06:00
info!("pre-slice: %?", (base, slice));
let slice : &[A] =
cast::transmute(slice);
2013-05-14 04:52:12 -05:00
info!("slice: %?", (base, slice.len(), end - base));
assert_eq!(slice.len(), end - base);
f(base, slice)
2012-05-30 12:36:29 -05:00
}
};
2013-02-15 01:30:30 -06:00
futures.push(f);
2012-05-30 12:36:29 -05:00
};
base += items_per_task;
}
2013-03-08 14:39:42 -06:00
info!("tasks spawned");
2012-05-30 12:36:29 -05:00
2013-03-08 14:39:42 -06:00
info!("num_tasks: %?", (num_tasks, futures.len()));
assert_eq!(num_tasks, futures.len());
2012-05-30 12:36:29 -05:00
do futures.consume_iter().transform |ys| {
let mut ys = ys;
2012-05-30 12:36:29 -05:00
ys.get()
}.collect()
2012-05-30 12:36:29 -05:00
}
}
/// A parallel version of map.
pub fn map<A:Clone + Send,B:Clone + Send>(
xs: &[A], fn_factory: &fn() -> ~fn(&A) -> B) -> ~[B] {
2012-06-30 18:19:07 -05:00
vec::concat(map_slices(xs, || {
let f = fn_factory();
let result: ~fn(uint, &[A]) -> ~[B] =
|_, slice| slice.iter().transform(|x| f(x)).collect();
result
}))
2012-05-30 12:36:29 -05:00
}
/// A parallel version of mapi.
pub fn mapi<A:Clone + Send,B:Clone + Send>(
xs: &[A],
fn_factory: &fn() -> ~fn(uint, &A) -> B) -> ~[B] {
2012-06-30 18:19:07 -05:00
let slices = map_slices(xs, || {
let f = fn_factory();
let result: ~fn(uint, &[A]) -> ~[B] = |base, slice| {
slice.iter().enumerate().transform(|(i, x)| {
2012-09-28 15:00:25 -05:00
f(i + base, x)
}).collect()
};
result
});
let r = vec::concat(slices);
2013-03-08 14:39:42 -06:00
info!("%?", (r.len(), xs.len()));
assert_eq!(r.len(), xs.len());
r
}
/// Returns true if the function holds for all elements in the vector.
2013-07-02 14:47:32 -05:00
pub fn alli<A:Clone + Send>(
xs: &[A],
fn_factory: &fn() -> ~fn(uint, &A) -> bool) -> bool
{
let mapped = map_slices(xs, || {
let f = fn_factory();
let result: ~fn(uint, &[A]) -> bool = |base, slice| {
slice.iter().enumerate().all(|(i, x)| f(i + base, x))
};
result
});
mapped.iter().all(|&x| x)
2012-05-30 12:36:29 -05:00
}
/// Returns true if the function holds for any elements in the vector.
2013-07-02 14:47:32 -05:00
pub fn any<A:Clone + Send>(
xs: &[A],
fn_factory: &fn() -> ~fn(&A) -> bool) -> bool {
let mapped = map_slices(xs, || {
let f = fn_factory();
2013-07-04 21:13:26 -05:00
let result: ~fn(uint, &[A]) -> bool = |_, slice| slice.iter().any(f);
result
});
2013-07-04 21:13:26 -05:00
mapped.iter().any(|&x| x)
2012-05-30 12:36:29 -05:00
}