aboutsummaryrefslogtreecommitdiffstats
path: root/components/util/taskpool.rs
blob: 04059e00f7e27ede1c733e55115f7a940f44e2ca (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

//! A load-balancing task pool.
//!
//! This differs in implementation from std::sync::TaskPool in that each job is
//! up for grabs by any of the child tasks in the pool.
//!

//
// This is based on the cargo task pool.
// https://github.com/rust-lang/cargo/blob/master/src/cargo/util/pool.rs
//
// The only difference is that a normal channel is used instead of a sync_channel.
//

use std::boxed::FnBox;
use std::sync::mpsc::{channel, Sender, Receiver};
use std::sync::{Arc, Mutex};
use task::spawn_named;

pub struct TaskPool {
    tx: Sender<Box<FnBox() + Send + 'static>>,
}

impl TaskPool {
    pub fn new(tasks: u32) -> TaskPool {
        assert!(tasks > 0);
        let (tx, rx) = channel();

        let state = Arc::new(Mutex::new(rx));

        for i in 0..tasks {
            let state = state.clone();
            spawn_named(
                format!("TaskPoolWorker {}/{}", i + 1, tasks),
                move || worker(&*state));
        }

        return TaskPool { tx: tx };

        fn worker(rx: &Mutex<Receiver<Box<FnBox() + Send + 'static>>>) {
            loop {
                let job = rx.lock().unwrap().recv();
                match job {
                    Ok(job) => job.call_box(()),
                    Err(..) => break,
                }
            }
        }
    }

    pub fn execute<F>(&self, job: F)
        where F: FnOnce() + Send + 'static
    {
        self.tx.send(Box::new(job)).unwrap();
    }
}