Skip to content

Commit

Permalink
Better handle destroying textures and buffers
Browse files Browse the repository at this point in the history
Before this commit, explicitly destroying a texture or a buffer (without dropping it)
schedules the asynchronous destruction of its raw resources but does not actually mark
it as destroyed. This can cause some incorrect behavior, for example mapping a buffer
after destroying it does not cause a validation error (and later panics due to the
map callback being dropped without being called).

This Commit adds `Storage::take_and_mark_destroyed` for use in `destroy` methods.
Since it puts the resource in the error state, other methods properly catch that
the resource is no longer usable when attempting to access it and raise validation
errors.

There are other resource types that require similar treatment and will be addressed
in followup work.
  • Loading branch information
nical committed Nov 9, 2023
1 parent 4e65eca commit 221b2ee
Show file tree
Hide file tree
Showing 4 changed files with 84 additions and 5 deletions.
1 change: 1 addition & 0 deletions tests/tests/gpu.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ mod device;
mod encoder;
mod external_texture;
mod instance;
mod life_cycle;
mod occlusion_query;
mod partially_bounded_arrays;
mod pipeline;
Expand Down
63 changes: 63 additions & 0 deletions tests/tests/life_cycle.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
use wgpu_test::{gpu_test, FailureCase, GpuTestConfiguration, TestParameters};

#[gpu_test]
static BUFFER_DESTROY: GpuTestConfiguration = GpuTestConfiguration::new()
.parameters(TestParameters::default().expect_fail(FailureCase::always()))
.run_sync(|ctx| {
let buffer = ctx.device.create_buffer(&wgpu::BufferDescriptor {
label: None,
size: 256,
usage: wgpu::BufferUsages::MAP_WRITE | wgpu::BufferUsages::COPY_SRC,
mapped_at_creation: false,
});

buffer.destroy();

buffer.destroy();

ctx.device.poll(wgpu::MaintainBase::Wait);

buffer
.slice(..)
.map_async(wgpu::MapMode::Write, move |_| {});

buffer.destroy();

ctx.device.poll(wgpu::MaintainBase::Wait);

buffer.destroy();

buffer.destroy();
});

#[gpu_test]
static TEXTURE_DESTROY: GpuTestConfiguration = GpuTestConfiguration::new().run_sync(|ctx| {
let texture = ctx.device.create_texture(&wgpu::TextureDescriptor {
label: None,
size: wgpu::Extent3d {
width: 128,
height: 128,
depth_or_array_layers: 1,
},
mip_level_count: 1,
sample_count: 1, // multisampling is not supported for clear
dimension: wgpu::TextureDimension::D2,
format: wgpu::TextureFormat::Rgba8Snorm,
usage: wgpu::TextureUsages::COPY_DST | wgpu::TextureUsages::TEXTURE_BINDING,
view_formats: &[],
});

texture.destroy();

texture.destroy();

ctx.device.poll(wgpu::MaintainBase::Wait);

texture.destroy();

ctx.device.poll(wgpu::MaintainBase::Wait);

texture.destroy();

texture.destroy();
});
10 changes: 5 additions & 5 deletions wgpu-core/src/device/global.rs
Original file line number Diff line number Diff line change
Expand Up @@ -495,8 +495,8 @@ impl<G: GlobalIdentityHandlerFactory> Global<G> {

log::trace!("Buffer::destroy {buffer_id:?}");
let (mut buffer_guard, _) = hub.buffers.write(&mut token);
let buffer = buffer_guard
.get_mut(buffer_id)
let mut buffer = buffer_guard
.take_and_mark_destroyed(buffer_id)
.map_err(|_| resource::DestroyError::Invalid)?;

let device = &mut device_guard[buffer.device_id.value];
Expand All @@ -506,7 +506,7 @@ impl<G: GlobalIdentityHandlerFactory> Global<G> {
| &BufferMapState::Init { .. }
| &BufferMapState::Active { .. }
=> {
self.buffer_unmap_inner(buffer_id, buffer, device)
self.buffer_unmap_inner(buffer_id, &mut buffer, device)
.unwrap_or(None)
}
_ => None,
Expand Down Expand Up @@ -800,8 +800,8 @@ impl<G: GlobalIdentityHandlerFactory> Global<G> {
let (mut device_guard, mut token) = hub.devices.write(&mut token);

let (mut texture_guard, _) = hub.textures.write(&mut token);
let texture = texture_guard
.get_mut(texture_id)
let mut texture = texture_guard
.take_and_mark_destroyed(texture_id)
.map_err(|_| resource::DestroyError::Invalid)?;

let device = &mut device_guard[texture.device_id.value];
Expand Down
15 changes: 15 additions & 0 deletions wgpu-core/src/storage.rs
Original file line number Diff line number Diff line change
Expand Up @@ -169,6 +169,21 @@ impl<T, I: id::TypedId> Storage<T, I> {
self.insert_impl(index as usize, Element::Error(epoch, label.to_string()))
}

pub(crate) fn take_and_mark_destroyed(&mut self, id: I) -> Result<T, InvalidId> {
let (index, epoch, _) = id.unzip();
match std::mem::replace(
&mut self.map[index as usize],
Element::Error(epoch, String::new()),
) {
Element::Vacant => panic!("Cannot mark a vacant resource destroyed"),
Element::Occupied(value, storage_epoch) => {
assert_eq!(epoch, storage_epoch);
Ok(value)
}
_ => Err(InvalidId),
}
}

pub(crate) fn force_replace(&mut self, id: I, value: T) {
let (index, epoch, _) = id.unzip();
self.map[index as usize] = Element::Occupied(value, epoch);
Expand Down

0 comments on commit 221b2ee

Please sign in to comment.