Skip to content

Commit

Permalink
Merge pull request #74 from str4d/bellman-multicore
Browse files Browse the repository at this point in the history
Place bellman multicore operations behind a (default) feature flag
  • Loading branch information
str4d authored May 22, 2019
2 parents d7ba310 + 8c5cd4e commit c2d0a7d
Show file tree
Hide file tree
Showing 3 changed files with 151 additions and 83 deletions.
9 changes: 5 additions & 4 deletions bellman/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,16 +13,17 @@ rand = "0.4"
bit-vec = "0.4.4"
ff = { path = "../ff" }
futures = "0.1"
futures-cpupool = "0.1"
futures-cpupool = { version = "0.1", optional = true }
group = { path = "../group" }
num_cpus = "1"
crossbeam = "0.3"
num_cpus = { version = "1", optional = true }
crossbeam = { version = "0.3", optional = true }
pairing = { path = "../pairing", optional = true }
byteorder = "1"

[features]
groth16 = ["pairing"]
default = ["groth16"]
multicore = ["futures-cpupool", "crossbeam", "num_cpus"]
default = ["groth16", "multicore"]

[[test]]
name = "mimc"
Expand Down
11 changes: 8 additions & 3 deletions bellman/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,18 @@ extern crate group;
#[cfg(feature = "pairing")]
extern crate pairing;
extern crate rand;
extern crate num_cpus;

extern crate futures;
extern crate futures_cpupool;
extern crate bit_vec;
extern crate crossbeam;
extern crate byteorder;

#[cfg(feature = "multicore")]
extern crate crossbeam;
#[cfg(feature = "multicore")]
extern crate futures_cpupool;
#[cfg(feature = "multicore")]
extern crate num_cpus;

pub mod multicore;
mod multiexp;
pub mod domain;
Expand Down
214 changes: 138 additions & 76 deletions bellman/src/multicore.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,103 +4,165 @@
//! crossbeam but may be extended in the future to
//! allow for various parallelism strategies.
use num_cpus;
use futures::{Future, IntoFuture, Poll};
use futures_cpupool::{CpuPool, CpuFuture};
use crossbeam::{self, Scope};

