2020-07-10 04:18:35 +00:00
|
|
|
use crate::{
|
2020-07-17 02:20:51 +00:00
|
|
|
system::{System, ThreadLocalExecution, SystemId}, resource::Resources,
|
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
|
|
|
};
|
2020-07-17 02:20:51 +00:00
|
|
|
use hecs::World;
|
2020-07-10 04:18:35 +00:00
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct Schedule {
|
2020-07-14 02:29:34 +00:00
|
|
|
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>,
|
2020-07-16 00:20:36 +00:00
|
|
|
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 let Some(_) = self.stages.get(&stage) {
|
|
|
|
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 let Some(_) = self.stages.get(&stage) {
|
|
|
|
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 let Some(_) = self.stages.get(&stage) {
|
|
|
|
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());
|
2020-07-14 02:29:34 +00:00
|
|
|
systems.push(Arc::new(Mutex::new(system)));
|
2020-07-10 04:18:35 +00:00
|
|
|
|
2020-07-16 00:20:36 +00:00
|
|
|
self.generation += 1;
|
2020-07-10 04:18:35 +00:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
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() {
|
|
|
|
#[cfg(feature = "profiler")]
|
|
|
|
crate::profiler::profiler_start(resources, system.name().clone());
|
2020-07-14 02:29:34 +00:00
|
|
|
let mut system = system.lock().unwrap();
|
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")]
|
|
|
|
crate::profiler::profiler_stop(resources, system.name().clone());
|
|
|
|
}
|
|
|
|
|
|
|
|
// "flush"
|
|
|
|
// NOTE: when this is made parallel a full sync is required here
|
|
|
|
for system in stage_systems.iter_mut() {
|
2020-07-14 02:29:34 +00:00
|
|
|
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 */ }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 00:20:36 +00:00
|
|
|
// TODO: move this code to ParallelExecutor
|
2020-07-10 04:18:35 +00:00
|
|
|
pub fn initialize(&mut self, resources: &mut Resources) {
|
2020-07-16 00:20:36 +00:00
|
|
|
if self.last_initialize_generation == self.generation {
|
2020-07-10 04:18:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for stage in self.stages.values_mut() {
|
|
|
|
for system in stage.iter_mut() {
|
2020-07-14 02:29:34 +00:00
|
|
|
let mut system = system.lock().unwrap();
|
2020-07-10 04:18:35 +00:00
|
|
|
system.initialize(resources);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 00:20:36 +00:00
|
|
|
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
|
|
|
}
|