diff --git a/crates/bevy_ecs/macros/src/fetch.rs b/crates/bevy_ecs/macros/src/fetch.rs index 6d0e3d509428ca..734057b7bf463f 100644 --- a/crates/bevy_ecs/macros/src/fetch.rs +++ b/crates/bevy_ecs/macros/src/fetch.rs @@ -186,7 +186,6 @@ pub fn derive_world_query_impl(ast: DeriveInput) -> TokenStream { #(#(#ignored_field_attrs)* #ignored_field_visibilities #ignored_field_idents: #ignored_field_types,)* } - #[derive(Clone)] #[doc(hidden)] #visibility struct #fetch_struct_name #user_impl_generics_with_world #user_where_clauses_with_world { #(#field_idents: <#field_types as #path::query::WorldQueryGats<'__w>>::Fetch,)* @@ -239,6 +238,19 @@ pub fn derive_world_query_impl(ast: DeriveInput) -> TokenStream { } } + unsafe fn clone_fetch<'__w>( + _fetch: &>::Fetch + ) -> >::Fetch { + #fetch_struct_name { + #( + #field_idents: <#field_types>::clone_fetch(& _fetch. #field_idents), + )* + #( + #ignored_field_idents: Default::default(), + )* + } + } + const IS_DENSE: bool = true #(&& <#field_types>::IS_DENSE)*; const IS_ARCHETYPAL: bool = true #(&& <#field_types>::IS_ARCHETYPAL)*; diff --git a/crates/bevy_ecs/src/query/fetch.rs b/crates/bevy_ecs/src/query/fetch.rs index 21fab0db52367e..9cfdcf36db47a1 100644 --- a/crates/bevy_ecs/src/query/fetch.rs +++ b/crates/bevy_ecs/src/query/fetch.rs @@ -350,6 +350,16 @@ pub unsafe trait WorldQuery: for<'w> WorldQueryGats<'w> { change_tick: u32, ) -> >::Fetch; + /// This function is safe to call if `Self:ReadOnlyWorldQuery` holds. + /// + /// # Safety + /// While calling this method on its own cannot cause UB it is marked `unsafe` as the caller must ensure + /// that the returned value is not used in any way that would cause two `QueryItem` for the same + /// `archetype_index` or `table_row` to be alive at the same time. + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch; + /// 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, [`WorldQuery::set_table`] and [`WorldQuery::table_fetch`] @@ -482,7 +492,6 @@ pub type ROQueryFetch<'w, Q> = QueryFetch<'w, ::ReadOnly>; pub type ROQueryItem<'w, Q> = QueryItem<'w, ::ReadOnly>; #[doc(hidden)] -#[derive(Clone)] pub struct EntityFetch<'w> { entities: Option>, } @@ -509,6 +518,14 @@ unsafe impl WorldQuery for Entity { EntityFetch { entities: None } } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + EntityFetch { + entities: fetch.entities, + } + } + #[inline] unsafe fn set_archetype<'w>( fetch: &mut EntityFetch<'w>, @@ -616,6 +633,17 @@ unsafe impl WorldQuery for &T { } } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + ReadFetch { + table_components: fetch.table_components, + entity_table_rows: fetch.entity_table_rows, + entities: fetch.entities, + sparse_set: fetch.sparse_set, + } + } + #[inline] unsafe fn set_archetype<'w>( fetch: &mut ReadFetch<'w, T>, @@ -713,17 +741,6 @@ unsafe impl WorldQuery for &T { } } -impl Clone for ReadFetch<'_, T> { - fn clone(&self) -> Self { - Self { - table_components: self.table_components, - entity_table_rows: self.entity_table_rows, - entities: self.entities, - sparse_set: self.sparse_set, - } - } -} - /// SAFETY: access is read only unsafe impl ReadOnlyWorldQuery for &T {} @@ -782,6 +799,20 @@ unsafe impl<'__w, T: Component> WorldQuery for &'__w mut T { } } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + WriteFetch { + table_components: fetch.table_components, + table_ticks: fetch.table_ticks, + entities: fetch.entities, + entity_table_rows: fetch.entity_table_rows, + sparse_set: fetch.sparse_set, + last_change_tick: fetch.last_change_tick, + change_tick: fetch.change_tick, + } + } + #[inline] unsafe fn set_archetype<'w>( fetch: &mut WriteFetch<'w, T>, @@ -908,20 +939,6 @@ unsafe impl<'__w, T: Component> WorldQuery for &'__w mut T { } } -impl Clone for WriteFetch<'_, T> { - fn clone(&self) -> Self { - Self { - table_components: self.table_components, - table_ticks: self.table_ticks, - entities: self.entities, - entity_table_rows: self.entity_table_rows, - sparse_set: self.sparse_set, - last_change_tick: self.last_change_tick, - change_tick: self.change_tick, - } - } -} - impl<'w, T: Component> WorldQueryGats<'w> for &mut T { type Fetch = WriteFetch<'w, T>; type Item = Mut<'w, T>; @@ -932,17 +949,6 @@ pub struct OptionFetch<'w, T: WorldQuery> { fetch: >::Fetch, matches: bool, } -impl<'w, T: WorldQuery> Clone for OptionFetch<'w, T> -where - >::Fetch: Clone, -{ - fn clone(&self) -> Self { - Self { - fetch: self.fetch.clone(), - matches: self.matches, - } - } -} // SAFETY: defers to soundness of `T: WorldQuery` impl unsafe impl WorldQuery for Option { @@ -969,6 +975,15 @@ unsafe impl WorldQuery for Option { } } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + OptionFetch { + fetch: T::clone_fetch(&fetch.fetch), + matches: fetch.matches, + } + } + #[inline] unsafe fn set_archetype<'w>( fetch: &mut OptionFetch<'w, T>, @@ -1086,7 +1101,6 @@ impl<'w, T: WorldQuery> WorldQueryGats<'w> for Option { /// } /// # bevy_ecs::system::assert_is_system(print_moving_objects_system); /// ``` -#[derive(Clone)] pub struct ChangeTrackers { pub(crate) component_ticks: ComponentTicks, pub(crate) last_change_tick: u32, @@ -1094,6 +1108,18 @@ pub struct ChangeTrackers { marker: PhantomData, } +impl Clone for ChangeTrackers { + fn clone(&self) -> Self { + Self { + component_ticks: self.component_ticks, + last_change_tick: self.last_change_tick, + change_tick: self.change_tick, + marker: PhantomData, + } + } +} +impl Copy for ChangeTrackers {} + impl std::fmt::Debug for ChangeTrackers { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { f.debug_struct("ChangeTrackers") @@ -1132,20 +1158,6 @@ pub struct ChangeTrackersFetch<'w, T> { change_tick: u32, } -impl Clone for ChangeTrackersFetch<'_, T> { - fn clone(&self) -> Self { - Self { - table_ticks: self.table_ticks, - entity_table_rows: self.entity_table_rows, - entities: self.entities, - sparse_set: self.sparse_set, - marker: self.marker, - last_change_tick: self.last_change_tick, - change_tick: self.change_tick, - } - } -} - // SAFETY: `ROQueryFetch` is the same as `QueryFetch` unsafe impl WorldQuery for ChangeTrackers { type ReadOnly = Self; @@ -1182,6 +1194,20 @@ unsafe impl WorldQuery for ChangeTrackers { } } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + ChangeTrackersFetch { + table_ticks: fetch.table_ticks, + entity_table_rows: fetch.entity_table_rows, + entities: fetch.entities, + sparse_set: fetch.sparse_set, + marker: fetch.marker, + last_change_tick: fetch.last_change_tick, + change_tick: fetch.change_tick, + } + } + #[inline] unsafe fn set_archetype<'w>( fetch: &mut ChangeTrackersFetch<'w, T>, @@ -1338,6 +1364,13 @@ macro_rules! impl_tuple_fetch { ($($name::init_fetch(_world, $name, _last_change_tick, _change_tick),)*) } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + let ($($name,)*) = &fetch; + ($($name::clone_fetch($name),)*) + } + const IS_DENSE: bool = true $(&& $name::IS_DENSE)*; const IS_ARCHETYPAL: bool = true $(&& $name::IS_ARCHETYPAL)*; @@ -1416,7 +1449,6 @@ macro_rules! impl_tuple_fetch { /// `Query>` is equivalent to `Query<(Option<&A>, Option<&B>, Option<&mut C>), (Or(With, With, With)>`. /// Each of the components in `T` is returned as an `Option`, as with `Option` queries. /// Entities are guaranteed to have at least one of the components in `T`. -#[derive(Clone)] pub struct AnyOf(PhantomData); macro_rules! impl_anytuple_fetch { @@ -1448,6 +1480,13 @@ macro_rules! impl_anytuple_fetch { ($(($name::init_fetch(_world, $name, _last_change_tick, _change_tick), false),)*) } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + let ($($name,)*) = &fetch; + ($(($name::clone_fetch(& $name.0), $name.1),)*) + } + const IS_DENSE: bool = true $(&& $name::IS_DENSE)*; const IS_ARCHETYPAL: bool = true $(&& $name::IS_ARCHETYPAL)*; @@ -1580,6 +1619,11 @@ unsafe impl WorldQuery for NopWorldQuery { ) { } + unsafe fn clone_fetch<'w>( + _fetch: &>::Fetch, + ) -> >::Fetch { + } + #[inline(always)] unsafe fn set_archetype( _fetch: &mut (), diff --git a/crates/bevy_ecs/src/query/filter.rs b/crates/bevy_ecs/src/query/filter.rs index 960c6d4bcd7f53..238964aaa909b8 100644 --- a/crates/bevy_ecs/src/query/filter.rs +++ b/crates/bevy_ecs/src/query/filter.rs @@ -66,6 +66,11 @@ unsafe impl WorldQuery for With { ) { } + unsafe fn clone_fetch<'w>( + _fetch: &>::Fetch, + ) -> >::Fetch { + } + const IS_DENSE: bool = { match T::Storage::STORAGE_TYPE { StorageType::Table => true, @@ -173,6 +178,11 @@ unsafe impl WorldQuery for Without { ) { } + unsafe fn clone_fetch<'w>( + _fetch: &>::Fetch, + ) -> >::Fetch { + } + const IS_DENSE: bool = { match T::Storage::STORAGE_TYPE { StorageType::Table => true, @@ -271,7 +281,6 @@ unsafe impl ReadOnlyWorldQuery for Without {} /// } /// # bevy_ecs::system::assert_is_system(print_cool_entity_system); /// ``` -#[derive(Clone, Copy)] pub struct Or(pub T); #[doc(hidden)] @@ -279,18 +288,6 @@ pub struct OrFetch<'w, T: WorldQuery> { fetch: QueryFetch<'w, T>, matches: bool, } -impl<'w, T: WorldQuery> Copy for OrFetch<'w, T> where QueryFetch<'w, T>: Copy {} -impl<'w, T: WorldQuery> Clone for OrFetch<'w, T> -where - QueryFetch<'w, T>: Clone, -{ - fn clone(&self) -> Self { - Self { - fetch: self.fetch.clone(), - matches: self.matches, - } - } -} macro_rules! impl_query_filter_tuple { ($(($filter: ident, $state: ident)),*) => { @@ -326,6 +323,18 @@ macro_rules! impl_query_filter_tuple { },)*) } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + let ($($filter,)*) = &fetch; + ($( + OrFetch { + fetch: $filter::clone_fetch(&$filter.fetch), + matches: $filter.matches, + }, + )*) + } + #[inline] unsafe fn set_table<'w>(fetch: &mut >::Fetch, state: &Self::State, table: &'w Table) { let ($($filter,)*) = fetch; @@ -471,6 +480,20 @@ macro_rules! impl_tick_filter { } } + unsafe fn clone_fetch<'w>( + fetch: &>::Fetch, + ) -> >::Fetch { + $fetch_name { + table_ticks: fetch.table_ticks, + entity_table_rows: fetch.entity_table_rows, + entities: fetch.entities, + sparse_set: fetch.sparse_set, + last_change_tick: fetch.last_change_tick, + change_tick: fetch.change_tick, + marker: PhantomData, + } + } + const IS_DENSE: bool = { match T::Storage::STORAGE_TYPE { StorageType::Table => true, @@ -565,22 +588,6 @@ macro_rules! impl_tick_filter { /// SAFETY: read-only access unsafe impl ReadOnlyWorldQuery for $name {} - - impl Clone for $fetch_name<'_, T> { - fn clone(&self) -> Self { - Self { - table_ticks: self.table_ticks.clone(), - entity_table_rows: self.entity_table_rows.clone(), - marker: self.marker.clone(), - entities: self.entities.clone(), - sparse_set: self.sparse_set.clone(), - last_change_tick: self.last_change_tick.clone(), - change_tick: self.change_tick.clone(), - } - } - } - - impl Copy for $fetch_name<'_, T> {} }; } diff --git a/crates/bevy_ecs/src/query/iter.rs b/crates/bevy_ecs/src/query/iter.rs index 1c8f7e92562d07..c10cd7c714b543 100644 --- a/crates/bevy_ecs/src/query/iter.rs +++ b/crates/bevy_ecs/src/query/iter.rs @@ -271,11 +271,7 @@ impl<'w, 's, Q: WorldQuery, F: WorldQuery, const K: usize> QueryCombinationIter< /// references to the same component, leading to unique reference aliasing. ///. /// It is always safe for shared access. - unsafe fn fetch_next_aliased_unchecked(&mut self) -> Option<[QueryItem<'w, Q>; K]> - where - QueryFetch<'w, Q>: Clone, - QueryFetch<'w, F>: Clone, - { + unsafe fn fetch_next_aliased_unchecked(&mut self) -> Option<[QueryItem<'w, Q>; K]> { if K == 0 { return None; } @@ -286,7 +282,7 @@ impl<'w, 's, Q: WorldQuery, F: WorldQuery, const K: usize> QueryCombinationIter< Some(_) => { // walk forward up to last element, propagating cursor state forward for j in (i + 1)..K { - self.cursors[j] = self.cursors[j - 1].clone(); + self.cursors[j] = self.cursors[j - 1].clone_cursor(); match self.cursors[j].next(self.tables, self.archetypes, self.query_state) { Some(_) => {} None if i > 0 => continue 'outer, @@ -312,11 +308,7 @@ impl<'w, 's, Q: WorldQuery, F: WorldQuery, const K: usize> QueryCombinationIter< /// Get next combination of queried components #[inline] - pub fn fetch_next(&mut self) -> Option<[QueryItem<'_, Q>; K]> - where - for<'a> QueryFetch<'a, Q>: Clone, - for<'a> QueryFetch<'a, F>: Clone, - { + pub fn fetch_next(&mut self) -> Option<[QueryItem<'_, Q>; K]> { // SAFETY: we are limiting the returned reference to self, // making sure this method cannot be called multiple times without getting rid // of any previously returned unique references first, thus preventing aliasing. @@ -332,9 +324,6 @@ impl<'w, 's, Q: WorldQuery, F: WorldQuery, const K: usize> QueryCombinationIter< // multiple times would allow multiple owned references to the same data to exist. impl<'w, 's, Q: ReadOnlyWorldQuery, F: ReadOnlyWorldQuery, const K: usize> Iterator for QueryCombinationIter<'w, 's, Q, F, K> -where - QueryFetch<'w, Q>: Clone, - QueryFetch<'w, F>: Clone, { type Item = [QueryItem<'w, Q>; K]; @@ -398,9 +387,6 @@ where impl<'w, 's, Q: ReadOnlyWorldQuery, F: ReadOnlyWorldQuery + ArchetypeFilter, const K: usize> ExactSizeIterator for QueryCombinationIter<'w, 's, Q, F, K> -where - QueryFetch<'w, Q>: Clone, - QueryFetch<'w, F>: Clone, { /// Returns the exact length of the iterator. /// @@ -414,9 +400,6 @@ where // This is correct as [`QueryCombinationIter`] always returns `None` once exhausted. impl<'w, 's, Q: ReadOnlyWorldQuery, F: ReadOnlyWorldQuery, const K: usize> FusedIterator for QueryCombinationIter<'w, 's, Q, F, K> -where - QueryFetch<'w, Q>: Clone, - QueryFetch<'w, F>: Clone, { } @@ -432,17 +415,20 @@ struct QueryIterationCursor<'w, 's, Q: WorldQuery, F: WorldQuery> { phantom: PhantomData, } -impl<'w, 's, Q: WorldQuery, F: WorldQuery> Clone for QueryIterationCursor<'w, 's, Q, F> -where - QueryFetch<'w, Q>: Clone, - QueryFetch<'w, F>: Clone, -{ - fn clone(&self) -> Self { +impl<'w, 's, Q: WorldQuery, F: WorldQuery> QueryIterationCursor<'w, 's, Q, F> { + /// This function is safe to call if `(Q, F): ReadOnlyWorldQuery` holds. + /// + /// # Safety + /// While calling this method on its own cannot cause UB it is marked `unsafe` as the caller must ensure + /// that the returned value is not used in any way that would cause two `QueryItem` for the same + /// `archetype_index` or `table_row` to be alive at the same time. + unsafe fn clone_cursor(&self) -> Self { Self { table_id_iter: self.table_id_iter.clone(), archetype_id_iter: self.archetype_id_iter.clone(), - fetch: self.fetch.clone(), - filter: self.filter.clone(), + // SAFETY: upheld by caller invariants + fetch: Q::clone_fetch(&self.fetch), + filter: F::clone_fetch(&self.filter), current_len: self.current_len, current_index: self.current_index, phantom: PhantomData, diff --git a/crates/bevy_ecs/src/query/mod.rs b/crates/bevy_ecs/src/query/mod.rs index 3efb6a5dc10176..8d2e282504d759 100644 --- a/crates/bevy_ecs/src/query/mod.rs +++ b/crates/bevy_ecs/src/query/mod.rs @@ -21,7 +21,7 @@ pub(crate) unsafe fn debug_checked_unreachable() -> ! { mod tests { use super::WorldQuery; use crate::prelude::{AnyOf, Entity, Or, QueryState, With, Without}; - use crate::query::{ArchetypeFilter, QueryCombinationIter, QueryFetch}; + use crate::query::{ArchetypeFilter, QueryCombinationIter}; use crate::system::{IntoSystem, Query, System, SystemState}; use crate::{self as bevy_ecs, component::Component, world::World}; use std::any::type_name; @@ -70,8 +70,6 @@ mod tests { Q: WorldQuery, F: WorldQuery, F::ReadOnly: ArchetypeFilter, - for<'w> QueryFetch<'w, Q::ReadOnly>: Clone, - for<'w> QueryFetch<'w, F::ReadOnly>: Clone, { let mut query = world.query_filtered::(); let iter = query.iter_combinations::(world); @@ -83,8 +81,6 @@ mod tests { Q: WorldQuery, F: WorldQuery, F::ReadOnly: ArchetypeFilter, - for<'w> QueryFetch<'w, Q::ReadOnly>: Clone, - for<'w> QueryFetch<'w, F::ReadOnly>: Clone, { let mut query = world.query_filtered::(); let iter = query.iter(world);