2020-11-08 20:34:05 +00:00
|
|
|
use crate::{
|
2020-11-16 04:32:23 +00:00
|
|
|
ArchetypeComponent, ChangedRes, Commands, Fetch, FromResources, Local, Or, Query, QueryAccess,
|
|
|
|
QueryFilter, QuerySet, QueryTuple, Res, ResMut, Resource, ResourceIndex, Resources,
|
|
|
|
SystemState, TypeAccess, World, WorldQuery,
|
2020-11-08 20:34:05 +00:00
|
|
|
};
|
|
|
|
use parking_lot::Mutex;
|
|
|
|
use std::{any::TypeId, sync::Arc};
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
pub struct In<Input>(pub Input);
|
|
|
|
|
|
|
|
impl<Input> SystemParam<Input> for In<Input> {
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
|
|
|
input: &mut Option<Input>,
|
|
|
|
_system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
_resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
Some(In(input.take().unwrap()))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn init(_system_state: &mut SystemState, _world: &World, _resources: &mut Resources) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait SystemParam<Input>: Sized {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, world: &World, resources: &mut Resources);
|
|
|
|
/// # Safety
|
|
|
|
/// This call might access any of the input parameters in an unsafe way. Make sure the data access is safe in
|
|
|
|
/// the context of the system scheduler
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self>;
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<'a, Q: WorldQuery, F: QueryFilter, Input> SystemParam<Input> for Query<'a, Q, F> {
|
2020-11-08 20:34:05 +00:00
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
world: &World,
|
|
|
|
_resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
let query_index = system_state.current_query_index;
|
|
|
|
let world: &'a World = std::mem::transmute(world);
|
|
|
|
let archetype_component_access: &'a TypeAccess<ArchetypeComponent> =
|
|
|
|
std::mem::transmute(&system_state.query_archetype_component_accesses[query_index]);
|
|
|
|
system_state.current_query_index += 1;
|
|
|
|
Some(Query::new(world, archetype_component_access))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn init(system_state: &mut SystemState, _world: &World, _resources: &mut Resources) {
|
|
|
|
system_state
|
|
|
|
.query_archetype_component_accesses
|
|
|
|
.push(TypeAccess::default());
|
2020-11-11 04:48:34 +00:00
|
|
|
let access = QueryAccess::union(vec![Q::Fetch::access(), F::access()]);
|
|
|
|
system_state.query_accesses.push(vec![access]);
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state
|
|
|
|
.query_type_names
|
|
|
|
.push(std::any::type_name::<Q>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<T: QueryTuple, Input> SystemParam<Input> for QuerySet<T> {
|
2020-11-08 20:34:05 +00:00
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
world: &World,
|
|
|
|
_resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
let query_index = system_state.current_query_index;
|
|
|
|
system_state.current_query_index += 1;
|
|
|
|
Some(QuerySet::new(
|
|
|
|
world,
|
|
|
|
&system_state.query_archetype_component_accesses[query_index],
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn init(system_state: &mut SystemState, _world: &World, _resources: &mut Resources) {
|
|
|
|
system_state
|
|
|
|
.query_archetype_component_accesses
|
|
|
|
.push(TypeAccess::default());
|
|
|
|
system_state.query_accesses.push(T::get_accesses());
|
|
|
|
system_state
|
|
|
|
.query_type_names
|
|
|
|
.push(std::any::type_name::<T>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<'a, Input> SystemParam<Input> for &'a mut Commands {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, world: &World, _resources: &mut Resources) {
|
|
|
|
system_state
|
|
|
|
.commands
|
|
|
|
.set_entity_reserver(world.get_entity_reserver())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
_resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
let commands: &'a mut Commands = std::mem::transmute(&mut system_state.commands);
|
|
|
|
Some(commands)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<Input> SystemParam<Input> for Arc<Mutex<Commands>> {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, world: &World, _resources: &mut Resources) {
|
|
|
|
system_state.arc_commands.get_or_insert_with(|| {
|
|
|
|
let mut commands = Commands::default();
|
|
|
|
commands.set_entity_reserver(world.get_entity_reserver());
|
|
|
|
Arc::new(Mutex::new(commands))
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
_resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
Some(system_state.arc_commands.as_ref().unwrap().clone())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<'a, T: Resource, Input> SystemParam<Input> for Res<'a, T> {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, _world: &World, _resources: &mut Resources) {
|
2020-11-15 20:42:43 +00:00
|
|
|
if system_state.resource_access.is_write(&TypeId::of::<T>()) {
|
|
|
|
panic!(
|
|
|
|
"System `{}` has a `Res<{res}>` parameter that conflicts with \
|
|
|
|
another parameter with mutable access to the same `{res}` resource.",
|
|
|
|
system_state.name,
|
|
|
|
res = std::any::type_name::<T>()
|
|
|
|
);
|
|
|
|
}
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state.resource_access.add_read(TypeId::of::<T>());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
_system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
Some(Res::new(
|
|
|
|
resources.get_unsafe_ref::<T>(ResourceIndex::Global),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<'a, T: Resource, Input> SystemParam<Input> for ResMut<'a, T> {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, _world: &World, _resources: &mut Resources) {
|
2020-11-15 20:42:43 +00:00
|
|
|
// If a system already has access to the resource in another parameter, then we fail early.
|
|
|
|
// e.g. `fn(Res<Foo>, ResMut<Foo>)` or `fn(ResMut<Foo>, ResMut<Foo>)` must not be allowed.
|
|
|
|
if system_state
|
|
|
|
.resource_access
|
|
|
|
.is_read_or_write(&TypeId::of::<T>())
|
|
|
|
{
|
|
|
|
panic!(
|
|
|
|
"System `{}` has a `ResMut<{res}>` parameter that conflicts with \
|
|
|
|
another parameter to the same `{res}` resource. `ResMut` must have unique access.",
|
|
|
|
system_state.name,
|
|
|
|
res = std::any::type_name::<T>()
|
|
|
|
);
|
|
|
|
}
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state.resource_access.add_write(TypeId::of::<T>());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
_system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
2020-11-17 00:55:59 +00:00
|
|
|
let (value, _added, mutated) =
|
|
|
|
resources.get_unsafe_ref_with_added_and_mutated::<T>(ResourceIndex::Global);
|
|
|
|
Some(ResMut::new(value, mutated))
|
2020-11-08 20:34:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<'a, T: Resource, Input> SystemParam<Input> for ChangedRes<'a, T> {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, _world: &World, _resources: &mut Resources) {
|
2020-11-15 20:42:43 +00:00
|
|
|
if system_state.resource_access.is_write(&TypeId::of::<T>()) {
|
|
|
|
panic!(
|
|
|
|
"System `{}` has a `ChangedRes<{res}>` parameter that conflicts with \
|
|
|
|
another parameter with mutable access to the same `{res}` resource.",
|
|
|
|
system_state.name,
|
|
|
|
res = std::any::type_name::<T>()
|
|
|
|
);
|
|
|
|
}
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state.resource_access.add_read(TypeId::of::<T>());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
_system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
2020-11-17 00:55:59 +00:00
|
|
|
let (value, added, mutated) =
|
|
|
|
resources.get_unsafe_ref_with_added_and_mutated::<T>(ResourceIndex::Global);
|
2020-11-08 20:34:05 +00:00
|
|
|
if *added.as_ptr() || *mutated.as_ptr() {
|
2020-11-17 00:55:59 +00:00
|
|
|
Some(ChangedRes::new(value))
|
2020-11-08 20:34:05 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<'a, T: Resource + FromResources, Input> SystemParam<Input> for Local<'a, T> {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, _world: &World, resources: &mut Resources) {
|
2020-11-15 20:42:43 +00:00
|
|
|
if system_state
|
|
|
|
.local_resource_access
|
|
|
|
.is_read_or_write(&TypeId::of::<T>())
|
|
|
|
{
|
|
|
|
panic!(
|
|
|
|
"System `{}` has multiple parameters requesting access to a local resource of type `{}`. \
|
|
|
|
There may be at most one `Local` parameter per resource type.",
|
|
|
|
system_state.name,
|
|
|
|
std::any::type_name::<T>()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// A resource could have been already initialized by another system with
|
|
|
|
// `Commands::insert_local_resource` or `Resources::insert_local`
|
2020-11-08 20:34:05 +00:00
|
|
|
if resources.get_local::<T>(system_state.id).is_none() {
|
|
|
|
let value = T::from_resources(resources);
|
|
|
|
resources.insert_local(system_state.id, value);
|
|
|
|
}
|
2020-11-15 20:42:43 +00:00
|
|
|
|
|
|
|
system_state
|
|
|
|
.local_resource_access
|
|
|
|
.add_write(TypeId::of::<T>());
|
2020-11-08 20:34:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
_input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
_world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
Some(Local::new(resources, system_state.id))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! impl_system_param_tuple {
|
|
|
|
($($param: ident),*) => {
|
|
|
|
#[allow(unused_variables)]
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<Input, $($param: SystemParam<Input>),*> SystemParam<Input> for ($($param,)*) {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, world: &World, resources: &mut Resources) {
|
|
|
|
$($param::init(system_state, world, resources);)*
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
2020-11-17 02:18:00 +00:00
|
|
|
Some(($($param::get_param(input, system_state, world, resources)?,)*))
|
2020-11-08 20:34:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused_variables)]
|
|
|
|
#[allow(unused_mut)]
|
|
|
|
#[allow(non_snake_case)]
|
2020-11-17 02:18:00 +00:00
|
|
|
impl<Input, $($param: SystemParam<Input>),*> SystemParam<Input> for Or<($(Option<$param>,)*)> {
|
2020-11-08 20:34:05 +00:00
|
|
|
fn init(system_state: &mut SystemState, world: &World, resources: &mut Resources) {
|
|
|
|
$($param::init(system_state, world, resources);)*
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_param(
|
2020-11-17 02:18:00 +00:00
|
|
|
input: &mut Option<Input>,
|
2020-11-08 20:34:05 +00:00
|
|
|
system_state: &mut SystemState,
|
|
|
|
world: &World,
|
|
|
|
resources: &Resources,
|
|
|
|
) -> Option<Self> {
|
|
|
|
let mut has_some = false;
|
|
|
|
$(
|
2020-11-17 02:18:00 +00:00
|
|
|
let $param = $param::get_param(input, system_state, world, resources);
|
2020-11-08 20:34:05 +00:00
|
|
|
if $param.is_some() {
|
|
|
|
has_some = true;
|
|
|
|
}
|
|
|
|
)*
|
|
|
|
|
|
|
|
if has_some {
|
|
|
|
Some(Or(($($param,)*)))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
impl_system_param_tuple!();
|
|
|
|
impl_system_param_tuple!(A);
|
|
|
|
impl_system_param_tuple!(A, B);
|
|
|
|
impl_system_param_tuple!(A, B, C);
|
|
|
|
impl_system_param_tuple!(A, B, C, D);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J, K);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J, K, L);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J, K, L, M);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J, K, L, M, N);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J, K, L, M, N, O);
|
|
|
|
impl_system_param_tuple!(A, B, C, D, E, F, G, H, I, J, K, L, M, N, O, P);
|