use std::{ fmt::{Debug, Write}, result::Result, }; use bevy_utils::default; #[cfg(feature = "trace")] use bevy_utils::tracing::info_span; use bevy_utils::{ petgraph::{algo::tarjan_scc, prelude::*}, thiserror::Error, tracing::{error, warn}, HashMap, HashSet, }; use fixedbitset::FixedBitSet; use crate::{ self as bevy_ecs, component::{ComponentId, Components}, schedule::*, system::{BoxedSystem, Resource, System}, world::World, }; /// Resource that stores [`Schedule`]s mapped to [`ScheduleLabel`]s. #[derive(Default, Resource)] pub struct Schedules { inner: HashMap, } impl Schedules { /// Constructs an empty `Schedules` with zero initial capacity. pub fn new() -> Self { Self { inner: HashMap::new(), } } /// Inserts a labeled schedule into the map. /// /// If the map already had an entry for `label`, `schedule` is inserted, /// and the old schedule is returned. Otherwise, `None` is returned. pub fn insert(&mut self, label: impl ScheduleLabel, schedule: Schedule) -> Option { let label = label.dyn_clone(); if self.inner.contains_key(&label) { warn!("schedule with label {:?} already exists", label); } self.inner.insert(label, schedule) } /// Removes the schedule corresponding to the `label` from the map, returning it if it existed. pub fn remove(&mut self, label: &dyn ScheduleLabel) -> Option { if !self.inner.contains_key(label) { warn!("schedule with label {:?} not found", label); } self.inner.remove(label) } /// Removes the (schedule, label) pair corresponding to the `label` from the map, returning it if it existed. pub fn remove_entry( &mut self, label: &dyn ScheduleLabel, ) -> Option<(Box, Schedule)> { if !self.inner.contains_key(label) { warn!("schedule with label {:?} not found", label); } self.inner.remove_entry(label) } /// Does a schedule with the provided label already exist? pub fn contains(&self, label: &dyn ScheduleLabel) -> bool { self.inner.contains_key(label) } /// Returns a reference to the schedule associated with `label`, if it exists. pub fn get(&self, label: &dyn ScheduleLabel) -> Option<&Schedule> { self.inner.get(label) } /// Returns a mutable reference to the schedule associated with `label`, if it exists. pub fn get_mut(&mut self, label: &dyn ScheduleLabel) -> Option<&mut Schedule> { self.inner.get_mut(label) } /// Returns an iterator over all schedules. Iteration order is undefined. pub fn iter(&self) -> impl Iterator { self.inner .iter() .map(|(label, schedule)| (&**label, schedule)) } /// Returns an iterator over mutable references to all schedules. Iteration order is undefined. pub fn iter_mut(&mut self) -> impl Iterator { self.inner .iter_mut() .map(|(label, schedule)| (&**label, schedule)) } /// Iterates the change ticks of all systems in all stored schedules and clamps any older than /// [`MAX_CHANGE_AGE`](crate::change_detection::MAX_CHANGE_AGE). /// This prevents overflow and thus prevents false positives. pub(crate) fn check_change_ticks(&mut self, change_tick: u32) { #[cfg(feature = "trace")] let _all_span = info_span!("check stored schedule ticks").entered(); // label used when trace feature is enabled #[allow(unused_variables)] for (label, schedule) in self.inner.iter_mut() { #[cfg(feature = "trace")] let name = format!("{label:?}"); #[cfg(feature = "trace")] let _one_span = info_span!("check schedule ticks", name = &name).entered(); schedule.check_change_ticks(change_tick); } } } /// A collection of systems, and the metadata and executor needed to run them /// in a certain order under certain conditions. pub struct Schedule { graph: ScheduleGraph, executable: SystemSchedule, executor: Box, executor_initialized: bool, } impl Default for Schedule { fn default() -> Self { Self::new() } } impl Schedule { /// Constructs an empty `Schedule`. pub fn new() -> Self { Self { graph: ScheduleGraph::new(), executable: SystemSchedule::new(), executor: Box::new(MultiThreadedExecutor::new()), executor_initialized: false, } } pub fn set_default_base_set(&mut self, default_base_set: impl SystemSet) -> &mut Self { self.graph .set_default_base_set(Some(Box::new(default_base_set))); self } /// Add a system to the schedule. pub fn add_system

(&mut self, system: impl IntoSystemConfig

) -> &mut Self { self.graph.add_system(system); self } /// Add a collection of systems to the schedule. pub fn add_systems

