2021-11-06 10:52:07 +11:00
|
|
|
// Copyright 2021 The piet-gpu authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
// Also licensed under MIT license, at your choice.
|
|
|
|
|
|
|
|
//! Test runner intended to make it easy to write tests.
|
|
|
|
|
2021-11-26 08:12:25 +11:00
|
|
|
use std::ops::RangeBounds;
|
|
|
|
|
2021-11-24 03:48:14 +11:00
|
|
|
use bytemuck::Pod;
|
2021-11-11 07:29:40 +11:00
|
|
|
use piet_gpu_hal::{
|
2022-04-21 03:21:49 +10:00
|
|
|
BackendType, BufReadGuard, BufWriteGuard, Buffer, BufferUsage, CmdBuf, ComputePass,
|
|
|
|
ComputePassDescriptor, Instance, InstanceFlags, QueryPool, Session,
|
2021-11-11 07:29:40 +11:00
|
|
|
};
|
2021-11-06 10:52:07 +11:00
|
|
|
|
|
|
|
pub struct Runner {
|
|
|
|
#[allow(unused)]
|
|
|
|
instance: Instance,
|
|
|
|
pub session: Session,
|
|
|
|
cmd_buf_pool: Vec<CmdBuf>,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A wrapper around command buffers.
|
|
|
|
pub struct Commands {
|
|
|
|
pub cmd_buf: CmdBuf,
|
|
|
|
query_pool: QueryPool,
|
|
|
|
}
|
|
|
|
|
2022-02-18 11:25:41 +11:00
|
|
|
/// Buffer for both uploading and downloading
|
|
|
|
pub struct BufStage {
|
2021-11-06 10:52:07 +11:00
|
|
|
pub stage_buf: Buffer,
|
|
|
|
pub dev_buf: Buffer,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Runner {
|
2021-11-10 15:28:06 +11:00
|
|
|
pub unsafe fn new(flags: InstanceFlags) -> Runner {
|
2022-07-15 04:46:46 +10:00
|
|
|
let instance = Instance::new(flags).unwrap();
|
|
|
|
let device = instance.device().unwrap();
|
2021-11-06 10:52:07 +11:00
|
|
|
let session = Session::new(device);
|
|
|
|
let cmd_buf_pool = Vec::new();
|
|
|
|
Runner {
|
|
|
|
instance,
|
|
|
|
session,
|
|
|
|
cmd_buf_pool,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn commands(&mut self) -> Commands {
|
|
|
|
let mut cmd_buf = self
|
|
|
|
.cmd_buf_pool
|
|
|
|
.pop()
|
|
|
|
.unwrap_or_else(|| self.session.cmd_buf().unwrap());
|
|
|
|
cmd_buf.begin();
|
|
|
|
// TODO: also pool these. But we might sort by size, as
|
|
|
|
// we might not always be doing two.
|
|
|
|
let query_pool = self.session.create_query_pool(2).unwrap();
|
|
|
|
cmd_buf.reset_query_pool(&query_pool);
|
|
|
|
Commands {
|
|
|
|
cmd_buf,
|
|
|
|
query_pool,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn submit(&mut self, commands: Commands) -> f64 {
|
|
|
|
let mut cmd_buf = commands.cmd_buf;
|
|
|
|
let query_pool = commands.query_pool;
|
2021-11-07 15:43:09 +11:00
|
|
|
cmd_buf.finish_timestamps(&query_pool);
|
2021-11-06 10:52:07 +11:00
|
|
|
cmd_buf.host_barrier();
|
|
|
|
cmd_buf.finish();
|
|
|
|
let submitted = self.session.run_cmd_buf(cmd_buf, &[], &[]).unwrap();
|
|
|
|
self.cmd_buf_pool.extend(submitted.wait().unwrap());
|
|
|
|
let timestamps = self.session.fetch_query_pool(&query_pool).unwrap();
|
2021-11-11 09:56:00 +11:00
|
|
|
timestamps.get(0).copied().unwrap_or_default()
|
2021-11-06 10:52:07 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
2022-02-18 11:25:41 +11:00
|
|
|
pub fn buf_up(&self, size: u64) -> BufStage {
|
2021-11-06 10:52:07 +11:00
|
|
|
let stage_buf = self
|
|
|
|
.session
|
|
|
|
.create_buffer(size, BufferUsage::MAP_WRITE | BufferUsage::COPY_SRC)
|
|
|
|
.unwrap();
|
|
|
|
let dev_buf = self
|
|
|
|
.session
|
|
|
|
.create_buffer(size, BufferUsage::COPY_DST | BufferUsage::STORAGE)
|
|
|
|
.unwrap();
|
2022-02-18 11:25:41 +11:00
|
|
|
BufStage { stage_buf, dev_buf }
|
2021-11-06 10:52:07 +11:00
|
|
|
}
|
|
|
|
|
2021-11-21 02:14:23 +11:00
|
|
|
/// Create a buffer for download (readback).
|
|
|
|
///
|
|
|
|
/// The `usage` parameter need not include COPY_SRC and STORAGE.
|
2022-02-18 11:25:41 +11:00
|
|
|
pub fn buf_down(&self, size: u64, usage: BufferUsage) -> BufStage {
|
2021-11-06 10:52:07 +11:00
|
|
|
let stage_buf = self
|
|
|
|
.session
|
|
|
|
.create_buffer(size, BufferUsage::MAP_READ | BufferUsage::COPY_DST)
|
|
|
|
.unwrap();
|
|
|
|
let dev_buf = self
|
|
|
|
.session
|
2021-11-21 02:14:23 +11:00
|
|
|
.create_buffer(size, usage | BufferUsage::COPY_SRC | BufferUsage::STORAGE)
|
2021-11-06 10:52:07 +11:00
|
|
|
.unwrap();
|
2022-02-18 11:25:41 +11:00
|
|
|
BufStage { stage_buf, dev_buf }
|
2021-11-06 10:52:07 +11:00
|
|
|
}
|
2021-11-10 15:28:06 +11:00
|
|
|
|
|
|
|
pub fn backend_type(&self) -> BackendType {
|
|
|
|
self.session.backend_type()
|
|
|
|
}
|
2021-11-06 10:52:07 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Commands {
|
2022-04-21 03:21:49 +10:00
|
|
|
/// Start a compute pass with timer queries.
|
|
|
|
pub unsafe fn compute_pass(&mut self, start_query: u32, end_query: u32) -> ComputePass {
|
|
|
|
self.cmd_buf
|
|
|
|
.begin_compute_pass(&ComputePassDescriptor::timer(
|
|
|
|
&self.query_pool,
|
|
|
|
start_query,
|
|
|
|
end_query,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
2022-02-18 11:25:41 +11:00
|
|
|
pub unsafe fn upload(&mut self, buf: &BufStage) {
|
2021-11-06 10:52:07 +11:00
|
|
|
self.cmd_buf.copy_buffer(&buf.stage_buf, &buf.dev_buf);
|
|
|
|
}
|
|
|
|
|
2022-02-18 11:25:41 +11:00
|
|
|
pub unsafe fn download(&mut self, buf: &BufStage) {
|
2021-11-06 10:52:07 +11:00
|
|
|
self.cmd_buf.copy_buffer(&buf.dev_buf, &buf.stage_buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-18 11:25:41 +11:00
|
|
|
impl BufStage {
|
2021-11-24 03:48:14 +11:00
|
|
|
pub unsafe fn read(&self, dst: &mut Vec<impl Pod>) {
|
2021-11-06 10:52:07 +11:00
|
|
|
self.stage_buf.read(dst).unwrap()
|
|
|
|
}
|
2021-11-26 08:12:25 +11:00
|
|
|
|
|
|
|
pub unsafe fn map_read<'a>(&'a self, range: impl RangeBounds<usize>) -> BufReadGuard<'a> {
|
|
|
|
self.stage_buf.map_read(range).unwrap()
|
|
|
|
}
|
2022-02-18 11:25:41 +11:00
|
|
|
|
|
|
|
pub unsafe fn map_write<'a>(&'a mut self, range: impl RangeBounds<usize>) -> BufWriteGuard {
|
|
|
|
self.stage_buf.map_write(range).unwrap()
|
|
|
|
}
|
2021-11-06 10:52:07 +11:00
|
|
|
}
|