bevy/crates/bevy_ecs/src/schedule/schedule.rs

196 lines
6.7 KiB
Rust
Raw Normal View History

2020-07-10 04:18:35 +00:00
use crate::{
2020-07-26 19:10:18 +00:00
resource::Resources,
schedule::ParallelExecutorOptions,
2020-07-26 19:10:18 +00:00
system::{System, SystemId, ThreadLocalExecution},
2020-07-10 04:18:35 +00:00
};
2020-08-10 00:58:56 +00:00
use bevy_hecs::World;
2020-07-10 04:18:35 +00:00
use std::{
borrow::Cow,
2020-07-17 00:23:50 +00:00
collections::{HashMap, HashSet},
sync::{Arc, Mutex},
2020-07-10 04:18:35 +00:00
};
/// An ordered collection of stages, which each contain an ordered list of [System]s.
/// Schedules are essentially the "execution plan" for an App's systems.
/// They are run on a given [World] and [Resources] reference.
2020-07-10 04:18:35 +00:00
#[derive(Default)]
pub struct Schedule {
pub(crate) stages: HashMap<Cow<'static, str>, Vec<Arc<Mutex<Box<dyn System>>>>>,
pub(crate) stage_order: Vec<Cow<'static, str>>,
pub(crate) system_ids: HashSet<SystemId>,
generation: usize,
last_initialize_generation: usize,
2020-07-10 04:18:35 +00:00
}
impl Schedule {
pub fn add_stage(&mut self, stage: impl Into<Cow<'static, str>>) {
let stage: Cow<str> = stage.into();
if self.stages.get(&stage).is_some() {
2020-07-10 04:18:35 +00:00
panic!("Stage already exists: {}", stage);
} else {
self.stages.insert(stage.clone(), Vec::new());
self.stage_order.push(stage);
}
}
pub fn add_stage_after(
&mut self,
target: impl Into<Cow<'static, str>>,
stage: impl Into<Cow<'static, str>>,
) {
let target: Cow<str> = target.into();
let stage: Cow<str> = stage.into();
if self.stages.get(&stage).is_some() {
2020-07-10 04:18:35 +00:00
panic!("Stage already exists: {}", stage);
}
let target_index = self
.stage_order
.iter()
.enumerate()
.find(|(_i, stage)| **stage == target)
.map(|(i, _)| i)
.unwrap_or_else(|| panic!("Target stage does not exist: {}", target));
self.stages.insert(stage.clone(), Vec::new());
self.stage_order.insert(target_index + 1, stage);
}
pub fn add_stage_before(
&mut self,
target: impl Into<Cow<'static, str>>,
stage: impl Into<Cow<'static, str>>,
) {
let target: Cow<str> = target.into();
let stage: Cow<str> = stage.into();
if self.stages.get(&stage).is_some() {
2020-07-10 04:18:35 +00:00
panic!("Stage already exists: {}", stage);
}
let target_index = self
.stage_order
.iter()
.enumerate()
.find(|(_i, stage)| **stage == target)
.map(|(i, _)| i)
.unwrap_or_else(|| panic!("Target stage does not exist: {}", target));
self.stages.insert(stage.clone(), Vec::new());
self.stage_order.insert(target_index, stage);
}
pub fn add_system_to_stage(
&mut self,
stage_name: impl Into<Cow<'static, str>>,
system: Box<dyn System>,
) -> &mut Self {
let stage_name = stage_name.into();
let systems = self
.stages
.get_mut(&stage_name)
.unwrap_or_else(|| panic!("Stage does not exist: {}", stage_name));
if self.system_ids.contains(&system.id()) {
2020-07-10 08:37:06 +00:00
panic!(
"System with id {:?} ({}) already exists",
system.id(),
system.name()
);
2020-07-10 04:18:35 +00:00
}
self.system_ids.insert(system.id());
systems.push(Arc::new(Mutex::new(system)));
2020-07-10 04:18:35 +00:00
self.generation += 1;
2020-07-10 04:18:35 +00:00
self
}
2020-07-24 01:26:08 +00:00
pub fn add_system_to_stage_front(
&mut self,
stage_name: impl Into<Cow<'static, str>>,
system: Box<dyn System>,
) -> &mut Self {
let stage_name = stage_name.into();
let systems = self
.stages
.get_mut(&stage_name)
.unwrap_or_else(|| panic!("Stage does not exist: {}", stage_name));
if self.system_ids.contains(&system.id()) {
panic!(
"System with id {:?} ({}) already exists",
system.id(),
system.name()
);
}
self.system_ids.insert(system.id());
systems.insert(0, Arc::new(Mutex::new(system)));
self.generation += 1;
self
}
2020-07-10 04:18:35 +00:00
pub fn run(&mut self, world: &mut World, resources: &mut Resources) {
for stage_name in self.stage_order.iter() {
if let Some(stage_systems) = self.stages.get_mut(stage_name) {
for system in stage_systems.iter_mut() {
let mut system = system.lock().unwrap();
2020-08-16 09:05:05 +00:00
#[cfg(feature = "profiler")]
crate::profiler_start(resources, system.name().clone());
2020-07-15 02:05:39 +00:00
system.update_archetype_access(world);
2020-07-10 04:18:35 +00:00
match system.thread_local_execution() {
ThreadLocalExecution::NextFlush => system.run(world, resources),
ThreadLocalExecution::Immediate => {
system.run(world, resources);
// NOTE: when this is made parallel a full sync is required here
system.run_thread_local(world, resources);
}
}
#[cfg(feature = "profiler")]
2020-08-16 09:05:05 +00:00
crate::profiler_stop(resources, system.name().clone());
2020-07-10 04:18:35 +00:00
}
// "flush"
// NOTE: when this is made parallel a full sync is required here
for system in stage_systems.iter_mut() {
let mut system = system.lock().unwrap();
2020-07-10 04:18:35 +00:00
match system.thread_local_execution() {
ThreadLocalExecution::NextFlush => {
system.run_thread_local(world, resources)
}
ThreadLocalExecution::Immediate => { /* already ran immediate */ }
}
}
}
}
world.clear_trackers();
2020-07-10 04:18:35 +00:00
}
// TODO: move this code to ParallelExecutor
2020-07-10 04:18:35 +00:00
pub fn initialize(&mut self, resources: &mut Resources) {
if self.last_initialize_generation == self.generation {
2020-07-10 04:18:35 +00:00
return;
}
let thread_pool_builder = resources
.get::<ParallelExecutorOptions>()
.map(|options| (*options).clone())
.unwrap_or_else(ParallelExecutorOptions::default)
.create_builder();
// For now, bevy_ecs only uses the global thread pool so it is sufficient to configure it once here.
// Dont call .unwrap() as the function is called twice..
let _ = thread_pool_builder.build_global();
2020-07-10 04:18:35 +00:00
for stage in self.stages.values_mut() {
for system in stage.iter_mut() {
let mut system = system.lock().unwrap();
2020-07-10 04:18:35 +00:00
system.initialize(resources);
}
}
self.last_initialize_generation = self.generation;
}
pub fn generation(&self) -> usize {
self.generation
2020-07-10 04:18:35 +00:00
}
2020-07-17 00:23:50 +00:00
}