use std::sync::Arc;
use ahash::{HashMap, HashMapExt};
use smallvec::smallvec;
use crate::{
draw_phases::{DrawPhase, OutlineMaskProcessor},
include_shader_module,
mesh::{gpu_data::MaterialUniformBuffer, mesh_vertices, GpuMesh},
view_builder::ViewBuilder,
wgpu_resources::{
BindGroupLayoutDesc, BufferDesc, GpuBindGroupLayoutHandle, GpuBuffer,
GpuRenderPipelineHandle, GpuRenderPipelinePoolAccessor, PipelineLayoutDesc,
RenderPipelineDesc,
},
Color32, CpuWriteGpuReadError, OutlineMaskPreference, PickingLayerId, PickingLayerProcessor,
};
use super::{DrawData, DrawError, RenderContext, Renderer};
mod gpu_data {
use ecolor::Color32;
use crate::{mesh::mesh_vertices, wgpu_resources::VertexBufferLayout};
#[repr(C)]
#[derive(Clone, Copy, bytemuck::Pod, bytemuck::Zeroable)]
pub struct InstanceData {
pub world_from_mesh_row_0: [f32; 4],
pub world_from_mesh_row_1: [f32; 4],
pub world_from_mesh_row_2: [f32; 4],
pub world_from_mesh_normal_row_0: [f32; 3],
pub world_from_mesh_normal_row_1: [f32; 3],
pub world_from_mesh_normal_row_2: [f32; 3],
pub additive_tint: Color32,
pub picking_layer_id: [u32; 4],
pub outline_mask_ids: [u8; 4],
}
impl InstanceData {
pub fn vertex_buffer_layout() -> VertexBufferLayout {
let shader_start_location = mesh_vertices::next_free_shader_location();
VertexBufferLayout {
array_stride: std::mem::size_of::<Self>() as _,
step_mode: wgpu::VertexStepMode::Instance,
attributes: VertexBufferLayout::attributes_from_formats(
shader_start_location,
[
wgpu::VertexFormat::Float32x4,
wgpu::VertexFormat::Float32x4,
wgpu::VertexFormat::Float32x4,
wgpu::VertexFormat::Float32x3,
wgpu::VertexFormat::Float32x3,
wgpu::VertexFormat::Float32x3,
wgpu::VertexFormat::Unorm8x4,
wgpu::VertexFormat::Uint32x4,
wgpu::VertexFormat::Uint8x2,
]
.into_iter(),
),
}
}
}
}
#[derive(Clone)]
struct MeshBatch {
mesh: Arc<GpuMesh>,
count: u32,
count_with_outlines: u32,
}
#[derive(Clone)]
pub struct MeshDrawData {
instance_buffer: Option<GpuBuffer>,
batches: Vec<MeshBatch>,
}
impl DrawData for MeshDrawData {
type Renderer = MeshRenderer;
}
pub struct GpuMeshInstance {
pub gpu_mesh: Arc<GpuMesh>,
pub world_from_mesh: glam::Affine3A,
pub additive_tint: Color32,
pub outline_mask_ids: OutlineMaskPreference,
pub picking_layer_id: PickingLayerId,
}
impl GpuMeshInstance {
pub fn new(gpu_mesh: Arc<GpuMesh>) -> Self {
Self {
gpu_mesh,
world_from_mesh: glam::Affine3A::IDENTITY,
additive_tint: Color32::TRANSPARENT,
outline_mask_ids: OutlineMaskPreference::NONE,
picking_layer_id: PickingLayerId::default(),
}
}
}
impl MeshDrawData {
pub fn new(
ctx: &RenderContext,
instances: &[GpuMeshInstance],
) -> Result<Self, CpuWriteGpuReadError> {
re_tracing::profile_function!();
let _mesh_renderer = ctx.renderer::<MeshRenderer>();
if instances.is_empty() {
return Ok(Self {
batches: Vec::new(),
instance_buffer: None,
});
}
let instance_buffer_size =
(std::mem::size_of::<gpu_data::InstanceData>() * instances.len()) as _;
let instance_buffer = ctx.gpu_resources.buffers.alloc(
&ctx.device,
&BufferDesc {
label: "MeshDrawData::instance_buffer".into(),
size: instance_buffer_size,
usage: wgpu::BufferUsages::VERTEX | wgpu::BufferUsages::COPY_DST,
mapped_at_creation: false,
},
);
let mut instances_by_mesh: HashMap<_, Vec<_>> = HashMap::new();
for instance in instances {
instances_by_mesh
.entry(Arc::as_ptr(&instance.gpu_mesh))
.or_insert_with(|| Vec::with_capacity(instances.len()))
.push(instance);
}
let mut batches = Vec::new();
{
let mut instance_buffer_staging = ctx
.cpu_write_gpu_read_belt
.lock()
.allocate::<gpu_data::InstanceData>(
&ctx.device,
&ctx.gpu_resources.buffers,
instances.len(),
)?;
let mut num_processed_instances = 0;
for (_mesh_ptr, mut instances) in instances_by_mesh {
let mut count = 0;
let mut count_with_outlines = 0;
instances.sort_by(|a, b| {
a.outline_mask_ids
.is_none()
.cmp(&b.outline_mask_ids.is_none())
});
let mut mesh = None;
for instance in instances {
if mesh.is_none() {
mesh = Some(instance.gpu_mesh.clone());
}
count += 1;
count_with_outlines += instance.outline_mask_ids.is_some() as u32;
let world_from_mesh_mat3 = instance.world_from_mesh.matrix3;
let world_from_mesh_normal =
if instance.world_from_mesh.matrix3.determinant() != 0.0 {
instance.world_from_mesh.matrix3.inverse().transpose()
} else {
glam::Mat3A::ZERO
};
instance_buffer_staging.push(gpu_data::InstanceData {
world_from_mesh_row_0: world_from_mesh_mat3
.row(0)
.extend(instance.world_from_mesh.translation.x)
.to_array(),
world_from_mesh_row_1: world_from_mesh_mat3
.row(1)
.extend(instance.world_from_mesh.translation.y)
.to_array(),
world_from_mesh_row_2: world_from_mesh_mat3
.row(2)
.extend(instance.world_from_mesh.translation.z)
.to_array(),
world_from_mesh_normal_row_0: world_from_mesh_normal.row(0).to_array(),
world_from_mesh_normal_row_1: world_from_mesh_normal.row(1).to_array(),
world_from_mesh_normal_row_2: world_from_mesh_normal.row(2).to_array(),
additive_tint: instance.additive_tint,
outline_mask_ids: instance
.outline_mask_ids
.0
.map_or([0, 0, 0, 0], |mask| [mask[0], mask[1], 0, 0]),
picking_layer_id: instance.picking_layer_id.into(),
})?;
}
num_processed_instances += count;
if let Some(mesh) = mesh {
batches.push(MeshBatch {
mesh,
count: count as _,
count_with_outlines,
});
}
}
assert_eq!(num_processed_instances, instances.len());
instance_buffer_staging.copy_to_buffer(
ctx.active_frame.before_view_builder_encoder.lock().get(),
&instance_buffer,
0,
)?;
}
Ok(Self {
batches,
instance_buffer: Some(instance_buffer),
})
}
}
pub struct MeshRenderer {
render_pipeline_shaded: GpuRenderPipelineHandle,
render_pipeline_picking_layer: GpuRenderPipelineHandle,
render_pipeline_outline_mask: GpuRenderPipelineHandle,
pub bind_group_layout: GpuBindGroupLayoutHandle,
}
impl Renderer for MeshRenderer {
type RendererDrawData = MeshDrawData;
fn participated_phases() -> &'static [DrawPhase] {
&[
DrawPhase::Opaque,
DrawPhase::OutlineMask,
DrawPhase::PickingLayer,
]
}
fn create_renderer(ctx: &RenderContext) -> Self {
re_tracing::profile_function!();
let render_pipelines = &ctx.gpu_resources.render_pipelines;
let bind_group_layout = ctx.gpu_resources.bind_group_layouts.get_or_create(
&ctx.device,
&BindGroupLayoutDesc {
label: "MeshRenderer::bind_group_layout".into(),
entries: vec![
wgpu::BindGroupLayoutEntry {
binding: 0,
visibility: wgpu::ShaderStages::FRAGMENT,
ty: wgpu::BindingType::Texture {
sample_type: wgpu::TextureSampleType::Float { filterable: true },
view_dimension: wgpu::TextureViewDimension::D2,
multisampled: false,
},
count: None,
},
wgpu::BindGroupLayoutEntry {
binding: 1,
visibility: wgpu::ShaderStages::FRAGMENT,
ty: wgpu::BindingType::Buffer {
ty: wgpu::BufferBindingType::Uniform,
has_dynamic_offset: false,
min_binding_size: (std::mem::size_of::<MaterialUniformBuffer>() as u64)
.try_into()
.ok(),
},
count: None,
},
],
},
);
let pipeline_layout = ctx.gpu_resources.pipeline_layouts.get_or_create(
ctx,
&PipelineLayoutDesc {
label: "MeshRenderer::pipeline_layout".into(),
entries: vec![ctx.global_bindings.layout, bind_group_layout],
},
);
let shader_module = ctx.gpu_resources.shader_modules.get_or_create(
ctx,
&include_shader_module!("../../shader/instanced_mesh.wgsl"),
);
let primitive = wgpu::PrimitiveState {
topology: wgpu::PrimitiveTopology::TriangleList,
cull_mode: None, ..Default::default()
};
let vertex_buffers: smallvec::SmallVec<[_; 4]> =
std::iter::once(gpu_data::InstanceData::vertex_buffer_layout())
.chain(mesh_vertices::vertex_buffer_layouts())
.collect();
let render_pipeline_shaded_desc = RenderPipelineDesc {
label: "MeshRenderer::render_pipeline_shaded".into(),
pipeline_layout,
vertex_entrypoint: "vs_main".into(),
vertex_handle: shader_module,
fragment_entrypoint: "fs_main_shaded".into(),
fragment_handle: shader_module,
vertex_buffers,
render_targets: smallvec![Some(ViewBuilder::MAIN_TARGET_COLOR_FORMAT.into())],
primitive,
depth_stencil: ViewBuilder::MAIN_TARGET_DEFAULT_DEPTH_STATE,
multisample: ViewBuilder::MAIN_TARGET_DEFAULT_MSAA_STATE,
};
let render_pipeline_shaded =
render_pipelines.get_or_create(ctx, &render_pipeline_shaded_desc);
let render_pipeline_picking_layer = render_pipelines.get_or_create(
ctx,
&RenderPipelineDesc {
label: "MeshRenderer::render_pipeline_picking_layer".into(),
fragment_entrypoint: "fs_main_picking_layer".into(),
render_targets: smallvec![Some(PickingLayerProcessor::PICKING_LAYER_FORMAT.into())],
depth_stencil: PickingLayerProcessor::PICKING_LAYER_DEPTH_STATE,
multisample: PickingLayerProcessor::PICKING_LAYER_MSAA_STATE,
..render_pipeline_shaded_desc.clone()
},
);
let render_pipeline_outline_mask = render_pipelines.get_or_create(
ctx,
&RenderPipelineDesc {
label: "MeshRenderer::render_pipeline_outline_mask".into(),
fragment_entrypoint: "fs_main_outline_mask".into(),
render_targets: smallvec![Some(OutlineMaskProcessor::MASK_FORMAT.into())],
depth_stencil: OutlineMaskProcessor::MASK_DEPTH_STATE,
multisample: OutlineMaskProcessor::mask_default_msaa_state(ctx.device_caps().tier),
..render_pipeline_shaded_desc
},
);
Self {
render_pipeline_shaded,
render_pipeline_picking_layer,
render_pipeline_outline_mask,
bind_group_layout,
}
}
fn draw(
&self,
render_pipelines: &GpuRenderPipelinePoolAccessor<'_>,
phase: DrawPhase,
pass: &mut wgpu::RenderPass<'_>,
draw_data: &Self::RendererDrawData,
) -> Result<(), DrawError> {
re_tracing::profile_function!();
let Some(instance_buffer) = &draw_data.instance_buffer else {
return Ok(()); };
let pipeline_handle = match phase {
DrawPhase::OutlineMask => self.render_pipeline_outline_mask,
DrawPhase::Opaque => self.render_pipeline_shaded,
DrawPhase::PickingLayer => self.render_pipeline_picking_layer,
_ => unreachable!("We were called on a phase we weren't subscribed to: {phase:?}"),
};
let pipeline = render_pipelines.get(pipeline_handle)?;
pass.set_pipeline(pipeline);
pass.set_vertex_buffer(0, instance_buffer.slice(..));
let mut instance_start_index = 0;
for mesh_batch in &draw_data.batches {
if phase == DrawPhase::OutlineMask && mesh_batch.count_with_outlines == 0 {
instance_start_index += mesh_batch.count;
continue;
}
let vertex_buffer_combined = &mesh_batch.mesh.vertex_buffer_combined;
let index_buffer = &mesh_batch.mesh.index_buffer;
pass.set_vertex_buffer(
1,
vertex_buffer_combined.slice(mesh_batch.mesh.vertex_buffer_positions_range.clone()),
);
pass.set_vertex_buffer(
2,
vertex_buffer_combined.slice(mesh_batch.mesh.vertex_buffer_colors_range.clone()),
);
pass.set_vertex_buffer(
3,
vertex_buffer_combined.slice(mesh_batch.mesh.vertex_buffer_normals_range.clone()),
);
pass.set_vertex_buffer(
4,
vertex_buffer_combined.slice(mesh_batch.mesh.vertex_buffer_texcoord_range.clone()),
);
pass.set_index_buffer(
index_buffer.slice(mesh_batch.mesh.index_buffer_range.clone()),
wgpu::IndexFormat::Uint32,
);
let num_meshes_to_draw = if phase == DrawPhase::OutlineMask {
mesh_batch.count_with_outlines
} else {
mesh_batch.count
};
let instance_range = instance_start_index..(instance_start_index + num_meshes_to_draw);
for material in &mesh_batch.mesh.materials {
debug_assert!(num_meshes_to_draw > 0);
pass.set_bind_group(1, &material.bind_group, &[]);
pass.draw_indexed(material.index_range.clone(), 0, instance_range.clone());
}
instance_start_index += mesh_batch.count;
}
Ok(())
}
}