mirror of
https://github.com/bevyengine/bevy
synced 2025-01-14 06:04:04 +00:00
af9c945f40
![image](https://github.com/bevyengine/bevy/assets/47158642/dbb62645-f639-4f2b-b84b-26fd915c186d)
# Objective
- Add Screen space ambient occlusion (SSAO). SSAO approximates
small-scale, local occlusion of _indirect_ diffuse light between
objects. SSAO does not apply to direct lighting, such as point or
directional lights.
- This darkens creases, e.g. on staircases, and gives nice contact
shadows where objects meet, giving entities a more "grounded" feel.
- Closes https://github.com/bevyengine/bevy/issues/3632.
## Solution
- Implement the GTAO algorithm.
-
https://www.activision.com/cdn/research/Practical_Real_Time_Strategies_for_Accurate_Indirect_Occlusion_NEW%20VERSION_COLOR.pdf
-
https://blog.selfshadow.com/publications/s2016-shading-course/activision/s2016_pbs_activision_occlusion.pdf
- Source code heavily based on [Intel's
XeGTAO](0d177ce06b/Source/Rendering/Shaders/XeGTAO.hlsli
).
- Add an SSAO bevy example.
## Algorithm Overview
* Run a depth and normal prepass
* Create downscaled mips of the depth texture (preprocess_depths pass)
* GTAO pass - for each pixel, take several random samples from the
depth+normal buffers, reconstruct world position, raytrace in screen
space to estimate occlusion. Rather then doing completely random samples
on a hemisphere, you choose random _slices_ of the hemisphere, and then
can analytically compute the full occlusion of that slice. Also compute
edges based on depth differences here.
* Spatial denoise pass - bilateral blur, using edge detection to not
blur over edges. This is the final SSAO result.
* Main pass - if SSAO exists, sample the SSAO texture, and set occlusion
to be the minimum of ssao/material occlusion. This then feeds into the
rest of the PBR shader as normal.
---
## Future Improvements
- Maybe remove the low quality preset for now (too noisy)
- WebGPU fallback (see below)
- Faster depth->world position (see reverted code)
- Bent normals
- Try interleaved gradient noise or spatiotemporal blue noise
- Replace the spatial denoiser with a combined spatial+temporal denoiser
- Render at half resolution and use a bilateral upsample
- Better multibounce approximation
(https://drive.google.com/file/d/1SyagcEVplIm2KkRD3WQYSO9O0Iyi1hfy/view)
## Far-Future Performance Improvements
- F16 math (missing naga-wgsl support
https://github.com/gfx-rs/naga/issues/1884)
- Faster coordinate space conversion for normals
- Faster depth mipchain creation
(https://github.com/GPUOpen-Effects/FidelityFX-SPD) (wgpu/naga does not
currently support subgroup ops)
- Deinterleaved SSAO for better cache efficiency
(https://developer.nvidia.com/sites/default/files/akamai/gameworks/samples/DeinterleavedTexturing.pdf)
## Other Interesting Papers
- Visibility bitmask
(https://link.springer.com/article/10.1007/s00371-022-02703-y,
https://cdrinmatane.github.io/posts/cgspotlight-slides/)
- Screen space diffuse lighting
(https://github.com/Patapom/GodComplex/blob/master/Tests/TestHBIL/2018%20Mayaux%20-%20Horizon-Based%20Indirect%20Lighting%20(HBIL).pdf)
## Platform Support
* SSAO currently does not work on DirectX12 due to issues with wgpu and
naga:
* https://github.com/gfx-rs/wgpu/pull/3798
* https://github.com/gfx-rs/naga/pull/2353
* SSAO currently does not work on WebGPU because r16float is not a valid
storage texture format
https://gpuweb.github.io/gpuweb/wgsl/#storage-texel-formats. We can fix
this with a fallback to r32float.
---
## Changelog
- Added ScreenSpaceAmbientOcclusionSettings,
ScreenSpaceAmbientOcclusionQualityLevel, and
ScreenSpaceAmbientOcclusionBundle
---------
Co-authored-by: IceSentry <c.giguere42@gmail.com>
Co-authored-by: IceSentry <IceSentry@users.noreply.github.com>
Co-authored-by: Daniel Chia <danstryder@gmail.com>
Co-authored-by: Elabajaba <Elabajaba@users.noreply.github.com>
Co-authored-by: Robert Swain <robert.swain@gmail.com>
Co-authored-by: robtfm <50659922+robtfm@users.noreply.github.com>
Co-authored-by: Brandon Dyer <brandondyer64@gmail.com>
Co-authored-by: Edgar Geier <geieredgar@gmail.com>
Co-authored-by: Nicola Papale <nicopap@users.noreply.github.com>
Co-authored-by: Carter Anderson <mcanders1@gmail.com>
989 lines
36 KiB
Rust
989 lines
36 KiB
Rust
use bevy_app::{App, Plugin};
|
|
use bevy_asset::{load_internal_asset, HandleUntyped};
|
|
use bevy_core_pipeline::{
|
|
core_3d::CORE_3D,
|
|
prelude::Camera3d,
|
|
prepass::{DepthPrepass, NormalPrepass, ViewPrepassTextures},
|
|
};
|
|
use bevy_ecs::{
|
|
prelude::{Bundle, Component, Entity},
|
|
query::{QueryItem, With},
|
|
reflect::ReflectComponent,
|
|
schedule::IntoSystemConfigs,
|
|
system::{Commands, Query, Res, ResMut, Resource},
|
|
world::{FromWorld, World},
|
|
};
|
|
use bevy_reflect::{Reflect, TypeUuid};
|
|
use bevy_render::{
|
|
camera::{ExtractedCamera, TemporalJitter},
|
|
extract_component::ExtractComponent,
|
|
globals::{GlobalsBuffer, GlobalsUniform},
|
|
prelude::Camera,
|
|
render_graph::{NodeRunError, RenderGraphApp, RenderGraphContext, ViewNode, ViewNodeRunner},
|
|
render_resource::{
|
|
AddressMode, BindGroup, BindGroupDescriptor, BindGroupEntry, BindGroupLayout,
|
|
BindGroupLayoutDescriptor, BindGroupLayoutEntry, BindingResource, BindingType,
|
|
BufferBindingType, CachedComputePipelineId, ComputePassDescriptor,
|
|
ComputePipelineDescriptor, Extent3d, FilterMode, PipelineCache, Sampler,
|
|
SamplerBindingType, SamplerDescriptor, Shader, ShaderDefVal, ShaderStages, ShaderType,
|
|
SpecializedComputePipeline, SpecializedComputePipelines, StorageTextureAccess,
|
|
TextureDescriptor, TextureDimension, TextureFormat, TextureSampleType, TextureUsages,
|
|
TextureView, TextureViewDescriptor, TextureViewDimension,
|
|
},
|
|
renderer::{RenderAdapter, RenderContext, RenderDevice, RenderQueue},
|
|
texture::{CachedTexture, TextureCache},
|
|
view::{Msaa, ViewUniform, ViewUniformOffset, ViewUniforms},
|
|
Extract, ExtractSchedule, Render, RenderApp, RenderSet,
|
|
};
|
|
use bevy_utils::{
|
|
prelude::default,
|
|
tracing::{error, warn},
|
|
};
|
|
use std::mem;
|
|
|
|
pub mod draw_3d_graph {
|
|
pub mod node {
|
|
/// Label for the screen space ambient occlusion render node.
|
|
pub const SCREEN_SPACE_AMBIENT_OCCLUSION: &str = "screen_space_ambient_occlusion";
|
|
}
|
|
}
|
|
|
|
const PREPROCESS_DEPTH_SHADER_HANDLE: HandleUntyped =
|
|
HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 102258915420479);
|
|
const GTAO_SHADER_HANDLE: HandleUntyped =
|
|
HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 253938746510568);
|
|
const SPATIAL_DENOISE_SHADER_HANDLE: HandleUntyped =
|
|
HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 466162052558226);
|
|
const GTAO_UTILS_SHADER_HANDLE: HandleUntyped =
|
|
HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 366465052568786);
|
|
|
|
/// Plugin for screen space ambient occlusion.
|
|
pub struct ScreenSpaceAmbientOcclusionPlugin;
|
|
|
|
impl Plugin for ScreenSpaceAmbientOcclusionPlugin {
|
|
fn build(&self, app: &mut App) {
|
|
load_internal_asset!(
|
|
app,
|
|
PREPROCESS_DEPTH_SHADER_HANDLE,
|
|
"preprocess_depth.wgsl",
|
|
Shader::from_wgsl
|
|
);
|
|
load_internal_asset!(app, GTAO_SHADER_HANDLE, "gtao.wgsl", Shader::from_wgsl);
|
|
load_internal_asset!(
|
|
app,
|
|
SPATIAL_DENOISE_SHADER_HANDLE,
|
|
"spatial_denoise.wgsl",
|
|
Shader::from_wgsl
|
|
);
|
|
load_internal_asset!(
|
|
app,
|
|
GTAO_UTILS_SHADER_HANDLE,
|
|
"gtao_utils.wgsl",
|
|
Shader::from_wgsl
|
|
);
|
|
|
|
app.register_type::<ScreenSpaceAmbientOcclusionSettings>();
|
|
}
|
|
|
|
fn finish(&self, app: &mut App) {
|
|
let Ok(render_app) = app.get_sub_app_mut(RenderApp) else { return };
|
|
|
|
if !render_app
|
|
.world
|
|
.resource::<RenderAdapter>()
|
|
.get_texture_format_features(TextureFormat::R16Float)
|
|
.allowed_usages
|
|
.contains(TextureUsages::STORAGE_BINDING)
|
|
{
|
|
warn!("ScreenSpaceAmbientOcclusionPlugin not loaded. GPU lacks support: TextureFormat::R16Float does not support TextureUsages::STORAGE_BINDING.");
|
|
return;
|
|
}
|
|
|
|
if render_app
|
|
.world
|
|
.resource::<RenderDevice>()
|
|
.limits()
|
|
.max_storage_textures_per_shader_stage
|
|
< 5
|
|
{
|
|
warn!("ScreenSpaceAmbientOcclusionPlugin not loaded. GPU lacks support: Limits::max_storage_textures_per_shader_stage is less than 5.");
|
|
return;
|
|
}
|
|
|
|
render_app
|
|
.init_resource::<SsaoPipelines>()
|
|
.init_resource::<SpecializedComputePipelines<SsaoPipelines>>()
|
|
.add_systems(ExtractSchedule, extract_ssao_settings)
|
|
.add_systems(Render, prepare_ssao_textures.in_set(RenderSet::Prepare))
|
|
.add_systems(Render, prepare_ssao_pipelines.in_set(RenderSet::Prepare))
|
|
.add_systems(Render, queue_ssao_bind_groups.in_set(RenderSet::Queue))
|
|
.add_render_graph_node::<ViewNodeRunner<SsaoNode>>(
|
|
CORE_3D,
|
|
draw_3d_graph::node::SCREEN_SPACE_AMBIENT_OCCLUSION,
|
|
)
|
|
.add_render_graph_edges(
|
|
CORE_3D,
|
|
&[
|
|
// PREPASS -> SCREEN_SPACE_AMBIENT_OCCLUSION -> MAIN_PASS
|
|
bevy_core_pipeline::core_3d::graph::node::PREPASS,
|
|
draw_3d_graph::node::SCREEN_SPACE_AMBIENT_OCCLUSION,
|
|
bevy_core_pipeline::core_3d::graph::node::START_MAIN_PASS,
|
|
],
|
|
);
|
|
}
|
|
}
|
|
|
|
/// Bundle to apply screen space ambient occlusion.
|
|
#[derive(Bundle, Default)]
|
|
pub struct ScreenSpaceAmbientOcclusionBundle {
|
|
pub settings: ScreenSpaceAmbientOcclusionSettings,
|
|
pub depth_prepass: DepthPrepass,
|
|
pub normal_prepass: NormalPrepass,
|
|
}
|
|
|
|
/// Component to apply screen space ambient occlusion to a 3d camera.
|
|
///
|
|
/// Screen space ambient occlusion (SSAO) approximates small-scale,
|
|
/// local occlusion of _indirect_ diffuse light between objects, based on what's visible on-screen.
|
|
/// SSAO does not apply to direct lighting, such as point or directional lights.
|
|
///
|
|
/// This darkens creases, e.g. on staircases, and gives nice contact shadows
|
|
/// where objects meet, giving entities a more "grounded" feel.
|
|
///
|
|
/// # Usage Notes
|
|
///
|
|
/// Requires that you add [`ScreenSpaceAmbientOcclusionPlugin`] to your app,
|
|
/// and add the [`DepthPrepass`] and [`NormalPrepass`] components to your camera.
|
|
///
|
|
/// It strongly recommended that you use SSAO in conjunction with
|
|
/// TAA ([`bevy_core_pipeline::experimental::taa::TemporalAntiAliasSettings`]).
|
|
/// Doing so greatly reduces SSAO noise.
|
|
///
|
|
/// SSAO is not supported on `WebGL2`, and is not currently supported on `WebGPU` or `DirectX12`.
|
|
#[derive(Component, ExtractComponent, Reflect, PartialEq, Eq, Hash, Clone, Default)]
|
|
#[reflect(Component)]
|
|
pub struct ScreenSpaceAmbientOcclusionSettings {
|
|
pub quality_level: ScreenSpaceAmbientOcclusionQualityLevel,
|
|
}
|
|
|
|
#[derive(Reflect, PartialEq, Eq, Hash, Clone, Copy, Default)]
|
|
pub enum ScreenSpaceAmbientOcclusionQualityLevel {
|
|
Low,
|
|
Medium,
|
|
#[default]
|
|
High,
|
|
Ultra,
|
|
Custom {
|
|
/// Higher slice count means less noise, but worse performance.
|
|
slice_count: u32,
|
|
/// Samples per slice side is also tweakable, but recommended to be left at 2 or 3.
|
|
samples_per_slice_side: u32,
|
|
},
|
|
}
|
|
|
|
impl ScreenSpaceAmbientOcclusionQualityLevel {
|
|
fn sample_counts(&self) -> (u32, u32) {
|
|
match self {
|
|
Self::Low => (1, 2), // 4 spp (1 * (2 * 2)), plus optional temporal samples
|
|
Self::Medium => (2, 2), // 8 spp (2 * (2 * 2)), plus optional temporal samples
|
|
Self::High => (3, 3), // 18 spp (3 * (3 * 2)), plus optional temporal samples
|
|
Self::Ultra => (9, 3), // 54 spp (9 * (3 * 2)), plus optional temporal samples
|
|
Self::Custom {
|
|
slice_count: slices,
|
|
samples_per_slice_side,
|
|
} => (*slices, *samples_per_slice_side),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Default)]
|
|
struct SsaoNode {}
|
|
|
|
impl ViewNode for SsaoNode {
|
|
type ViewQuery = (
|
|
&'static ExtractedCamera,
|
|
&'static SsaoPipelineId,
|
|
&'static SsaoBindGroups,
|
|
&'static ViewUniformOffset,
|
|
);
|
|
|
|
fn run(
|
|
&self,
|
|
_graph: &mut RenderGraphContext,
|
|
render_context: &mut RenderContext,
|
|
(camera, pipeline_id, bind_groups, view_uniform_offset): QueryItem<Self::ViewQuery>,
|
|
world: &World,
|
|
) -> Result<(), NodeRunError> {
|
|
let pipelines = world.resource::<SsaoPipelines>();
|
|
let pipeline_cache = world.resource::<PipelineCache>();
|
|
let (
|
|
Some(camera_size),
|
|
Some(preprocess_depth_pipeline),
|
|
Some(spatial_denoise_pipeline),
|
|
Some(gtao_pipeline),
|
|
) = (
|
|
camera.physical_viewport_size,
|
|
pipeline_cache.get_compute_pipeline(pipelines.preprocess_depth_pipeline),
|
|
pipeline_cache.get_compute_pipeline(pipelines.spatial_denoise_pipeline),
|
|
pipeline_cache.get_compute_pipeline(pipeline_id.0),
|
|
) else {
|
|
return Ok(());
|
|
};
|
|
|
|
render_context.command_encoder().push_debug_group("ssao");
|
|
|
|
{
|
|
let mut preprocess_depth_pass =
|
|
render_context
|
|
.command_encoder()
|
|
.begin_compute_pass(&ComputePassDescriptor {
|
|
label: Some("ssao_preprocess_depth_pass"),
|
|
});
|
|
preprocess_depth_pass.set_pipeline(preprocess_depth_pipeline);
|
|
preprocess_depth_pass.set_bind_group(0, &bind_groups.preprocess_depth_bind_group, &[]);
|
|
preprocess_depth_pass.set_bind_group(
|
|
1,
|
|
&bind_groups.common_bind_group,
|
|
&[view_uniform_offset.offset],
|
|
);
|
|
preprocess_depth_pass.dispatch_workgroups(
|
|
div_ceil(camera_size.x, 16),
|
|
div_ceil(camera_size.y, 16),
|
|
1,
|
|
);
|
|
}
|
|
|
|
{
|
|
let mut gtao_pass =
|
|
render_context
|
|
.command_encoder()
|
|
.begin_compute_pass(&ComputePassDescriptor {
|
|
label: Some("ssao_gtao_pass"),
|
|
});
|
|
gtao_pass.set_pipeline(gtao_pipeline);
|
|
gtao_pass.set_bind_group(0, &bind_groups.gtao_bind_group, &[]);
|
|
gtao_pass.set_bind_group(
|
|
1,
|
|
&bind_groups.common_bind_group,
|
|
&[view_uniform_offset.offset],
|
|
);
|
|
gtao_pass.dispatch_workgroups(
|
|
div_ceil(camera_size.x, 8),
|
|
div_ceil(camera_size.y, 8),
|
|
1,
|
|
);
|
|
}
|
|
|
|
{
|
|
let mut spatial_denoise_pass =
|
|
render_context
|
|
.command_encoder()
|
|
.begin_compute_pass(&ComputePassDescriptor {
|
|
label: Some("ssao_spatial_denoise_pass"),
|
|
});
|
|
spatial_denoise_pass.set_pipeline(spatial_denoise_pipeline);
|
|
spatial_denoise_pass.set_bind_group(0, &bind_groups.spatial_denoise_bind_group, &[]);
|
|
spatial_denoise_pass.set_bind_group(
|
|
1,
|
|
&bind_groups.common_bind_group,
|
|
&[view_uniform_offset.offset],
|
|
);
|
|
spatial_denoise_pass.dispatch_workgroups(
|
|
div_ceil(camera_size.x, 8),
|
|
div_ceil(camera_size.y, 8),
|
|
1,
|
|
);
|
|
}
|
|
|
|
render_context.command_encoder().pop_debug_group();
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Resource)]
|
|
struct SsaoPipelines {
|
|
preprocess_depth_pipeline: CachedComputePipelineId,
|
|
spatial_denoise_pipeline: CachedComputePipelineId,
|
|
|
|
common_bind_group_layout: BindGroupLayout,
|
|
preprocess_depth_bind_group_layout: BindGroupLayout,
|
|
gtao_bind_group_layout: BindGroupLayout,
|
|
spatial_denoise_bind_group_layout: BindGroupLayout,
|
|
|
|
hilbert_index_lut: TextureView,
|
|
point_clamp_sampler: Sampler,
|
|
}
|
|
|
|
impl FromWorld for SsaoPipelines {
|
|
fn from_world(world: &mut World) -> Self {
|
|
let render_device = world.resource::<RenderDevice>();
|
|
let render_queue = world.resource::<RenderQueue>();
|
|
let pipeline_cache = world.resource::<PipelineCache>();
|
|
|
|
let hilbert_index_lut = render_device
|
|
.create_texture_with_data(
|
|
render_queue,
|
|
&(TextureDescriptor {
|
|
label: Some("ssao_hilbert_index_lut"),
|
|
size: Extent3d {
|
|
width: HILBERT_WIDTH as u32,
|
|
height: HILBERT_WIDTH as u32,
|
|
depth_or_array_layers: 1,
|
|
},
|
|
mip_level_count: 1,
|
|
sample_count: 1,
|
|
dimension: TextureDimension::D2,
|
|
format: TextureFormat::R16Uint,
|
|
usage: TextureUsages::TEXTURE_BINDING,
|
|
view_formats: &[],
|
|
}),
|
|
bytemuck::cast_slice(&generate_hilbert_index_lut()),
|
|
)
|
|
.create_view(&TextureViewDescriptor::default());
|
|
|
|
let point_clamp_sampler = render_device.create_sampler(&SamplerDescriptor {
|
|
min_filter: FilterMode::Nearest,
|
|
mag_filter: FilterMode::Nearest,
|
|
mipmap_filter: FilterMode::Nearest,
|
|
address_mode_u: AddressMode::ClampToEdge,
|
|
address_mode_v: AddressMode::ClampToEdge,
|
|
..Default::default()
|
|
});
|
|
|
|
let common_bind_group_layout =
|
|
render_device.create_bind_group_layout(&BindGroupLayoutDescriptor {
|
|
label: Some("ssao_common_bind_group_layout"),
|
|
entries: &[
|
|
BindGroupLayoutEntry {
|
|
binding: 0,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Sampler(SamplerBindingType::NonFiltering),
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 1,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Buffer {
|
|
ty: BufferBindingType::Uniform,
|
|
has_dynamic_offset: true,
|
|
min_binding_size: Some(ViewUniform::min_size()),
|
|
},
|
|
count: None,
|
|
},
|
|
],
|
|
});
|
|
|
|
let mip_texture_entry = BindGroupLayoutEntry {
|
|
binding: 1,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::StorageTexture {
|
|
access: StorageTextureAccess::WriteOnly,
|
|
format: TextureFormat::R16Float,
|
|
view_dimension: TextureViewDimension::D2,
|
|
},
|
|
count: None,
|
|
};
|
|
let preprocess_depth_bind_group_layout =
|
|
render_device.create_bind_group_layout(&BindGroupLayoutDescriptor {
|
|
label: Some("ssao_preprocess_depth_bind_group_layout"),
|
|
entries: &[
|
|
BindGroupLayoutEntry {
|
|
binding: 0,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Texture {
|
|
sample_type: TextureSampleType::Depth,
|
|
view_dimension: TextureViewDimension::D2,
|
|
multisampled: false,
|
|
},
|
|
count: None,
|
|
},
|
|
mip_texture_entry,
|
|
BindGroupLayoutEntry {
|
|
binding: 2,
|
|
..mip_texture_entry
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 3,
|
|
..mip_texture_entry
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 4,
|
|
..mip_texture_entry
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 5,
|
|
..mip_texture_entry
|
|
},
|
|
],
|
|
});
|
|
|
|
let gtao_bind_group_layout =
|
|
render_device.create_bind_group_layout(&BindGroupLayoutDescriptor {
|
|
label: Some("ssao_gtao_bind_group_layout"),
|
|
entries: &[
|
|
BindGroupLayoutEntry {
|
|
binding: 0,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Texture {
|
|
sample_type: TextureSampleType::Float { filterable: false },
|
|
view_dimension: TextureViewDimension::D2,
|
|
multisampled: false,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 1,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Texture {
|
|
sample_type: TextureSampleType::Float { filterable: false },
|
|
view_dimension: TextureViewDimension::D2,
|
|
multisampled: false,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 2,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Texture {
|
|
sample_type: TextureSampleType::Uint,
|
|
view_dimension: TextureViewDimension::D2,
|
|
multisampled: false,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 3,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::StorageTexture {
|
|
access: StorageTextureAccess::WriteOnly,
|
|
format: TextureFormat::R16Float,
|
|
view_dimension: TextureViewDimension::D2,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 4,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::StorageTexture {
|
|
access: StorageTextureAccess::WriteOnly,
|
|
format: TextureFormat::R32Uint,
|
|
view_dimension: TextureViewDimension::D2,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 5,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Buffer {
|
|
ty: BufferBindingType::Uniform,
|
|
has_dynamic_offset: false,
|
|
min_binding_size: Some(GlobalsUniform::min_size()),
|
|
},
|
|
count: None,
|
|
},
|
|
],
|
|
});
|
|
|
|
let spatial_denoise_bind_group_layout =
|
|
render_device.create_bind_group_layout(&BindGroupLayoutDescriptor {
|
|
label: Some("ssao_spatial_denoise_bind_group_layout"),
|
|
entries: &[
|
|
BindGroupLayoutEntry {
|
|
binding: 0,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Texture {
|
|
sample_type: TextureSampleType::Float { filterable: false },
|
|
view_dimension: TextureViewDimension::D2,
|
|
multisampled: false,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 1,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::Texture {
|
|
sample_type: TextureSampleType::Uint,
|
|
view_dimension: TextureViewDimension::D2,
|
|
multisampled: false,
|
|
},
|
|
count: None,
|
|
},
|
|
BindGroupLayoutEntry {
|
|
binding: 2,
|
|
visibility: ShaderStages::COMPUTE,
|
|
ty: BindingType::StorageTexture {
|
|
access: StorageTextureAccess::WriteOnly,
|
|
format: TextureFormat::R16Float,
|
|
view_dimension: TextureViewDimension::D2,
|
|
},
|
|
count: None,
|
|
},
|
|
],
|
|
});
|
|
|
|
let preprocess_depth_pipeline =
|
|
pipeline_cache.queue_compute_pipeline(ComputePipelineDescriptor {
|
|
label: Some("ssao_preprocess_depth_pipeline".into()),
|
|
layout: vec![
|
|
preprocess_depth_bind_group_layout.clone(),
|
|
common_bind_group_layout.clone(),
|
|
],
|
|
push_constant_ranges: vec![],
|
|
shader: PREPROCESS_DEPTH_SHADER_HANDLE.typed(),
|
|
shader_defs: Vec::new(),
|
|
entry_point: "preprocess_depth".into(),
|
|
});
|
|
|
|
let spatial_denoise_pipeline =
|
|
pipeline_cache.queue_compute_pipeline(ComputePipelineDescriptor {
|
|
label: Some("ssao_spatial_denoise_pipeline".into()),
|
|
layout: vec![
|
|
spatial_denoise_bind_group_layout.clone(),
|
|
common_bind_group_layout.clone(),
|
|
],
|
|
push_constant_ranges: vec![],
|
|
shader: SPATIAL_DENOISE_SHADER_HANDLE.typed(),
|
|
shader_defs: Vec::new(),
|
|
entry_point: "spatial_denoise".into(),
|
|
});
|
|
|
|
Self {
|
|
preprocess_depth_pipeline,
|
|
spatial_denoise_pipeline,
|
|
|
|
common_bind_group_layout,
|
|
preprocess_depth_bind_group_layout,
|
|
gtao_bind_group_layout,
|
|
spatial_denoise_bind_group_layout,
|
|
|
|
hilbert_index_lut,
|
|
point_clamp_sampler,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(PartialEq, Eq, Hash, Clone)]
|
|
struct SsaoPipelineKey {
|
|
ssao_settings: ScreenSpaceAmbientOcclusionSettings,
|
|
temporal_noise: bool,
|
|
}
|
|
|
|
impl SpecializedComputePipeline for SsaoPipelines {
|
|
type Key = SsaoPipelineKey;
|
|
|
|
fn specialize(&self, key: Self::Key) -> ComputePipelineDescriptor {
|
|
let (slice_count, samples_per_slice_side) = key.ssao_settings.quality_level.sample_counts();
|
|
|
|
let mut shader_defs = vec![
|
|
ShaderDefVal::Int("SLICE_COUNT".to_string(), slice_count as i32),
|
|
ShaderDefVal::Int(
|
|
"SAMPLES_PER_SLICE_SIDE".to_string(),
|
|
samples_per_slice_side as i32,
|
|
),
|
|
];
|
|
|
|
if key.temporal_noise {
|
|
shader_defs.push("TEMPORAL_NOISE".into());
|
|
}
|
|
|
|
ComputePipelineDescriptor {
|
|
label: Some("ssao_gtao_pipeline".into()),
|
|
layout: vec![
|
|
self.gtao_bind_group_layout.clone(),
|
|
self.common_bind_group_layout.clone(),
|
|
],
|
|
push_constant_ranges: vec![],
|
|
shader: GTAO_SHADER_HANDLE.typed(),
|
|
shader_defs,
|
|
entry_point: "gtao".into(),
|
|
}
|
|
}
|
|
}
|
|
|
|
fn extract_ssao_settings(
|
|
mut commands: Commands,
|
|
cameras: Extract<
|
|
Query<
|
|
(Entity, &Camera, &ScreenSpaceAmbientOcclusionSettings),
|
|
(With<Camera3d>, With<DepthPrepass>, With<NormalPrepass>),
|
|
>,
|
|
>,
|
|
msaa: Extract<Res<Msaa>>,
|
|
) {
|
|
for (entity, camera, ssao_settings) in &cameras {
|
|
if **msaa != Msaa::Off {
|
|
error!(
|
|
"SSAO is being used which requires Msaa::Off, but Msaa is currently set to Msaa::{:?}",
|
|
**msaa
|
|
);
|
|
return;
|
|
}
|
|
|
|
if camera.is_active {
|
|
commands.get_or_spawn(entity).insert(ssao_settings.clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Component)]
|
|
pub struct ScreenSpaceAmbientOcclusionTextures {
|
|
preprocessed_depth_texture: CachedTexture,
|
|
ssao_noisy_texture: CachedTexture, // Pre-spatially denoised texture
|
|
pub screen_space_ambient_occlusion_texture: CachedTexture, // Spatially denoised texture
|
|
depth_differences_texture: CachedTexture,
|
|
}
|
|
|
|
fn prepare_ssao_textures(
|
|
mut commands: Commands,
|
|
mut texture_cache: ResMut<TextureCache>,
|
|
render_device: Res<RenderDevice>,
|
|
views: Query<(Entity, &ExtractedCamera), With<ScreenSpaceAmbientOcclusionSettings>>,
|
|
) {
|
|
for (entity, camera) in &views {
|
|
let Some(physical_viewport_size) = camera.physical_viewport_size else { continue };
|
|
let size = Extent3d {
|
|
width: physical_viewport_size.x,
|
|
height: physical_viewport_size.y,
|
|
depth_or_array_layers: 1,
|
|
};
|
|
|
|
let preprocessed_depth_texture = texture_cache.get(
|
|
&render_device,
|
|
TextureDescriptor {
|
|
label: Some("ssao_preprocessed_depth_texture"),
|
|
size,
|
|
mip_level_count: 5,
|
|
sample_count: 1,
|
|
dimension: TextureDimension::D2,
|
|
format: TextureFormat::R16Float,
|
|
usage: TextureUsages::STORAGE_BINDING | TextureUsages::TEXTURE_BINDING,
|
|
view_formats: &[],
|
|
},
|
|
);
|
|
|
|
let ssao_noisy_texture = texture_cache.get(
|
|
&render_device,
|
|
TextureDescriptor {
|
|
label: Some("ssao_noisy_texture"),
|
|
size,
|
|
mip_level_count: 1,
|
|
sample_count: 1,
|
|
dimension: TextureDimension::D2,
|
|
format: TextureFormat::R16Float,
|
|
usage: TextureUsages::STORAGE_BINDING | TextureUsages::TEXTURE_BINDING,
|
|
view_formats: &[],
|
|
},
|
|
);
|
|
|
|
let ssao_texture = texture_cache.get(
|
|
&render_device,
|
|
TextureDescriptor {
|
|
label: Some("ssao_texture"),
|
|
size,
|
|
mip_level_count: 1,
|
|
sample_count: 1,
|
|
dimension: TextureDimension::D2,
|
|
format: TextureFormat::R16Float,
|
|
usage: TextureUsages::STORAGE_BINDING | TextureUsages::TEXTURE_BINDING,
|
|
view_formats: &[],
|
|
},
|
|
);
|
|
|
|
let depth_differences_texture = texture_cache.get(
|
|
&render_device,
|
|
TextureDescriptor {
|
|
label: Some("ssao_depth_differences_texture"),
|
|
size,
|
|
mip_level_count: 1,
|
|
sample_count: 1,
|
|
dimension: TextureDimension::D2,
|
|
format: TextureFormat::R32Uint,
|
|
usage: TextureUsages::STORAGE_BINDING | TextureUsages::TEXTURE_BINDING,
|
|
view_formats: &[],
|
|
},
|
|
);
|
|
|
|
commands
|
|
.entity(entity)
|
|
.insert(ScreenSpaceAmbientOcclusionTextures {
|
|
preprocessed_depth_texture,
|
|
ssao_noisy_texture,
|
|
screen_space_ambient_occlusion_texture: ssao_texture,
|
|
depth_differences_texture,
|
|
});
|
|
}
|
|
}
|
|
|
|
#[derive(Component)]
|
|
struct SsaoPipelineId(CachedComputePipelineId);
|
|
|
|
fn prepare_ssao_pipelines(
|
|
mut commands: Commands,
|
|
pipeline_cache: Res<PipelineCache>,
|
|
mut pipelines: ResMut<SpecializedComputePipelines<SsaoPipelines>>,
|
|
pipeline: Res<SsaoPipelines>,
|
|
views: Query<(
|
|
Entity,
|
|
&ScreenSpaceAmbientOcclusionSettings,
|
|
Option<&TemporalJitter>,
|
|
)>,
|
|
) {
|
|
for (entity, ssao_settings, temporal_jitter) in &views {
|
|
let pipeline_id = pipelines.specialize(
|
|
&pipeline_cache,
|
|
&pipeline,
|
|
SsaoPipelineKey {
|
|
ssao_settings: ssao_settings.clone(),
|
|
temporal_noise: temporal_jitter.is_some(),
|
|
},
|
|
);
|
|
|
|
commands.entity(entity).insert(SsaoPipelineId(pipeline_id));
|
|
}
|
|
}
|
|
|
|
#[derive(Component)]
|
|
struct SsaoBindGroups {
|
|
common_bind_group: BindGroup,
|
|
preprocess_depth_bind_group: BindGroup,
|
|
gtao_bind_group: BindGroup,
|
|
spatial_denoise_bind_group: BindGroup,
|
|
}
|
|
|
|
fn queue_ssao_bind_groups(
|
|
mut commands: Commands,
|
|
render_device: Res<RenderDevice>,
|
|
pipelines: Res<SsaoPipelines>,
|
|
view_uniforms: Res<ViewUniforms>,
|
|
global_uniforms: Res<GlobalsBuffer>,
|
|
views: Query<(
|
|
Entity,
|
|
&ScreenSpaceAmbientOcclusionTextures,
|
|
&ViewPrepassTextures,
|
|
)>,
|
|
) {
|
|
let (Some(view_uniforms), Some(globals_uniforms)) = (
|
|
view_uniforms.uniforms.binding(),
|
|
global_uniforms.buffer.binding(),
|
|
) else {
|
|
return;
|
|
};
|
|
|
|
for (entity, ssao_textures, prepass_textures) in &views {
|
|
let common_bind_group = render_device.create_bind_group(&BindGroupDescriptor {
|
|
label: Some("ssao_common_bind_group"),
|
|
layout: &pipelines.common_bind_group_layout,
|
|
entries: &[
|
|
BindGroupEntry {
|
|
binding: 0,
|
|
resource: BindingResource::Sampler(&pipelines.point_clamp_sampler),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 1,
|
|
resource: view_uniforms.clone(),
|
|
},
|
|
],
|
|
});
|
|
|
|
let preprocess_depth_mip_view_descriptor = TextureViewDescriptor {
|
|
format: Some(TextureFormat::R16Float),
|
|
dimension: Some(TextureViewDimension::D2),
|
|
mip_level_count: Some(1),
|
|
..default()
|
|
};
|
|
let preprocess_depth_bind_group = render_device.create_bind_group(&BindGroupDescriptor {
|
|
label: Some("ssao_preprocess_depth_bind_group"),
|
|
layout: &pipelines.preprocess_depth_bind_group_layout,
|
|
entries: &[
|
|
BindGroupEntry {
|
|
binding: 0,
|
|
resource: BindingResource::TextureView(
|
|
&prepass_textures.depth.as_ref().unwrap().default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 1,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures
|
|
.preprocessed_depth_texture
|
|
.texture
|
|
.create_view(&TextureViewDescriptor {
|
|
label: Some("ssao_preprocessed_depth_texture_mip_view_0"),
|
|
base_mip_level: 0,
|
|
..preprocess_depth_mip_view_descriptor
|
|
}),
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 2,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures
|
|
.preprocessed_depth_texture
|
|
.texture
|
|
.create_view(&TextureViewDescriptor {
|
|
label: Some("ssao_preprocessed_depth_texture_mip_view_1"),
|
|
base_mip_level: 1,
|
|
..preprocess_depth_mip_view_descriptor
|
|
}),
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 3,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures
|
|
.preprocessed_depth_texture
|
|
.texture
|
|
.create_view(&TextureViewDescriptor {
|
|
label: Some("ssao_preprocessed_depth_texture_mip_view_2"),
|
|
base_mip_level: 2,
|
|
..preprocess_depth_mip_view_descriptor
|
|
}),
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 4,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures
|
|
.preprocessed_depth_texture
|
|
.texture
|
|
.create_view(&TextureViewDescriptor {
|
|
label: Some("ssao_preprocessed_depth_texture_mip_view_3"),
|
|
base_mip_level: 3,
|
|
..preprocess_depth_mip_view_descriptor
|
|
}),
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 5,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures
|
|
.preprocessed_depth_texture
|
|
.texture
|
|
.create_view(&TextureViewDescriptor {
|
|
label: Some("ssao_preprocessed_depth_texture_mip_view_4"),
|
|
base_mip_level: 4,
|
|
..preprocess_depth_mip_view_descriptor
|
|
}),
|
|
),
|
|
},
|
|
],
|
|
});
|
|
|
|
let gtao_bind_group = render_device.create_bind_group(&BindGroupDescriptor {
|
|
label: Some("ssao_gtao_bind_group"),
|
|
layout: &pipelines.gtao_bind_group_layout,
|
|
entries: &[
|
|
BindGroupEntry {
|
|
binding: 0,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures.preprocessed_depth_texture.default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 1,
|
|
resource: BindingResource::TextureView(
|
|
&prepass_textures.normal.as_ref().unwrap().default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 2,
|
|
resource: BindingResource::TextureView(&pipelines.hilbert_index_lut),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 3,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures.ssao_noisy_texture.default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 4,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures.depth_differences_texture.default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 5,
|
|
resource: globals_uniforms.clone(),
|
|
},
|
|
],
|
|
});
|
|
|
|
let spatial_denoise_bind_group = render_device.create_bind_group(&BindGroupDescriptor {
|
|
label: Some("ssao_spatial_denoise_bind_group"),
|
|
layout: &pipelines.spatial_denoise_bind_group_layout,
|
|
entries: &[
|
|
BindGroupEntry {
|
|
binding: 0,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures.ssao_noisy_texture.default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 1,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures.depth_differences_texture.default_view,
|
|
),
|
|
},
|
|
BindGroupEntry {
|
|
binding: 2,
|
|
resource: BindingResource::TextureView(
|
|
&ssao_textures
|
|
.screen_space_ambient_occlusion_texture
|
|
.default_view,
|
|
),
|
|
},
|
|
],
|
|
});
|
|
|
|
commands.entity(entity).insert(SsaoBindGroups {
|
|
common_bind_group,
|
|
preprocess_depth_bind_group,
|
|
gtao_bind_group,
|
|
spatial_denoise_bind_group,
|
|
});
|
|
}
|
|
}
|
|
|
|
#[allow(clippy::needless_range_loop)]
|
|
fn generate_hilbert_index_lut() -> [[u16; 64]; 64] {
|
|
let mut t = [[0; 64]; 64];
|
|
|
|
for x in 0..64 {
|
|
for y in 0..64 {
|
|
t[x][y] = hilbert_index(x as u16, y as u16);
|
|
}
|
|
}
|
|
|
|
t
|
|
}
|
|
|
|
// https://www.shadertoy.com/view/3tB3z3
|
|
const HILBERT_WIDTH: u16 = 64;
|
|
fn hilbert_index(mut x: u16, mut y: u16) -> u16 {
|
|
let mut index = 0;
|
|
|
|
let mut level: u16 = HILBERT_WIDTH / 2;
|
|
while level > 0 {
|
|
let region_x = (x & level > 0) as u16;
|
|
let region_y = (y & level > 0) as u16;
|
|
index += level * level * ((3 * region_x) ^ region_y);
|
|
|
|
if region_y == 0 {
|
|
if region_x == 1 {
|
|
x = HILBERT_WIDTH - 1 - x;
|
|
y = HILBERT_WIDTH - 1 - y;
|
|
}
|
|
|
|
mem::swap(&mut x, &mut y);
|
|
}
|
|
|
|
level /= 2;
|
|
}
|
|
|
|
index
|
|
}
|
|
|
|
/// Divide `numerator` by `denominator`, rounded up to the nearest multiple of `denominator`.
|
|
fn div_ceil(numerator: u32, denominator: u32) -> u32 {
|
|
(numerator + denominator - 1) / denominator
|
|
}
|