use crate::{ archetype::{Archetype, ArchetypeComponentId}, component::{Component, ComponentId, ComponentTicks, StorageType}, entity::Entity, query::{Access, FilteredAccess}, storage::{ComponentSparseSet, Table, Tables}, world::{Mut, World}, }; use bevy_ecs_macros::all_tuples; use std::{ marker::PhantomData, ptr::{self, NonNull}, }; pub trait WorldQuery { type Fetch: for<'a> Fetch<'a, State = Self::State>; type State: FetchState; } pub trait Fetch<'w>: Sized { type Item; type State: FetchState; /// Creates a new instance of this fetch. /// /// # Safety /// `state` must have been initialized (via [FetchState::init]) using the same `world` passed in /// to this function. unsafe fn init( world: &World, state: &Self::State, last_change_tick: u32, change_tick: u32, ) -> Self; /// Returns true if (and only if) every table of every archetype matched by this Fetch contains /// all of the matched components. This is used to select a more efficient "table iterator" /// for "dense" queries. If this returns true, [Fetch::set_table] and [Fetch::table_fetch] /// will be called for iterators If this returns false, [Fetch::set_archetype] and /// [Fetch::archetype_fetch] will be called for iterators fn is_dense(&self) -> bool; /// Adjusts internal state to account for the next [Archetype]. This will always be called on /// archetypes that match this [Fetch] /// /// # Safety /// `archetype` and `tables` must be from the [World] [Fetch::init] was called on. `state` must /// be the [Self::State] this was initialized with. unsafe fn set_archetype(&mut self, state: &Self::State, archetype: &Archetype, tables: &Tables); /// Adjusts internal state to account for the next [Table]. This will always be called on tables /// that match this [Fetch] /// /// # Safety /// `table` must be from the [World] [Fetch::init] was called on. `state` must be the /// [Self::State] this was initialized with. unsafe fn set_table(&mut self, state: &Self::State, table: &Table); /// Fetch [Self::Item] for the given `archetype_index` in the current [Archetype]. This must /// always be called after [Fetch::set_archetype] with an `archetype_index` in the range of /// the current [Archetype] /// /// # Safety /// Must always be called _after_ [Fetch::set_archetype]. `archetype_index` must be in the range /// of the current archetype unsafe fn archetype_fetch(&mut self, archetype_index: usize) -> Self::Item; /// Fetch [Self::Item] for the given `table_row` in the current [Table]. This must always be /// called after [Fetch::set_table] with a `table_row` in the range of the current [Table] /// /// # Safety /// Must always be called _after_ [Fetch::set_table]. `table_row` must be in the range of the /// current table unsafe fn table_fetch(&mut self, table_row: usize) -> Self::Item; } /// State used to construct a Fetch. This will be cached inside QueryState, so it is best to move as /// much data / computation here as possible to reduce the cost of constructing Fetch. /// SAFETY: /// Implementor must ensure that [FetchState::update_component_access] and /// [FetchState::update_archetype_component_access] exactly reflects the results of /// [FetchState::matches_archetype], [FetchState::matches_table], [Fetch::archetype_fetch], and /// [Fetch::table_fetch] pub unsafe trait FetchState: Send + Sync + Sized { fn init(world: &mut World) -> Self; fn update_component_access(&self, access: &mut FilteredAccess); fn update_archetype_component_access( &self, archetype: &Archetype, access: &mut Access, ); fn matches_archetype(&self, archetype: &Archetype) -> bool; fn matches_table(&self, table: &Table) -> bool; } /// A fetch that is read only. This must only be implemented for read-only fetches. pub unsafe trait ReadOnlyFetch {} impl WorldQuery for Entity { type Fetch = EntityFetch; type State = EntityState; } pub struct EntityFetch { entities: *const Entity, } /// SAFE: access is read only unsafe impl ReadOnlyFetch for EntityFetch {} pub struct EntityState; // SAFE: no component or archetype access unsafe impl FetchState for EntityState { fn init(_world: &mut World) -> Self { Self } fn update_component_access(&self, _access: &mut FilteredAccess) {} fn update_archetype_component_access( &self, _archetype: &Archetype, _access: &mut Access, ) { } #[inline] fn matches_archetype(&self, _archetype: &Archetype) -> bool { true } #[inline] fn matches_table(&self, _table: &Table) -> bool { true } } impl<'w> Fetch<'w> for EntityFetch { type Item = Entity; type State = EntityState; #[inline] fn is_dense(&self) -> bool { true } unsafe fn init( _world: &World, _state: &Self::State, _last_change_tick: u32, _change_tick: u32, ) -> Self { Self { entities: std::ptr::null::(), } } #[inline] unsafe fn set_archetype( &mut self, _state: &Self::State, archetype: &Archetype, _tables: &Tables, ) { self.entities = archetype.entities().as_ptr(); } #[inline] unsafe fn set_table(&mut self, _state: &Self::State, table: &Table) { self.entities = table.entities().as_ptr(); } #[inline] unsafe fn table_fetch(&mut self, table_row: usize) -> Self::Item { *self.entities.add(table_row) } #[inline] unsafe fn archetype_fetch(&mut self, archetype_index: usize) -> Self::Item { *self.entities.add(archetype_index) } } impl WorldQuery for &T { type Fetch = ReadFetch; type State = ReadState; } pub struct ReadState { component_id: ComponentId, storage_type: StorageType, marker: PhantomData, } // SAFE: component access and archetype component access are properly updated to reflect that T is // read unsafe impl FetchState for ReadState { fn init(world: &mut World) -> Self { let component_info = world.components.get_or_insert_info::(); ReadState { component_id: component_info.id(), storage_type: component_info.storage_type(), marker: PhantomData, } } fn update_component_access(&self, access: &mut FilteredAccess) { if access.access().has_write(self.component_id) { panic!("&{} conflicts with a previous access in this query. Shared access cannot coincide with exclusive access.", std::any::type_name::()); } access.add_read(self.component_id) } fn update_archetype_component_access( &self, archetype: &Archetype, access: &mut Access, ) { if let Some(archetype_component_id) = archetype.get_archetype_component_id(self.component_id) { access.add_read(archetype_component_id); } } fn matches_archetype(&self, archetype: &Archetype) -> bool { archetype.contains(self.component_id) } fn matches_table(&self, table: &Table) -> bool { table.has_column(self.component_id) } } pub struct ReadFetch { storage_type: StorageType, table_components: NonNull, entity_table_rows: *const usize, entities: *const Entity, sparse_set: *const ComponentSparseSet, } /// SAFE: access is read only unsafe impl ReadOnlyFetch for ReadFetch {} impl<'w, T: Component> Fetch<'w> for ReadFetch { type Item = &'w T; type State = ReadState; #[inline] fn is_dense(&self) -> bool { match self.storage_type { StorageType::Table => true, StorageType::SparseSet => false, } } unsafe fn init( world: &World, state: &Self::State, _last_change_tick: u32, _change_tick: u32, ) -> Self { let mut value = Self { storage_type: state.storage_type, table_components: NonNull::dangling(), entities: ptr::null::(), entity_table_rows: ptr::null::(), sparse_set: ptr::null::(), }; if state.storage_type == StorageType::SparseSet { value.sparse_set = world .storages() .sparse_sets .get(state.component_id) .unwrap(); } value } #[inline] unsafe fn set_archetype( &mut self, state: &Self::State, archetype: &Archetype, tables: &Tables, ) { match state.storage_type { StorageType::Table => { self.entity_table_rows = archetype.entity_table_rows().as_ptr(); let column = tables[archetype.table_id()] .get_column(state.component_id) .unwrap(); self.table_components = column.get_ptr().cast::(); } StorageType::SparseSet => self.entities = archetype.entities().as_ptr(), } } #[inline] unsafe fn set_table(&mut self, state: &Self::State, table: &Table) { self.table_components = table .get_column(state.component_id) .unwrap() .get_ptr() .cast::(); } #[inline] unsafe fn archetype_fetch(&mut self, archetype_index: usize) -> Self::Item { match self.storage_type { StorageType::Table => { let table_row = *self.entity_table_rows.add(archetype_index); &*self.table_components.as_ptr().add(table_row) } StorageType::SparseSet => { let entity = *self.entities.add(archetype_index); &*(*self.sparse_set).get(entity).unwrap().cast::() } } } #[inline] unsafe fn table_fetch(&mut self, table_row: usize) -> Self::Item { &*self.table_components.as_ptr().add(table_row) } } impl WorldQuery for &mut T { type Fetch = WriteFetch; type State = WriteState; } pub struct WriteFetch { storage_type: StorageType, table_components: NonNull, table_ticks: *mut ComponentTicks, entities: *const Entity, entity_table_rows: *const usize, sparse_set: *const ComponentSparseSet, last_change_tick: u32, change_tick: u32, } pub struct WriteState { component_id: ComponentId, storage_type: StorageType, marker: PhantomData, } // SAFE: component access and archetype component access are properly updated to reflect that T is // written unsafe impl FetchState for WriteState { fn init(world: &mut World) -> Self { let component_info = world.components.get_or_insert_info::(); WriteState { component_id: component_info.id(), storage_type: component_info.storage_type(), marker: PhantomData, } } fn update_component_access(&self, access: &mut FilteredAccess) { if access.access().has_read(self.component_id) { panic!("&mut {} conflicts with a previous access in this query. Mutable component access must be unique.", std::any::type_name::()); } access.add_write(self.component_id); } fn update_archetype_component_access( &self, archetype: &Archetype, access: &mut Access, ) { if let Some(archetype_component_id) = archetype.get_archetype_component_id(self.component_id) { access.add_write(archetype_component_id); } } fn matches_archetype(&self, archetype: &Archetype) -> bool { archetype.contains(self.component_id) } fn matches_table(&self, table: &Table) -> bool { table.has_column(self.component_id) } } impl<'w, T: Component> Fetch<'w> for WriteFetch { type Item = Mut<'w, T>; type State = WriteState; #[inline] fn is_dense(&self) -> bool { match self.storage_type { StorageType::Table => true, StorageType::SparseSet => false, } } unsafe fn init( world: &World, state: &Self::State, last_change_tick: u32, change_tick: u32, ) -> Self { let mut value = Self { storage_type: state.storage_type, table_components: NonNull::dangling(), entities: ptr::null::(), entity_table_rows: ptr::null::(), sparse_set: ptr::null::(), table_ticks: ptr::null_mut::(), last_change_tick, change_tick, }; if state.storage_type == StorageType::SparseSet { value.sparse_set = world .storages() .sparse_sets .get(state.component_id) .unwrap(); } value } #[inline] unsafe fn set_archetype( &mut self, state: &Self::State, archetype: &Archetype, tables: &Tables, ) { match state.storage_type { StorageType::Table => { self.entity_table_rows = archetype.entity_table_rows().as_ptr(); let column = tables[archetype.table_id()] .get_column(state.component_id) .unwrap(); self.table_components = column.get_ptr().cast::(); self.table_ticks = column.get_ticks_mut_ptr(); } StorageType::SparseSet => self.entities = archetype.entities().as_ptr(), } } #[inline] unsafe fn set_table(&mut self, state: &Self::State, table: &Table) { let column = table.get_column(state.component_id).unwrap(); self.table_components = column.get_ptr().cast::(); self.table_ticks = column.get_ticks_mut_ptr(); } #[inline] unsafe fn archetype_fetch(&mut self, archetype_index: usize) -> Self::Item { match self.storage_type { StorageType::Table => { let table_row = *self.entity_table_rows.add(archetype_index); Mut { value: &mut *self.table_components.as_ptr().add(table_row), component_ticks: &mut *self.table_ticks.add(table_row), change_tick: self.change_tick, last_change_tick: self.last_change_tick, } } StorageType::SparseSet => { let entity = *self.entities.add(archetype_index); let (component, component_ticks) = (*self.sparse_set).get_with_ticks(entity).unwrap(); Mut { value: &mut *component.cast::(), component_ticks: &mut *component_ticks, change_tick: self.change_tick, last_change_tick: self.last_change_tick, } } } } #[inline] unsafe fn table_fetch(&mut self, table_row: usize) -> Self::Item { Mut { value: &mut *self.table_components.as_ptr().add(table_row), component_ticks: &mut *self.table_ticks.add(table_row), change_tick: self.change_tick, last_change_tick: self.last_change_tick, } } } impl WorldQuery for Option { type Fetch = OptionFetch; type State = OptionState; } pub struct OptionFetch { fetch: T, matches: bool, } /// SAFE: OptionFetch is read only because T is read only unsafe impl ReadOnlyFetch for OptionFetch {} pub struct OptionState { state: T, } // SAFE: component access and archetype component access are properly updated according to the // internal Fetch unsafe impl FetchState for OptionState { fn init(world: &mut World) -> Self { Self { state: T::init(world), } } fn update_component_access(&self, access: &mut FilteredAccess) { self.state.update_component_access(access); } fn update_archetype_component_access( &self, archetype: &Archetype, access: &mut Access, ) { if self.state.matches_archetype(archetype) { self.state .update_archetype_component_access(archetype, access) } } fn matches_archetype(&self, _archetype: &Archetype) -> bool { true } fn matches_table(&self, _table: &Table) -> bool { true } } impl<'w, T: Fetch<'w>> Fetch<'w> for OptionFetch { type Item = Option; type State = OptionState; #[inline] fn is_dense(&self) -> bool { self.fetch.is_dense() } unsafe fn init( world: &World, state: &Self::State, last_change_tick: u32, change_tick: u32, ) -> Self { Self { fetch: T::init(world, &state.state, last_change_tick, change_tick), matches: false, } } #[inline] unsafe fn set_archetype( &mut self, state: &Self::State, archetype: &Archetype, tables: &Tables, ) { self.matches = state.state.matches_archetype(archetype); if self.matches { self.fetch.set_archetype(&state.state, archetype, tables); } } #[inline] unsafe fn set_table(&mut self, state: &Self::State, table: &Table) { self.matches = state.state.matches_table(table); if self.matches { self.fetch.set_table(&state.state, table); } } #[inline] unsafe fn archetype_fetch(&mut self, archetype_index: usize) -> Self::Item { if self.matches { Some(self.fetch.archetype_fetch(archetype_index)) } else { None } } #[inline] unsafe fn table_fetch(&mut self, table_row: usize) -> Self::Item { if self.matches { Some(self.fetch.table_fetch(table_row)) } else { None } } } /// Change trackers for component `T` #[derive(Clone)] pub struct ChangeTrackers { pub(crate) component_ticks: ComponentTicks, pub(crate) last_change_tick: u32, pub(crate) change_tick: u32, marker: PhantomData, } impl std::fmt::Debug for ChangeTrackers { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { f.debug_struct("ChangeTrackers") .field("component_ticks", &self.component_ticks) .field("last_change_tick", &self.last_change_tick) .field("change_tick", &self.change_tick) .finish() } } impl ChangeTrackers { /// Has this component been added since the last execution of this system. pub fn is_added(&self) -> bool { self.component_ticks .is_added(self.last_change_tick, self.change_tick) } /// Has this component been changed since the last execution of this system. pub fn is_changed(&self) -> bool { self.component_ticks .is_changed(self.last_change_tick, self.change_tick) } } impl WorldQuery for ChangeTrackers { type Fetch = ChangeTrackersFetch; type State = ChangeTrackersState; } pub struct ChangeTrackersState { component_id: ComponentId, storage_type: StorageType, marker: PhantomData, } // SAFE: component access and archetype component access are properly updated to reflect that T is // read unsafe impl FetchState for ChangeTrackersState { fn init(world: &mut World) -> Self { let component_info = world.components.get_or_insert_info::(); Self { component_id: component_info.id(), storage_type: component_info.storage_type(), marker: PhantomData, } } fn update_component_access(&self, access: &mut FilteredAccess) { if access.access().has_write(self.component_id) { panic!("ChangeTrackers<{}> conflicts with a previous access in this query. Shared access cannot coincide with exclusive access.", std::any::type_name::()); } access.add_read(self.component_id) } fn update_archetype_component_access( &self, archetype: &Archetype, access: &mut Access, ) { if let Some(archetype_component_id) = archetype.get_archetype_component_id(self.component_id) { access.add_read(archetype_component_id); } } fn matches_archetype(&self, archetype: &Archetype) -> bool { archetype.contains(self.component_id) } fn matches_table(&self, table: &Table) -> bool { table.has_column(self.component_id) } } pub struct ChangeTrackersFetch { storage_type: StorageType, table_ticks: *const ComponentTicks, entity_table_rows: *const usize, entities: *const Entity, sparse_set: *const ComponentSparseSet, marker: PhantomData, last_change_tick: u32, change_tick: u32, } /// SAFE: access is read only unsafe impl ReadOnlyFetch for ChangeTrackersFetch {} impl<'w, T: Component> Fetch<'w> for ChangeTrackersFetch { type Item = ChangeTrackers; type State = ChangeTrackersState; #[inline] fn is_dense(&self) -> bool { match self.storage_type { StorageType::Table => true, StorageType::SparseSet => false, } } unsafe fn init( world: &World, state: &Self::State, last_change_tick: u32, change_tick: u32, ) -> Self { let mut value = Self { storage_type: state.storage_type, table_ticks: ptr::null::(), entities: ptr::null::(), entity_table_rows: ptr::null::(), sparse_set: ptr::null::(), marker: PhantomData, last_change_tick, change_tick, }; if state.storage_type == StorageType::SparseSet { value.sparse_set = world .storages() .sparse_sets .get(state.component_id) .unwrap(); } value } #[inline] unsafe fn set_archetype( &mut self, state: &Self::State, archetype: &Archetype, tables: &Tables, ) { match state.storage_type { StorageType::Table => { self.entity_table_rows = archetype.entity_table_rows().as_ptr(); let column = tables[archetype.table_id()] .get_column(state.component_id) .unwrap(); self.table_ticks = column.get_ticks_mut_ptr().cast::(); } StorageType::SparseSet => self.entities = archetype.entities().as_ptr(), } } #[inline] unsafe fn set_table(&mut self, state: &Self::State, table: &Table) { self.table_ticks = table .get_column(state.component_id) .unwrap() .get_ticks_mut_ptr() .cast::(); } #[inline] unsafe fn archetype_fetch(&mut self, archetype_index: usize) -> Self::Item { match self.storage_type { StorageType::Table => { let table_row = *self.entity_table_rows.add(archetype_index); ChangeTrackers { component_ticks: *self.table_ticks.add(table_row), marker: PhantomData, last_change_tick: self.last_change_tick, change_tick: self.change_tick, } } StorageType::SparseSet => { let entity = *self.entities.add(archetype_index); ChangeTrackers { component_ticks: *(*self.sparse_set).get_ticks(entity).unwrap(), marker: PhantomData, last_change_tick: self.last_change_tick, change_tick: self.change_tick, } } } } #[inline] unsafe fn table_fetch(&mut self, table_row: usize) -> Self::Item { ChangeTrackers { component_ticks: *self.table_ticks.add(table_row), marker: PhantomData, last_change_tick: self.last_change_tick, change_tick: self.change_tick, } } } macro_rules! impl_tuple_fetch { ($(($name: ident, $state: ident)),*) => { #[allow(non_snake_case)] impl<'a, $($name: Fetch<'a>),*> Fetch<'a> for ($($name,)*) { type Item = ($($name::Item,)*); type State = ($($name::State,)*); unsafe fn init(_world: &World, state: &Self::State, _last_change_tick: u32, _change_tick: u32) -> Self { let ($($name,)*) = state; ($($name::init(_world, $name, _last_change_tick, _change_tick),)*) } #[inline] fn is_dense(&self) -> bool { let ($($name,)*) = self; true $(&& $name.is_dense())* } #[inline] unsafe fn set_archetype(&mut self, _state: &Self::State, _archetype: &Archetype, _tables: &Tables) { let ($($name,)*) = self; let ($($state,)*) = _state; $($name.set_archetype($state, _archetype, _tables);)* } #[inline] unsafe fn set_table(&mut self, _state: &Self::State, _table: &Table) { let ($($name,)*) = self; let ($($state,)*) = _state; $($name.set_table($state, _table);)* } #[inline] unsafe fn table_fetch(&mut self, _table_row: usize) -> Self::Item { let ($($name,)*) = self; ($($name.table_fetch(_table_row),)*) } #[inline] unsafe fn archetype_fetch(&mut self, _archetype_index: usize) -> Self::Item { let ($($name,)*) = self; ($($name.archetype_fetch(_archetype_index),)*) } } // SAFE: update_component_access and update_archetype_component_access are called for each item in the tuple #[allow(non_snake_case)] unsafe impl<$($name: FetchState),*> FetchState for ($($name,)*) { fn init(_world: &mut World) -> Self { ($($name::init(_world),)*) } fn update_component_access(&self, _access: &mut FilteredAccess) { let ($($name,)*) = self; $($name.update_component_access(_access);)* } fn update_archetype_component_access(&self, _archetype: &Archetype, _access: &mut Access) { let ($($name,)*) = self; $($name.update_archetype_component_access(_archetype, _access);)* } fn matches_archetype(&self, _archetype: &Archetype) -> bool { let ($($name,)*) = self; true $(&& $name.matches_archetype(_archetype))* } fn matches_table(&self, _table: &Table) -> bool { let ($($name,)*) = self; true $(&& $name.matches_table(_table))* } } impl<$($name: WorldQuery),*> WorldQuery for ($($name,)*) { type Fetch = ($($name::Fetch,)*); type State = ($($name::State,)*); } /// SAFE: each item in the tuple is read only unsafe impl<$($name: ReadOnlyFetch),*> ReadOnlyFetch for ($($name,)*) {} }; } all_tuples!(impl_tuple_fetch, 0, 15, F, S);