[hal] Add timer queries

Note: also make dispatch size programmable.
This commit is contained in:
Raph Levien 2020-04-12 22:28:03 -07:00
parent 265d990cbe
commit 487d948217
3 changed files with 85 additions and 4 deletions

View file

@ -14,15 +14,20 @@ fn main() {
let code = include_bytes!("./shader/collatz.spv"); let code = include_bytes!("./shader/collatz.spv");
let pipeline = device.create_simple_compute_pipeline(code, 1).unwrap(); let pipeline = device.create_simple_compute_pipeline(code, 1).unwrap();
let descriptor_set = device.create_descriptor_set(&pipeline, &[&buffer]).unwrap(); let descriptor_set = device.create_descriptor_set(&pipeline, &[&buffer]).unwrap();
let query_pool = device.create_query_pool(2).unwrap();
let mut cmd_buf = device.create_cmd_buf().unwrap(); let mut cmd_buf = device.create_cmd_buf().unwrap();
cmd_buf.begin(); cmd_buf.begin();
cmd_buf.dispatch(&pipeline, &descriptor_set); cmd_buf.write_timestamp(&query_pool, 0);
cmd_buf.dispatch(&pipeline, &descriptor_set, (256, 1, 1));
cmd_buf.write_timestamp(&query_pool, 1);
cmd_buf.finish(); cmd_buf.finish();
device.run_cmd_buf(&cmd_buf).unwrap(); device.run_cmd_buf(&cmd_buf).unwrap();
let timestamps = device.reap_query_pool(query_pool);
let mut dst: Vec<u32> = Default::default(); let mut dst: Vec<u32> = Default::default();
device.read_buffer(&buffer, &mut dst).unwrap(); device.read_buffer(&buffer, &mut dst).unwrap();
for (i, val) in dst.iter().enumerate().take(16) { for (i, val) in dst.iter().enumerate().take(16) {
println!("{}: {}", i, val); println!("{}: {}", i, val);
} }
println!("{:?}", timestamps);
} }
} }

View file

