2023-10-04 08:30:33 +00:00
|
|
|
use std::thread;
|
2023-10-04 08:55:44 +00:00
|
|
|
use std::sync::{mpsc,Arc};
|
|
|
|
use parking_lot::{Mutex,Condvar};
|
2023-10-04 08:30:33 +00:00
|
|
|
|
2023-10-04 10:33:06 +00:00
|
|
|
//The goal here is to have a worker thread that parks itself when it runs out of work.
|
|
|
|
//The worker thread publishes the result of its work back to the worker object for every item in the work queue.
|
|
|
|
//The physics (target use case) knows when it has not changed the body, so not updating the value is also an option.
|
|
|
|
|
|
|
|
struct Worker<Task:Send,Value:Clone> {
|
2023-10-04 09:58:02 +00:00
|
|
|
sender: mpsc::Sender<Task>,
|
|
|
|
receiver: Arc<(Mutex<mpsc::Receiver<Task>>,Condvar)>,
|
2023-10-04 10:33:06 +00:00
|
|
|
value:Arc<Mutex<Value>>,
|
2023-10-04 08:30:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-04 10:33:06 +00:00
|
|
|
impl<Task:Send+'static,Value:Clone+Send+'static> Worker<Task,Value> {
|
|
|
|
fn new<F:Fn(Task)->Value+Send+'static>(value:Value,f:F) -> Self {
|
2023-10-04 09:58:02 +00:00
|
|
|
let (sender, receiver) = mpsc::channel::<Task>();
|
2023-10-04 10:33:06 +00:00
|
|
|
let ret=Self {
|
2023-10-04 09:58:02 +00:00
|
|
|
sender,
|
|
|
|
receiver:Arc::new((Mutex::new(receiver),Condvar::new())),
|
2023-10-04 10:33:06 +00:00
|
|
|
value:Arc::new(Mutex::new(value)),
|
|
|
|
};
|
|
|
|
let receiver=ret.receiver.clone();
|
|
|
|
let value=ret.value.clone();
|
2023-10-04 08:30:33 +00:00
|
|
|
thread::spawn(move || {
|
2023-10-04 09:58:02 +00:00
|
|
|
loop{
|
|
|
|
loop {
|
|
|
|
match receiver.0.lock().recv() {
|
2023-10-04 10:33:06 +00:00
|
|
|
Ok(task) => {
|
2023-10-04 09:58:02 +00:00
|
|
|
println!("Worker got a task");
|
|
|
|
// Process the task
|
2023-10-04 10:33:06 +00:00
|
|
|
*value.lock()=f(task);
|
2023-10-04 09:58:02 +00:00
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
println!("Worker stopping.",);
|
|
|
|
break;
|
|
|
|
}
|
2023-10-04 08:30:33 +00:00
|
|
|
}
|
|
|
|
}
|
2023-10-04 09:58:02 +00:00
|
|
|
receiver.1.wait(&mut receiver.0.lock());
|
2023-10-04 08:30:33 +00:00
|
|
|
}
|
|
|
|
});
|
2023-10-04 10:33:06 +00:00
|
|
|
ret
|
|
|
|
}
|
|
|
|
|
|
|
|
fn send(&self,task:Task)->Result<(), mpsc::SendError<Task>>{
|
|
|
|
let ret=self.sender.send(task);
|
|
|
|
self.receiver.1.notify_one();
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
|
|
|
|
fn grab_clone(&self)->Value{
|
|
|
|
self.value.lock().clone()
|
2023-10-04 08:30:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-04 09:22:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_worker() {
|
2023-10-04 10:33:06 +00:00
|
|
|
println!("hiiiii");
|
2023-10-04 08:57:18 +00:00
|
|
|
// Create the worker thread
|
2023-10-04 10:33:06 +00:00
|
|
|
let worker = Worker::new(crate::body::Body::with_pva(glam::Vec3::ZERO,glam::Vec3::ZERO,glam::Vec3::ZERO),
|
|
|
|
|_|crate::body::Body::with_pva(glam::Vec3::ONE,glam::Vec3::ONE,glam::Vec3::ONE)
|
|
|
|
);
|
2023-10-04 08:30:33 +00:00
|
|
|
|
2023-10-04 08:57:18 +00:00
|
|
|
// Send tasks to the worker
|
2023-10-04 08:30:33 +00:00
|
|
|
for i in 0..5 {
|
2023-10-04 09:58:02 +00:00
|
|
|
let task = crate::instruction::TimedInstruction{
|
|
|
|
time:0,
|
|
|
|
instruction:crate::body::PhysicsInstruction::StrafeTick,
|
|
|
|
};
|
|
|
|
worker.send(task).unwrap();
|
2023-10-04 08:30:33 +00:00
|
|
|
}
|
|
|
|
|
2023-10-04 08:57:18 +00:00
|
|
|
// Optional: Signal the worker to stop (in a real-world scenario)
|
2023-10-04 08:30:33 +00:00
|
|
|
// sender.send("STOP".to_string()).unwrap();
|
|
|
|
|
2023-10-04 08:57:18 +00:00
|
|
|
// Sleep to allow the worker thread to finish processing
|
2023-10-04 08:30:33 +00:00
|
|
|
thread::sleep(std::time::Duration::from_secs(2));
|
2023-10-04 08:34:24 +00:00
|
|
|
|
2023-10-04 09:58:02 +00:00
|
|
|
// Send a new task
|
|
|
|
let task = crate::instruction::TimedInstruction{
|
|
|
|
time:0,
|
|
|
|
instruction:crate::body::PhysicsInstruction::StrafeTick,
|
|
|
|
};
|
|
|
|
worker.send(task).unwrap();
|
2023-10-04 10:33:06 +00:00
|
|
|
|
|
|
|
println!("value={:?}",worker.grab_clone());
|
2023-10-04 08:34:24 +00:00
|
|
|
}
|