(&mut self, systems: impl IntoSystemConfigs

) -> &mut Self { self.graph.add_systems(systems); self } /// Configures a system set in this schedule, adding it if it does not exist. pub fn configure_set(&mut self, set: impl IntoSystemSetConfig) -> &mut Self { self.graph.configure_set(set); self } /// Configures a collection of system sets in this schedule, adding them if they does not exist. pub fn configure_sets(&mut self, sets: impl IntoSystemSetConfigs) -> &mut Self { self.graph.configure_sets(sets); self } /// Changes miscellaneous build settings. pub fn set_build_settings(&mut self, settings: ScheduleBuildSettings) -> &mut Self { self.graph.settings = settings; self } /// Returns the schedule's current execution strategy. pub fn get_executor_kind(&self) -> ExecutorKind { self.executor.kind() } /// Sets the schedule's execution strategy. pub fn set_executor_kind(&mut self, executor: ExecutorKind) -> &mut Self { if executor != self.executor.kind() { self.executor = match executor { ExecutorKind::Simple => Box::new(SimpleExecutor::new()), ExecutorKind::SingleThreaded => Box::new(SingleThreadedExecutor::new()), ExecutorKind::MultiThreaded => Box::new(MultiThreadedExecutor::new()), }; self.executor_initialized = false; } self } /// Set whether the schedule applies buffers on final time or not. This is a catchall /// incase a system uses commands but was not explicitly ordered after a /// [`apply_system_buffers`](crate::prelude::apply_system_buffers). By default this /// setting is true, but may be disabled if needed. pub fn set_apply_final_buffers(&mut self, apply_final_buffers: bool) -> &mut Self { self.executor.set_apply_final_buffers(apply_final_buffers); self } /// Runs all systems in this schedule on the `world`, using its current execution strategy. pub fn run(&mut self, world: &mut World) { world.check_change_ticks(); self.initialize(world).unwrap(); self.executor.run(&mut self.executable, world); } /// Initializes any newly-added systems and conditions, rebuilds the executable schedule, /// and re-initializes the executor. pub fn initialize(&mut self, world: &mut World) -> Result<(), ScheduleBuildError> { if self.graph.changed { self.graph.initialize(world); self.graph .update_schedule(&mut self.executable, world.components())?; self.graph.changed = false; self.executor_initialized = false; } if !self.executor_initialized { self.executor.init(&self.executable); self.executor_initialized = true; } Ok(()) } /// Returns the [`ScheduleGraph`]. pub fn graph(&self) -> &ScheduleGraph { &self.graph } /// Iterates the change ticks of all systems in the schedule and clamps any older than /// [`MAX_CHANGE_AGE`](crate::change_detection::MAX_CHANGE_AGE). /// This prevents overflow and thus prevents false positives. pub(crate) fn check_change_ticks(&mut self, change_tick: u32) { for system in &mut self.executable.systems { system.check_change_tick(change_tick); } for conditions in &mut self.executable.system_conditions { for system in conditions.iter_mut() { system.check_change_tick(change_tick); } } for conditions in &mut self.executable.set_conditions { for system in conditions.iter_mut() { system.check_change_tick(change_tick); } } } /// Directly applies any accumulated system buffers (like [`Commands`](crate::prelude::Commands)) to the `world`. /// /// Like always, system buffers are applied in the "topological sort order" of the schedule graph. /// As a result, buffers from one system are only guaranteed to be applied before those of other systems /// if there is an explicit system ordering between the two systems. /// /// This is used in rendering to extract data from the main world, storing the data in system buffers, /// before applying their buffers in a different world. pub fn apply_system_buffers(&mut self, world: &mut World) { for system in &mut self.executable.systems { system.apply_buffers(world); } } } /// A directed acylic graph structure. #[derive(Default)] pub struct Dag { /// A directed graph. graph: DiGraphMap, /// A cached topological ordering of the graph. topsort: Vec, } impl Dag { fn new() -> Self { Self { graph: DiGraphMap::new(), topsort: Vec::new(), } } /// The directed graph of the stored systems, connected by their ordering dependencies. pub fn graph(&self) -> &DiGraphMap { &self.graph } /// A cached topological ordering of the graph. /// /// The order is determined by the ordering dependencies between systems. pub fn cached_topsort(&self) -> &[NodeId] { &self.topsort } } /// Describes which base set (i.e. [`SystemSet`] where [`SystemSet::is_base`] returns true) /// a system belongs to. /// /// Note that this is only populated once [`ScheduleGraph::build_schedule`] is called. #[derive(Copy, Clone, Eq, PartialEq, Debug)] pub enum BaseSetMembership { Uncalculated, None, Some(NodeId), } /// A [`SystemSet`] with metadata, stored in a [`ScheduleGraph`]. struct SystemSetNode { inner: BoxedSystemSet, base_set_membership: BaseSetMembership, } impl SystemSetNode { pub fn new(set: BoxedSystemSet) -> Self { Self { inner: set, base_set_membership: BaseSetMembership::Uncalculated, } } pub fn name(&self) -> String { format!("{:?}", &self.inner) } pub fn is_system_type(&self) -> bool { self.inner.is_system_type() } } /// A [`BoxedSystem`] with metadata, stored in a [`ScheduleGraph`]. struct SystemNode { inner: Option, base_set_membership: BaseSetMembership, } impl SystemNode { pub fn new(system: BoxedSystem) -> Self { Self { inner: Some(system), base_set_membership: BaseSetMembership::Uncalculated, } } pub fn get(&self) -> Option<&BoxedSystem> { self.inner.as_ref() } pub fn get_mut(&mut self) -> Option<&mut BoxedSystem> { self.inner.as_mut() } pub fn name(&self) -> String { format!("{:?}", &self.inner) } } /// Metadata for a [`Schedule`]. #[derive(Default)] pub struct ScheduleGraph { systems: Vec, system_conditions: Vec>>, system_sets: Vec, system_set_conditions: Vec>>, system_set_ids: HashMap, uninit: Vec<(NodeId, usize)>, maybe_default_base_set: Vec, hierarchy: Dag, dependency: Dag, dependency_flattened: Dag, ambiguous_with: UnGraphMap, ambiguous_with_flattened: UnGraphMap, ambiguous_with_all: HashSet, changed: bool, settings: ScheduleBuildSettings, default_base_set: Option, } impl ScheduleGraph { pub fn new() -> Self { Self { systems: Vec::new(), system_conditions: Vec::new(), system_sets: Vec::new(), system_set_conditions: Vec::new(), system_set_ids: HashMap::new(), maybe_default_base_set: Vec::new(), uninit: Vec::new(), hierarchy: Dag::new(), dependency: Dag::new(), dependency_flattened: Dag::new(), ambiguous_with: UnGraphMap::new(), ambiguous_with_flattened: UnGraphMap::new(), ambiguous_with_all: HashSet::new(), changed: false, settings: default(), default_base_set: None, } } /// Returns the system at the given [`NodeId`], if it exists. pub fn get_system_at(&self, id: NodeId) -> Option<&dyn System> { if !id.is_system() { return None; } self.systems .get(id.index()) .and_then(|system| system.inner.as_deref()) } /// Returns the system at the given [`NodeId`]. /// /// Panics if it doesn't exist. #[track_caller] pub fn system_at(&self, id: NodeId) -> &dyn System { self.get_system_at(id) .ok_or_else(|| format!("system with id {id:?} does not exist in this Schedule")) .unwrap() } /// Returns the set at the given [`NodeId`], if it exists. pub fn get_set_at(&self, id: NodeId) -> Option<&dyn SystemSet> { if !id.is_set() { return None; } self.system_sets.get(id.index()).map(|set| &*set.inner) } /// Returns the set at the given [`NodeId`]. /// /// Panics if it doesn't exist. #[track_caller] pub fn set_at(&self, id: NodeId) -> &dyn SystemSet { self.get_set_at(id) .ok_or_else(|| format!("set with id {id:?} does not exist in this Schedule")) .unwrap() } /// Returns an iterator over all systems in this schedule. /// /// Note that the [`BaseSetMembership`] will only be initialized after [`ScheduleGraph::build_schedule`] is called. pub fn systems( &self, ) -> impl Iterator< Item = ( NodeId, &dyn System, BaseSetMembership, &[BoxedCondition], ), > { self.systems .iter() .zip(self.system_conditions.iter()) .enumerate() .filter_map(|(i, (system_node, condition))| { let system = system_node.inner.as_deref()?; let base_set_membership = system_node.base_set_membership; let condition = condition.as_ref()?.as_slice(); Some((NodeId::System(i), system, base_set_membership, condition)) }) } /// Returns an iterator over all system sets in this schedule. /// /// Note that the [`BaseSetMembership`] will only be initialized after [`ScheduleGraph::build_schedule`] is called. pub fn system_sets( &self, ) -> impl Iterator { self.system_set_ids.iter().map(|(_, node_id)| { let set_node = &self.system_sets[node_id.index()]; let set = &*set_node.inner; let base_set_membership = set_node.base_set_membership; let conditions = self.system_set_conditions[node_id.index()] .as_deref() .unwrap_or(&[]); (*node_id, set, base_set_membership, conditions) }) } /// Returns the [`Dag`] of the hierarchy. /// /// The hierarchy is a directed acyclic graph of the systems and sets, /// where an edge denotes that a system or set is the child of another set. pub fn hierarchy(&self) -> &Dag { &self.hierarchy } /// Returns the [`Dag`] of the dependencies in the schedule. /// /// Nodes in this graph are systems and sets, and edges denote that /// a system or set has to run before another system or set. pub fn dependency(&self) -> &Dag { &self.dependency } fn add_systems