@ -12,6 +12,7 @@ pub trait Device: Sized {
type MemFlags: MemFlags; type MemFlags: MemFlags;
type Pipeline; type Pipeline;
type DescriptorSet; type DescriptorSet;
type QueryPool;
type CmdBuf: CmdBuf<Self>; type CmdBuf: CmdBuf<Self>;
fn create_buffer(&self, size: u64, mem_flags: Self::MemFlags) -> Result<Self::Buffer, Error>; fn create_buffer(&self, size: u64, mem_flags: Self::MemFlags) -> Result<Self::Buffer, Error>;
@ -30,6 +31,17 @@ pub trait Device: Sized {
fn create_cmd_buf(&self) -> Result<Self::CmdBuf, Error>; fn create_cmd_buf(&self) -> Result<Self::CmdBuf, Error>;
fn create_query_pool(&self, n_queries: u32) -> Result<Self::QueryPool, Error>;
/// Get results from query pool, destroying it in the process.
///
/// The returned vector is one less than the number of queries; the first is used as
/// a baseline.
///
/// # Safety
/// All submitted commands that refer to this query pool must have completed.
unsafe fn reap_query_pool(&self, pool: Self::QueryPool) -> Result<Vec<f64>, Error>;
unsafe fn run_cmd_buf(&self, cmd_buf: &Self::CmdBuf) -> Result<(), Error>; unsafe fn run_cmd_buf(&self, cmd_buf: &Self::CmdBuf) -> Result<(), Error>;
unsafe fn read_buffer<T: Sized>( unsafe fn read_buffer<T: Sized>(
@ -50,9 +62,16 @@ pub trait CmdBuf<D: Device> {
unsafe fn finish(&mut self); unsafe fn finish(&mut self);
unsafe fn dispatch(&mut self, pipeline: &D::Pipeline, descriptor_set: &D::DescriptorSet); unsafe fn dispatch(
&mut self,
pipeline: &D::Pipeline,
descriptor_set: &D::DescriptorSet,
size: (u32, u32, u32),
);
unsafe fn memory_barrier(&mut self); unsafe fn memory_barrier(&mut self);
unsafe fn write_timestamp(&mut self, pool: &D::QueryPool, query: u32);
} }
pub trait MemFlags: Sized { pub trait MemFlags: Sized {

View file

@ -21,6 +21,7 @@ pub struct VkDevice {
device_mem_props: vk::PhysicalDeviceMemoryProperties, device_mem_props: vk::PhysicalDeviceMemoryProperties,
queue: vk::Queue, queue: vk::Queue,
qfi: u32, qfi: u32,
timestamp_period: f32,
} }
struct RawDevice { struct RawDevice {
@ -52,6 +53,11 @@ pub struct CmdBuf {
device: Arc<RawDevice>, device: Arc<RawDevice>,
} }
pub struct QueryPool {
pool: vk::QueryPool,
n_queries: u32,
}
pub struct MemFlags(vk::MemoryPropertyFlags); pub struct MemFlags(vk::MemoryPropertyFlags);
impl VkInstance { impl VkInstance {
@ -108,11 +114,15 @@ impl VkInstance {
let device = Arc::new(RawDevice { device }); let device = Arc::new(RawDevice { device });
let props = self.instance.get_physical_device_properties(pdevice);
let timestamp_period = props.limits.timestamp_period;
Ok(VkDevice { Ok(VkDevice {
device, device,
device_mem_props, device_mem_props,
qfi, qfi,
queue, queue,
timestamp_period,
}) })
} }
} }
@ -122,6 +132,7 @@ impl crate::Device for VkDevice {
type CmdBuf = CmdBuf; type CmdBuf = CmdBuf;
type DescriptorSet = DescriptorSet; type DescriptorSet = DescriptorSet;
type Pipeline = Pipeline; type Pipeline = Pipeline;
type QueryPool = QueryPool;
type MemFlags = MemFlags; type MemFlags = MemFlags;
fn create_buffer(&self, size: u64, mem_flags: MemFlags) -> Result<Buffer, Error> { fn create_buffer(&self, size: u64, mem_flags: MemFlags) -> Result<Buffer, Error> {
@ -283,6 +294,37 @@ impl crate::Device for VkDevice {
} }
} }
/// Create a query pool for timestamp queries.
fn create_query_pool(&self, n_queries: u32) -> Result<QueryPool, Error> {
unsafe {
let device = &self.device.device;
let pool = device.create_query_pool(
&vk::QueryPoolCreateInfo::builder()
.query_type(vk::QueryType::TIMESTAMP)
.query_count(n_queries),
None,
)?;
Ok(QueryPool { pool, n_queries })
}
}
unsafe fn reap_query_pool(&self, pool: Self::QueryPool) -> Result<Vec<f64>, Error> {
let device = &self.device.device;
let mut buf = vec![0u64; pool.n_queries as usize];
device.get_query_pool_results(
pool.pool,
0,
pool.n_queries,
&mut buf,
vk::QueryResultFlags::TYPE_64,
)?;
device.destroy_query_pool(pool.pool, None);
let ts0 = buf[0];
let tsp = self.timestamp_period as f64 * 1e-9;
let result = buf[1..].iter().map(|ts| ts.wrapping_sub(ts0) as f64 * tsp).collect();
Ok(result)
}
/// Run the command buffer. /// Run the command buffer.
/// ///
/// This version simply blocks until it's complete. /// This version simply blocks until it's complete.
@ -358,7 +400,12 @@ impl crate::CmdBuf<VkDevice> for CmdBuf {
self.device.device.end_command_buffer(self.cmd_buf).unwrap(); self.device.device.end_command_buffer(self.cmd_buf).unwrap();
} }
unsafe fn dispatch(&mut self, pipeline: &Pipeline, descriptor_set: &DescriptorSet) { unsafe fn dispatch(
&mut self,
pipeline: &Pipeline,
descriptor_set: &DescriptorSet,
size: (u32, u32, u32),
) {
let device = &self.device.device; let device = &self.device.device;
device.cmd_bind_pipeline( device.cmd_bind_pipeline(
self.cmd_buf, self.cmd_buf,
@ -373,7 +420,7 @@ impl crate::CmdBuf<VkDevice> for CmdBuf {
&[descriptor_set.descriptor_set], &[descriptor_set.descriptor_set],
&[], &[],
); );
device.cmd_dispatch(self.cmd_buf, 256, 1, 1); device.cmd_dispatch(self.cmd_buf, size.0, size.1, size.2);
} }
/// Insert a pipeline barrier for all memory accesses. /// Insert a pipeline barrier for all memory accesses.
@ -392,6 +439,16 @@ impl crate::CmdBuf<VkDevice> for CmdBuf {
&[], &[],
); );
} }
unsafe fn write_timestamp(&mut self, pool: &QueryPool, query: u32) {
let device = &self.device.device;
device.cmd_write_timestamp(
self.cmd_buf,
vk::PipelineStageFlags::COMPUTE_SHADER,
pool.pool,
query,
);
}
} }
impl crate::MemFlags for MemFlags { impl crate::MemFlags for MemFlags {