Skip to content

Commit

Permalink
Remove panics from SparseSet construction
Browse files Browse the repository at this point in the history
  • Loading branch information
james7132 committed Mar 17, 2024
1 parent 6304b0f commit 1c0b8da
Show file tree
Hide file tree
Showing 3 changed files with 17 additions and 12 deletions.
1 change: 0 additions & 1 deletion crates/bevy_ecs/src/archetype.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,6 @@ use crate::{
query::{DebugCheckedUnwrap, UnsafeVecExtensions},
storage::{ImmutableSparseSet, SparseArray, SparseSet, SparseSetIndex, TableId, TableRow},
};
use core::arch;
use std::{
hash::Hash,
ops::{Index, RangeFrom},
Expand Down
24 changes: 15 additions & 9 deletions crates/bevy_ecs/src/storage/sparse_set.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
use crate::{
component::{ComponentId, ComponentInfo, ComponentTicks, Tick, TickCells},
entity::Entity,
query::UnsafeVecExtensions,
query::{DebugCheckedUnwrap, UnsafeVecExtensions},
storage::{Column, TableRow},
};
use bevy_ptr::{OwningPtr, Ptr};
Expand Down Expand Up @@ -289,12 +289,14 @@ impl ComponentSparseSet {
// SAFETY: dense_index was just removed from `sparse`, which ensures that it is valid
let (value, _) = unsafe { self.dense.swap_remove_and_forget_unchecked(dense_index) };
if !is_last {
let swapped_entity = self.entities[dense_index.as_usize()];
// SAFETY: This index was just swapped to above, it must be valid.
let swapped_entity = unsafe { self.entities.get_unchecked(dense_index.as_usize()) };
#[cfg(not(debug_assertions))]
let index = swapped_entity;
let index = *swapped_entity;
#[cfg(debug_assertions)]
let index = swapped_entity.index();
*self.sparse.get_mut(index).unwrap() = dense_index;
// SAFETY: if the sparse index points to something in the dense vec, it exists
unsafe { *self.sparse.get_mut(index).debug_checked_unwrap() = dense_index }
}
value
})
Expand All @@ -316,12 +318,14 @@ impl ComponentSparseSet {
self.dense.swap_remove_unchecked(dense_index);
}
if !is_last {
let swapped_entity = self.entities[dense_index.as_usize()];
// SAFETY: This index was just swapped to above, it must be valid.
let swapped_entity = unsafe { self.entities.get_unchecked(dense_index.as_usize()) };
#[cfg(not(debug_assertions))]
let index = swapped_entity;
let index = *swapped_entity;
#[cfg(debug_assertions)]
let index = swapped_entity.index();
*self.sparse.get_mut(index).unwrap() = dense_index;
// SAFETY: if the sparse index points to something in the dense vec, it exists
unsafe { *self.sparse.get_mut(index).debug_checked_unwrap() = dense_index }
}
true
} else {
Expand Down Expand Up @@ -508,8 +512,10 @@ impl<I: SparseSetIndex, V> SparseSet<I, V> {
// in the indicies Vec.
unsafe { self.indices.swap_remove_unchecked(index) };
if !is_last {
let swapped_index = self.indices[index].clone();
*self.sparse.get_mut(swapped_index).unwrap() = dense_index;
// SAFETY: This index was just swapped to above, it must be valid.
let swapped_index = unsafe { self.indices.get_unchecked(index).clone() };
// SAFETY: The swapped index must be valid in the sparse array.
unsafe { *self.sparse.get_mut(swapped_index).debug_checked_unwrap() = dense_index };
}
value
})
Expand Down
4 changes: 2 additions & 2 deletions crates/bevy_ecs/src/world/entity_ref.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1562,8 +1562,8 @@ impl<'w, 'a, T: Component> OccupiedEntry<'w, 'a, T> {
/// ```
#[inline]
pub fn into_mut(self) -> Mut<'a, T> {
// This shouldn't panic because if we have an OccupiedEntry the component must exist.
self.entity_world.get_mut().unwrap()
// SAFETY: If we have an OccupiedEntry the component must exist.
unsafe { self.entity_world.get_mut().debug_checked_unwrap() }
}

/// Replaces the component of the entry.
Expand Down

0 comments on commit 1c0b8da

Please sign in to comment.