mirror of
https://github.com/bevyengine/bevy
synced 2025-01-06 10:18:59 +00:00
368 lines
13 KiB
Rust
368 lines
13 KiB
Rust
|
mod light;
|
||
|
pub use light::*;
|
||
|
|
||
|
use crate::StandardMaterial;
|
||
|
use bevy_asset::{Assets, Handle};
|
||
|
use bevy_ecs::{prelude::*, system::SystemState};
|
||
|
use bevy_math::Mat4;
|
||
|
use bevy_render2::{
|
||
|
core_pipeline::Transparent3dPhase,
|
||
|
mesh::Mesh,
|
||
|
pipeline::*,
|
||
|
render_graph::{Node, NodeRunError, RenderGraphContext},
|
||
|
render_phase::{Draw, DrawFunctions, Drawable, RenderPhase, TrackedRenderPass},
|
||
|
render_resource::{BindGroupBuilder, BindGroupId, BufferId, DynamicUniformVec},
|
||
|
renderer::{RenderContext, RenderResources},
|
||
|
shader::{Shader, ShaderStage, ShaderStages},
|
||
|
texture::{TextureFormat, TextureSampleType},
|
||
|
view::{ViewMeta, ViewUniform},
|
||
|
};
|
||
|
use bevy_transform::components::GlobalTransform;
|
||
|
|
||
|
pub struct PbrShaders {
|
||
|
pipeline: PipelineId,
|
||
|
pipeline_descriptor: RenderPipelineDescriptor,
|
||
|
}
|
||
|
|
||
|
// TODO: this pattern for initializing the shaders / pipeline isn't ideal. this should be handled by the asset system
|
||
|
impl FromWorld for PbrShaders {
|
||
|
fn from_world(world: &mut World) -> Self {
|
||
|
let render_resources = world.get_resource::<RenderResources>().unwrap();
|
||
|
let vertex_shader = Shader::from_glsl(ShaderStage::Vertex, include_str!("pbr.vert"))
|
||
|
.get_spirv_shader(None)
|
||
|
.unwrap();
|
||
|
let fragment_shader = Shader::from_glsl(ShaderStage::Fragment, include_str!("pbr.frag"))
|
||
|
.get_spirv_shader(None)
|
||
|
.unwrap();
|
||
|
|
||
|
let vertex_layout = vertex_shader.reflect_layout(true).unwrap();
|
||
|
let fragment_layout = fragment_shader.reflect_layout(true).unwrap();
|
||
|
|
||
|
let mut pipeline_layout =
|
||
|
PipelineLayout::from_shader_layouts(&mut [vertex_layout, fragment_layout]);
|
||
|
|
||
|
let vertex = render_resources.create_shader_module(&vertex_shader);
|
||
|
let fragment = render_resources.create_shader_module(&fragment_shader);
|
||
|
|
||
|
pipeline_layout.vertex_buffer_descriptors = vec![VertexBufferLayout {
|
||
|
stride: 32,
|
||
|
name: "Vertex".into(),
|
||
|
step_mode: InputStepMode::Vertex,
|
||
|
attributes: vec![
|
||
|
// GOTCHA! Vertex_Position isn't first in the buffer due to how Mesh sorts attributes (alphabetically)
|
||
|
VertexAttribute {
|
||
|
name: "Vertex_Position".into(),
|
||
|
format: VertexFormat::Float32x3,
|
||
|
offset: 12,
|
||
|
shader_location: 0,
|
||
|
},
|
||
|
VertexAttribute {
|
||
|
name: "Vertex_Normals".into(),
|
||
|
format: VertexFormat::Float32x3,
|
||
|
offset: 0,
|
||
|
shader_location: 1,
|
||
|
},
|
||
|
VertexAttribute {
|
||
|
name: "Vertex_Uv".into(),
|
||
|
format: VertexFormat::Float32x2,
|
||
|
offset: 24,
|
||
|
shader_location: 2,
|
||
|
},
|
||
|
],
|
||
|
}];
|
||
|
|
||
|
pipeline_layout.bind_group_mut(0).bindings[0].set_dynamic(true);
|
||
|
pipeline_layout.bind_group_mut(0).bindings[1].set_dynamic(true);
|
||
|
if let BindType::Texture { sample_type, .. } =
|
||
|
&mut pipeline_layout.bind_group_mut(0).bindings[2].bind_type
|
||
|
{
|
||
|
*sample_type = TextureSampleType::Depth;
|
||
|
}
|
||
|
if let BindType::Sampler { comparison, .. } =
|
||
|
&mut pipeline_layout.bind_group_mut(0).bindings[3].bind_type
|
||
|
{
|
||
|
*comparison = true;
|
||
|
}
|
||
|
pipeline_layout.bind_group_mut(1).bindings[0].set_dynamic(true);
|
||
|
|
||
|
pipeline_layout.update_bind_group_ids();
|
||
|
|
||
|
let pipeline_descriptor = RenderPipelineDescriptor {
|
||
|
depth_stencil: Some(DepthStencilState {
|
||
|
format: TextureFormat::Depth32Float,
|
||
|
depth_write_enabled: true,
|
||
|
depth_compare: CompareFunction::Less,
|
||
|
stencil: StencilState {
|
||
|
front: StencilFaceState::IGNORE,
|
||
|
back: StencilFaceState::IGNORE,
|
||
|
read_mask: 0,
|
||
|
write_mask: 0,
|
||
|
},
|
||
|
bias: DepthBiasState {
|
||
|
constant: 0,
|
||
|
slope_scale: 0.0,
|
||
|
clamp: 0.0,
|
||
|
},
|
||
|
}),
|
||
|
color_target_states: vec![ColorTargetState {
|
||
|
format: TextureFormat::default(),
|
||
|
blend: Some(BlendState {
|
||
|
color: BlendComponent {
|
||
|
src_factor: BlendFactor::SrcAlpha,
|
||
|
dst_factor: BlendFactor::OneMinusSrcAlpha,
|
||
|
operation: BlendOperation::Add,
|
||
|
},
|
||
|
alpha: BlendComponent {
|
||
|
src_factor: BlendFactor::One,
|
||
|
dst_factor: BlendFactor::One,
|
||
|
operation: BlendOperation::Add,
|
||
|
},
|
||
|
}),
|
||
|
write_mask: ColorWrite::ALL,
|
||
|
}],
|
||
|
..RenderPipelineDescriptor::new(
|
||
|
ShaderStages {
|
||
|
vertex,
|
||
|
fragment: Some(fragment),
|
||
|
},
|
||
|
pipeline_layout,
|
||
|
)
|
||
|
};
|
||
|
|
||
|
let pipeline = render_resources.create_render_pipeline(&pipeline_descriptor);
|
||
|
|
||
|
PbrShaders {
|
||
|
pipeline,
|
||
|
pipeline_descriptor,
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
struct ExtractedMesh {
|
||
|
transform: Mat4,
|
||
|
vertex_buffer: BufferId,
|
||
|
index_info: Option<IndexInfo>,
|
||
|
transform_binding_offset: u32,
|
||
|
}
|
||
|
|
||
|
struct IndexInfo {
|
||
|
buffer: BufferId,
|
||
|
count: u32,
|
||
|
}
|
||
|
|
||
|
pub struct ExtractedMeshes {
|
||
|
meshes: Vec<ExtractedMesh>,
|
||
|
}
|
||
|
|
||
|
pub fn extract_meshes(
|
||
|
mut commands: Commands,
|
||
|
meshes: Res<Assets<Mesh>>,
|
||
|
_materials: Res<Assets<StandardMaterial>>,
|
||
|
query: Query<(&GlobalTransform, &Handle<Mesh>, &Handle<StandardMaterial>)>,
|
||
|
) {
|
||
|
let mut extracted_meshes = Vec::new();
|
||
|
for (transform, mesh_handle, _material_handle) in query.iter() {
|
||
|
if let Some(mesh) = meshes.get(mesh_handle) {
|
||
|
if let Some(gpu_data) = &mesh.gpu_data() {
|
||
|
extracted_meshes.push(ExtractedMesh {
|
||
|
transform: transform.compute_matrix(),
|
||
|
vertex_buffer: gpu_data.vertex_buffer,
|
||
|
index_info: gpu_data.index_buffer.map(|i| IndexInfo {
|
||
|
buffer: i,
|
||
|
count: mesh.indices().unwrap().len() as u32,
|
||
|
}),
|
||
|
transform_binding_offset: 0,
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
commands.insert_resource(ExtractedMeshes {
|
||
|
meshes: extracted_meshes,
|
||
|
});
|
||
|
}
|
||
|
|
||
|
#[derive(Default)]
|
||
|
pub struct MeshMeta {
|
||
|
transform_uniforms: DynamicUniformVec<Mat4>,
|
||
|
}
|
||
|
|
||
|
pub fn prepare_meshes(
|
||
|
render_resources: Res<RenderResources>,
|
||
|
mut mesh_meta: ResMut<MeshMeta>,
|
||
|
mut extracted_meshes: ResMut<ExtractedMeshes>,
|
||
|
) {
|
||
|
mesh_meta
|
||
|
.transform_uniforms
|
||
|
.reserve_and_clear(extracted_meshes.meshes.len(), &render_resources);
|
||
|
for extracted_mesh in extracted_meshes.meshes.iter_mut() {
|
||
|
extracted_mesh.transform_binding_offset =
|
||
|
mesh_meta.transform_uniforms.push(extracted_mesh.transform);
|
||
|
}
|
||
|
|
||
|
mesh_meta
|
||
|
.transform_uniforms
|
||
|
.write_to_staging_buffer(&render_resources);
|
||
|
}
|
||
|
|
||
|
// TODO: This is temporary. Once we expose BindGroupLayouts directly, we can create view bind groups without specific shader context
|
||
|
struct MeshViewBindGroups {
|
||
|
view_bind_group: BindGroupId,
|
||
|
mesh_transform_bind_group: BindGroupId,
|
||
|
}
|
||
|
|
||
|
pub fn queue_meshes(
|
||
|
mut commands: Commands,
|
||
|
draw_functions: Res<DrawFunctions>,
|
||
|
render_resources: Res<RenderResources>,
|
||
|
pbr_shaders: Res<PbrShaders>,
|
||
|
shadow_shaders: Res<ShadowShaders>,
|
||
|
mesh_meta: Res<MeshMeta>,
|
||
|
light_meta: Res<LightMeta>,
|
||
|
view_meta: Res<ViewMeta>,
|
||
|
extracted_meshes: Res<ExtractedMeshes>,
|
||
|
mut views: Query<(Entity, &ViewLights, &mut RenderPhase<Transparent3dPhase>)>,
|
||
|
mut view_light_shadow_phases: Query<&mut RenderPhase<ShadowPhase>>,
|
||
|
) {
|
||
|
if extracted_meshes.meshes.is_empty() {
|
||
|
return;
|
||
|
}
|
||
|
for (entity, view_lights, mut transparent_phase) in views.iter_mut() {
|
||
|
let layout = &pbr_shaders.pipeline_descriptor.layout;
|
||
|
let view_bind_group = BindGroupBuilder::default()
|
||
|
.add_binding(0, view_meta.uniforms.binding())
|
||
|
.add_binding(1, light_meta.view_gpu_lights.binding())
|
||
|
.add_binding(2, view_lights.light_depth_texture_view)
|
||
|
.add_binding(3, shadow_shaders.light_sampler)
|
||
|
.finish();
|
||
|
|
||
|
// TODO: this will only create the bind group if it isn't already created. this is a bit nasty
|
||
|
render_resources.create_bind_group(layout.bind_group(0).id, &view_bind_group);
|
||
|
|
||
|
let mesh_transform_bind_group = BindGroupBuilder::default()
|
||
|
.add_binding(0, mesh_meta.transform_uniforms.binding())
|
||
|
.finish();
|
||
|
render_resources.create_bind_group(layout.bind_group(1).id, &mesh_transform_bind_group);
|
||
|
|
||
|
commands.entity(entity).insert(MeshViewBindGroups {
|
||
|
view_bind_group: view_bind_group.id,
|
||
|
mesh_transform_bind_group: mesh_transform_bind_group.id,
|
||
|
});
|
||
|
|
||
|
let draw_pbr = draw_functions.read().get_id::<DrawPbr>().unwrap();
|
||
|
for i in 0..extracted_meshes.meshes.len() {
|
||
|
// TODO: currently there is only "transparent phase". this should pick transparent vs opaque according to the mesh material
|
||
|
transparent_phase.add(Drawable {
|
||
|
draw_function: draw_pbr,
|
||
|
draw_key: i,
|
||
|
sort_key: 0, // TODO: sort back-to-front
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// ultimately lights should check meshes for relevancy (ex: light views can "see" different meshes than the main view can)
|
||
|
let draw_shadow_mesh = draw_functions.read().get_id::<DrawShadowMesh>().unwrap();
|
||
|
for view_light_entity in view_lights.lights.iter().copied() {
|
||
|
let mut shadow_phase = view_light_shadow_phases.get_mut(view_light_entity).unwrap();
|
||
|
let layout = &shadow_shaders.pipeline_descriptor.layout;
|
||
|
let shadow_view_bind_group = BindGroupBuilder::default()
|
||
|
.add_binding(0, view_meta.uniforms.binding())
|
||
|
.finish();
|
||
|
|
||
|
render_resources.create_bind_group(layout.bind_group(0).id, &shadow_view_bind_group);
|
||
|
// TODO: this should only queue up meshes that are actually visible by each "light view"
|
||
|
for i in 0..extracted_meshes.meshes.len() {
|
||
|
shadow_phase.add(Drawable {
|
||
|
draw_function: draw_shadow_mesh,
|
||
|
draw_key: i,
|
||
|
sort_key: 0, // TODO: sort back-to-front
|
||
|
})
|
||
|
}
|
||
|
|
||
|
commands
|
||
|
.entity(view_light_entity)
|
||
|
.insert(MeshViewBindGroups {
|
||
|
view_bind_group: shadow_view_bind_group.id,
|
||
|
mesh_transform_bind_group: mesh_transform_bind_group.id,
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: this logic can be moved to prepare_meshes once wgpu::Queue is exposed directly
|
||
|
pub struct PbrNode;
|
||
|
|
||
|
impl Node for PbrNode {
|
||
|
fn run(
|
||
|
&self,
|
||
|
_graph: &mut RenderGraphContext,
|
||
|
render_context: &mut dyn RenderContext,
|
||
|
world: &World,
|
||
|
) -> Result<(), NodeRunError> {
|
||
|
let mesh_meta = world.get_resource::<MeshMeta>().unwrap();
|
||
|
let light_meta = world.get_resource::<LightMeta>().unwrap();
|
||
|
mesh_meta
|
||
|
.transform_uniforms
|
||
|
.write_to_uniform_buffer(render_context);
|
||
|
light_meta
|
||
|
.view_gpu_lights
|
||
|
.write_to_uniform_buffer(render_context);
|
||
|
Ok(())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type DrawPbrParams<'a> = (
|
||
|
Res<'a, PbrShaders>,
|
||
|
Res<'a, ExtractedMeshes>,
|
||
|
Query<'a, (&'a ViewUniform, &'a MeshViewBindGroups, &'a ViewLights)>,
|
||
|
);
|
||
|
pub struct DrawPbr {
|
||
|
params: SystemState<DrawPbrParams<'static>>,
|
||
|
}
|
||
|
|
||
|
impl DrawPbr {
|
||
|
pub fn new(world: &mut World) -> Self {
|
||
|
Self {
|
||
|
params: SystemState::new(world),
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl Draw for DrawPbr {
|
||
|
fn draw(
|
||
|
&mut self,
|
||
|
world: &World,
|
||
|
pass: &mut TrackedRenderPass,
|
||
|
view: Entity,
|
||
|
draw_key: usize,
|
||
|
_sort_key: usize,
|
||
|
) {
|
||
|
let (pbr_shaders, extracted_meshes, views) = self.params.get(world);
|
||
|
let (view_uniforms, mesh_view_bind_groups, view_lights) = views.get(view).unwrap();
|
||
|
let layout = &pbr_shaders.pipeline_descriptor.layout;
|
||
|
let extracted_mesh = &extracted_meshes.meshes[draw_key];
|
||
|
pass.set_pipeline(pbr_shaders.pipeline);
|
||
|
pass.set_bind_group(
|
||
|
0,
|
||
|
layout.bind_group(0).id,
|
||
|
mesh_view_bind_groups.view_bind_group,
|
||
|
Some(&[
|
||
|
view_uniforms.view_uniform_offset,
|
||
|
view_lights.gpu_light_binding_index,
|
||
|
]),
|
||
|
);
|
||
|
pass.set_bind_group(
|
||
|
1,
|
||
|
layout.bind_group(1).id,
|
||
|
mesh_view_bind_groups.mesh_transform_bind_group,
|
||
|
Some(&[extracted_mesh.transform_binding_offset]),
|
||
|
);
|
||
|
pass.set_vertex_buffer(0, extracted_mesh.vertex_buffer, 0);
|
||
|
if let Some(index_info) = &extracted_mesh.index_info {
|
||
|
pass.set_index_buffer(index_info.buffer, 0, IndexFormat::Uint32);
|
||
|
pass.draw_indexed(0..index_info.count, 0, 0..1);
|
||
|
} else {
|
||
|
panic!("non-indexed drawing not supported yet")
|
||
|
}
|
||
|
}
|
||
|
}
|