#[derive(Clone)]
pub struct Worker {
cpus: usize,
pool: CpuPool
}
#[cfg(feature = "multicore")]
mod implementation {
use num_cpus;
use futures::{Future, IntoFuture, Poll};
use futures_cpupool::{CpuPool, CpuFuture};
use crossbeam::{self, Scope};

#[derive(Clone)]
pub struct Worker {
cpus: usize,
pool: CpuPool
}

impl Worker {
// We don't expose this outside the library so that
// all `Worker` instances have the same number of
// CPUs configured.
pub(crate) fn new_with_cpus(cpus: usize) -> Worker {
Worker {
cpus: cpus,
pool: CpuPool::new(cpus)
}
}

pub fn new() -> Worker {
Self::new_with_cpus(num_cpus::get())
}

pub fn log_num_cpus(&self) -> u32 {
log2_floor(self.cpus)
}

impl Worker {
// We don't expose this outside the library so that
// all `Worker` instances have the same number of
// CPUs configured.
pub(crate) fn new_with_cpus(cpus: usize) -> Worker {
Worker {
cpus: cpus,
pool: CpuPool::new(cpus)
pub fn compute<F, R>(
&self, f: F
) -> WorkerFuture<R::Item, R::Error>
where F: FnOnce() -> R + Send + 'static,
R: IntoFuture + 'static,
R::Future: Send + 'static,
R::Item: Send + 'static,
R::Error: Send + 'static
{
WorkerFuture {
future: self.pool.spawn_fn(f)
}
}

pub fn scope<'a, F, R>(
&self,
elements: usize,
f: F
) -> R
where F: FnOnce(&Scope<'a>, usize) -> R
{
let chunk_size = if elements < self.cpus {
1
} else {
elements / self.cpus
};

crossbeam::scope(|scope| {
f(scope, chunk_size)
})
}
}

pub fn new() -> Worker {
Self::new_with_cpus(num_cpus::get())
pub struct WorkerFuture<T, E> {
future: CpuFuture<T, E>
}

pub fn log_num_cpus(&self) -> u32 {
log2_floor(self.cpus)
impl<T: Send + 'static, E: Send + 'static> Future for WorkerFuture<T, E> {
type Item = T;
type Error = E;

fn poll(&mut self) -> Poll<Self::Item, Self::Error>
{
self.future.poll()
}
}

pub fn compute<F, R>(
&self, f: F
) -> WorkerFuture<R::Item, R::Error>
where F: FnOnce() -> R + Send + 'static,
R: IntoFuture + 'static,
R::Future: Send + 'static,
R::Item: Send + 'static,
R::Error: Send + 'static
{
WorkerFuture {
future: self.pool.spawn_fn(f)
fn log2_floor(num: usize) -> u32 {
assert!(num > 0);

let mut pow = 0;

while (1 << (pow+1)) <= num {
pow += 1;
}

pow
}

pub fn scope<'a, F, R>(
&self,
elements: usize,
f: F
) -> R
where F: FnOnce(&Scope<'a>, usize) -> R
{
let chunk_size = if elements < self.cpus {
1
} else {
elements / self.cpus
};

crossbeam::scope(|scope| {
f(scope, chunk_size)
})
#[test]
fn test_log2_floor() {
assert_eq!(log2_floor(1), 0);
assert_eq!(log2_floor(2), 1);
assert_eq!(log2_floor(3), 1);
assert_eq!(log2_floor(4), 2);
assert_eq!(log2_floor(5), 2);
assert_eq!(log2_floor(6), 2);
assert_eq!(log2_floor(7), 2);
assert_eq!(log2_floor(8), 3);
}
}

pub struct WorkerFuture<T, E> {
future: CpuFuture<T, E>
}
#[cfg(not(feature = "multicore"))]
mod implementation {
use futures::{future, Future, IntoFuture, Poll};

#[derive(Clone)]
pub struct Worker;

impl Worker {
pub fn new() -> Worker {
Worker
}

impl<T: Send + 'static, E: Send + 'static> Future for WorkerFuture<T, E> {
type Item = T;
type Error = E;
pub fn log_num_cpus(&self) -> u32 {
0
}

pub fn compute<F, R>(&self, f: F) -> R::Future
where
F: FnOnce() -> R + Send + 'static,
R: IntoFuture + 'static,
R::Future: Send + 'static,
R::Item: Send + 'static,
R::Error: Send + 'static,
{
f().into_future()
}

fn poll(&mut self) -> Poll<Self::Item, Self::Error>
{
self.future.poll()
pub fn scope<F, R>(&self, elements: usize, f: F) -> R
where
F: FnOnce(&DummyScope, usize) -> R,
{
f(&DummyScope, elements)
}
}
}

fn log2_floor(num: usize) -> u32 {
assert!(num > 0);
pub struct WorkerFuture<T, E> {
future: future::FutureResult<T, E>,
}

let mut pow = 0;
impl<T: Send + 'static, E: Send + 'static> Future for WorkerFuture<T, E> {
type Item = T;
type Error = E;

while (1 << (pow+1)) <= num {
pow += 1;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
self.future.poll()
}
}

pow
}
pub struct DummyScope;

#[test]
fn test_log2_floor() {
assert_eq!(log2_floor(1), 0);
assert_eq!(log2_floor(2), 1);
assert_eq!(log2_floor(3), 1);
assert_eq!(log2_floor(4), 2);
assert_eq!(log2_floor(5), 2);
assert_eq!(log2_floor(6), 2);
assert_eq!(log2_floor(7), 2);
assert_eq!(log2_floor(8), 3);
impl DummyScope {
pub fn spawn<F: FnOnce()>(&self, f: F) {
f();
}
}
}

pub use self::implementation::*;

0 comments on commit c2d0a7d

Please sign in to comment.