1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
use crate::{
wgpu_buffer_types,
wgpu_resources::{
BindGroupDesc, BindGroupEntry, BindGroupLayoutDesc, GpuBindGroup, GpuBindGroupLayoutHandle,
GpuSamplerHandle, SamplerDesc, WgpuResourcePools,
},
};
use bytemuck::{Pod, Zeroable};
use smallvec::smallvec;
/// Mirrors the GPU contents of a frame-global uniform buffer.
///
/// Contains information that is constant for a single frame like camera.
/// (does not contain information that is special to a particular renderer)
#[repr(C)]
#[derive(Clone, Copy, Zeroable, Pod)]
pub struct FrameUniformBuffer {
pub view_from_world: wgpu_buffer_types::Mat4x3,
pub projection_from_view: wgpu_buffer_types::Mat4,
pub projection_from_world: wgpu_buffer_types::Mat4,
/// Camera position in world space.
pub camera_position: glam::Vec3,
/// For perspective: Multiply this with a camera distance to get a measure of how wide a pixel is in world units.
/// For orthographic: This is the world size value, independent of distance.
pub pixel_world_size_from_camera_distance: f32,
/// Camera direction in world space.
/// Same as `-view_from_world.row(2).truncate()`
pub camera_forward: glam::Vec3,
/// How many pixels there are per point.
/// I.e. the UI zoom factor
pub pixels_per_point: f32,
/// `(tan(fov_y / 2) * aspect_ratio, tan(fov_y /2))`, i.e. half ratio of screen dimension to screen distance in x & y.
/// Both values are set to f32max for orthographic projection
pub tan_half_fov: wgpu_buffer_types::Vec2RowPadded,
/// `re_renderer` defined device tier.
pub device_tier: wgpu_buffer_types::U32RowPadded,
}
pub(crate) struct GlobalBindings {
pub(crate) layout: GpuBindGroupLayoutHandle,
nearest_neighbor_sampler_repeat: GpuSamplerHandle,
nearest_neighbor_sampler_clamped: GpuSamplerHandle,
trilinear_sampler_repeat: GpuSamplerHandle,
}
impl GlobalBindings {
pub fn new(pools: &WgpuResourcePools, device: &wgpu::Device) -> Self {
Self {
layout: pools.bind_group_layouts.get_or_create(
device,
&BindGroupLayoutDesc {
label: "GlobalBindings::layout".into(),
// Needs to be kept in sync with `global_bindings.wgsl` / `create_bind_group`
entries: vec![
// The global per-frame uniform buffer.
wgpu::BindGroupLayoutEntry {
binding: 0,
visibility: wgpu::ShaderStages::all(),
ty: wgpu::BindingType::Buffer {
ty: wgpu::BufferBindingType::Uniform,
has_dynamic_offset: false,
min_binding_size: (std::mem::size_of::<FrameUniformBuffer>()
as u64)
.try_into()
.ok(),
},
count: None,
},
// Sampler without any filtering, repeat.
wgpu::BindGroupLayoutEntry {
binding: 1,
visibility: wgpu::ShaderStages::FRAGMENT,
ty: wgpu::BindingType::Sampler(wgpu::SamplerBindingType::NonFiltering),
count: None,
},
// Sampler without any filtering, clamped.
wgpu::BindGroupLayoutEntry {
binding: 2,
visibility: wgpu::ShaderStages::FRAGMENT,
ty: wgpu::BindingType::Sampler(wgpu::SamplerBindingType::NonFiltering),
count: None,
},
// Trilinear sampler, repeat.
wgpu::BindGroupLayoutEntry {
binding: 3,
visibility: wgpu::ShaderStages::FRAGMENT,
ty: wgpu::BindingType::Sampler(wgpu::SamplerBindingType::Filtering),
count: None,
},
],
},
),
nearest_neighbor_sampler_repeat: pools.samplers.get_or_create(
device,
&SamplerDesc {
label: "GlobalBindings::nearest_neighbor_sampler".into(),
address_mode_u: wgpu::AddressMode::Repeat,
address_mode_v: wgpu::AddressMode::Repeat,
address_mode_w: wgpu::AddressMode::Repeat,
..Default::default()
},
),
nearest_neighbor_sampler_clamped: pools.samplers.get_or_create(
device,
&SamplerDesc {
label: "GlobalBindings::nearest_neighbor_sampler_clamped".into(),
address_mode_u: wgpu::AddressMode::ClampToEdge,
address_mode_v: wgpu::AddressMode::ClampToEdge,
address_mode_w: wgpu::AddressMode::ClampToEdge,
..Default::default()
},
),
trilinear_sampler_repeat: pools.samplers.get_or_create(
device,
&SamplerDesc {
label: "GlobalBindings::trilinear_sampler".into(),
mag_filter: wgpu::FilterMode::Linear,
min_filter: wgpu::FilterMode::Linear,
mipmap_filter: wgpu::FilterMode::Linear,
address_mode_u: wgpu::AddressMode::Repeat,
address_mode_v: wgpu::AddressMode::Repeat,
address_mode_w: wgpu::AddressMode::Repeat,
..Default::default()
},
),
}
}
/// Creates a bind group that follows the global bind group layout.
pub fn create_bind_group(
&self,
pools: &WgpuResourcePools,
device: &wgpu::Device,
frame_uniform_buffer_binding: BindGroupEntry,
) -> GpuBindGroup {
pools.bind_groups.alloc(
device,
pools,
// Needs to be kept in sync with `global_bindings.wgsl` / `self.layout`
&BindGroupDesc {
label: "GlobalBindings::create_bind_group".into(),
entries: smallvec![
frame_uniform_buffer_binding,
BindGroupEntry::Sampler(self.nearest_neighbor_sampler_repeat),
BindGroupEntry::Sampler(self.nearest_neighbor_sampler_clamped),
BindGroupEntry::Sampler(self.trilinear_sampler_repeat),
],
layout: self.layout,
},
)
}
}