2024-03-25 19:08:27 +00:00
|
|
|
use super::{
|
|
|
|
gpu_scene::{MeshletViewBindGroups, MeshletViewResources},
|
|
|
|
material_draw_prepare::{
|
|
|
|
MeshletViewMaterialsDeferredGBufferPrepass, MeshletViewMaterialsMainOpaquePass,
|
|
|
|
MeshletViewMaterialsPrepass,
|
|
|
|
},
|
|
|
|
MeshletGpuScene,
|
|
|
|
};
|
|
|
|
use crate::{
|
2024-07-19 15:00:50 +00:00
|
|
|
MeshViewBindGroup, PrepassViewBindGroup, ViewEnvironmentMapUniformOffset, ViewFogUniformOffset,
|
|
|
|
ViewLightProbesUniformOffset, ViewLightsUniformOffset, ViewScreenSpaceReflectionsUniformOffset,
|
2024-03-25 19:08:27 +00:00
|
|
|
};
|
2024-06-21 19:06:08 +00:00
|
|
|
use bevy_core_pipeline::prepass::{
|
|
|
|
MotionVectorPrepass, PreviousViewUniformOffset, ViewPrepassTextures,
|
|
|
|
};
|
|
|
|
use bevy_ecs::{
|
|
|
|
query::{Has, QueryItem},
|
|
|
|
world::World,
|
|
|
|
};
|
2024-03-25 19:08:27 +00:00
|
|
|
use bevy_render::{
|
|
|
|
camera::ExtractedCamera,
|
|
|
|
render_graph::{NodeRunError, RenderGraphContext, ViewNode},
|
|
|
|
render_resource::{
|
|
|
|
LoadOp, Operations, PipelineCache, RenderPassDepthStencilAttachment, RenderPassDescriptor,
|
|
|
|
StoreOp,
|
|
|
|
},
|
|
|
|
renderer::RenderContext,
|
|
|
|
view::{ViewTarget, ViewUniformOffset},
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Fullscreen shading pass based on the visibility buffer generated from rasterizing meshlets.
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct MeshletMainOpaquePass3dNode;
|
|
|
|
impl ViewNode for MeshletMainOpaquePass3dNode {
|
|
|
|
type ViewQuery = (
|
|
|
|
&'static ExtractedCamera,
|
|
|
|
&'static ViewTarget,
|
|
|
|
&'static MeshViewBindGroup,
|
|
|
|
&'static ViewUniformOffset,
|
|
|
|
&'static ViewLightsUniformOffset,
|
|
|
|
&'static ViewFogUniformOffset,
|
|
|
|
&'static ViewLightProbesUniformOffset,
|
Implement opt-in sharp screen-space reflections for the deferred renderer, with improved raymarching code. (#13418)
This commit, a revamp of #12959, implements screen-space reflections
(SSR), which approximate real-time reflections based on raymarching
through the depth buffer and copying samples from the final rendered
frame. This patch is a relatively minimal implementation of SSR, so as
to provide a flexible base on which to customize and build in the
future. However, it's based on the production-quality [raymarching code
by Tomasz
Stachowiak](https://gist.github.com/h3r2tic/9c8356bdaefbe80b1a22ae0aaee192db).
For a general basic overview of screen-space reflections, see
[1](https://lettier.github.io/3d-game-shaders-for-beginners/screen-space-reflection.html).
The raymarching shader uses the basic algorithm of tracing forward in
large steps, refining that trace in smaller increments via binary
search, and then using the secant method. No temporal filtering or
roughness blurring, is performed at all; for this reason, SSR currently
only operates on very shiny surfaces. No acceleration via the
hierarchical Z-buffer is implemented (though note that
https://github.com/bevyengine/bevy/pull/12899 will add the
infrastructure for this). Reflections are traced at full resolution,
which is often considered slow. All of these improvements and more can
be follow-ups.
SSR is built on top of the deferred renderer and is currently only
supported in that mode. Forward screen-space reflections are possible
albeit uncommon (though e.g. *Doom Eternal* uses them); however, they
require tracing from the previous frame, which would add complexity.
This patch leaves the door open to implementing SSR in the forward
rendering path but doesn't itself have such an implementation.
Screen-space reflections aren't supported in WebGL 2, because they
require sampling from the depth buffer, which Naga can't do because of a
bug (`sampler2DShadow` is incorrectly generated instead of `sampler2D`;
this is the same reason why depth of field is disabled on that
platform).
To add screen-space reflections to a camera, use the
`ScreenSpaceReflectionsBundle` bundle or the
`ScreenSpaceReflectionsSettings` component. In addition to
`ScreenSpaceReflectionsSettings`, `DepthPrepass` and `DeferredPrepass`
must also be present for the reflections to show up. The
`ScreenSpaceReflectionsSettings` component contains several settings
that artists can tweak, and also comes with sensible defaults.
A new example, `ssr`, has been added. It's loosely based on the
[three.js ocean
sample](https://threejs.org/examples/webgl_shaders_ocean.html), but all
the assets are original. Note that the three.js demo has no screen-space
reflections and instead renders a mirror world. In contrast to #12959,
this demo tests not only a cube but also a more complex model (the
flight helmet).
## Changelog
### Added
* Screen-space reflections can be enabled for very smooth surfaces by
adding the `ScreenSpaceReflections` component to a camera. Deferred
rendering must be enabled for the reflections to appear.
![Screenshot 2024-05-18
143555](https://github.com/bevyengine/bevy/assets/157897/b8675b39-8a89-433e-a34e-1b9ee1233267)
![Screenshot 2024-05-18
143606](https://github.com/bevyengine/bevy/assets/157897/cc9e1cd0-9951-464a-9a08-e589210e5606)
2024-05-27 13:43:40 +00:00
|
|
|
&'static ViewScreenSpaceReflectionsUniformOffset,
|
2024-07-19 15:00:50 +00:00
|
|
|
&'static ViewEnvironmentMapUniformOffset,
|
2024-03-25 19:08:27 +00:00
|
|
|
&'static MeshletViewMaterialsMainOpaquePass,
|
|
|
|
&'static MeshletViewBindGroups,
|
|
|
|
&'static MeshletViewResources,
|
|
|
|
);
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&self,
|
|
|
|
_graph: &mut RenderGraphContext,
|
|
|
|
render_context: &mut RenderContext,
|
|
|
|
(
|
|
|
|
camera,
|
|
|
|
target,
|
|
|
|
mesh_view_bind_group,
|
|
|
|
view_uniform_offset,
|
|
|
|
view_lights_offset,
|
|
|
|
view_fog_offset,
|
|
|
|
view_light_probes_offset,
|
Implement opt-in sharp screen-space reflections for the deferred renderer, with improved raymarching code. (#13418)
This commit, a revamp of #12959, implements screen-space reflections
(SSR), which approximate real-time reflections based on raymarching
through the depth buffer and copying samples from the final rendered
frame. This patch is a relatively minimal implementation of SSR, so as
to provide a flexible base on which to customize and build in the
future. However, it's based on the production-quality [raymarching code
by Tomasz
Stachowiak](https://gist.github.com/h3r2tic/9c8356bdaefbe80b1a22ae0aaee192db).
For a general basic overview of screen-space reflections, see
[1](https://lettier.github.io/3d-game-shaders-for-beginners/screen-space-reflection.html).
The raymarching shader uses the basic algorithm of tracing forward in
large steps, refining that trace in smaller increments via binary
search, and then using the secant method. No temporal filtering or
roughness blurring, is performed at all; for this reason, SSR currently
only operates on very shiny surfaces. No acceleration via the
hierarchical Z-buffer is implemented (though note that
https://github.com/bevyengine/bevy/pull/12899 will add the
infrastructure for this). Reflections are traced at full resolution,
which is often considered slow. All of these improvements and more can
be follow-ups.
SSR is built on top of the deferred renderer and is currently only
supported in that mode. Forward screen-space reflections are possible
albeit uncommon (though e.g. *Doom Eternal* uses them); however, they
require tracing from the previous frame, which would add complexity.
This patch leaves the door open to implementing SSR in the forward
rendering path but doesn't itself have such an implementation.
Screen-space reflections aren't supported in WebGL 2, because they
require sampling from the depth buffer, which Naga can't do because of a
bug (`sampler2DShadow` is incorrectly generated instead of `sampler2D`;
this is the same reason why depth of field is disabled on that
platform).
To add screen-space reflections to a camera, use the
`ScreenSpaceReflectionsBundle` bundle or the
`ScreenSpaceReflectionsSettings` component. In addition to
`ScreenSpaceReflectionsSettings`, `DepthPrepass` and `DeferredPrepass`
must also be present for the reflections to show up. The
`ScreenSpaceReflectionsSettings` component contains several settings
that artists can tweak, and also comes with sensible defaults.
A new example, `ssr`, has been added. It's loosely based on the
[three.js ocean
sample](https://threejs.org/examples/webgl_shaders_ocean.html), but all
the assets are original. Note that the three.js demo has no screen-space
reflections and instead renders a mirror world. In contrast to #12959,
this demo tests not only a cube but also a more complex model (the
flight helmet).
## Changelog
### Added
* Screen-space reflections can be enabled for very smooth surfaces by
adding the `ScreenSpaceReflections` component to a camera. Deferred
rendering must be enabled for the reflections to appear.
![Screenshot 2024-05-18
143555](https://github.com/bevyengine/bevy/assets/157897/b8675b39-8a89-433e-a34e-1b9ee1233267)
![Screenshot 2024-05-18
143606](https://github.com/bevyengine/bevy/assets/157897/cc9e1cd0-9951-464a-9a08-e589210e5606)
2024-05-27 13:43:40 +00:00
|
|
|
view_ssr_offset,
|
2024-07-19 15:00:50 +00:00
|
|
|
view_environment_map_offset,
|
2024-03-25 19:08:27 +00:00
|
|
|
meshlet_view_materials,
|
|
|
|
meshlet_view_bind_groups,
|
|
|
|
meshlet_view_resources,
|
|
|
|
): QueryItem<Self::ViewQuery>,
|
|
|
|
world: &World,
|
|
|
|
) -> Result<(), NodeRunError> {
|
|
|
|
if meshlet_view_materials.is_empty() {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let (
|
|
|
|
Some(meshlet_gpu_scene),
|
|
|
|
Some(pipeline_cache),
|
|
|
|
Some(meshlet_material_depth),
|
|
|
|
Some(meshlet_material_draw_bind_group),
|
|
|
|
) = (
|
|
|
|
world.get_resource::<MeshletGpuScene>(),
|
|
|
|
world.get_resource::<PipelineCache>(),
|
|
|
|
meshlet_view_resources.material_depth.as_ref(),
|
|
|
|
meshlet_view_bind_groups.material_draw.as_ref(),
|
|
|
|
)
|
|
|
|
else {
|
|
|
|
return Ok(());
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut render_pass = render_context.begin_tracked_render_pass(RenderPassDescriptor {
|
|
|
|
label: Some("meshlet_main_opaque_pass_3d"),
|
|
|
|
color_attachments: &[Some(target.get_color_attachment())],
|
|
|
|
depth_stencil_attachment: Some(RenderPassDepthStencilAttachment {
|
|
|
|
view: &meshlet_material_depth.default_view,
|
|
|
|
depth_ops: Some(Operations {
|
|
|
|
load: LoadOp::Load,
|
|
|
|
store: StoreOp::Store,
|
|
|
|
}),
|
|
|
|
stencil_ops: None,
|
|
|
|
}),
|
|
|
|
timestamp_writes: None,
|
|
|
|
occlusion_query_set: None,
|
|
|
|
});
|
|
|
|
if let Some(viewport) = camera.viewport.as_ref() {
|
|
|
|
render_pass.set_camera_viewport(viewport);
|
|
|
|
}
|
|
|
|
|
|
|
|
render_pass.set_bind_group(
|
|
|
|
0,
|
|
|
|
&mesh_view_bind_group.value,
|
|
|
|
&[
|
|
|
|
view_uniform_offset.offset,
|
|
|
|
view_lights_offset.offset,
|
|
|
|
view_fog_offset.offset,
|
|
|
|
**view_light_probes_offset,
|
Implement opt-in sharp screen-space reflections for the deferred renderer, with improved raymarching code. (#13418)
This commit, a revamp of #12959, implements screen-space reflections
(SSR), which approximate real-time reflections based on raymarching
through the depth buffer and copying samples from the final rendered
frame. This patch is a relatively minimal implementation of SSR, so as
to provide a flexible base on which to customize and build in the
future. However, it's based on the production-quality [raymarching code
by Tomasz
Stachowiak](https://gist.github.com/h3r2tic/9c8356bdaefbe80b1a22ae0aaee192db).
For a general basic overview of screen-space reflections, see
[1](https://lettier.github.io/3d-game-shaders-for-beginners/screen-space-reflection.html).
The raymarching shader uses the basic algorithm of tracing forward in
large steps, refining that trace in smaller increments via binary
search, and then using the secant method. No temporal filtering or
roughness blurring, is performed at all; for this reason, SSR currently
only operates on very shiny surfaces. No acceleration via the
hierarchical Z-buffer is implemented (though note that
https://github.com/bevyengine/bevy/pull/12899 will add the
infrastructure for this). Reflections are traced at full resolution,
which is often considered slow. All of these improvements and more can
be follow-ups.
SSR is built on top of the deferred renderer and is currently only
supported in that mode. Forward screen-space reflections are possible
albeit uncommon (though e.g. *Doom Eternal* uses them); however, they
require tracing from the previous frame, which would add complexity.
This patch leaves the door open to implementing SSR in the forward
rendering path but doesn't itself have such an implementation.
Screen-space reflections aren't supported in WebGL 2, because they
require sampling from the depth buffer, which Naga can't do because of a
bug (`sampler2DShadow` is incorrectly generated instead of `sampler2D`;
this is the same reason why depth of field is disabled on that
platform).
To add screen-space reflections to a camera, use the
`ScreenSpaceReflectionsBundle` bundle or the
`ScreenSpaceReflectionsSettings` component. In addition to
`ScreenSpaceReflectionsSettings`, `DepthPrepass` and `DeferredPrepass`
must also be present for the reflections to show up. The
`ScreenSpaceReflectionsSettings` component contains several settings
that artists can tweak, and also comes with sensible defaults.
A new example, `ssr`, has been added. It's loosely based on the
[three.js ocean
sample](https://threejs.org/examples/webgl_shaders_ocean.html), but all
the assets are original. Note that the three.js demo has no screen-space
reflections and instead renders a mirror world. In contrast to #12959,
this demo tests not only a cube but also a more complex model (the
flight helmet).
## Changelog
### Added
* Screen-space reflections can be enabled for very smooth surfaces by
adding the `ScreenSpaceReflections` component to a camera. Deferred
rendering must be enabled for the reflections to appear.
![Screenshot 2024-05-18
143555](https://github.com/bevyengine/bevy/assets/157897/b8675b39-8a89-433e-a34e-1b9ee1233267)
![Screenshot 2024-05-18
143606](https://github.com/bevyengine/bevy/assets/157897/cc9e1cd0-9951-464a-9a08-e589210e5606)
2024-05-27 13:43:40 +00:00
|
|
|
**view_ssr_offset,
|
2024-07-19 15:00:50 +00:00
|
|
|
**view_environment_map_offset,
|
2024-03-25 19:08:27 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
render_pass.set_bind_group(1, meshlet_material_draw_bind_group, &[]);
|
|
|
|
|
|
|
|
// 1 fullscreen triangle draw per material
|
|
|
|
for (material_id, material_pipeline_id, material_bind_group) in
|
|
|
|
meshlet_view_materials.iter()
|
|
|
|
{
|
|
|
|
if meshlet_gpu_scene.material_present_in_scene(material_id) {
|
|
|
|
if let Some(material_pipeline) =
|
|
|
|
pipeline_cache.get_render_pipeline(*material_pipeline_id)
|
|
|
|
{
|
|
|
|
let x = *material_id * 3;
|
|
|
|
render_pass.set_render_pipeline(material_pipeline);
|
2024-05-04 19:56:19 +00:00
|
|
|
render_pass.set_bind_group(2, material_bind_group, &[]);
|
2024-03-25 19:08:27 +00:00
|
|
|
render_pass.draw(x..(x + 3), 0..1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Fullscreen pass to generate prepass textures based on the visibility buffer generated from rasterizing meshlets.
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct MeshletPrepassNode;
|
|
|
|
impl ViewNode for MeshletPrepassNode {
|
|
|
|
type ViewQuery = (
|
|
|
|
&'static ExtractedCamera,
|
|
|
|
&'static ViewPrepassTextures,
|
|
|
|
&'static ViewUniformOffset,
|
2024-06-21 19:06:08 +00:00
|
|
|
&'static PreviousViewUniformOffset,
|
|
|
|
Has<MotionVectorPrepass>,
|
2024-03-25 19:08:27 +00:00
|
|
|
&'static MeshletViewMaterialsPrepass,
|
|
|
|
&'static MeshletViewBindGroups,
|
|
|
|
&'static MeshletViewResources,
|
|
|
|
);
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&self,
|
|
|
|
_graph: &mut RenderGraphContext,
|
|
|
|
render_context: &mut RenderContext,
|
|
|
|
(
|
|
|
|
camera,
|
|
|
|
view_prepass_textures,
|
|
|
|
view_uniform_offset,
|
2024-04-07 18:59:16 +00:00
|
|
|
previous_view_uniform_offset,
|
2024-06-21 19:06:08 +00:00
|
|
|
view_has_motion_vector_prepass,
|
2024-03-25 19:08:27 +00:00
|
|
|
meshlet_view_materials,
|
|
|
|
meshlet_view_bind_groups,
|
|
|
|
meshlet_view_resources,
|
|
|
|
): QueryItem<Self::ViewQuery>,
|
|
|
|
world: &World,
|
|
|
|
) -> Result<(), NodeRunError> {
|
|
|
|
if meshlet_view_materials.is_empty() {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let (
|
|
|
|
Some(prepass_view_bind_group),
|
|
|
|
Some(meshlet_gpu_scene),
|
|
|
|
Some(pipeline_cache),
|
|
|
|
Some(meshlet_material_depth),
|
|
|
|
Some(meshlet_material_draw_bind_group),
|
|
|
|
) = (
|
|
|
|
world.get_resource::<PrepassViewBindGroup>(),
|
|
|
|
world.get_resource::<MeshletGpuScene>(),
|
|
|
|
world.get_resource::<PipelineCache>(),
|
|
|
|
meshlet_view_resources.material_depth.as_ref(),
|
|
|
|
meshlet_view_bind_groups.material_draw.as_ref(),
|
|
|
|
)
|
|
|
|
else {
|
|
|
|
return Ok(());
|
|
|
|
};
|
|
|
|
|
|
|
|
let color_attachments = vec![
|
|
|
|
view_prepass_textures
|
|
|
|
.normal
|
|
|
|
.as_ref()
|
|
|
|
.map(|normals_texture| normals_texture.get_attachment()),
|
|
|
|
view_prepass_textures
|
|
|
|
.motion_vectors
|
|
|
|
.as_ref()
|
|
|
|
.map(|motion_vectors_texture| motion_vectors_texture.get_attachment()),
|
|
|
|
// Use None in place of Deferred attachments
|
|
|
|
None,
|
|
|
|
None,
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut render_pass = render_context.begin_tracked_render_pass(RenderPassDescriptor {
|
|
|
|
label: Some("meshlet_prepass"),
|
|
|
|
color_attachments: &color_attachments,
|
|
|
|
depth_stencil_attachment: Some(RenderPassDepthStencilAttachment {
|
|
|
|
view: &meshlet_material_depth.default_view,
|
|
|
|
depth_ops: Some(Operations {
|
|
|
|
load: LoadOp::Load,
|
|
|
|
store: StoreOp::Store,
|
|
|
|
}),
|
|
|
|
stencil_ops: None,
|
|
|
|
}),
|
|
|
|
timestamp_writes: None,
|
|
|
|
occlusion_query_set: None,
|
|
|
|
});
|
|
|
|
if let Some(viewport) = camera.viewport.as_ref() {
|
|
|
|
render_pass.set_camera_viewport(viewport);
|
|
|
|
}
|
|
|
|
|
2024-06-21 19:06:08 +00:00
|
|
|
if view_has_motion_vector_prepass {
|
2024-03-25 19:08:27 +00:00
|
|
|
render_pass.set_bind_group(
|
|
|
|
0,
|
|
|
|
prepass_view_bind_group.motion_vectors.as_ref().unwrap(),
|
|
|
|
&[
|
|
|
|
view_uniform_offset.offset,
|
2024-04-07 18:59:16 +00:00
|
|
|
previous_view_uniform_offset.offset,
|
2024-03-25 19:08:27 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
render_pass.set_bind_group(
|
|
|
|
0,
|
|
|
|
prepass_view_bind_group.no_motion_vectors.as_ref().unwrap(),
|
|
|
|
&[view_uniform_offset.offset],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
render_pass.set_bind_group(1, meshlet_material_draw_bind_group, &[]);
|
|
|
|
|
|
|
|
// 1 fullscreen triangle draw per material
|
|
|
|
for (material_id, material_pipeline_id, material_bind_group) in
|
|
|
|
meshlet_view_materials.iter()
|
|
|
|
{
|
|
|
|
if meshlet_gpu_scene.material_present_in_scene(material_id) {
|
|
|
|
if let Some(material_pipeline) =
|
|
|
|
pipeline_cache.get_render_pipeline(*material_pipeline_id)
|
|
|
|
{
|
|
|
|
let x = *material_id * 3;
|
|
|
|
render_pass.set_render_pipeline(material_pipeline);
|
2024-05-04 19:56:19 +00:00
|
|
|
render_pass.set_bind_group(2, material_bind_group, &[]);
|
2024-03-25 19:08:27 +00:00
|
|
|
render_pass.draw(x..(x + 3), 0..1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Fullscreen pass to generate a gbuffer based on the visibility buffer generated from rasterizing meshlets.
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct MeshletDeferredGBufferPrepassNode;
|
|
|
|
impl ViewNode for MeshletDeferredGBufferPrepassNode {
|
|
|
|
type ViewQuery = (
|
|
|
|
&'static ExtractedCamera,
|
|
|
|
&'static ViewPrepassTextures,
|
|
|
|
&'static ViewUniformOffset,
|
2024-06-21 19:06:08 +00:00
|
|
|
&'static PreviousViewUniformOffset,
|
|
|
|
Has<MotionVectorPrepass>,
|
2024-03-25 19:08:27 +00:00
|
|
|
&'static MeshletViewMaterialsDeferredGBufferPrepass,
|
|
|
|
&'static MeshletViewBindGroups,
|
|
|
|
&'static MeshletViewResources,
|
|
|
|
);
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&self,
|
|
|
|
_graph: &mut RenderGraphContext,
|
|
|
|
render_context: &mut RenderContext,
|
|
|
|
(
|
|
|
|
camera,
|
|
|
|
view_prepass_textures,
|
|
|
|
view_uniform_offset,
|
2024-04-07 18:59:16 +00:00
|
|
|
previous_view_uniform_offset,
|
2024-06-21 19:06:08 +00:00
|
|
|
view_has_motion_vector_prepass,
|
2024-03-25 19:08:27 +00:00
|
|
|
meshlet_view_materials,
|
|
|
|
meshlet_view_bind_groups,
|
|
|
|
meshlet_view_resources,
|
|
|
|
): QueryItem<Self::ViewQuery>,
|
|
|
|
world: &World,
|
|
|
|
) -> Result<(), NodeRunError> {
|
|
|
|
if meshlet_view_materials.is_empty() {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let (
|
|
|
|
Some(prepass_view_bind_group),
|
|
|
|
Some(meshlet_gpu_scene),
|
|
|
|
Some(pipeline_cache),
|
|
|
|
Some(meshlet_material_depth),
|
|
|
|
Some(meshlet_material_draw_bind_group),
|
|
|
|
) = (
|
|
|
|
world.get_resource::<PrepassViewBindGroup>(),
|
|
|
|
world.get_resource::<MeshletGpuScene>(),
|
|
|
|
world.get_resource::<PipelineCache>(),
|
|
|
|
meshlet_view_resources.material_depth.as_ref(),
|
|
|
|
meshlet_view_bind_groups.material_draw.as_ref(),
|
|
|
|
)
|
|
|
|
else {
|
|
|
|
return Ok(());
|
|
|
|
};
|
|
|
|
|
|
|
|
let color_attachments = vec![
|
|
|
|
view_prepass_textures
|
|
|
|
.normal
|
|
|
|
.as_ref()
|
|
|
|
.map(|normals_texture| normals_texture.get_attachment()),
|
|
|
|
view_prepass_textures
|
|
|
|
.motion_vectors
|
|
|
|
.as_ref()
|
|
|
|
.map(|motion_vectors_texture| motion_vectors_texture.get_attachment()),
|
|
|
|
view_prepass_textures
|
|
|
|
.deferred
|
|
|
|
.as_ref()
|
|
|
|
.map(|deferred_texture| deferred_texture.get_attachment()),
|
|
|
|
view_prepass_textures
|
|
|
|
.deferred_lighting_pass_id
|
|
|
|
.as_ref()
|
|
|
|
.map(|deferred_lighting_pass_id| deferred_lighting_pass_id.get_attachment()),
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut render_pass = render_context.begin_tracked_render_pass(RenderPassDescriptor {
|
|
|
|
label: Some("meshlet_deferred_prepass"),
|
|
|
|
color_attachments: &color_attachments,
|
|
|
|
depth_stencil_attachment: Some(RenderPassDepthStencilAttachment {
|
|
|
|
view: &meshlet_material_depth.default_view,
|
|
|
|
depth_ops: Some(Operations {
|
|
|
|
load: LoadOp::Load,
|
|
|
|
store: StoreOp::Store,
|
|
|
|
}),
|
|
|
|
stencil_ops: None,
|
|
|
|
}),
|
|
|
|
timestamp_writes: None,
|
|
|
|
occlusion_query_set: None,
|
|
|
|
});
|
|
|
|
if let Some(viewport) = camera.viewport.as_ref() {
|
|
|
|
render_pass.set_camera_viewport(viewport);
|
|
|
|
}
|
|
|
|
|
2024-06-21 19:06:08 +00:00
|
|
|
if view_has_motion_vector_prepass {
|
2024-03-25 19:08:27 +00:00
|
|
|
render_pass.set_bind_group(
|
|
|
|
0,
|
|
|
|
prepass_view_bind_group.motion_vectors.as_ref().unwrap(),
|
|
|
|
&[
|
|
|
|
view_uniform_offset.offset,
|
2024-04-07 18:59:16 +00:00
|
|
|
previous_view_uniform_offset.offset,
|
2024-03-25 19:08:27 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
render_pass.set_bind_group(
|
|
|
|
0,
|
|
|
|
prepass_view_bind_group.no_motion_vectors.as_ref().unwrap(),
|
|
|
|
&[view_uniform_offset.offset],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
render_pass.set_bind_group(1, meshlet_material_draw_bind_group, &[]);
|
|
|
|
|
|
|
|
// 1 fullscreen triangle draw per material
|
|
|
|
for (material_id, material_pipeline_id, material_bind_group) in
|
|
|
|
meshlet_view_materials.iter()
|
|
|
|
{
|
|
|
|
if meshlet_gpu_scene.material_present_in_scene(material_id) {
|
|
|
|
if let Some(material_pipeline) =
|
|
|
|
pipeline_cache.get_render_pipeline(*material_pipeline_id)
|
|
|
|
{
|
|
|
|
let x = *material_id * 3;
|
|
|
|
render_pass.set_render_pipeline(material_pipeline);
|
2024-05-04 19:56:19 +00:00
|
|
|
render_pass.set_bind_group(2, material_bind_group, &[]);
|
2024-03-25 19:08:27 +00:00
|
|
|
render_pass.draw(x..(x + 3), 0..1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|