
# Objective - Some methods are stabilised with Rust 1.70 https://blog.rust-lang.org/2023/06/01/Rust-1.70.0.html#oncecell-and-oncelock ## Solution - Remove `once_cell` when possible and use std instead
141 lines
4.6 KiB
Rust
141 lines
4.6 KiB
Rust
//! Definitions for a few common task pools that we want. Generally the determining factor for what
|
|
//! kind of work should go in each pool is latency requirements.
|
|
//!
|
|
//! For CPU-intensive work (tasks that generally spin until completion) we have a standard
|
|
//! [`ComputeTaskPool`] and an [`AsyncComputeTaskPool`]. Work that does not need to be completed to
|
|
//! present the next frame should go to the [`AsyncComputeTaskPool`]
|
|
//!
|
|
//! For IO-intensive work (tasks that spend very little time in a "woken" state) we have an IO
|
|
//! task pool. The tasks here are expected to complete very quickly. Generally they should just
|
|
//! await receiving data from somewhere (i.e. disk) and signal other systems when the data is ready
|
|
//! for consumption. (likely via channels)
|
|
|
|
use super::TaskPool;
|
|
use std::{ops::Deref, sync::OnceLock};
|
|
|
|
static COMPUTE_TASK_POOL: OnceLock<ComputeTaskPool> = OnceLock::new();
|
|
static ASYNC_COMPUTE_TASK_POOL: OnceLock<AsyncComputeTaskPool> = OnceLock::new();
|
|
static IO_TASK_POOL: OnceLock<IoTaskPool> = OnceLock::new();
|
|
|
|
/// A newtype for a task pool for CPU-intensive work that must be completed to deliver the next
|
|
/// frame
|
|
#[derive(Debug)]
|
|
pub struct ComputeTaskPool(TaskPool);
|
|
|
|
impl ComputeTaskPool {
|
|
/// Initializes the global [`ComputeTaskPool`] instance.
|
|
pub fn init(f: impl FnOnce() -> TaskPool) -> &'static Self {
|
|
COMPUTE_TASK_POOL.get_or_init(|| Self(f()))
|
|
}
|
|
|
|
/// Gets the global [`ComputeTaskPool`] instance.
|
|
///
|
|
/// # Panics
|
|
/// Panics if no pool has been initialized yet.
|
|
pub fn get() -> &'static Self {
|
|
COMPUTE_TASK_POOL.get().expect(
|
|
"A ComputeTaskPool has not been initialized yet. Please call \
|
|
ComputeTaskPool::init beforehand.",
|
|
)
|
|
}
|
|
}
|
|
|
|
impl Deref for ComputeTaskPool {
|
|
type Target = TaskPool;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
/// A newtype for a task pool for CPU-intensive work that may span across multiple frames
|
|
#[derive(Debug)]
|
|
pub struct AsyncComputeTaskPool(TaskPool);
|
|
|
|
impl AsyncComputeTaskPool {
|
|
/// Initializes the global [`AsyncComputeTaskPool`] instance.
|
|
pub fn init(f: impl FnOnce() -> TaskPool) -> &'static Self {
|
|
ASYNC_COMPUTE_TASK_POOL.get_or_init(|| Self(f()))
|
|
}
|
|
|
|
/// Gets the global [`AsyncComputeTaskPool`] instance.
|
|
///
|
|
/// # Panics
|
|
/// Panics if no pool has been initialized yet.
|
|
pub fn get() -> &'static Self {
|
|
ASYNC_COMPUTE_TASK_POOL.get().expect(
|
|
"A AsyncComputeTaskPool has not been initialized yet. Please call \
|
|
AsyncComputeTaskPool::init beforehand.",
|
|
)
|
|
}
|
|
}
|
|
|
|
impl Deref for AsyncComputeTaskPool {
|
|
type Target = TaskPool;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
/// A newtype for a task pool for IO-intensive work (i.e. tasks that spend very little time in a
|
|
/// "woken" state)
|
|
#[derive(Debug)]
|
|
pub struct IoTaskPool(TaskPool);
|
|
|
|
impl IoTaskPool {
|
|
/// Initializes the global [`IoTaskPool`] instance.
|
|
pub fn init(f: impl FnOnce() -> TaskPool) -> &'static Self {
|
|
IO_TASK_POOL.get_or_init(|| Self(f()))
|
|
}
|
|
|
|
/// Gets the global [`IoTaskPool`] instance.
|
|
///
|
|
/// # Panics
|
|
/// Panics if no pool has been initialized yet.
|
|
pub fn get() -> &'static Self {
|
|
IO_TASK_POOL.get().expect(
|
|
"A IoTaskPool has not been initialized yet. Please call \
|
|
IoTaskPool::init beforehand.",
|
|
)
|
|
}
|
|
}
|
|
|
|
impl Deref for IoTaskPool {
|
|
type Target = TaskPool;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
/// A function used by `bevy_core` to tick the global tasks pools on the main thread.
|
|
/// This will run a maximum of 100 local tasks per executor per call to this function.
|
|
///
|
|
/// # Warning
|
|
///
|
|
/// This function *must* be called on the main thread, or the task pools will not be updated appropriately.
|
|
#[cfg(not(target_arch = "wasm32"))]
|
|
pub fn tick_global_task_pools_on_main_thread() {
|
|
COMPUTE_TASK_POOL
|
|
.get()
|
|
.unwrap()
|
|
.with_local_executor(|compute_local_executor| {
|
|
ASYNC_COMPUTE_TASK_POOL
|
|
.get()
|
|
.unwrap()
|
|
.with_local_executor(|async_local_executor| {
|
|
IO_TASK_POOL
|
|
.get()
|
|
.unwrap()
|
|
.with_local_executor(|io_local_executor| {
|
|
for _ in 0..100 {
|
|
compute_local_executor.try_tick();
|
|
async_local_executor.try_tick();
|
|
io_local_executor.try_tick();
|
|
}
|
|
});
|
|
});
|
|
});
|
|
}
|