2012-12-03 18:48:01 -06:00
|
|
|
// 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.
|
|
|
|
|
2012-12-23 16:41:37 -06:00
|
|
|
use core::cast;
|
2013-01-08 21:37:25 -06:00
|
|
|
use core::prelude::*;
|
2012-12-23 16:41:37 -06:00
|
|
|
use core::ptr;
|
|
|
|
use core::sys;
|
|
|
|
use core::uint;
|
|
|
|
use core::vec;
|
|
|
|
|
2012-09-04 13:23:53 -05:00
|
|
|
use future_spawn = future::spawn;
|
2012-05-30 12:36:29 -05:00
|
|
|
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* The maximum number of tasks this module will spawn for a single
|
|
|
|
* operation.
|
|
|
|
*/
|
2013-03-22 16:00:15 -05:00
|
|
|
static max_tasks : uint = 32u;
|
2012-05-30 12:36:29 -05:00
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// The minimum number of elements each task will process.
|
2013-03-22 16:00:15 -05:00
|
|
|
static min_granularity : uint = 1024u;
|
2012-05-30 12:36:29 -05:00
|
|
|
|
2012-07-04 16:53:12 -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-02-20 19:07:17 -06:00
|
|
|
fn map_slices<A:Copy + Owned,B:Copy + Owned>(
|
2012-08-28 17:54:45 -05:00
|
|
|
xs: &[A],
|
2013-01-31 19:12:29 -06:00
|
|
|
f: &fn() -> ~fn(uint, v: &[A]) -> B)
|
2012-06-29 18:26:56 -05:00
|
|
|
-> ~[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.
|
2012-06-29 18:26:56 -05:00
|
|
|
~[f()(0u, xs)]
|
2012-05-30 12:36:29 -05:00
|
|
|
}
|
|
|
|
else {
|
2012-08-30 14:54:50 -05:00
|
|
|
let num_tasks = uint::min(max_tasks, len / min_granularity);
|
2012-05-30 12:36:29 -05:00
|
|
|
|
|
|
|
let items_per_task = len / num_tasks;
|
|
|
|
|
2012-06-29 18:26:56 -05:00
|
|
|
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 {
|
2012-08-30 14:54:50 -05:00
|
|
|
let end = uint::min(len, base + items_per_task);
|
2012-09-13 13:46:10 -05:00
|
|
|
do vec::as_imm_buf(xs) |p, _len| {
|
2012-05-30 17:18:45 -05:00
|
|
|
let f = f();
|
2013-01-10 12:59:58 -06:00
|
|
|
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));
|
2012-06-29 18:26:56 -05:00
|
|
|
let slice : &[A] =
|
2012-09-18 19:34:08 -05:00
|
|
|
cast::reinterpret_cast(&slice);
|
2013-03-08 14:39:42 -06:00
|
|
|
info!("slice: %?",
|
|
|
|
(base, vec::len(slice), end - base));
|
2013-03-06 15:58:02 -06:00
|
|
|
fail_unless!((vec::len(slice) == end - base));
|
2012-05-30 17:18:45 -05:00
|
|
|
f(base, slice)
|
2012-05-30 12:36:29 -05:00
|
|
|
}
|
2012-06-25 12:55: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()));
|
2013-03-06 15:58:02 -06:00
|
|
|
fail_unless!((num_tasks == futures.len()));
|
2012-05-30 12:36:29 -05:00
|
|
|
|
2012-06-30 18:19:07 -05:00
|
|
|
let r = do futures.map() |ys| {
|
2012-05-30 12:36:29 -05:00
|
|
|
ys.get()
|
|
|
|
};
|
2013-03-06 15:58:02 -06:00
|
|
|
fail_unless!((r.len() == futures.len()));
|
2012-05-30 12:36:29 -05:00
|
|
|
r
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// A parallel version of map.
|
2013-02-20 19:07:17 -06:00
|
|
|
pub fn map<A:Copy + Owned,B:Copy + Owned>(
|
2013-01-31 19:12:29 -06:00
|
|
|
xs: &[A], fn_factory: &fn() -> ~fn(&A) -> B) -> ~[B] {
|
2012-06-30 18:19:07 -05:00
|
|
|
vec::concat(map_slices(xs, || {
|
2013-01-31 19:12:29 -06:00
|
|
|
let f = fn_factory();
|
2013-03-01 16:15:15 -06:00
|
|
|
let result: ~fn(uint, &[A]) -> ~[B] =
|
|
|
|
|_, slice| vec::map(slice, |x| f(x));
|
|
|
|
result
|
2012-06-26 15:55:56 -05:00
|
|
|
}))
|
2012-05-30 12:36:29 -05:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// A parallel version of mapi.
|
2013-02-20 19:07:17 -06:00
|
|
|
pub fn mapi<A:Copy + Owned,B:Copy + Owned>(
|
2013-03-01 16:15:15 -06:00
|
|
|
xs: &[A],
|
|
|
|
fn_factory: &fn() -> ~fn(uint, &A) -> B) -> ~[B] {
|
2012-06-30 18:19:07 -05:00
|
|
|
let slices = map_slices(xs, || {
|
2013-01-31 19:12:29 -06:00
|
|
|
let f = fn_factory();
|
2013-03-01 16:15:15 -06:00
|
|
|
let result: ~fn(uint, &[A]) -> ~[B] = |base, slice| {
|
2012-06-30 18:19:07 -05:00
|
|
|
vec::mapi(slice, |i, x| {
|
2012-09-28 15:00:25 -05:00
|
|
|
f(i + base, x)
|
2012-06-26 15:55:56 -05:00
|
|
|
})
|
2013-03-01 16:15:15 -06:00
|
|
|
};
|
|
|
|
result
|
2012-06-26 15:55:56 -05:00
|
|
|
});
|
2012-05-30 17:18:45 -05:00
|
|
|
let r = vec::concat(slices);
|
2013-03-08 14:39:42 -06:00
|
|
|
info!("%?", (r.len(), xs.len()));
|
2013-03-06 15:58:02 -06:00
|
|
|
fail_unless!((r.len() == xs.len()));
|
2012-05-30 17:18:45 -05:00
|
|
|
r
|
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns true if the function holds for all elements in the vector.
|
2013-02-20 19:07:17 -06:00
|
|
|
pub fn alli<A:Copy + Owned>(
|
2013-01-31 19:12:29 -06:00
|
|
|
xs: &[A],
|
|
|
|
fn_factory: &fn() -> ~fn(uint, &A) -> bool) -> bool
|
|
|
|
{
|
2012-06-30 18:19:07 -05:00
|
|
|
do vec::all(map_slices(xs, || {
|
2013-01-31 19:12:29 -06:00
|
|
|
let f = fn_factory();
|
2013-03-01 16:15:15 -06:00
|
|
|
let result: ~fn(uint, &[A]) -> bool = |base, slice| {
|
2012-06-30 18:19:07 -05:00
|
|
|
vec::alli(slice, |i, x| {
|
2012-09-28 15:00:25 -05:00
|
|
|
f(i + base, x)
|
2012-06-26 15:55:56 -05:00
|
|
|
})
|
2013-03-01 16:15:15 -06:00
|
|
|
};
|
|
|
|
result
|
2012-09-28 00:20:47 -05:00
|
|
|
})) |x| { *x }
|
2012-05-30 12:36:29 -05:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns true if the function holds for any elements in the vector.
|
2013-02-20 19:07:17 -06:00
|
|
|
pub fn any<A:Copy + Owned>(
|
2013-01-31 19:12:29 -06:00
|
|
|
xs: &[A],
|
|
|
|
fn_factory: &fn() -> ~fn(&A) -> bool) -> bool {
|
2012-06-30 18:19:07 -05:00
|
|
|
do vec::any(map_slices(xs, || {
|
2013-01-31 19:12:29 -06:00
|
|
|
let f = fn_factory();
|
2013-03-01 16:15:15 -06:00
|
|
|
let result: ~fn(uint, &[A]) -> bool =
|
|
|
|
|_, slice| vec::any(slice, |x| f(x));
|
|
|
|
result
|
2012-09-28 00:20:47 -05:00
|
|
|
})) |x| { *x }
|
2012-05-30 12:36:29 -05:00
|
|
|
}
|