(&mut self, systems: impl IntoSystemConfigs

) { let SystemConfigs { systems, chained } = systems.into_configs(); let mut system_iter = systems.into_iter(); if chained { let Some(prev) = system_iter.next() else { return }; let mut prev_id = self.add_system_inner(prev).unwrap(); for next in system_iter { let next_id = self.add_system_inner(next).unwrap(); self.dependency.graph.add_edge(prev_id, next_id, ()); prev_id = next_id; } } else { for system in system_iter { self.add_system_inner(system).unwrap(); } } } fn add_system

(&mut self, system: impl IntoSystemConfig

) { self.add_system_inner(system).unwrap(); } fn add_system_inner

( &mut self, system: impl IntoSystemConfig

, ) -> Result { let SystemConfig { system, graph_info, conditions, } = system.into_config(); let id = NodeId::System(self.systems.len()); // graph updates are immediate self.update_graphs(id, graph_info, false)?; // system init has to be deferred (need `&mut World`) self.uninit.push((id, 0)); self.systems.push(SystemNode::new(system)); self.system_conditions.push(Some(conditions)); Ok(id) } fn configure_sets(&mut self, sets: impl IntoSystemSetConfigs) { let SystemSetConfigs { sets, chained } = sets.into_configs(); let mut set_iter = sets.into_iter(); if chained { let Some(prev) = set_iter.next() else { return }; let mut prev_id = self.configure_set_inner(prev).unwrap(); for next in set_iter { let next_id = self.configure_set_inner(next).unwrap(); self.dependency.graph.add_edge(prev_id, next_id, ()); prev_id = next_id; } } else { for set in set_iter { self.configure_set_inner(set).unwrap(); } } } fn configure_set(&mut self, set: impl IntoSystemSetConfig) { self.configure_set_inner(set).unwrap(); } fn configure_set_inner( &mut self, set: impl IntoSystemSetConfig, ) -> Result { let SystemSetConfig { set, graph_info, mut conditions, } = set.into_config(); let id = match self.system_set_ids.get(&set) { Some(&id) => id, None => self.add_set(set.dyn_clone()), }; // graph updates are immediate self.update_graphs(id, graph_info, set.is_base())?; // system init has to be deferred (need `&mut World`) let system_set_conditions = self.system_set_conditions[id.index()].get_or_insert_with(Vec::new); self.uninit.push((id, system_set_conditions.len())); system_set_conditions.append(&mut conditions); Ok(id) } fn add_set(&mut self, set: BoxedSystemSet) -> NodeId { let id = NodeId::Set(self.system_sets.len()); self.system_sets.push(SystemSetNode::new(set.dyn_clone())); self.system_set_conditions.push(None); self.system_set_ids.insert(set, id); id } fn check_set(&mut self, id: &NodeId, set: &dyn SystemSet) -> Result<(), ScheduleBuildError> { match self.system_set_ids.get(set) { Some(set_id) => { if id == set_id { let string = format!("{:?}", &set); return Err(ScheduleBuildError::HierarchyLoop(string)); } } None => { self.add_set(set.dyn_clone()); } } Ok(()) } fn check_sets( &mut self, id: &NodeId, graph_info: &GraphInfo, ) -> Result<(), ScheduleBuildError> { for set in &graph_info.sets { self.check_set(id, &**set)?; } if let Some(base_set) = &graph_info.base_set { self.check_set(id, &**base_set)?; } Ok(()) } fn check_edges( &mut self, id: &NodeId, graph_info: &GraphInfo, ) -> Result<(), ScheduleBuildError> { for Dependency { kind: _, set } in &graph_info.dependencies { match self.system_set_ids.get(set) { Some(set_id) => { if id == set_id { let string = format!("{:?}", &set); return Err(ScheduleBuildError::DependencyLoop(string)); } } None => { self.add_set(set.dyn_clone()); } } } if let Ambiguity::IgnoreWithSet(ambiguous_with) = &graph_info.ambiguous_with { for set in ambiguous_with { if !self.system_set_ids.contains_key(set) { self.add_set(set.dyn_clone()); } } } Ok(()) } fn update_graphs( &mut self, id: NodeId, graph_info: GraphInfo, is_base_set: bool, ) -> Result<(), ScheduleBuildError> { self.check_sets(&id, &graph_info)?; self.check_edges(&id, &graph_info)?; self.changed = true; let GraphInfo { sets, dependencies, ambiguous_with, base_set, add_default_base_set, .. } = graph_info; self.hierarchy.graph.add_node(id); self.dependency.graph.add_node(id); for set in sets.into_iter().map(|set| self.system_set_ids[&set]) { self.hierarchy.graph.add_edge(set, id, ()); // ensure set also appears in dependency graph self.dependency.graph.add_node(set); } // If the current node is not a base set, set the base set if it was configured if !is_base_set { if let Some(base_set) = base_set { let set_id = self.system_set_ids[&base_set]; self.hierarchy.graph.add_edge(set_id, id, ()); } else if let Some(default_base_set) = &self.default_base_set { if add_default_base_set { match id { NodeId::System(_) => { // Queue the default base set. We queue systems instead of adding directly to allow // sets to define base sets, which will override the default inheritance behavior self.maybe_default_base_set.push(id); } NodeId::Set(_) => { // Sets should be added automatically because developers explicitly called // in_default_base_set() let set_id = self.system_set_ids[default_base_set]; self.hierarchy.graph.add_edge(set_id, id, ()); } } } } } if !self.dependency.graph.contains_node(id) { self.dependency.graph.add_node(id); } for (kind, set) in dependencies .into_iter() .map(|Dependency { kind, set }| (kind, self.system_set_ids[&set])) { let (lhs, rhs) = match kind { DependencyKind::Before => (id, set), DependencyKind::After => (set, id), }; self.dependency.graph.add_edge(lhs, rhs, ()); // ensure set also appears in hierarchy graph self.hierarchy.graph.add_node(set); } match ambiguous_with { Ambiguity::Check => (), Ambiguity::IgnoreWithSet(ambiguous_with) => { for set in ambiguous_with .into_iter() .map(|set| self.system_set_ids[&set]) { self.ambiguous_with.add_edge(id, set, ()); } } Ambiguity::IgnoreAll => { self.ambiguous_with_all.insert(id); } } Ok(()) } fn initialize(&mut self, world: &mut World) { for (id, i) in self.uninit.drain(..) { match id { NodeId::System(index) => { self.systems[index].get_mut().unwrap().initialize(world); if let Some(v) = self.system_conditions[index].as_mut() { for condition in v.iter_mut() { condition.initialize(world); } } } NodeId::Set(index) => { if let Some(v) = self.system_set_conditions[index].as_mut() { for condition in v.iter_mut().skip(i) { condition.initialize(world); } } } } } } /// Calculates the base set for each node and caches the results on the node fn calculate_base_sets_and_detect_cycles(&mut self) -> Result<(), ScheduleBuildError> { let set_ids = (0..self.system_sets.len()).map(NodeId::Set); let system_ids = (0..self.systems.len()).map(NodeId::System); let mut visited_sets = vec![false; self.system_sets.len()]; // reset base set membership, as this can change when the schedule updates for system in &mut self.systems { system.base_set_membership = BaseSetMembership::Uncalculated; } for system_set in &mut self.system_sets { system_set.base_set_membership = BaseSetMembership::Uncalculated; } for node_id in set_ids.chain(system_ids) { Self::calculate_base_set( &self.hierarchy, &mut self.system_sets, &mut self.systems, &mut visited_sets, node_id, )?; } Ok(()) } fn calculate_base_set( hierarchy: &Dag, system_sets: &mut [SystemSetNode], systems: &mut [SystemNode], visited_sets: &mut [bool], node_id: NodeId, ) -> Result, ScheduleBuildError> { let base_set_membership = match node_id { // systems only have NodeId::System(_) => BaseSetMembership::Uncalculated, NodeId::Set(index) => { let set_node = &mut system_sets[index]; if set_node.inner.is_base() { set_node.base_set_membership = BaseSetMembership::Some(node_id); } set_node.base_set_membership } }; let base_set = match base_set_membership { BaseSetMembership::None => None, BaseSetMembership::Some(node_id) => Some(node_id), BaseSetMembership::Uncalculated => { let mut base_set: Option = None; if let NodeId::Set(index) = node_id { if visited_sets[index] { return Err(ScheduleBuildError::HierarchyCycle); } visited_sets[index] = true; } for neighbor in hierarchy .graph .neighbors_directed(node_id, Direction::Incoming) { if let Some(calculated_base_set) = Self::calculate_base_set( hierarchy, system_sets, systems, visited_sets, neighbor, )? { if let Some(first_set) = base_set { return Err(match node_id { NodeId::System(index) => { ScheduleBuildError::SystemInMultipleBaseSets { system: systems[index].name(), first_set: system_sets[first_set.index()].name(), second_set: system_sets[calculated_base_set.index()].name(), } } NodeId::Set(index) => ScheduleBuildError::SetInMultipleBaseSets { set: system_sets[index].name(), first_set: system_sets[first_set.index()].name(), second_set: system_sets[calculated_base_set.index()].name(), }, }); } base_set = Some(calculated_base_set); } } match node_id { NodeId::System(index) => { systems[index].base_set_membership = if let Some(base_set) = base_set { BaseSetMembership::Some(base_set) } else { BaseSetMembership::None }; } NodeId::Set(index) => { system_sets[index].base_set_membership = if let Some(base_set) = base_set { BaseSetMembership::Some(base_set) } else { BaseSetMembership::None }; } } base_set } }; Ok(base_set) } /// Build a [`SystemSchedule`] optimized for scheduler access from the [`ScheduleGraph`]. /// /// This method also /// - calculates [`BaseSetMembership`] /// - checks for dependency or hierarchy cycles /// - checks for system access conflicts and reports ambiguities pub fn build_schedule( &mut self, components: &Components, ) -> Result { self.calculate_base_sets_and_detect_cycles()?; // Add missing base set membership to systems that defaulted to using the // default base set and weren't added to a set that belongs to a base set. if let Some(default_base_set) = &self.default_base_set { let default_set_id = self.system_set_ids[default_base_set]; for system_id in std::mem::take(&mut self.maybe_default_base_set) { let system_node = &mut self.systems[system_id.index()]; if system_node.base_set_membership == BaseSetMembership::None { self.hierarchy.graph.add_edge(default_set_id, system_id, ()); system_node.base_set_membership = BaseSetMembership::Some(default_set_id); } debug_assert_ne!( system_node.base_set_membership, BaseSetMembership::Uncalculated, "base set membership should have been calculated" ); } } // check hierarchy for cycles let hier_scc = tarjan_scc(&self.hierarchy.graph); // PERF: in theory we can skip this contains_cycles because we've already detected cycles // using calculate_base_sets_and_detect_cycles if self.contains_cycles(&hier_scc) { self.report_cycles(&hier_scc); return Err(ScheduleBuildError::HierarchyCycle); } self.hierarchy.topsort = hier_scc.into_iter().flatten().rev().collect::>(); let hier_results = check_graph(&self.hierarchy.graph, &self.hierarchy.topsort); if self.settings.hierarchy_detection != LogLevel::Ignore && self.contains_hierarchy_conflicts(&hier_results.transitive_edges) { self.report_hierarchy_conflicts(&hier_results.transitive_edges); if matches!(self.settings.hierarchy_detection, LogLevel::Error) { return Err(ScheduleBuildError::HierarchyRedundancy); } } // remove redundant edges self.hierarchy.graph = hier_results.transitive_reduction; // check dependencies for cycles let dep_scc = tarjan_scc(&self.dependency.graph); if self.contains_cycles(&dep_scc) { self.report_cycles(&dep_scc); return Err(ScheduleBuildError::DependencyCycle); } self.dependency.topsort = dep_scc.into_iter().flatten().rev().collect::>(); // check for systems or system sets depending on sets they belong to let dep_results = check_graph(&self.dependency.graph, &self.dependency.topsort); for &(a, b) in dep_results.connected.iter() { if hier_results.connected.contains(&(a, b)) || hier_results.connected.contains(&(b, a)) { let name_a = self.get_node_name(&a); let name_b = self.get_node_name(&b); return Err(ScheduleBuildError::CrossDependency(name_a, name_b)); } } // map all system sets to their systems // go in reverse topological order (bottom-up) for efficiency let mut set_systems: HashMap> = HashMap::with_capacity(self.system_sets.len()); let mut set_system_bitsets = HashMap::with_capacity(self.system_sets.len()); for &id in self.hierarchy.topsort.iter().rev() { if id.is_system() { continue; } let mut systems = Vec::new(); let mut system_bitset = FixedBitSet::with_capacity(self.systems.len()); for child in self .hierarchy .graph .neighbors_directed(id, Direction::Outgoing) { match child { NodeId::System(_) => { systems.push(child); system_bitset.insert(child.index()); } NodeId::Set(_) => { let child_systems = set_systems.get(&child).unwrap(); let child_system_bitset = set_system_bitsets.get(&child).unwrap(); systems.extend_from_slice(child_systems); system_bitset.union_with(child_system_bitset); } } } set_systems.insert(id, systems); set_system_bitsets.insert(id, system_bitset); } // check that there is no ordering between system sets that intersect for (a, b) in dep_results.connected.iter() { if !(a.is_set() && b.is_set()) { continue; } let a_systems = set_system_bitsets.get(a).unwrap(); let b_systems = set_system_bitsets.get(b).unwrap(); if !(a_systems.is_disjoint(b_systems)) { return Err(ScheduleBuildError::SetsHaveOrderButIntersect( self.get_node_name(a), self.get_node_name(b), )); } } // check that there are no edges to system-type sets that have multiple instances for (&id, systems) in set_systems.iter() { let set = &self.system_sets[id.index()]; if set.is_system_type() { let instances = systems.len(); let ambiguous_with = self.ambiguous_with.edges(id); let before = self .dependency .graph .edges_directed(id, Direction::Incoming); let after = self .dependency .graph .edges_directed(id, Direction::Outgoing); let relations = before.count() + after.count() + ambiguous_with.count(); if instances > 1 && relations > 0 { return Err(ScheduleBuildError::SystemTypeSetAmbiguity( self.get_node_name(&id), )); } } } // flatten: combine `in_set` with `before` and `after` information // have to do it like this to preserve transitivity let mut dependency_flattened = self.dependency.graph.clone(); let mut temp = Vec::new(); for (&set, systems) in set_systems.iter() { if systems.is_empty() { for a in dependency_flattened.neighbors_directed(set, Direction::Incoming) { for b in dependency_flattened.neighbors_directed(set, Direction::Outgoing) { temp.push((a, b)); } } } else { for a in dependency_flattened.neighbors_directed(set, Direction::Incoming) { for &sys in systems { temp.push((a, sys)); } } for b in dependency_flattened.neighbors_directed(set, Direction::Outgoing) { for &sys in systems { temp.push((sys, b)); } } } dependency_flattened.remove_node(set); for (a, b) in temp.drain(..) { dependency_flattened.add_edge(a, b, ()); } } // topsort let flat_scc = tarjan_scc(&dependency_flattened); if self.contains_cycles(&flat_scc) { self.report_cycles(&flat_scc); return Err(ScheduleBuildError::DependencyCycle); } self.dependency_flattened.graph = dependency_flattened; self.dependency_flattened.topsort = flat_scc.into_iter().flatten().rev().collect::>(); let flat_results = check_graph( &self.dependency_flattened.graph, &self.dependency_flattened.topsort, ); // remove redundant edges self.dependency_flattened.graph = flat_results.transitive_reduction; // flatten: combine `in_set` with `ambiguous_with` information let mut ambiguous_with_flattened = UnGraphMap::new(); for (lhs, rhs, _) in self.ambiguous_with.all_edges() { match (lhs, rhs) { (NodeId::System(_), NodeId::System(_)) => { ambiguous_with_flattened.add_edge(lhs, rhs, ()); } (NodeId::Set(_), NodeId::System(_)) => { for &lhs_ in set_systems.get(&lhs).unwrap() { ambiguous_with_flattened.add_edge(lhs_, rhs, ()); } } (NodeId::System(_), NodeId::Set(_)) => { for &rhs_ in set_systems.get(&rhs).unwrap() { ambiguous_with_flattened.add_edge(lhs, rhs_, ()); } } (NodeId::Set(_), NodeId::Set(_)) => { for &lhs_ in set_systems.get(&lhs).unwrap() { for &rhs_ in set_systems.get(&rhs).unwrap() { ambiguous_with_flattened.add_edge(lhs_, rhs_, ()); } } } } } self.ambiguous_with_flattened = ambiguous_with_flattened; // check for conflicts let mut conflicting_systems = Vec::new(); for &(a, b) in flat_results.disconnected.iter() { if self.ambiguous_with_flattened.contains_edge(a, b) || self.ambiguous_with_all.contains(&a) || self.ambiguous_with_all.contains(&b) { continue; } let system_a = self.systems[a.index()].get().unwrap(); let system_b = self.systems[b.index()].get().unwrap(); if system_a.is_exclusive() || system_b.is_exclusive() { conflicting_systems.push((a, b, Vec::new())); } else { let access_a = system_a.component_access(); let access_b = system_b.component_access(); if !access_a.is_compatible(access_b) { let conflicts = access_a.get_conflicts(access_b); conflicting_systems.push((a, b, conflicts)); } } } if self.settings.ambiguity_detection != LogLevel::Ignore && self.contains_conflicts(&conflicting_systems) { self.report_conflicts(&conflicting_systems, components); if matches!(self.settings.ambiguity_detection, LogLevel::Error) { return Err(ScheduleBuildError::Ambiguity); } } // build the schedule let dg_system_ids = self.dependency_flattened.topsort.clone(); let dg_system_idx_map = dg_system_ids .iter() .cloned() .enumerate() .map(|(i, id)| (id, i)) .collect::>(); let hg_systems = self .hierarchy .topsort .iter() .cloned() .enumerate() .filter(|&(_i, id)| id.is_system()) .collect::>(); let (hg_set_with_conditions_idxs, hg_set_ids): (Vec<_>, Vec<_>) = self .hierarchy .topsort .iter() .cloned() .enumerate() .filter(|&(_i, id)| { // ignore system sets that have no conditions // ignore system type sets (already covered, they don't have conditions) id.is_set() && self.system_set_conditions[id.index()] .as_ref() .filter(|v| !v.is_empty()) .is_some() }) .unzip(); let sys_count = self.systems.len(); let set_with_conditions_count = hg_set_ids.len(); let node_count = self.systems.len() + self.system_sets.len(); // get the number of dependencies and the immediate dependents of each system // (needed by multi-threaded executor to run systems in the correct order) let mut system_dependencies = Vec::with_capacity(sys_count); let mut system_dependents = Vec::with_capacity(sys_count); for &sys_id in &dg_system_ids { let num_dependencies = self .dependency_flattened .graph .neighbors_directed(sys_id, Direction::Incoming) .count(); let dependents = self .dependency_flattened .graph .neighbors_directed(sys_id, Direction::Outgoing) .map(|dep_id| dg_system_idx_map[&dep_id]) .collect::>(); system_dependencies.push(num_dependencies); system_dependents.push(dependents); } // get the rows and columns of the hierarchy graph's reachability matrix // (needed to we can evaluate conditions in the correct order) let mut systems_in_sets_with_conditions = vec![FixedBitSet::with_capacity(sys_count); set_with_conditions_count]; for (i, &row) in hg_set_with_conditions_idxs.iter().enumerate() { let bitset = &mut systems_in_sets_with_conditions[i]; for &(col, sys_id) in &hg_systems { let idx = dg_system_idx_map[&sys_id]; let is_descendant = hier_results.reachable[index(row, col, node_count)]; bitset.set(idx, is_descendant); } } let mut sets_with_conditions_of_systems = vec![FixedBitSet::with_capacity(set_with_conditions_count); sys_count]; for &(col, sys_id) in &hg_systems { let i = dg_system_idx_map[&sys_id]; let bitset = &mut sets_with_conditions_of_systems[i]; for (idx, &row) in hg_set_with_conditions_idxs .iter() .enumerate() .take_while(|&(_idx, &row)| row < col) { let is_ancestor = hier_results.reachable[index(row, col, node_count)]; bitset.set(idx, is_ancestor); } } Ok(SystemSchedule { systems: Vec::with_capacity(sys_count), system_conditions: Vec::with_capacity(sys_count), set_conditions: Vec::with_capacity(set_with_conditions_count), system_ids: dg_system_ids, set_ids: hg_set_ids, system_dependencies, system_dependents, sets_with_conditions_of_systems, systems_in_sets_with_conditions, }) } fn update_schedule( &mut self, schedule: &mut SystemSchedule, components: &Components, ) -> Result<(), ScheduleBuildError> { if !self.uninit.is_empty() { return Err(ScheduleBuildError::Uninitialized); } // move systems out of old schedule for ((id, system), conditions) in schedule .system_ids .drain(..) .zip(schedule.systems.drain(..)) .zip(schedule.system_conditions.drain(..)) { self.systems[id.index()].inner = Some(system); self.system_conditions[id.index()] = Some(conditions); } for (id, conditions) in schedule .set_ids .drain(..) .zip(schedule.set_conditions.drain(..)) { self.system_set_conditions[id.index()] = Some(conditions); } *schedule = self.build_schedule(components)?; // move systems into new schedule for &id in &schedule.system_ids { let system = self.systems[id.index()].inner.take().unwrap(); let conditions = self.system_conditions[id.index()].take().unwrap(); schedule.systems.push(system); schedule.system_conditions.push(conditions); } for &id in &schedule.set_ids { let conditions = self.system_set_conditions[id.index()].take().unwrap(); schedule.set_conditions.push(conditions); } Ok(()) } fn set_default_base_set(&mut self, set: Option) { if let Some(set) = set { self.default_base_set = Some(set.dyn_clone()); if self.system_set_ids.get(&set).is_none() { self.add_set(set); } } else { self.default_base_set = None; } } } // methods for reporting errors impl ScheduleGraph { fn get_node_name(&self, id: &NodeId) -> String { match id { NodeId::System(_) => self.systems[id.index()].get().unwrap().name().to_string(), NodeId::Set(_) => self.system_sets[id.index()].name(), } } fn get_node_kind(id: &NodeId) -> &'static str { match id { NodeId::System(_) => "system", NodeId::Set(_) => "system set", } } fn contains_hierarchy_conflicts(&self, transitive_edges: &[(NodeId, NodeId)]) -> bool { if transitive_edges.is_empty() { return false; } true } fn report_hierarchy_conflicts(&self, transitive_edges: &[(NodeId, NodeId)]) { let mut message = String::from("hierarchy contains redundant edge(s)"); for (parent, child) in transitive_edges { writeln!( message, " -- {:?} '{:?}' cannot be child of set '{:?}', longer path exists", Self::get_node_kind(child), self.get_node_name(child), self.get_node_name(parent), ) .unwrap(); } error!("{}", message); } fn contains_cycles(&self, strongly_connected_components: &[Vec]) -> bool { if strongly_connected_components .iter() .all(|scc| scc.len() == 1) { return false; } true } fn report_cycles(&self, strongly_connected_components: &[Vec]) { let components_with_cycles = strongly_connected_components .iter() .filter(|scc| scc.len() > 1) .cloned() .collect::>(); let mut message = format!( "schedule contains at least {} cycle(s)", components_with_cycles.len() ); writeln!(message, " -- cycle(s) found within:").unwrap(); for (i, scc) in components_with_cycles.into_iter().enumerate() { let names = scc .iter() .map(|id| self.get_node_name(id)) .collect::>(); writeln!(message, " ---- {i}: {names:?}").unwrap(); } error!("{}", message); } fn contains_conflicts(&self, conflicts: &[(NodeId, NodeId, Vec)]) -> bool { if conflicts.is_empty() { return false; } true } fn report_conflicts( &self, ambiguities: &[(NodeId, NodeId, Vec)], components: &Components, ) { let n_ambiguities = ambiguities.len(); let mut string = format!( "{n_ambiguities} pairs of systems with conflicting data access have indeterminate execution order. \ Consider adding `before`, `after`, or `ambiguous_with` relationships between these:\n", ); for (system_a, system_b, conflicts) in ambiguities { let name_a = self.get_node_name(system_a); let name_b = self.get_node_name(system_b); debug_assert!(system_a.is_system(), "{name_a} is not a system."); debug_assert!(system_b.is_system(), "{name_b} is not a system."); writeln!(string, " -- {name_a} and {name_b}").unwrap(); if !conflicts.is_empty() { let conflict_names: Vec<_> = conflicts .iter() .map(|id| components.get_name(*id).unwrap()) .collect(); writeln!(string, " conflict on: {conflict_names:?}").unwrap(); } else { // one or both systems must be exclusive let world = std::any::type_name::(); writeln!(string, " conflict on: {world}").unwrap(); } } warn!("{}", string); } } /// Category of errors encountered during schedule construction. #[derive(Error, Debug)] #[non_exhaustive] pub enum ScheduleBuildError { /// A system set contains itself. #[error("`{0:?}` contains itself.")] HierarchyLoop(String), /// The hierarchy of system sets contains a cycle. #[error("System set hierarchy contains cycle(s).")] HierarchyCycle, /// The hierarchy of system sets contains redundant edges. /// /// This error is disabled by default, but can be opted-in using [`ScheduleBuildSettings`]. #[error("System set hierarchy contains redundant edges.")] HierarchyRedundancy, /// A system (set) has been told to run before itself. #[error("`{0:?}` depends on itself.")] DependencyLoop(String), /// The dependency graph contains a cycle. #[error("System dependencies contain cycle(s).")] DependencyCycle, /// Tried to order a system (set) relative to a system set it belongs to. #[error("`{0:?}` and `{1:?}` have both `in_set` and `before`-`after` relationships (these might be transitive). This combination is unsolvable as a system cannot run before or after a set it belongs to.")] CrossDependency(String, String), /// Tried to order system sets that share systems. #[error("`{0:?}` and `{1:?}` have a `before`-`after` relationship (which may be transitive) but share systems.")] SetsHaveOrderButIntersect(String, String), /// Tried to order a system (set) relative to all instances of some system function. #[error("Tried to order against `fn {0:?}` in a schedule that has more than one `{0:?}` instance. `fn {0:?}` is a `SystemTypeSet` and cannot be used for ordering if ambiguous. Use a different set without this restriction.")] SystemTypeSetAmbiguity(String), /// Systems with conflicting access have indeterminate run order. /// /// This error is disabled by default, but can be opted-in using [`ScheduleBuildSettings`]. #[error("Systems with conflicting access have indeterminate run order.")] Ambiguity, /// Tried to run a schedule before all of its systems have been initialized. #[error("Systems in schedule have not been initialized.")] Uninitialized, /// Tried to add a system to multiple base sets. #[error("System `{system:?}` is in the base sets {first_set:?} and {second_set:?}, but systems can only belong to one base set.")] SystemInMultipleBaseSets { system: String, first_set: String, second_set: String, }, /// Tried to add a set to multiple base sets. #[error("Set `{set:?}` is in the base sets {first_set:?} and {second_set:?}, but sets can only belong to one base set.")] SetInMultipleBaseSets { set: String, first_set: String, second_set: String, }, } /// Specifies how schedule construction should respond to detecting a certain kind of issue. #[derive(Debug, Clone, PartialEq)] pub enum LogLevel { /// Occurences are completely ignored. Ignore, /// Occurrences are logged only. Warn, /// Occurrences are logged and result in errors. Error, } /// Specifies miscellaneous settings for schedule construction. #[derive(Clone, Debug)] pub struct ScheduleBuildSettings { ambiguity_detection: LogLevel, hierarchy_detection: LogLevel, } impl Default for ScheduleBuildSettings { fn default() -> Self { Self::new() } } impl ScheduleBuildSettings { pub const fn new() -> Self { Self { ambiguity_detection: LogLevel::Ignore, hierarchy_detection: LogLevel::Warn, } } /// Determines whether the presence of ambiguities (systems with conflicting access but indeterminate order) /// is only logged or also results in an [`Ambiguity`](ScheduleBuildError::Ambiguity) error. pub fn with_ambiguity_detection(mut self, level: LogLevel) -> Self { self.ambiguity_detection = level; self } /// Determines whether the presence of redundant edges in the hierarchy of system sets is only /// logged or also results in a [`HierarchyRedundancy`](ScheduleBuildError::HierarchyRedundancy) /// error. pub fn with_hierarchy_detection(mut self, level: LogLevel) -> Self { self.hierarchy_detection = level; self } }