From 75c4b62730682d694b7577b08ee79374ee37c0f4 Mon Sep 17 00:00:00 2001 From: Raph Levien Date: Tue, 17 Nov 2020 08:04:25 -0800 Subject: [PATCH] Add hub abstraction The hub does a little better lifetime tracking of resources (so Rust-side references can be dropped), and in the future will be used for dynamic selection of backend. The migration is still a bit half-baked, as there are a bunch of Vulkan-specific types in the signatures, but it shouldn't be too much work to sort that out. Perhaps it can wait until there is a second backend though. The main motivation for this is to create image objects with lifetime tracking, one of the things required for #38. --- piet-gpu-hal/examples/collatz.rs | 29 ++- piet-gpu-hal/src/hub.rs | 293 +++++++++++++++++++++++++++++++ piet-gpu-hal/src/lib.rs | 23 ++- piet-gpu-hal/src/vulkan.rs | 29 ++- piet-gpu/bin/cli.rs | 23 +-- piet-gpu/bin/winit.rs | 37 ++-- piet-gpu/src/lib.rs | 193 ++++++++++---------- 7 files changed, 486 insertions(+), 141 deletions(-) create mode 100644 piet-gpu-hal/src/hub.rs diff --git a/piet-gpu-hal/examples/collatz.rs b/piet-gpu-hal/examples/collatz.rs index a017662..3a261ad 100644 --- a/piet-gpu-hal/examples/collatz.rs +++ b/piet-gpu-hal/examples/collatz.rs @@ -1,24 +1,25 @@ +use piet_gpu_hal::hub; use piet_gpu_hal::vulkan::VkInstance; -use piet_gpu_hal::{CmdBuf, Device, MemFlags}; +use piet_gpu_hal::{CmdBuf, MemFlags}; fn main() { let (instance, _) = VkInstance::new(None).unwrap(); unsafe { let device = instance.device(None).unwrap(); - let fence = device.create_fence(false).unwrap(); + let session = hub::Session::new(device); let mem_flags = MemFlags::host_coherent(); let src = (0..256).map(|x| x + 1).collect::>(); - let buffer = device + let mut buffer = session .create_buffer(std::mem::size_of_val(&src[..]) as u64, mem_flags) .unwrap(); - device.write_buffer(&buffer, &src).unwrap(); + buffer.write(&src).unwrap(); let code = include_bytes!("./shader/collatz.spv"); - let pipeline = device.create_simple_compute_pipeline(code, 1, 0).unwrap(); - let descriptor_set = device - .create_descriptor_set(&pipeline, &[&buffer], &[]) + let pipeline = session.create_simple_compute_pipeline(code, 1, 0).unwrap(); + let descriptor_set = session + .create_descriptor_set(&pipeline, &[buffer.vk_buffer()], &[]) .unwrap(); - let query_pool = device.create_query_pool(2).unwrap(); - let mut cmd_buf = device.create_cmd_buf().unwrap(); + let query_pool = session.create_query_pool(2).unwrap(); + let mut cmd_buf = session.cmd_buf().unwrap(); cmd_buf.begin(); cmd_buf.reset_query_pool(&query_pool); cmd_buf.write_timestamp(&query_pool, 0); @@ -26,13 +27,11 @@ fn main() { cmd_buf.write_timestamp(&query_pool, 1); cmd_buf.host_barrier(); cmd_buf.finish(); - device - .run_cmd_buf(&cmd_buf, &[], &[], Some(&fence)) - .unwrap(); - device.wait_and_reset(&[fence]).unwrap(); - let timestamps = device.reap_query_pool(&query_pool); + let submitted = session.run_cmd_buf(cmd_buf, &[], &[]).unwrap(); + submitted.wait().unwrap(); + let timestamps = session.fetch_query_pool(&query_pool); let mut dst: Vec = Default::default(); - device.read_buffer(&buffer, &mut dst).unwrap(); + buffer.read(&mut dst).unwrap(); for (i, val) in dst.iter().enumerate().take(16) { println!("{}: {}", i, val); } diff --git a/piet-gpu-hal/src/hub.rs b/piet-gpu-hal/src/hub.rs new file mode 100644 index 0000000..d082924 --- /dev/null +++ b/piet-gpu-hal/src/hub.rs @@ -0,0 +1,293 @@ +//! A convenience layer on top of raw hal. +//! +//! This layer takes care of some lifetime and synchronization bookkeeping. +//! It is likely that it will also take care of compile time and runtime +//! negotiation of backends (Vulkan, DX12), but right now it's Vulkan-only. + +use std::any::Any; +use std::sync::{Arc, Mutex, Weak}; + +use crate::vulkan; +use crate::{Device, Error}; + +pub type MemFlags = ::MemFlags; +pub type Semaphore = ::Semaphore; +pub type Pipeline = ::Pipeline; +pub type DescriptorSet = ::DescriptorSet; +pub type QueryPool = ::QueryPool; + +type Fence = ::Fence; + +type VkImage = ::Image; +type VkBuffer = ::Buffer; + +#[derive(Clone)] +pub struct Session(Arc); + +struct SessionInner { + device: vulkan::VkDevice, + cmd_buf_pool: Mutex>, + /// Command buffers that are still pending (so resources can't be freed). + pending: Mutex>, +} + +pub struct CmdBuf { + cmd_buf: vulkan::CmdBuf, + fence: Fence, + resources: Vec>, + session: Weak, +} + +// Maybe "pending" is a better name? +pub struct SubmittedCmdBuf(Option, Weak); + +struct SubmittedCmdBufInner { + cmd_buf: vulkan::CmdBuf, + fence: Fence, + resources: Vec>, +} + +#[derive(Clone)] +pub struct Image(Arc); + +struct ImageInner { + image: VkImage, + session: Weak, +} + +#[derive(Clone)] +pub struct Buffer(Arc); + +struct BufferInner { + buffer: VkBuffer, + session: Weak, +} + +impl Session { + pub fn new(device: vulkan::VkDevice) -> Session { + Session(Arc::new(SessionInner { + device, + cmd_buf_pool: Default::default(), + pending: Default::default(), + })) + } + + pub fn cmd_buf(&self) -> Result { + self.poll_cleanup(); + let (cmd_buf, fence) = if let Some(cf) = self.0.cmd_buf_pool.lock().unwrap().pop() { + cf + } else { + let cmd_buf = self.0.device.create_cmd_buf()?; + let fence = unsafe { self.0.device.create_fence(false)? }; + (cmd_buf, fence) + }; + Ok(CmdBuf { + cmd_buf, + fence, + resources: Vec::new(), + session: Arc::downgrade(&self.0), + }) + } + + fn poll_cleanup(&self) { + let mut pending = self.0.pending.lock().unwrap(); + unsafe { + let mut i = 0; + while i < pending.len() { + if let Ok(true) = self.0.device.get_fence_status(pending[i].fence) { + let item = pending.swap_remove(i); + // TODO: wait is superfluous, can just reset + let _ = self.0.device.wait_and_reset(&[item.fence]); + self.0 + .cmd_buf_pool + .lock() + .unwrap() + .push((item.cmd_buf, item.fence)); + std::mem::drop(item.resources); + } else { + i += 1; + } + } + } + } + + pub unsafe fn run_cmd_buf( + &self, + cmd_buf: CmdBuf, + wait_semaphores: &[Semaphore], + signal_semaphores: &[Semaphore], + ) -> Result { + self.0.device.run_cmd_buf( + &cmd_buf.cmd_buf, + wait_semaphores, + signal_semaphores, + Some(&cmd_buf.fence), + )?; + Ok(SubmittedCmdBuf( + Some(SubmittedCmdBufInner { + cmd_buf: cmd_buf.cmd_buf, + fence: cmd_buf.fence, + resources: cmd_buf.resources, + }), + cmd_buf.session, + )) + } + + pub fn create_buffer(&self, size: u64, mem_flags: MemFlags) -> Result { + let buffer = self.0.device.create_buffer(size, mem_flags)?; + Ok(Buffer(Arc::new(BufferInner { + buffer, + session: Arc::downgrade(&self.0), + }))) + } + + pub unsafe fn create_image2d( + &self, + width: u32, + height: u32, + mem_flags: MemFlags, + ) -> Result { + let image = self.0.device.create_image2d(width, height, mem_flags)?; + Ok(Image(Arc::new(ImageInner { + image, + session: Arc::downgrade(&self.0), + }))) + } + + pub unsafe fn create_semaphore(&self) -> Result { + self.0.device.create_semaphore() + } + + /// This creates a pipeline that runs over the buffer. + /// + /// The descriptor set layout is just some number of storage buffers and storage images (this might change). + pub unsafe fn create_simple_compute_pipeline( + &self, + code: &[u8], + n_buffers: u32, + n_images: u32, + ) -> Result { + self.0 + .device + .create_simple_compute_pipeline(code, n_buffers, n_images) + } + + /// Create a descriptor set for a simple pipeline that just references buffers and images. + /// + /// Note: when we do portability, the signature will change to not reference the Vulkan types + /// directly. + pub unsafe fn create_descriptor_set( + &self, + pipeline: &Pipeline, + bufs: &[&vulkan::Buffer], + images: &[&vulkan::Image], + ) -> Result { + self.0.device.create_descriptor_set(pipeline, bufs, images) + } + + /// Create a query pool for timestamp queries. + pub fn create_query_pool(&self, n_queries: u32) -> Result { + self.0.device.create_query_pool(n_queries) + } + + pub unsafe fn fetch_query_pool(&self, pool: &QueryPool) -> Result, Error> { + self.0.device.fetch_query_pool(pool) + } +} + +impl CmdBuf { + pub fn add_resource(&mut self, resource: &T) { + self.resources.push(Box::new(resource.clone())); + } +} + +impl SubmittedCmdBuf { + pub fn wait(mut self) -> Result<(), Error> { + let item = self.0.take().unwrap(); + if let Some(session) = Weak::upgrade(&self.1) { + unsafe { + session.device.wait_and_reset(&[item.fence])?; + } + session + .cmd_buf_pool + .lock() + .unwrap() + .push((item.cmd_buf, item.fence)); + std::mem::drop(item.resources); + } + // else session dropped error? + Ok(()) + } +} + +impl Drop for SubmittedCmdBuf { + fn drop(&mut self) { + if let Some(inner) = self.0.take() { + if let Some(session) = Weak::upgrade(&self.1) { + session.pending.lock().unwrap().push(inner); + } + } + } +} + +impl Drop for BufferInner { + fn drop(&mut self) { + if let Some(session) = Weak::upgrade(&self.session) { + unsafe { + let _ = session.device.destroy_buffer(&self.buffer); + } + } + } +} + +impl Drop for ImageInner { + fn drop(&mut self) { + if let Some(session) = Weak::upgrade(&self.session) { + unsafe { + let _ = session.device.destroy_image(&self.image); + } + } + } +} + +/// For now, we deref, but for runtime backend switching we'll need to wrap +/// all methods. +impl std::ops::Deref for CmdBuf { + type Target = vulkan::CmdBuf; + fn deref(&self) -> &Self::Target { + &self.cmd_buf + } +} + +impl std::ops::DerefMut for CmdBuf { + fn deref_mut(&mut self) -> &mut Self::Target { + &mut self.cmd_buf + } +} + +impl Image { + pub fn vk_image(&self) -> &vulkan::Image { + &self.0.image + } +} + +impl Buffer { + pub fn vk_buffer(&self) -> &vulkan::Buffer { + &self.0.buffer + } + + pub unsafe fn write(&mut self, contents: &[T]) -> Result<(), Error> { + if let Some(session) = Weak::upgrade(&self.0.session) { + session.device.write_buffer(&self.0.buffer, contents)?; + } + // else session lost error? + Ok(()) + } + pub unsafe fn read(&self, result: &mut Vec) -> Result<(), Error> { + if let Some(session) = Weak::upgrade(&self.0.session) { + session.device.read_buffer(&self.0.buffer, result)?; + } + // else session lost error? + Ok(()) + } +} diff --git a/piet-gpu-hal/src/lib.rs b/piet-gpu-hal/src/lib.rs index 2924841..698ae43 100644 --- a/piet-gpu-hal/src/lib.rs +++ b/piet-gpu-hal/src/lib.rs @@ -2,6 +2,8 @@ /// /// This abstraction is inspired by gfx-hal, but is specialized to the needs of piet-gpu. /// In time, it may go away and be replaced by either gfx-hal or wgpu. +pub mod hub; + pub mod vulkan; /// This isn't great but is expedient. @@ -29,6 +31,14 @@ pub trait Device: Sized { fn create_buffer(&self, size: u64, mem_flags: Self::MemFlags) -> Result; + /// Destroy a buffer. + /// + /// The same safety requirements hold as in Vulkan: the buffer cannot be used + /// after this call, and all commands referencing this buffer must have completed. + /// + /// Maybe doesn't need result return? + unsafe fn destroy_buffer(&self, buffer: &Self::Buffer) -> Result<(), Error>; + unsafe fn create_image2d( &self, width: u32, @@ -36,6 +46,16 @@ pub trait Device: Sized { mem_flags: Self::MemFlags, ) -> Result; + /// Destroy an image. + /// + /// The same safety requirements hold as in Vulkan: the image cannot be used + /// after this call, and all commands referencing this image must have completed. + /// + /// Use this only with images we created, not for swapchain images. + /// + /// Maybe doesn't need result return? + unsafe fn destroy_image(&self, image: &Self::Image) -> Result<(), Error>; + unsafe fn create_simple_compute_pipeline( &self, code: &[u8], @@ -61,7 +81,7 @@ pub trait Device: Sized { /// /// # Safety /// All submitted commands that refer to this query pool must have completed. - unsafe fn reap_query_pool(&self, pool: &Self::QueryPool) -> Result, Error>; + unsafe fn fetch_query_pool(&self, pool: &Self::QueryPool) -> Result, Error>; unsafe fn run_cmd_buf( &self, @@ -86,6 +106,7 @@ pub trait Device: Sized { unsafe fn create_semaphore(&self) -> Result; unsafe fn create_fence(&self, signaled: bool) -> Result; unsafe fn wait_and_reset(&self, fences: &[Self::Fence]) -> Result<(), Error>; + unsafe fn get_fence_status(&self, fence: Self::Fence) -> Result; } pub trait CmdBuf { diff --git a/piet-gpu-hal/src/vulkan.rs b/piet-gpu-hal/src/vulkan.rs index dee01f2..1905daa 100644 --- a/piet-gpu-hal/src/vulkan.rs +++ b/piet-gpu-hal/src/vulkan.rs @@ -62,8 +62,7 @@ pub struct Buffer { pub struct Image { image: vk::Image, - // Not used now but probably will be for destruction. - _image_memory: vk::DeviceMemory, + image_memory: vk::DeviceMemory, image_view: vk::ImageView, extent: vk::Extent3D, } @@ -413,6 +412,13 @@ impl crate::Device for VkDevice { } } + unsafe fn destroy_buffer(&self, buffer: &Self::Buffer) -> Result<(), Error> { + let device = &self.device.device; + device.destroy_buffer(buffer.buffer, None); + device.free_memory(buffer.buffer_memory, None); + Ok(()) + } + unsafe fn create_image2d( &self, width: u32, @@ -476,12 +482,20 @@ impl crate::Device for VkDevice { )?; Ok(Image { image, - _image_memory: image_memory, + image_memory, image_view, extent, }) } + unsafe fn destroy_image(&self, image: &Self::Image) -> Result<(), Error> { + let device = &self.device.device; + device.destroy_image(image.image, None); + device.destroy_image_view(image.image_view, None); + device.free_memory(image.image_memory, None); + Ok(()) + } + unsafe fn create_fence(&self, signaled: bool) -> Result { let device = &self.device.device; let mut flags = vk::FenceCreateFlags::empty(); @@ -503,6 +517,11 @@ impl crate::Device for VkDevice { Ok(()) } + unsafe fn get_fence_status(&self, fence: Self::Fence) -> Result { + let device = &self.device.device; + Ok(device.get_fence_status(fence)?) + } + /// This creates a pipeline that runs over the buffer. /// /// The descriptor set layout is just some number of storage buffers and storage images (this might change). @@ -686,7 +705,7 @@ impl crate::Device for VkDevice { } } - unsafe fn reap_query_pool(&self, pool: &Self::QueryPool) -> Result, Error> { + unsafe fn fetch_query_pool(&self, pool: &Self::QueryPool) -> Result, Error> { let device = &self.device.device; let mut buf = vec![0u64; pool.n_queries as usize]; device.get_query_pool_results( @@ -1025,7 +1044,7 @@ impl VkSwapchain { pub unsafe fn image(&self, idx: usize) -> Image { Image { image: self.images[idx], - _image_memory: vk::DeviceMemory::null(), + image_memory: vk::DeviceMemory::null(), image_view: vk::ImageView::null(), extent: vk::Extent3D { width: self.extent.width, diff --git a/piet-gpu/bin/cli.rs b/piet-gpu/bin/cli.rs index 5052dc6..7876f9e 100644 --- a/piet-gpu/bin/cli.rs +++ b/piet-gpu/bin/cli.rs @@ -4,8 +4,9 @@ use std::path::Path; use clap::{App, Arg}; +use piet_gpu_hal::hub; use piet_gpu_hal::vulkan::VkInstance; -use piet_gpu_hal::{CmdBuf, Device, Error, MemFlags}; +use piet_gpu_hal::{CmdBuf, Error, MemFlags}; use piet_gpu::{render_scene, render_svg, PietGpuRenderContext, Renderer, HEIGHT, WIDTH}; @@ -195,10 +196,10 @@ fn main() -> Result<(), Error> { let (instance, _) = VkInstance::new(None)?; unsafe { let device = instance.device(None)?; + let session = hub::Session::new(device); - let fence = device.create_fence(false)?; - let mut cmd_buf = device.create_cmd_buf()?; - let query_pool = device.create_query_pool(8)?; + let mut cmd_buf = session.cmd_buf()?; + let query_pool = session.create_query_pool(8)?; let mut ctx = PietGpuRenderContext::new(); if let Some(input) = matches.value_of("INPUT") { @@ -218,20 +219,20 @@ fn main() -> Result<(), Error> { let scene = ctx.get_scene_buf(); //dump_scene(&scene); - let renderer = Renderer::new(&device, scene, n_paths, n_pathseg)?; + let renderer = Renderer::new(&session, scene, n_paths, n_pathseg)?; let image_buf = - device.create_buffer((WIDTH * HEIGHT * 4) as u64, MemFlags::host_coherent())?; + session.create_buffer((WIDTH * HEIGHT * 4) as u64, MemFlags::host_coherent())?; cmd_buf.begin(); renderer.record(&mut cmd_buf, &query_pool); - cmd_buf.copy_image_to_buffer(&renderer.image_dev, &image_buf); + cmd_buf.copy_image_to_buffer(renderer.image_dev.vk_image(), image_buf.vk_buffer()); cmd_buf.host_barrier(); cmd_buf.finish(); let start = std::time::Instant::now(); - device.run_cmd_buf(&cmd_buf, &[], &[], Some(&fence))?; - device.wait_and_reset(&[fence])?; + let submitted = session.run_cmd_buf(cmd_buf, &[], &[])?; + submitted.wait()?; println!("elapsed = {:?}", start.elapsed()); - let ts = device.reap_query_pool(&query_pool).unwrap(); + let ts = session.fetch_query_pool(&query_pool).unwrap(); println!("Element kernel time: {:.3}ms", ts[0] * 1e3); println!( "Tile allocation kernel time: {:.3}ms", @@ -253,7 +254,7 @@ fn main() -> Result<(), Error> { let mut img_data: Vec = Default::default(); // Note: because png can use a `&[u8]` slice, we could avoid an extra copy // (probably passing a slice into a closure). But for now: keep it simple. - device.read_buffer(&image_buf, &mut img_data).unwrap(); + image_buf.read(&mut img_data).unwrap(); // Write image as PNG file. let path = Path::new("image.png"); diff --git a/piet-gpu/bin/winit.rs b/piet-gpu/bin/winit.rs index 0b471b3..644abbe 100644 --- a/piet-gpu/bin/winit.rs +++ b/piet-gpu/bin/winit.rs @@ -1,5 +1,6 @@ +use piet_gpu_hal::hub; use piet_gpu_hal::vulkan::VkInstance; -use piet_gpu_hal::{CmdBuf, Device, Error, ImageLayout}; +use piet_gpu_hal::{CmdBuf, Error, ImageLayout}; use piet_gpu::{render_scene, PietGpuRenderContext, Renderer, HEIGHT, WIDTH}; @@ -25,19 +26,14 @@ fn main() -> Result<(), Error> { unsafe { let device = instance.device(surface.as_ref())?; let mut swapchain = instance.swapchain(&device, surface.as_ref().unwrap())?; + let session = hub::Session::new(device); let mut current_frame = 0; let present_semaphores = (0..NUM_FRAMES) - .map(|_| device.create_semaphore()) - .collect::, Error>>()?; - let frame_fences = (0..NUM_FRAMES) - .map(|_| device.create_fence(false)) - .collect::, Error>>()?; - let mut cmd_buffers = (0..NUM_FRAMES) - .map(|_| device.create_cmd_buf()) + .map(|_| session.create_semaphore()) .collect::, Error>>()?; let query_pools = (0..NUM_FRAMES) - .map(|_| device.create_query_pool(8)) + .map(|_| session.create_query_pool(8)) .collect::, Error>>()?; let mut ctx = PietGpuRenderContext::new(); @@ -46,7 +42,9 @@ fn main() -> Result<(), Error> { let n_pathseg = ctx.pathseg_count(); let scene = ctx.get_scene_buf(); - let renderer = Renderer::new(&device, scene, n_paths, n_pathseg)?; + let renderer = Renderer::new(&session, scene, n_paths, n_pathseg)?; + + let mut submitted: Option = None; event_loop.run(move |event, _, control_flow| { *control_flow = ControlFlow::Poll; // `ControlFlow::Wait` if only re-render on event @@ -67,10 +65,10 @@ fn main() -> Result<(), Error> { let frame_idx = current_frame % NUM_FRAMES; let query_pool = &query_pools[frame_idx]; - if current_frame >= NUM_FRAMES { - device.wait_and_reset(&[frame_fences[frame_idx]]).unwrap(); + if let Some(submitted) = submitted.take() { + submitted.wait().unwrap(); - let ts = device.reap_query_pool(query_pool).unwrap(); + let ts = session.fetch_query_pool(query_pool).unwrap(); window.set_title(&format!( "{:.3}ms :: e:{:.3}ms|alloc:{:.3}ms|cp:{:.3}ms|bd:{:.3}ms|bin:{:.3}ms|cr:{:.3}ms|r:{:.3}ms", ts[6] * 1e3, @@ -86,9 +84,9 @@ fn main() -> Result<(), Error> { let (image_idx, acquisition_semaphore) = swapchain.next().unwrap(); let swap_image = swapchain.image(image_idx); - let cmd_buf = &mut cmd_buffers[frame_idx]; + let mut cmd_buf = session.cmd_buf().unwrap(); cmd_buf.begin(); - renderer.record(cmd_buf, &query_pool); + renderer.record(&mut cmd_buf, &query_pool); // Image -> Swapchain cmd_buf.image_barrier( @@ -96,18 +94,17 @@ fn main() -> Result<(), Error> { ImageLayout::Undefined, ImageLayout::BlitDst, ); - cmd_buf.blit_image(&renderer.image_dev, &swap_image); + cmd_buf.blit_image(renderer.image_dev.vk_image(), &swap_image); cmd_buf.image_barrier(&swap_image, ImageLayout::BlitDst, ImageLayout::Present); cmd_buf.finish(); - device + submitted = Some(session .run_cmd_buf( - &cmd_buf, + cmd_buf, &[acquisition_semaphore], &[present_semaphores[frame_idx]], - Some(&frame_fences[frame_idx]), ) - .unwrap(); + .unwrap()); swapchain .present(image_idx, &[present_semaphores[frame_idx]]) diff --git a/piet-gpu/src/lib.rs b/piet-gpu/src/lib.rs index 29bdb09..c02bf56 100644 --- a/piet-gpu/src/lib.rs +++ b/piet-gpu/src/lib.rs @@ -10,7 +10,8 @@ use piet::{Color, RenderContext}; use piet_gpu_types::encoder::Encode; -use piet_gpu_hal::{CmdBuf, Device, Error, ImageLayout, MemFlags}; +use piet_gpu_hal::hub; +use piet_gpu_hal::{CmdBuf, Error, ImageLayout, MemFlags}; use pico_svg::PicoSvg; @@ -113,57 +114,57 @@ pub fn dump_k1_data(k1_buf: &[u32]) { } } -pub struct Renderer { - pub image_dev: D::Image, // resulting image +pub struct Renderer { + pub image_dev: hub::Image, // resulting image - scene_buf: D::Buffer, - scene_dev: D::Buffer, + scene_buf: hub::Buffer, + scene_dev: hub::Buffer, - pub state_buf: D::Buffer, - pub anno_buf: D::Buffer, - pub pathseg_buf: D::Buffer, - pub tile_buf: D::Buffer, - pub bin_buf: D::Buffer, - pub ptcl_buf: D::Buffer, + pub state_buf: hub::Buffer, + pub anno_buf: hub::Buffer, + pub pathseg_buf: hub::Buffer, + pub tile_buf: hub::Buffer, + pub bin_buf: hub::Buffer, + pub ptcl_buf: hub::Buffer, - el_pipeline: D::Pipeline, - el_ds: D::DescriptorSet, + el_pipeline: hub::Pipeline, + el_ds: hub::DescriptorSet, - tile_pipeline: D::Pipeline, - tile_ds: D::DescriptorSet, + tile_pipeline: hub::Pipeline, + tile_ds: hub::DescriptorSet, - path_pipeline: D::Pipeline, - path_ds: D::DescriptorSet, + path_pipeline: hub::Pipeline, + path_ds: hub::DescriptorSet, - backdrop_pipeline: D::Pipeline, - backdrop_ds: D::DescriptorSet, + backdrop_pipeline: hub::Pipeline, + backdrop_ds: hub::DescriptorSet, - tile_alloc_buf_host: D::Buffer, - tile_alloc_buf_dev: D::Buffer, + tile_alloc_buf_host: hub::Buffer, + tile_alloc_buf_dev: hub::Buffer, - bin_pipeline: D::Pipeline, - bin_ds: D::DescriptorSet, + bin_pipeline: hub::Pipeline, + bin_ds: hub::DescriptorSet, - bin_alloc_buf_host: D::Buffer, - bin_alloc_buf_dev: D::Buffer, + bin_alloc_buf_host: hub::Buffer, + bin_alloc_buf_dev: hub::Buffer, - coarse_pipeline: D::Pipeline, - coarse_ds: D::DescriptorSet, + coarse_pipeline: hub::Pipeline, + coarse_ds: hub::DescriptorSet, - coarse_alloc_buf_host: D::Buffer, - coarse_alloc_buf_dev: D::Buffer, + coarse_alloc_buf_host: hub::Buffer, + coarse_alloc_buf_dev: hub::Buffer, - k4_pipeline: D::Pipeline, - k4_ds: D::DescriptorSet, + k4_pipeline: hub::Pipeline, + k4_ds: hub::DescriptorSet, n_elements: usize, n_paths: usize, n_pathseg: usize, } -impl Renderer { +impl Renderer { pub unsafe fn new( - device: &D, + session: &hub::Session, scene: &[u8], n_paths: usize, n_pathseg: usize, @@ -177,107 +178,108 @@ impl Renderer { n_elements, n_paths, n_pathseg ); - let scene_buf = device + let mut scene_buf = session .create_buffer(std::mem::size_of_val(&scene[..]) as u64, host) .unwrap(); - let scene_dev = device + let scene_dev = session .create_buffer(std::mem::size_of_val(&scene[..]) as u64, dev) .unwrap(); - device.write_buffer(&scene_buf, &scene)?; + scene_buf.write(&scene)?; - let state_buf = device.create_buffer(1 * 1024 * 1024, dev)?; - let anno_buf = device.create_buffer(64 * 1024 * 1024, dev)?; - let pathseg_buf = device.create_buffer(64 * 1024 * 1024, dev)?; - let tile_buf = device.create_buffer(64 * 1024 * 1024, dev)?; - let bin_buf = device.create_buffer(64 * 1024 * 1024, dev)?; - let ptcl_buf = device.create_buffer(48 * 1024 * 1024, dev)?; - let image_dev = device.create_image2d(WIDTH as u32, HEIGHT as u32, dev)?; + let state_buf = session.create_buffer(1 * 1024 * 1024, dev)?; + let anno_buf = session.create_buffer(64 * 1024 * 1024, dev)?; + let pathseg_buf = session.create_buffer(64 * 1024 * 1024, dev)?; + let tile_buf = session.create_buffer(64 * 1024 * 1024, dev)?; + let bin_buf = session.create_buffer(64 * 1024 * 1024, dev)?; + let ptcl_buf = session.create_buffer(48 * 1024 * 1024, dev)?; + let image_dev = session.create_image2d(WIDTH as u32, HEIGHT as u32, dev)?; let el_code = include_bytes!("../shader/elements.spv"); - let el_pipeline = device.create_simple_compute_pipeline(el_code, 4, 0)?; - let el_ds = device.create_descriptor_set( + let el_pipeline = session.create_simple_compute_pipeline(el_code, 4, 0)?; + let el_ds = session.create_descriptor_set( &el_pipeline, - &[&scene_dev, &state_buf, &anno_buf, &pathseg_buf], + &[ + scene_dev.vk_buffer(), + state_buf.vk_buffer(), + anno_buf.vk_buffer(), + pathseg_buf.vk_buffer(), + ], &[], )?; - let tile_alloc_buf_host = device.create_buffer(12, host)?; - let tile_alloc_buf_dev = device.create_buffer(12, dev)?; + let mut tile_alloc_buf_host = session.create_buffer(12, host)?; + let tile_alloc_buf_dev = session.create_buffer(12, dev)?; // TODO: constants const PATH_SIZE: usize = 12; let tile_alloc_start = ((n_paths + 31) & !31) * PATH_SIZE; - device.write_buffer( - &tile_alloc_buf_host, + tile_alloc_buf_host.write( &[n_paths as u32, n_pathseg as u32, tile_alloc_start as u32], )?; let tile_alloc_code = include_bytes!("../shader/tile_alloc.spv"); - let tile_pipeline = device.create_simple_compute_pipeline(tile_alloc_code, 3, 0)?; - let tile_ds = device.create_descriptor_set( + let tile_pipeline = session.create_simple_compute_pipeline(tile_alloc_code, 3, 0)?; + let tile_ds = session.create_descriptor_set( &tile_pipeline, - &[&anno_buf, &tile_alloc_buf_dev, &tile_buf], + &[anno_buf.vk_buffer(), tile_alloc_buf_dev.vk_buffer(), tile_buf.vk_buffer()], &[], )?; let path_alloc_code = include_bytes!("../shader/path_coarse.spv"); - let path_pipeline = device.create_simple_compute_pipeline(path_alloc_code, 3, 0)?; - let path_ds = device.create_descriptor_set( + let path_pipeline = session.create_simple_compute_pipeline(path_alloc_code, 3, 0)?; + let path_ds = session.create_descriptor_set( &path_pipeline, - &[&pathseg_buf, &tile_alloc_buf_dev, &tile_buf], + &[pathseg_buf.vk_buffer(), tile_alloc_buf_dev.vk_buffer(), tile_buf.vk_buffer()], &[], )?; let backdrop_alloc_code = include_bytes!("../shader/backdrop.spv"); - let backdrop_pipeline = device.create_simple_compute_pipeline(backdrop_alloc_code, 3, 0)?; - let backdrop_ds = device.create_descriptor_set( + let backdrop_pipeline = + session.create_simple_compute_pipeline(backdrop_alloc_code, 3, 0)?; + let backdrop_ds = session.create_descriptor_set( &backdrop_pipeline, - &[&anno_buf, &tile_alloc_buf_dev, &tile_buf], + &[anno_buf.vk_buffer(), tile_alloc_buf_dev.vk_buffer(), tile_buf.vk_buffer()], &[], )?; - let bin_alloc_buf_host = device.create_buffer(8, host)?; - let bin_alloc_buf_dev = device.create_buffer(8, dev)?; + let mut bin_alloc_buf_host = session.create_buffer(8, host)?; + let bin_alloc_buf_dev = session.create_buffer(8, dev)?; // TODO: constants let bin_alloc_start = ((n_paths + 255) & !255) * 8; - device.write_buffer( - &bin_alloc_buf_host, - &[n_paths as u32, bin_alloc_start as u32], - )?; + bin_alloc_buf_host.write(&[n_paths as u32, bin_alloc_start as u32])?; let bin_code = include_bytes!("../shader/binning.spv"); - let bin_pipeline = device.create_simple_compute_pipeline(bin_code, 3, 0)?; - let bin_ds = device.create_descriptor_set( + let bin_pipeline = session.create_simple_compute_pipeline(bin_code, 3, 0)?; + let bin_ds = session.create_descriptor_set( &bin_pipeline, - &[&anno_buf, &bin_alloc_buf_dev, &bin_buf], + &[anno_buf.vk_buffer(), bin_alloc_buf_dev.vk_buffer(), bin_buf.vk_buffer()], &[], )?; - let coarse_alloc_buf_host = device.create_buffer(8, host)?; - let coarse_alloc_buf_dev = device.create_buffer(8, dev)?; + let mut coarse_alloc_buf_host = session.create_buffer(8, host)?; + let coarse_alloc_buf_dev = session.create_buffer(8, dev)?; let coarse_alloc_start = WIDTH_IN_TILES * HEIGHT_IN_TILES * PTCL_INITIAL_ALLOC; - device.write_buffer( - &coarse_alloc_buf_host, + coarse_alloc_buf_host.write( &[n_paths as u32, coarse_alloc_start as u32], )?; let coarse_code = include_bytes!("../shader/coarse.spv"); - let coarse_pipeline = device.create_simple_compute_pipeline(coarse_code, 5, 0)?; - let coarse_ds = device.create_descriptor_set( + let coarse_pipeline = session.create_simple_compute_pipeline(coarse_code, 5, 0)?; + let coarse_ds = session.create_descriptor_set( &coarse_pipeline, &[ - &anno_buf, - &bin_buf, - &tile_buf, - &coarse_alloc_buf_dev, - &ptcl_buf, + anno_buf.vk_buffer(), + bin_buf.vk_buffer(), + tile_buf.vk_buffer(), + coarse_alloc_buf_dev.vk_buffer(), + ptcl_buf.vk_buffer(), ], &[], )?; let k4_code = include_bytes!("../shader/kernel4.spv"); - let k4_pipeline = device.create_simple_compute_pipeline(k4_code, 2, 1)?; + let k4_pipeline = session.create_simple_compute_pipeline(k4_code, 2, 1)?; let k4_ds = - device.create_descriptor_set(&k4_pipeline, &[&ptcl_buf, &tile_buf], &[&image_dev])?; + session.create_descriptor_set(&k4_pipeline, &[ptcl_buf.vk_buffer(), tile_buf.vk_buffer()], &[image_dev.vk_image()])?; Ok(Renderer { scene_buf, @@ -315,15 +317,24 @@ impl Renderer { }) } - pub unsafe fn record(&self, cmd_buf: &mut impl CmdBuf, query_pool: &D::QueryPool) { - cmd_buf.copy_buffer(&self.scene_buf, &self.scene_dev); - cmd_buf.copy_buffer(&self.tile_alloc_buf_host, &self.tile_alloc_buf_dev); - cmd_buf.copy_buffer(&self.bin_alloc_buf_host, &self.bin_alloc_buf_dev); - cmd_buf.copy_buffer(&self.coarse_alloc_buf_host, &self.coarse_alloc_buf_dev); - cmd_buf.clear_buffer(&self.state_buf); + pub unsafe fn record(&self, cmd_buf: &mut hub::CmdBuf, query_pool: &hub::QueryPool) { + cmd_buf.copy_buffer(self.scene_buf.vk_buffer(), self.scene_dev.vk_buffer()); + cmd_buf.copy_buffer( + self.tile_alloc_buf_host.vk_buffer(), + self.tile_alloc_buf_dev.vk_buffer(), + ); + cmd_buf.copy_buffer( + self.bin_alloc_buf_host.vk_buffer(), + self.bin_alloc_buf_dev.vk_buffer(), + ); + cmd_buf.copy_buffer( + self.coarse_alloc_buf_host.vk_buffer(), + self.coarse_alloc_buf_dev.vk_buffer(), + ); + cmd_buf.clear_buffer(self.state_buf.vk_buffer()); cmd_buf.memory_barrier(); cmd_buf.image_barrier( - &self.image_dev, + self.image_dev.vk_image(), ImageLayout::Undefined, ImageLayout::General, ); @@ -381,6 +392,10 @@ impl Renderer { ); cmd_buf.write_timestamp(&query_pool, 7); cmd_buf.memory_barrier(); - cmd_buf.image_barrier(&self.image_dev, ImageLayout::General, ImageLayout::BlitSrc); + cmd_buf.image_barrier( + self.image_dev.vk_image(), + ImageLayout::General, + ImageLayout::BlitSrc, + ); } }