From 4907186de47a4f3e77027662ac0efcdb4f3eaf12 Mon Sep 17 00:00:00 2001 From: Raph Levien Date: Thu, 12 Jan 2023 18:56:48 -0800 Subject: [PATCH 1/2] Prototype of buffer reuse This helps performance but not all performance issues have been resolved. Nontrivial CPU goes into write_buffer, and it's also possible that there isn't enough overlapping between CPU and GPU work. --- shader/coarse.wgsl | 4 +- src/engine.rs | 100 ++++++++++++++++++++++++++++++--------------- 2 files changed, 70 insertions(+), 34 deletions(-) diff --git a/shader/coarse.wgsl b/shader/coarse.wgsl index dec040e..8728903 100644 --- a/shader/coarse.wgsl +++ b/shader/coarse.wgsl @@ -399,8 +399,8 @@ fn main( } workgroupBarrier(); } - if bin_tile_x < config.width_in_tiles && bin_tile_y < config.height_in_tiles { - //ptcl[cmd_offset] = CMD_END; + if bin_tile_x + tile_x < config.width_in_tiles && bin_tile_y + tile_y < config.height_in_tiles { + ptcl[cmd_offset] = CMD_END; // TODO: blend stack allocation } } diff --git a/src/engine.rs b/src/engine.rs index 5c3f6b9..f965bb6 100644 --- a/src/engine.rs +++ b/src/engine.rs @@ -26,7 +26,7 @@ use parking_lot::RawMutex; use wgpu::{ util::DeviceExt, BindGroup, BindGroupLayout, Buffer, BufferAsyncError, BufferSlice, BufferView, ComputePipeline, Device, Queue, Texture, TextureAspect, TextureFormat, TextureUsages, - TextureView, TextureViewDimension, + TextureView, TextureViewDimension, BufferUsages, }; pub type Error = Box; @@ -41,6 +41,7 @@ static ID_COUNTER: AtomicU64 = AtomicU64::new(0); pub struct Engine { shaders: Vec, + pool: ResourcePool, } struct Shader { @@ -123,9 +124,17 @@ struct BindMap { image_map: HashMap, } +#[derive(Default)] +struct ResourcePool { + bufs: HashMap<(u64, BufferUsages), Vec>, +} + impl Engine { pub fn new() -> Engine { - Engine { shaders: vec![] } + Engine { + shaders: vec![], + pool: Default::default(), + } } /// Add a shader. @@ -233,21 +242,18 @@ impl Engine { for command in &recording.commands { match command { Command::Upload(buf_proxy, bytes) => { - let buf = device.create_buffer_init(&wgpu::util::BufferInitDescriptor { - label: None, - contents: &bytes, - usage: wgpu::BufferUsages::STORAGE - | wgpu::BufferUsages::COPY_DST - | wgpu::BufferUsages::COPY_SRC, - }); + let usage = BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; + let buf = self.pool.get_buf(bytes.len() as u64, usage, device); + // TODO: if buffer is newly created, might be better to make it mapped at creation + // and copy. However, we expect reuse will be most common. + queue.write_buffer(&buf, 0, bytes); bind_map.insert_buf(buf_proxy.id, buf); } Command::UploadUniform(buf_proxy, bytes) => { - let buf = device.create_buffer_init(&wgpu::util::BufferInitDescriptor { - label: None, - contents: &bytes, - usage: wgpu::BufferUsages::UNIFORM, - }); + let usage = BufferUsages::UNIFORM | BufferUsages::COPY_DST; + // Same consideration as above + let buf = self.pool.get_buf(bytes.len() as u64, usage, device); + queue.write_buffer(&buf, 0, bytes); bind_map.insert_buf(buf_proxy.id, buf); } Command::UploadImage(image_proxy, bytes) => { @@ -310,6 +316,7 @@ impl Engine { &shader.bind_group_layout, bindings, external_resources, + &mut self.pool, )?; let mut cpass = encoder.begin_compute_pass(&Default::default()); cpass.set_pipeline(&shader.pipeline); @@ -328,12 +335,13 @@ impl Engine { downloads.buf_map.insert(proxy.id, buf); } Command::Clear(proxy, offset, size) => { - let buffer = bind_map.get_or_create(*proxy, device)?; + let buffer = bind_map.get_or_create(*proxy, device, &mut self.pool)?; encoder.clear_buffer(buffer, *offset, *size); } } } queue.submit(Some(encoder.finish())); + self.pool.reap_bindmap(bind_map); Ok(downloads) } } @@ -481,6 +489,7 @@ impl BindMap { layout: &BindGroupLayout, bindings: &[ResourceProxy], external_resources: &[ExternalResource], + pool: &mut ResourcePool, ) -> Result { // These functions are ugly and linear, but the remap array should generally be // small. Should find a better solution for this. @@ -519,14 +528,8 @@ impl BindMap { continue; } if let Entry::Vacant(v) = self.buf_map.entry(proxy.id) { - let buf = device.create_buffer(&wgpu::BufferDescriptor { - label: None, - size: proxy.size, - usage: wgpu::BufferUsages::STORAGE - | wgpu::BufferUsages::COPY_DST - | wgpu::BufferUsages::COPY_SRC, - mapped_at_creation: false, - }); + let usage = BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; + let buf = pool.get_buf(proxy.size, usage, device); v.insert(buf); } } @@ -595,18 +598,12 @@ impl BindMap { Ok(bind_group) } - fn get_or_create(&mut self, proxy: BufProxy, device: &Device) -> Result<&Buffer, Error> { + fn get_or_create(&mut self, proxy: BufProxy, device: &Device, pool: &mut ResourcePool) -> Result<&Buffer, Error> { match self.buf_map.entry(proxy.id) { Entry::Occupied(occupied) => Ok(occupied.into_mut()), Entry::Vacant(vacant) => { - let buf = device.create_buffer(&wgpu::BufferDescriptor { - label: None, - size: proxy.size, - usage: wgpu::BufferUsages::STORAGE - | wgpu::BufferUsages::COPY_DST - | wgpu::BufferUsages::COPY_SRC, - mapped_at_creation: false, - }); + let usage = BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; + let buf = pool.get_buf(proxy.size, usage, device); Ok(vacant.insert(buf)) } } @@ -648,3 +645,42 @@ impl<'a> DownloadsMapped<'a> { Ok(slice.get_mapped_range()) } } + +const SIZE_CLASS_BITS: u32 = 1; + +impl ResourcePool { + /// Get a buffer from the pool or create one. + fn get_buf(&mut self, size: u64, usage: BufferUsages, device: &Device) -> Buffer { + let rounded_size = Self::size_class(size, SIZE_CLASS_BITS); + if let Some(buf_vec) = self.bufs.get_mut(&(rounded_size, usage)) { + if let Some(buf) = buf_vec.pop() { + return buf; + } + } + device.create_buffer(&wgpu::BufferDescriptor { + label: None, + size: rounded_size, + usage, + mapped_at_creation: false, + }) + } + + fn reap_bindmap(&mut self, bind_map: BindMap) { + for (_id, buf) in bind_map.buf_map { + let size = buf.size(); + let usage = buf.usage(); + self.bufs.entry((size, usage)).or_default().push(buf); + } + } + + /// Quantize a size up to the nearest size class. + fn size_class(x: u64, bits: u32) -> u64 { + if x > 1 << bits { + let a = (x - 1).leading_zeros(); + let b = (x - 1) | (((u64::MAX / 2) >> bits) >> a); + b + 1 + } else { + 1 << bits + } + } +} From ed437f7ffc7f84dc7bdea0c2dd3429f2693c5bba Mon Sep 17 00:00:00 2001 From: Raph Levien Date: Thu, 12 Jan 2023 21:11:54 -0800 Subject: [PATCH 2/2] Rustfmt --- src/engine.rs | 19 +++++++++++++------ 1 file changed, 13 insertions(+), 6 deletions(-) diff --git a/src/engine.rs b/src/engine.rs index f965bb6..a823339 100644 --- a/src/engine.rs +++ b/src/engine.rs @@ -24,9 +24,9 @@ use std::{ use futures_intrusive::channel::shared::GenericOneshotReceiver; use parking_lot::RawMutex; use wgpu::{ - util::DeviceExt, BindGroup, BindGroupLayout, Buffer, BufferAsyncError, BufferSlice, BufferView, - ComputePipeline, Device, Queue, Texture, TextureAspect, TextureFormat, TextureUsages, - TextureView, TextureViewDimension, BufferUsages, + util::DeviceExt, BindGroup, BindGroupLayout, Buffer, BufferAsyncError, BufferSlice, + BufferUsages, BufferView, ComputePipeline, Device, Queue, Texture, TextureAspect, + TextureFormat, TextureUsages, TextureView, TextureViewDimension, }; pub type Error = Box; @@ -242,7 +242,8 @@ impl Engine { for command in &recording.commands { match command { Command::Upload(buf_proxy, bytes) => { - let usage = BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; + let usage = + BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; let buf = self.pool.get_buf(bytes.len() as u64, usage, device); // TODO: if buffer is newly created, might be better to make it mapped at creation // and copy. However, we expect reuse will be most common. @@ -528,7 +529,8 @@ impl BindMap { continue; } if let Entry::Vacant(v) = self.buf_map.entry(proxy.id) { - let usage = BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; + let usage = + BufferUsages::COPY_SRC | BufferUsages::COPY_DST | BufferUsages::STORAGE; let buf = pool.get_buf(proxy.size, usage, device); v.insert(buf); } @@ -598,7 +600,12 @@ impl BindMap { Ok(bind_group) } - fn get_or_create(&mut self, proxy: BufProxy, device: &Device, pool: &mut ResourcePool) -> Result<&Buffer, Error> { + fn get_or_create( + &mut self, + proxy: BufProxy, + device: &Device, + pool: &mut ResourcePool, + ) -> Result<&Buffer, Error> { match self.buf_map.entry(proxy.id) { Entry::Occupied(occupied) => Ok(occupied.into_mut()), Entry::Vacant(vacant) => {