use super::system_param::FetchSystemParam; use crate::{ ArchetypeComponent, Commands, QueryAccess, Resources, System, SystemId, SystemParam, ThreadLocalExecution, TypeAccess, World, }; use parking_lot::Mutex; use std::{any::TypeId, borrow::Cow, cell::UnsafeCell, sync::Arc}; pub struct SystemState { pub(crate) id: SystemId, pub(crate) name: Cow<'static, str>, pub(crate) archetype_component_access: TypeAccess, pub(crate) resource_access: TypeAccess, pub(crate) local_resource_access: TypeAccess, pub(crate) query_archetype_component_accesses: Vec>, pub(crate) query_accesses: Vec>, pub(crate) query_type_names: Vec<&'static str>, pub(crate) commands: UnsafeCell, pub(crate) arc_commands: Option>>, pub(crate) current_query_index: UnsafeCell, } // SAFE: UnsafeCell and UnsafeCell only accessed from the thread they are scheduled on unsafe impl Sync for SystemState {} impl SystemState { pub fn reset_indices(&mut self) { // SAFE: done with unique mutable access to Self unsafe { *self.current_query_index.get() = 0; } } pub fn update(&mut self, world: &World) { self.archetype_component_access.clear(); let mut conflict_index = None; let mut conflict_name = None; for (i, (query_accesses, component_access)) in self .query_accesses .iter() .zip(self.query_archetype_component_accesses.iter_mut()) .enumerate() { component_access.clear(); for query_access in query_accesses.iter() { query_access.get_world_archetype_access(world, Some(component_access)); } if !component_access.is_compatible(&self.archetype_component_access) { conflict_index = Some(i); conflict_name = component_access .get_conflict(&self.archetype_component_access) .and_then(|archetype_component| { query_accesses .iter() .filter_map(|query_access| { query_access.get_type_name(archetype_component.component) }) .next() }); break; } self.archetype_component_access.union(component_access); } if let Some(conflict_index) = conflict_index { let mut conflicts_with_index = None; for prior_index in 0..conflict_index { if !self.query_archetype_component_accesses[conflict_index] .is_compatible(&self.query_archetype_component_accesses[prior_index]) { conflicts_with_index = Some(prior_index); } } panic!("System {} has conflicting queries. {} conflicts with the component access [{}] in this prior query: {}.", self.name, self.query_type_names[conflict_index], conflict_name.unwrap_or("Unknown"), conflicts_with_index.map(|index| self.query_type_names[index]).unwrap_or("Unknown")); } } } pub struct FuncSystem { func: Box Option + Send + Sync + 'static>, thread_local_func: Box, init_func: Box, state: SystemState, } impl System for FuncSystem { type In = (); type Out = Out; fn name(&self) -> std::borrow::Cow<'static, str> { self.state.name.clone() } fn id(&self) -> SystemId { self.state.id } fn update(&mut self, world: &World) { self.state.update(world); } fn archetype_component_access(&self) -> &TypeAccess { &self.state.archetype_component_access } fn resource_access(&self) -> &TypeAccess { &self.state.resource_access } fn thread_local_execution(&self) -> ThreadLocalExecution { ThreadLocalExecution::NextFlush } unsafe fn run_unsafe( &mut self, _input: Self::In, world: &World, resources: &Resources, ) -> Option { (self.func)(&mut self.state, world, resources) } fn run_thread_local(&mut self, world: &mut World, resources: &mut Resources) { (self.thread_local_func)(&mut self.state, world, resources) } fn initialize(&mut self, world: &mut World, resources: &mut Resources) { (self.init_func)(&mut self.state, world, resources); } } pub struct InputFuncSystem { func: Box< dyn FnMut(In, &mut SystemState, &World, &Resources) -> Option + Send + Sync + 'static, >, thread_local_func: Box, init_func: Box, state: SystemState, } impl System for InputFuncSystem { type In = In; type Out = Out; fn name(&self) -> std::borrow::Cow<'static, str> { self.state.name.clone() } fn id(&self) -> SystemId { self.state.id } fn update(&mut self, world: &World) { self.state.update(world); } fn archetype_component_access(&self) -> &TypeAccess { &self.state.archetype_component_access } fn resource_access(&self) -> &TypeAccess { &self.state.resource_access } fn thread_local_execution(&self) -> ThreadLocalExecution { ThreadLocalExecution::NextFlush } unsafe fn run_unsafe( &mut self, input: In, world: &World, resources: &Resources, ) -> Option { (self.func)(input, &mut self.state, world, resources) } fn run_thread_local(&mut self, world: &mut World, resources: &mut Resources) { (self.thread_local_func)(&mut self.state, world, resources) } fn initialize(&mut self, world: &mut World, resources: &mut Resources) { (self.init_func)(&mut self.state, world, resources); } } pub trait IntoSystem { fn system(self) -> SystemType; } // Systems implicitly implement IntoSystem impl IntoSystem<(), Sys> for Sys { fn system(self) -> Sys { self } } pub struct In(pub In); macro_rules! impl_into_system { ($($param: ident),*) => { impl IntoSystem<($($param,)*), FuncSystem> for Func where Func: FnMut($($param),*) -> Out + FnMut($(<<$param as SystemParam>::Fetch as FetchSystemParam>::Item),*) -> Out + Send + Sync + 'static, Out: 'static { #[allow(unused_variables)] #[allow(unused_unsafe)] #[allow(non_snake_case)] fn system(mut self) -> FuncSystem { FuncSystem { state: SystemState { name: std::any::type_name::().into(), archetype_component_access: TypeAccess::default(), resource_access: TypeAccess::default(), local_resource_access: TypeAccess::default(), id: SystemId::new(), commands: Default::default(), arc_commands: Default::default(), current_query_index: Default::default(), query_archetype_component_accesses: Vec::new(), query_accesses: Vec::new(), query_type_names: Vec::new(), }, func: Box::new(move |state, world, resources| { state.reset_indices(); // let mut input = Some(input); unsafe { if let Some(($($param,)*)) = <<($($param,)*) as SystemParam>::Fetch as FetchSystemParam>::get_param(state, world, resources) { Some(self($($param),*)) } else { None } } }), thread_local_func: Box::new(|state, world, resources| { // SAFE: this is called with unique access to SystemState unsafe { (&mut *state.commands.get()).apply(world, resources); } if let Some(ref commands) = state.arc_commands { let mut commands = commands.lock(); commands.apply(world, resources); } }), init_func: Box::new(|state, world, resources| { <<($($param,)*) as SystemParam>::Fetch as FetchSystemParam>::init(state, world, resources) }), } } } impl IntoSystem<(Input, $($param,)*), InputFuncSystem> for Func where Func: FnMut(In, $($param),*) -> Out + FnMut(In, $(<<$param as SystemParam>::Fetch as FetchSystemParam>::Item),*) -> Out + Send + Sync + 'static, Input: 'static, Out: 'static { #[allow(unused_variables)] #[allow(unused_unsafe)] #[allow(non_snake_case)] fn system(mut self) -> InputFuncSystem { InputFuncSystem { state: SystemState { name: std::any::type_name::().into(), archetype_component_access: TypeAccess::default(), resource_access: TypeAccess::default(), local_resource_access: TypeAccess::default(), id: SystemId::new(), commands: Default::default(), arc_commands: Default::default(), current_query_index: Default::default(), query_archetype_component_accesses: Vec::new(), query_accesses: Vec::new(), query_type_names: Vec::new(), }, func: Box::new(move |input, state, world, resources| { state.reset_indices(); // let mut input = Some(input); unsafe { if let Some(($($param,)*)) = <<($($param,)*) as SystemParam>::Fetch as FetchSystemParam>::get_param(state, world, resources) { Some(self(In(input), $($param),*)) } else { None } } }), thread_local_func: Box::new(|state, world, resources| { // SAFE: this is called with unique access to SystemState unsafe { (&mut *state.commands.get()).apply(world, resources); } if let Some(ref commands) = state.arc_commands { let mut commands = commands.lock(); commands.apply(world, resources); } }), init_func: Box::new(|state, world, resources| { <<($($param,)*) as SystemParam>::Fetch as FetchSystemParam>::init(state, world, resources) }), } } } }; } impl_into_system!(); impl_into_system!(A); impl_into_system!(A, B); impl_into_system!(A, B, C); impl_into_system!(A, B, C, D); impl_into_system!(A, B, C, D, E); impl_into_system!(A, B, C, D, E, F); impl_into_system!(A, B, C, D, E, F, G); impl_into_system!(A, B, C, D, E, F, G, H); impl_into_system!(A, B, C, D, E, F, G, H, I); impl_into_system!(A, B, C, D, E, F, G, H, I, J); impl_into_system!(A, B, C, D, E, F, G, H, I, J, K); impl_into_system!(A, B, C, D, E, F, G, H, I, J, K, L); impl_into_system!(A, B, C, D, E, F, G, H, I, J, K, L, M); impl_into_system!(A, B, C, D, E, F, G, H, I, J, K, L, M, N); impl_into_system!(A, B, C, D, E, F, G, H, I, J, K, L, M, N, O); impl_into_system!(A, B, C, D, E, F, G, H, I, J, K, L, M, N, O, P); #[cfg(test)] mod tests { use super::IntoSystem; use crate::{ clear_trackers_system, resource::{Res, ResMut, Resources}, schedule::Schedule, ChangedRes, Entity, Local, Or, Query, QuerySet, System, SystemStage, With, World, }; #[derive(Debug, Eq, PartialEq, Default)] struct A; struct B; struct C; struct D; #[test] fn query_system_gets() { fn query_system( mut ran: ResMut, entity_query: Query>, b_query: Query<&B>, a_c_query: Query<(&A, &C)>, d_query: Query<&D>, ) { let entities = entity_query.iter().collect::>(); assert!( b_query.get_component::(entities[0]).is_err(), "entity 0 should not have B" ); assert!( b_query.get_component::(entities[1]).is_ok(), "entity 1 should have B" ); assert!( b_query.get_component::(entities[1]).is_err(), "entity 1 should have A, but b_query shouldn't have access to it" ); assert!( b_query.get_component::(entities[3]).is_err(), "entity 3 should have D, but it shouldn't be accessible from b_query" ); assert!( b_query.get_component::(entities[2]).is_err(), "entity 2 has C, but it shouldn't be accessible from b_query" ); assert!( a_c_query.get_component::(entities[2]).is_ok(), "entity 2 has C, and it should be accessible from a_c_query" ); assert!( a_c_query.get_component::(entities[3]).is_err(), "entity 3 should have D, but it shouldn't be accessible from b_query" ); assert!( d_query.get_component::(entities[3]).is_ok(), "entity 3 should have D" ); *ran = true; } let mut world = World::default(); let mut resources = Resources::default(); resources.insert(false); world.spawn((A,)); world.spawn((A, B)); world.spawn((A, C)); world.spawn((A, D)); run_system(&mut world, &mut resources, query_system.system()); assert!(*resources.get::().unwrap(), "system ran"); } #[test] fn or_query_set_system() { // Regression test for issue #762 use crate::{Added, Changed, Mutated, Or}; fn query_system( mut ran: ResMut, set: QuerySet<( Query<(), Or<(Changed, Changed)>>, Query<(), Or<(Added, Added)>>, Query<(), Or<(Mutated, Mutated)>>, )>, ) { let changed = set.q0().iter().count(); let added = set.q1().iter().count(); let mutated = set.q2().iter().count(); assert_eq!(changed, 1); assert_eq!(added, 1); assert_eq!(mutated, 0); *ran = true; }; let mut world = World::default(); let mut resources = Resources::default(); resources.insert(false); world.spawn((A, B)); run_system(&mut world, &mut resources, query_system.system()); assert!(*resources.get::().unwrap(), "system ran"); } #[test] fn changed_resource_system() { fn incr_e_on_flip(_run_on_flip: ChangedRes, mut query: Query<&mut i32>) { for mut i in query.iter_mut() { *i += 1; } } let mut world = World::default(); let mut resources = Resources::default(); resources.insert(false); let ent = world.spawn((0,)); let mut schedule = Schedule::default(); let mut update = SystemStage::parallel(); update.add_system(incr_e_on_flip.system()); schedule.add_stage("update", update); schedule.add_stage( "clear_trackers", SystemStage::single(clear_trackers_system.system()), ); schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 1); schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 1); *resources.get_mut::().unwrap() = true; schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 2); } #[test] fn changed_resource_or_system() { fn incr_e_on_flip( _or: Or<(Option>, Option>)>, mut query: Query<&mut i32>, ) { for mut i in query.iter_mut() { *i += 1; } } let mut world = World::default(); let mut resources = Resources::default(); resources.insert(false); resources.insert::(10); let ent = world.spawn((0,)); let mut schedule = Schedule::default(); let mut update = SystemStage::parallel(); update.add_system(incr_e_on_flip.system()); schedule.add_stage("update", update); schedule.add_stage( "clear_trackers", SystemStage::single(clear_trackers_system.system()), ); schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 1); schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 1); *resources.get_mut::().unwrap() = true; schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 2); schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 2); *resources.get_mut::().unwrap() = 20; schedule.initialize_and_run(&mut world, &mut resources); assert_eq!(*(world.get::(ent).unwrap()), 3); } #[test] #[should_panic] fn conflicting_query_mut_system() { fn sys(_q1: Query<&mut A>, _q2: Query<&mut A>) {} let mut world = World::default(); let mut resources = Resources::default(); world.spawn((A,)); run_system(&mut world, &mut resources, sys.system()); } #[test] #[should_panic] fn conflicting_query_immut_system() { fn sys(_q1: Query<&A>, _q2: Query<&mut A>) {} let mut world = World::default(); let mut resources = Resources::default(); world.spawn((A,)); run_system(&mut world, &mut resources, sys.system()); } #[test] fn query_set_system() { fn sys(_set: QuerySet<(Query<&mut A>, Query<&B>)>) {} let mut world = World::default(); let mut resources = Resources::default(); world.spawn((A,)); run_system(&mut world, &mut resources, sys.system()); } #[test] #[should_panic] fn conflicting_query_with_query_set_system() { fn sys(_query: Query<&mut A>, _set: QuerySet<(Query<&mut A>, Query<&B>)>) {} let mut world = World::default(); let mut resources = Resources::default(); world.spawn((A,)); run_system(&mut world, &mut resources, sys.system()); } #[test] #[should_panic] fn conflicting_query_sets_system() { fn sys(_set_1: QuerySet<(Query<&mut A>,)>, _set_2: QuerySet<(Query<&mut A>, Query<&B>)>) {} let mut world = World::default(); let mut resources = Resources::default(); world.spawn((A,)); run_system(&mut world, &mut resources, sys.system()); } fn run_system>( world: &mut World, resources: &mut Resources, system: S, ) { let mut schedule = Schedule::default(); let mut update = SystemStage::parallel(); update.add_system(system); schedule.add_stage("update", update); schedule.initialize_and_run(world, resources); } #[derive(Default)] struct BufferRes { _buffer: Vec, } fn test_for_conflicting_resources>(sys: S) { let mut world = World::default(); let mut resources = Resources::default(); resources.insert(BufferRes::default()); resources.insert(A); resources.insert(B); run_system(&mut world, &mut resources, sys.system()); } #[test] #[should_panic] fn conflicting_system_resources() { fn sys(_: ResMut, _: Res) {} test_for_conflicting_resources(sys.system()) } #[test] #[should_panic] fn conflicting_system_resources_reverse_order() { fn sys(_: Res, _: ResMut) {} test_for_conflicting_resources(sys.system()) } #[test] #[should_panic] fn conflicting_system_resources_multiple_mutable() { fn sys(_: ResMut, _: ResMut) {} test_for_conflicting_resources(sys.system()) } #[test] #[should_panic] fn conflicting_changed_and_mutable_resource() { // A tempting pattern, but unsound if allowed. fn sys(_: ResMut, _: ChangedRes) {} test_for_conflicting_resources(sys.system()) } #[test] #[should_panic] fn conflicting_system_local_resources() { fn sys(_: Local, _: Local) {} test_for_conflicting_resources(sys.system()) } #[test] fn nonconflicting_system_resources() { fn sys(_: Local, _: ResMut, _: Local, _: ResMut) {} test_for_conflicting_resources(sys.system()) } }