use crate::{ archetype::ArchetypeComponentId, change_detection::MAX_CHANGE_AGE, component::ComponentId, query::Access, system::{ check_system_change_tick, ExclusiveSystemParam, ExclusiveSystemParamItem, In, InputMarker, IntoSystem, System, SystemMeta, }, world::{World, WorldId}, }; use bevy_ecs_macros::all_tuples; use std::{any::TypeId, borrow::Cow, marker::PhantomData}; /// A function system that runs with exclusive [`World`] access. /// /// You get this by calling [`IntoSystem::into_system`] on a function that only accepts /// [`ExclusiveSystemParam`]s. /// /// [`ExclusiveFunctionSystem`] must be `.initialized` before they can be run. pub struct ExclusiveFunctionSystem where F: ExclusiveSystemParamFunction, { func: F, param_state: Option<::State>, system_meta: SystemMeta, world_id: Option, // NOTE: PhantomData T> gives this safe Send/Sync impls marker: PhantomData Marker>, } pub struct IsExclusiveFunctionSystem; impl IntoSystem for F where Marker: 'static, F: ExclusiveSystemParamFunction, { type System = ExclusiveFunctionSystem; fn into_system(func: Self) -> Self::System { ExclusiveFunctionSystem { func, param_state: None, system_meta: SystemMeta::new::(), world_id: None, marker: PhantomData, } } } const PARAM_MESSAGE: &str = "System's param_state was not found. Did you forget to initialize this system before running it?"; impl System for ExclusiveFunctionSystem where Marker: 'static, F: ExclusiveSystemParamFunction, { type In = F::In; type Out = F::Out; #[inline] fn name(&self) -> Cow<'static, str> { self.system_meta.name.clone() } #[inline] fn type_id(&self) -> TypeId { TypeId::of::() } #[inline] fn component_access(&self) -> &Access { self.system_meta.component_access_set.combined_access() } #[inline] fn archetype_component_access(&self) -> &Access { &self.system_meta.archetype_component_access } #[inline] fn is_send(&self) -> bool { // exclusive systems should have access to non-send resources // the executor runs exclusive systems on the main thread, so this // field reflects that constraint false } #[inline] unsafe fn run_unsafe(&mut self, _input: Self::In, _world: &World) -> Self::Out { panic!("Cannot run exclusive systems with a shared World reference"); } fn run(&mut self, input: Self::In, world: &mut World) -> Self::Out { let saved_last_tick = world.last_change_tick; world.last_change_tick = self.system_meta.last_change_tick; let params = F::Param::get_param( self.param_state.as_mut().expect(PARAM_MESSAGE), &self.system_meta, ); let out = self.func.run(world, input, params); let change_tick = world.change_tick.get_mut(); self.system_meta.last_change_tick = *change_tick; *change_tick = change_tick.wrapping_add(1); world.last_change_tick = saved_last_tick; out } #[inline] fn is_exclusive(&self) -> bool { true } fn get_last_change_tick(&self) -> u32 { self.system_meta.last_change_tick } fn set_last_change_tick(&mut self, last_change_tick: u32) { self.system_meta.last_change_tick = last_change_tick; } #[inline] fn apply_buffers(&mut self, _world: &mut World) { // "pure" exclusive systems do not have any buffers to apply. // Systems made by piping a normal system with an exclusive system // might have buffers to apply, but this is handled by `PipeSystem`. } #[inline] fn initialize(&mut self, world: &mut World) { self.world_id = Some(world.id()); self.system_meta.last_change_tick = world.change_tick().wrapping_sub(MAX_CHANGE_AGE); self.param_state = Some(F::Param::init(world, &mut self.system_meta)); } fn update_archetype_component_access(&mut self, _world: &World) {} #[inline] fn check_change_tick(&mut self, change_tick: u32) { check_system_change_tick( &mut self.system_meta.last_change_tick, change_tick, self.system_meta.name.as_ref(), ); } fn default_system_sets(&self) -> Vec> { let set = crate::schedule::SystemTypeSet::::new(); vec![Box::new(set)] } } /// A trait implemented for all exclusive system functions that can be used as [`System`]s. /// /// This trait can be useful for making your own systems which accept other systems, /// sometimes called higher order systems. pub trait ExclusiveSystemParamFunction: Send + Sync + 'static { /// The input type to this system. See [`System::In`]. type In; /// The return type of this system. See [`System::Out`]. type Out; /// The [`ExclusiveSystemParam`]/s defined by this system's `fn` parameters. type Param: ExclusiveSystemParam; /// Executes this system once. See [`System::run`]. fn run( &mut self, world: &mut World, input: Self::In, param_value: ExclusiveSystemParamItem, ) -> Self::Out; } macro_rules! impl_exclusive_system_function { ($($param: ident),*) => { #[allow(non_snake_case)] impl ExclusiveSystemParamFunction<((), Out, $($param,)*)> for Func where for <'a> &'a mut Func: FnMut(&mut World, $($param),*) -> Out + FnMut(&mut World, $(ExclusiveSystemParamItem<$param>),*) -> Out, Out: 'static, { type In = (); type Out = Out; type Param = ($($param,)*); #[inline] fn run(&mut self, world: &mut World, _in: (), param_value: ExclusiveSystemParamItem< ($($param,)*)>) -> Out { // Yes, this is strange, but `rustc` fails to compile this impl // without using this function. It fails to recognise that `func` // is a function, potentially because of the multiple impls of `FnMut` #[allow(clippy::too_many_arguments)] fn call_inner( mut f: impl FnMut(&mut World, $($param,)*) -> Out, world: &mut World, $($param: $param,)* ) -> Out { f(world, $($param,)*) } let ($($param,)*) = param_value; call_inner(self, world, $($param),*) } } #[allow(non_snake_case)] impl ExclusiveSystemParamFunction<(Input, Out, $($param,)* InputMarker)> for Func where for <'a> &'a mut Func: FnMut(In, &mut World, $($param),*) -> Out + FnMut(In, &mut World, $(ExclusiveSystemParamItem<$param>),*) -> Out, Out: 'static, { type In = Input; type Out = Out; type Param = ($($param,)*); #[inline] fn run(&mut self, world: &mut World, input: Input, param_value: ExclusiveSystemParamItem< ($($param,)*)>) -> Out { // Yes, this is strange, but `rustc` fails to compile this impl // without using this function. It fails to recognise that `func` // is a function, potentially because of the multiple impls of `FnMut` #[allow(clippy::too_many_arguments)] fn call_inner( mut f: impl FnMut(In, &mut World, $($param,)*) -> Out, input: Input, world: &mut World, $($param: $param,)* ) -> Out { f(In(input), world, $($param,)*) } let ($($param,)*) = param_value; call_inner(self, input, world, $($param),*) } } }; } // Note that we rely on the highest impl to be <= the highest order of the tuple impls // of `SystemParam` created. all_tuples!(impl_exclusive_system_function, 0, 16, F);