Skip to content

Commit be9b38e

Browse files
authored
implement UniqueEntitySlice (#17589)
# Objective Follow-up to #17549 and #16547. A large part of `Vec`s usefulness is behind its ability to be sliced, like sorting f.e., so we want the same to be possible for `UniqueEntityVec`. ## Solution Add a `UniqueEntitySlice` type. It is a wrapper around `[T]`, and itself a DST. Because `mem::swap` has a `Sized` bound, DSTs cannot be swapped, and we can freely hand out mutable subslices without worrying about the uniqueness invariant of the backing collection! `UniqueEntityVec` and the relevant `UniqueEntityIter`s now have methods and trait impls that return `UniqueEntitySlice`s. `UniqueEntitySlice` itself can deref into normal slices, which means we can avoid implementing the vast majority of immutable slice methods. Most of the remaining methods: - split a slice/collection in further unique subsections/slices - reorder the slice: `sort`, `rotate_*`, `swap` - construct/deconstruct/convert pointer-like types: `Box`, `Arc`, `Rc`, `Cow` - are comparison trait impls As this PR is already larger than I'd like, we leave several things to follow-ups: - `UniqueEntityArray` and the related slice methods that would return it - denoted by "chunk", "array_*" for iterators - Methods that return iterators with `UniqueEntitySlice` as their item - `windows`, `chunks` and `split` families - All methods that are capable of actively mutating individual elements. While they could be offered unsafely, subslicing makes their safety contract weird enough to warrant its own discussion. - `fill_with`, `swap_with_slice`, `iter_mut`, `split_first/last_mut`, `select_nth_unstable_*` Note that `Arc`, `Rc` and `Cow` are not fundamental types, so even if they contain `UniqueEntitySlice`, we cannot write direct trait impls for them. On top of that, `Cow` is not a receiver (like `self: Arc<Self>` is) so we cannot write inherent methods for it either.
1 parent 69b2ae8 commit be9b38e

File tree

4 files changed

+1305
-14
lines changed

4 files changed

+1305
-14
lines changed

crates/bevy_ecs/src/entity/entity_set.rs

+40-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ use core::{
1313
option, result,
1414
};
1515

16-
use super::Entity;
16+
use super::{Entity, UniqueEntitySlice};
1717

1818
use bevy_platform_support::sync::Arc;
1919

@@ -350,6 +350,7 @@ impl<I: EntitySetIterator> UniqueEntityIter<I> {
350350
Self { iter }
351351
}
352352
}
353+
353354
impl<I: Iterator<Item: TrustedEntityBorrow>> UniqueEntityIter<I> {
354355
/// Constructs a [`UniqueEntityIter`] from an iterator unsafely.
355356
///
@@ -359,6 +360,26 @@ impl<I: Iterator<Item: TrustedEntityBorrow>> UniqueEntityIter<I> {
359360
pub unsafe fn from_iterator_unchecked(iter: I) -> Self {
360361
Self { iter }
361362
}
363+
364+
/// Returns the inner `I`.
365+
pub fn into_inner(self) -> I {
366+
self.iter
367+
}
368+
369+
/// Returns a reference to the inner `I`.
370+
pub fn as_inner(&self) -> &I {
371+
&self.iter
372+
}
373+
374+
/// Returns a mutable reference to the inner `I`.
375+
///
376+
/// # Safety
377+
///
378+
/// `self` must always contain an iterator that yields unique elements,
379+
/// even while this reference is live.
380+
pub unsafe fn as_mut_inner(&mut self) -> &mut I {
381+
&mut self.iter
382+
}
362383
}
363384

364385
impl<I: Iterator<Item: TrustedEntityBorrow>> Iterator for UniqueEntityIter<I> {
@@ -394,6 +415,24 @@ impl<T, I: Iterator<Item: TrustedEntityBorrow> + AsRef<[T]>> AsRef<[T]> for Uniq
394415
}
395416
}
396417

418+
impl<T: TrustedEntityBorrow, I: Iterator<Item: TrustedEntityBorrow> + AsRef<[T]>>
419+
AsRef<UniqueEntitySlice<T>> for UniqueEntityIter<I>
420+
{
421+
fn as_ref(&self) -> &UniqueEntitySlice<T> {
422+
// SAFETY: All elements in the original slice are unique.
423+
unsafe { UniqueEntitySlice::from_slice_unchecked(self.iter.as_ref()) }
424+
}
425+
}
426+
427+
impl<T: TrustedEntityBorrow, I: Iterator<Item: TrustedEntityBorrow> + AsMut<[T]>>
428+
AsMut<UniqueEntitySlice<T>> for UniqueEntityIter<I>
429+
{
430+
fn as_mut(&mut self) -> &mut UniqueEntitySlice<T> {
431+
// SAFETY: All elements in the original slice are unique.
432+
unsafe { UniqueEntitySlice::from_slice_unchecked_mut(self.iter.as_mut()) }
433+
}
434+
}
435+
397436
// Default does not guarantee uniqueness, meaning `I` needs to be EntitySetIterator.
398437
impl<I: EntitySetIterator + Default> Default for UniqueEntityIter<I> {
399438
fn default() -> Self {

crates/bevy_ecs/src/entity/mod.rs

+4
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,10 @@ mod index_set;
6666
pub use index_map::EntityIndexMap;
6767
pub use index_set::EntityIndexSet;
6868

69+
mod unique_slice;
70+
71+
pub use unique_slice::*;
72+
6973
use crate::{
7074
archetype::{ArchetypeId, ArchetypeRow},
7175
identifier::{

0 commit comments

Comments
 (0)