use crate::{ system::{System, ThreadLocalExecution}, Resources, World, SystemId, }; use rayon::prelude::*; use std::{ borrow::Cow, collections::{HashMap, HashSet}, }; #[derive(Default)] pub struct Schedule { stages: HashMap, Vec>>, stage_order: Vec>, system_ids: HashSet, is_dirty: bool, } impl Schedule { pub fn add_stage(&mut self, stage: impl Into>) { let stage: Cow = 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>, stage: impl Into>, ) { let target: Cow = target.into(); let stage: Cow = 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>, stage: impl Into>, ) { let target: Cow = target.into(); let stage: Cow = 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>, system: Box, ) -> &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.push(system); self.is_dirty = true; 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()); 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() { match system.thread_local_execution() { ThreadLocalExecution::NextFlush => { system.run_thread_local(world, resources) } ThreadLocalExecution::Immediate => { /* already ran immediate */ } } } } } } pub fn dumb_par_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) { stage_systems.par_iter_mut().for_each(|system| { match system.thread_local_execution() { ThreadLocalExecution::NextFlush => system.run(world, resources), ThreadLocalExecution::Immediate => { system.run(world, resources); } } }); // "flush" // NOTE: when this is made parallel a full sync is required here for system in stage_systems.iter_mut() { match system.thread_local_execution() { ThreadLocalExecution::NextFlush => { system.run_thread_local(world, resources) } ThreadLocalExecution::Immediate => { // TODO: this should happen immediately after thread local systems system.run_thread_local(world, resources) } } } } } } pub fn initialize(&mut self, resources: &mut Resources) { if !self.is_dirty { return; } for stage in self.stages.values_mut() { for system in stage.iter_mut() { system.initialize(resources); } } self.is_dirty = false; } } // #[cfg(test)] // mod tests { // use crate::{Resources, Schedule, World}; // use crate::{IntoForEachSystem, IntoQuerySystem}; // #[test] // fn schedule() { // let mut world = World::new(); // let mut resources = Resources::default(); // world.spawn((1u32, 2u64)); // let mut schedule = Schedule::default(); // schedule.add_stage("A"); // schedule.add_stage("B"); // let xy_system = (|_x: &u32, _y: &mut u64| { // }).system(); // } // }