Skip to content

Commit

Permalink
Invalidate some bind group on buffer reallocate
Browse files Browse the repository at this point in the history
Invalidate the sim bind group when the particle buffer is re-allocated. Also
always re-create that bind group for now, as we don't invalidate on binding
offset/size changes. With that, revert the fix for #330, which shouldn't be
needed because the buffer is indeed contiguous.
  • Loading branch information
djeedai committed Oct 17, 2024
1 parent ee39c5a commit fa39723
Show file tree
Hide file tree
Showing 6 changed files with 88 additions and 60 deletions.
4 changes: 2 additions & 2 deletions examples/utils/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,16 @@

use std::{fmt::Display, num::NonZeroU8};

use crate::prelude::*;
use bevy::{
log::LogPlugin,
prelude::*,
render::{settings::WgpuSettings, RenderPlugin},
};

#[cfg(feature = "examples_world_inspector")]
use bevy_inspector_egui::quick::WorldInspectorPlugin;

use crate::prelude::*;

/// Helper system to enable closing the example application by pressing the
/// escape key (ESC).
pub fn close_on_esc(mut ev_app_exit: EventWriter<AppExit>, input: Res<ButtonInput<KeyCode>>) {
Expand Down
8 changes: 3 additions & 5 deletions src/asset.rs
Original file line number Diff line number Diff line change
@@ -1,18 +1,16 @@
use std::ops::Deref;

#[cfg(feature = "serde")]
use bevy::asset::{io::Reader, AssetLoader, AsyncReadExt, LoadContext};
use bevy::{
asset::Asset,
reflect::Reflect,
utils::{default, HashSet},
};

#[cfg(feature = "serde")]
use bevy::asset::{io::Reader, AssetLoader, AsyncReadExt, LoadContext};
use serde::{Deserialize, Serialize};
#[cfg(feature = "serde")]
use thiserror::Error;

use serde::{Deserialize, Serialize};

use crate::{
modifier::{Modifier, RenderModifier},
ExprHandle, GroupedModifier, ModifierContext, Module, ParticleGroupSet, ParticleLayout,
Expand Down
5 changes: 2 additions & 3 deletions src/plugin.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ use bevy::{
time::{time_system, TimeSystem},
};

#[cfg(feature = "serde")]
use crate::asset::EffectAssetLoader;
use crate::{
asset::EffectAsset,
compile_effects, gather_removed_effects,
Expand All @@ -34,9 +36,6 @@ use crate::{
RemovedEffectsEvent, Spawner,
};

#[cfg(feature = "serde")]
use crate::asset::EffectAssetLoader;

/// Labels for the Hanabi systems.
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, SystemSet)]
pub enum EffectSystems {
Expand Down
42 changes: 38 additions & 4 deletions src/render/effect_cache.rs
Original file line number Diff line number Diff line change
Expand Up @@ -454,9 +454,14 @@ impl EffectBuffer {
render_device: &RenderDevice,
group_binding: BufferBinding,
) {
if self.simulate_bind_group.is_some() {
return;
}
// Note: We can't cache this bind group, because not only the group buffer can
// get reallocated (we do track that), but the offset/size of the group binding
// can change each frame, and we don't track them (we re-allocate them
// linearly each frame). So just re-create that bind group each frame
// too.
// FIXME: Check if Buffer::id is unique for the buffer lifetime. Also we could
// track the offset/size manually to check for a change, and
// auto-invalidate.

let layout = self.particle_layout_bind_group_sim();
let label = format!("hanabi:bind_group_sim_batch{}", buffer_index);
Expand Down Expand Up @@ -489,11 +494,28 @@ impl EffectBuffer {
self.simulate_bind_group = Some(bind_group);
}

/// Invalidate any existing simulate bind group.
///
/// Invalidate any existing bind group previously created by
/// [`create_sim_bind_group()`], generally because a buffer was
/// re-allocated. This forces a re-creation of the bind group
/// next time [`create_sim_bind_group()`] is called.
///
/// [`create_sim_bind_group()`]: self::EffectBuffer::create_sim_bind_group
fn invalidate_sim_bind_group(&mut self) {
self.simulate_bind_group = None;
}

/// Return the cached bind group for the init and update passes.
///
/// This is the per-buffer bind group at binding @1 which binds all
/// per-buffer resources shared by all effect instances batched in a single
/// buffer.
/// buffer. The bind group is created by [`create_sim_bind_group()`], and
/// cached until a call to [`invalidate_sim_bind_group()`] clears the cached
/// reference.
///
/// [`create_sim_bind_group()`]: self::EffectBuffer::create_sim_bind_group
/// [`invalidate_sim_bind_group()`]: self::EffectBuffer::invalidate_sim_bind_group
pub fn sim_bind_group(&self) -> Option<&BindGroup> {
self.simulate_bind_group.as_ref()
}
Expand Down Expand Up @@ -730,6 +752,18 @@ impl EffectCache {
&mut self.buffers
}

/// Invalidate all the "simulation" bind groups for all buffers.
///
/// This iterates over all valid buffers and calls
/// [`EffectBuffer::invalidate_sim_bind_group()`] on each one.
pub fn invalidate_sim_bind_groups(&mut self) {
for buffer in &mut self.buffers {
if let Some(buffer) = buffer {
buffer.invalidate_sim_bind_group();
}
}
}

pub fn insert(
&mut self,
asset: Handle<EffectAsset>,
Expand Down
78 changes: 40 additions & 38 deletions src/render/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2336,6 +2336,10 @@ pub(crate) fn prepare_effects(
.write_buffer(&render_device, &render_queue)
{
// The buffer changed; invalidate all bind groups for all effects.
effect_cache.invalidate_sim_bind_groups();

// Note: effects_meta.dr_indirect_bind_group is re-created each frame,
// no need to clear explicitly.
}

// Update simulation parameters
Expand Down Expand Up @@ -3249,7 +3253,6 @@ pub(crate) fn prepare_bind_groups(
};

// Bind group for the init compute shader to simulate particles.
// TODO - move this creation in RenderSet::PrepareBindGroups
effect_buffer.create_sim_bind_group(
effect_batches.buffer_index,
&render_device,
Expand Down Expand Up @@ -3644,7 +3647,7 @@ impl Node for VfxSimulateNode {
.write_buffer(render_context.command_encoder());

// Compute init pass
// let mut total_group_count = 0;
let mut total_group_count = 0;
{
let mut compute_pass =
render_context
Expand All @@ -3666,10 +3669,7 @@ impl Node for VfxSimulateNode {
.dispatch_buffer_indices
.first_render_group_dispatch_buffer_index;

// FIXME - Currently we unconditionally count all groups because the dispatch
// pass always runs on all groups. We should consider if it's worth skipping
// e.g. dormant or finished effects at the cost of extra complexity.
// total_group_count += batches.group_batches.len() as u32;
total_group_count += batches.group_batches.len() as u32;

let Some(init_pipeline) =
pipeline_cache.get_compute_pipeline(batches.init_pipeline_id)
Expand Down Expand Up @@ -3792,38 +3792,40 @@ impl Node for VfxSimulateNode {
timestamp_writes: None,
});

// Dispatch indirect dispatch compute job
trace!("record commands for indirect dispatch pipeline...");

// FIXME - The `vfx_indirect` shader assumes a contiguous array of ParticleGroup
// structures. So we need to pass the full array size, and we
// just update the unused groups for nothing. Otherwise we might
// update some unused group and miss some used ones, if there's any gap
// in the array.
const WORKGROUP_SIZE: u32 = 64;
let total_group_count = effects_meta.particle_group_buffer.len() as u32;
let workgroup_count = (total_group_count + WORKGROUP_SIZE - 1) / WORKGROUP_SIZE;

// Setup compute pass
compute_pass.set_pipeline(&dispatch_indirect_pipeline.pipeline);
compute_pass.set_bind_group(
0,
// FIXME - got some unwrap() panic here, investigate... possibly race
// condition!
effects_meta.dr_indirect_bind_group.as_ref().unwrap(),
&[],
);
compute_pass.set_bind_group(
1,
effects_meta.sim_params_bind_group.as_ref().unwrap(),
&[],
);
compute_pass.dispatch_workgroups(workgroup_count, 1, 1);
trace!(
"indirect dispatch compute dispatched: num_batches={} workgroup_count={}",
total_group_count,
workgroup_count
);
// Dispatch indirect dispatch compute job.
// FIXME - we do an explicit check again here because there was an unwrap()
// which triggered some panic. Because we also check above, this
// means there's likely a race condition hidden somewhere.
if let (
Some(dr_indirect_bind_group),
Some(sim_params_bind_group),
) = (
effects_meta.dr_indirect_bind_group.as_ref(),
effects_meta.sim_params_bind_group.as_ref(),
) {
trace!("record commands for indirect dispatch pipeline for {total_group_count} groups...");

const WORKGROUP_SIZE: u32 = 64;
let workgroup_count = (total_group_count + WORKGROUP_SIZE - 1) / WORKGROUP_SIZE;

// Setup compute pass
compute_pass.set_pipeline(&dispatch_indirect_pipeline.pipeline);
compute_pass.set_bind_group(0, dr_indirect_bind_group, &[]);
compute_pass.set_bind_group(1, sim_params_bind_group, &[]);
compute_pass.dispatch_workgroups(workgroup_count, 1, 1);
trace!(
"indirect dispatch compute dispatched: num_batches={} workgroup_count={}",
total_group_count,
workgroup_count
);
} else {
if effects_meta.dr_indirect_bind_group.is_none() {
warn!("Missing indirect dispatch bind group. This is a bug.");
}
if effects_meta.sim_params_bind_group.is_none() {
warn!("Missing indirect dispatch SimParams bind group. This is a bug.");
}
}
}

// Compute update pass
Expand Down
11 changes: 3 additions & 8 deletions src/render/vfx_indirect.wgsl
Original file line number Diff line number Diff line change
Expand Up @@ -21,14 +21,9 @@ fn main(@builtin(global_invocation_id) global_invocation_id: vec3<u32>) {

// Cap at maximum number of groups to process
let index = global_invocation_id.x;

// FIXME - the group_buffer array has gaps, we can't limit to the number of effects in use
// otherwise we'll miss some and process the unused gaps instead of some active ones.
// Since all writes below are idempotent, except the ping/pong one, there's no harm updating
// unused effect rows.
// if (index >= sim_params.num_groups) {
// return;
// }
if (index >= sim_params.num_groups) {
return;
}

// Cap at group array size, just for safety
if (index >= arrayLength(&group_buffer)) {
Expand Down

0 comments on commit fa39723

Please sign in to comment.