2020-11-20 04:29:57 +11:00
|
|
|
//! DX12 implemenation of HAL trait.
|
|
|
|
|
|
|
|
mod error;
|
|
|
|
mod wrappers;
|
|
|
|
|
2021-04-30 15:06:13 +10:00
|
|
|
use std::{cell::Cell, convert::TryInto};
|
|
|
|
|
2020-11-20 04:29:57 +11:00
|
|
|
use winapi::shared::dxgi1_3;
|
|
|
|
use winapi::um::d3d12;
|
|
|
|
|
2021-04-30 15:06:13 +10:00
|
|
|
use crate::Error;
|
2020-11-20 04:29:57 +11:00
|
|
|
|
2021-04-30 15:06:13 +10:00
|
|
|
use self::wrappers::{
|
|
|
|
CommandAllocator, CommandQueue, Device, Factory4, GraphicsCommandList, Resource,
|
|
|
|
};
|
2020-11-20 04:29:57 +11:00
|
|
|
|
|
|
|
pub struct Dx12Instance {
|
|
|
|
factory: Factory4,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Dx12Device {
|
|
|
|
device: Device,
|
2021-04-30 15:06:13 +10:00
|
|
|
command_allocator: CommandAllocator,
|
2020-11-20 04:29:57 +11:00
|
|
|
command_queue: CommandQueue,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Buffer {
|
|
|
|
resource: Resource,
|
2021-04-30 15:06:13 +10:00
|
|
|
size: u64,
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Image {
|
|
|
|
resource: Resource,
|
|
|
|
}
|
|
|
|
|
2021-04-30 15:06:13 +10:00
|
|
|
// TODO: this doesn't make an upload/download distinction. Probably
|
|
|
|
// we want to move toward webgpu-style usage flags, ie map_read and
|
|
|
|
// map_write are the main ones that affect buffer creation.
|
2020-11-20 04:29:57 +11:00
|
|
|
#[derive(Clone, Copy)]
|
|
|
|
pub enum MemFlags {
|
|
|
|
DeviceLocal,
|
|
|
|
HostCoherent,
|
|
|
|
}
|
|
|
|
|
2021-04-30 15:06:13 +10:00
|
|
|
pub struct CmdBuf(GraphicsCommandList);
|
2020-11-20 04:29:57 +11:00
|
|
|
|
|
|
|
pub struct Pipeline;
|
|
|
|
|
|
|
|
pub struct DescriptorSet;
|
|
|
|
|
|
|
|
pub struct QueryPool;
|
|
|
|
|
2021-04-30 15:06:13 +10:00
|
|
|
pub struct Fence {
|
|
|
|
fence: wrappers::Fence,
|
|
|
|
event: wrappers::Event,
|
|
|
|
// This could as well be an atomic, if we needed to cross threads.
|
|
|
|
val: Cell<u64>,
|
|
|
|
}
|
2020-11-20 04:29:57 +11:00
|
|
|
|
|
|
|
pub struct Semaphore;
|
|
|
|
|
|
|
|
impl Dx12Instance {
|
|
|
|
/// Create a new instance.
|
|
|
|
///
|
|
|
|
/// TODO: take a raw window handle.
|
|
|
|
/// TODO: can probably be a trait.
|
|
|
|
pub fn new() -> Result<Dx12Instance, Error> {
|
|
|
|
unsafe {
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
let factory_flags = dxgi1_3::DXGI_CREATE_FACTORY_DEBUG;
|
|
|
|
|
|
|
|
#[cfg(not(debug_assertions))]
|
|
|
|
let factory_flags: u32 = 0;
|
|
|
|
|
|
|
|
let factory = Factory4::create(factory_flags)?;
|
|
|
|
Ok(Dx12Instance { factory })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get a device suitable for compute workloads.
|
|
|
|
///
|
|
|
|
/// TODO: handle window.
|
|
|
|
/// TODO: probably can also be trait'ified.
|
|
|
|
pub fn device(&self) -> Result<Dx12Device, Error> {
|
|
|
|
unsafe {
|
|
|
|
let device = Device::create_device(&self.factory)?;
|
|
|
|
let list_type = d3d12::D3D12_COMMAND_LIST_TYPE_DIRECT;
|
2021-04-30 15:06:13 +10:00
|
|
|
let command_queue = device.create_command_queue(
|
|
|
|
list_type,
|
|
|
|
0,
|
|
|
|
d3d12::D3D12_COMMAND_QUEUE_FLAG_NONE,
|
|
|
|
0,
|
|
|
|
)?;
|
|
|
|
let command_allocator = device.create_command_allocator(list_type)?;
|
|
|
|
Ok(Dx12Device {
|
|
|
|
device,
|
|
|
|
command_queue,
|
|
|
|
command_allocator,
|
|
|
|
})
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl crate::Device for Dx12Device {
|
|
|
|
type Buffer = Buffer;
|
|
|
|
|
|
|
|
type Image = Image;
|
|
|
|
|
|
|
|
type MemFlags = MemFlags;
|
|
|
|
|
|
|
|
type Pipeline = Pipeline;
|
|
|
|
|
|
|
|
type DescriptorSet = DescriptorSet;
|
|
|
|
|
|
|
|
type QueryPool = QueryPool;
|
|
|
|
|
|
|
|
type CmdBuf = CmdBuf;
|
|
|
|
|
|
|
|
type Fence = Fence;
|
|
|
|
|
|
|
|
type Semaphore = Semaphore;
|
|
|
|
|
|
|
|
fn create_buffer(&self, size: u64, mem_flags: Self::MemFlags) -> Result<Self::Buffer, Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
unsafe {
|
|
|
|
let resource = match mem_flags {
|
|
|
|
MemFlags::DeviceLocal => self
|
|
|
|
.device
|
|
|
|
.create_gpu_only_byte_addressed_buffer(size.try_into()?)?,
|
|
|
|
MemFlags::HostCoherent => self
|
|
|
|
.device
|
|
|
|
.create_uploadable_byte_addressed_buffer(size.try_into()?)?,
|
|
|
|
};
|
|
|
|
Ok(Buffer { resource, size })
|
|
|
|
}
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn destroy_buffer(&self, buffer: &Self::Buffer) -> Result<(), Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
buffer.resource.destroy();
|
|
|
|
Ok(())
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn create_image2d(
|
|
|
|
&self,
|
|
|
|
width: u32,
|
|
|
|
height: u32,
|
|
|
|
mem_flags: Self::MemFlags,
|
|
|
|
) -> Result<Self::Image, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn destroy_image(&self, image: &Self::Image) -> Result<(), Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn create_simple_compute_pipeline(
|
|
|
|
&self,
|
|
|
|
code: &[u8],
|
|
|
|
n_buffers: u32,
|
|
|
|
n_images: u32,
|
|
|
|
) -> Result<Self::Pipeline, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn create_descriptor_set(
|
|
|
|
&self,
|
|
|
|
pipeline: &Self::Pipeline,
|
|
|
|
bufs: &[&Self::Buffer],
|
|
|
|
images: &[&Self::Image],
|
|
|
|
) -> Result<Self::DescriptorSet, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn create_cmd_buf(&self) -> Result<Self::CmdBuf, Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
let list_type = d3d12::D3D12_COMMAND_LIST_TYPE_DIRECT;
|
|
|
|
let node_mask = 0;
|
|
|
|
unsafe {
|
|
|
|
let cmd_buf = self.device.create_graphics_command_list(
|
|
|
|
list_type,
|
|
|
|
&self.command_allocator,
|
|
|
|
None,
|
|
|
|
node_mask,
|
|
|
|
)?;
|
|
|
|
Ok(CmdBuf(cmd_buf))
|
|
|
|
}
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
fn create_query_pool(&self, n_queries: u32) -> Result<Self::QueryPool, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn fetch_query_pool(&self, pool: &Self::QueryPool) -> Result<Vec<f64>, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn run_cmd_buf(
|
|
|
|
&self,
|
|
|
|
cmd_buf: &Self::CmdBuf,
|
|
|
|
wait_semaphores: &[Self::Semaphore],
|
|
|
|
signal_semaphores: &[Self::Semaphore],
|
|
|
|
fence: Option<&Self::Fence>,
|
|
|
|
) -> Result<(), Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
// TODO: handle semaphores
|
|
|
|
self.command_queue
|
|
|
|
.execute_command_lists(&[cmd_buf.0.as_raw_list()]);
|
|
|
|
if let Some(fence) = fence {
|
|
|
|
let val = fence.val.get() + 1;
|
|
|
|
fence.val.set(val);
|
|
|
|
self.command_queue.signal(&fence.fence, val)?;
|
|
|
|
fence.fence.set_event_on_completion(&fence.event, val)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn read_buffer<T: Sized>(
|
|
|
|
&self,
|
|
|
|
buffer: &Self::Buffer,
|
|
|
|
result: &mut Vec<T>,
|
|
|
|
) -> Result<(), Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
let len = buffer.size as usize / std::mem::size_of::<T>();
|
|
|
|
if len > result.len() {
|
|
|
|
result.reserve(len - result.len());
|
|
|
|
}
|
|
|
|
buffer.resource.read_resource(result.as_mut_ptr(), len)?;
|
|
|
|
result.set_len(len);
|
|
|
|
Ok(())
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn write_buffer<T: Sized>(
|
|
|
|
&self,
|
|
|
|
buffer: &Self::Buffer,
|
|
|
|
contents: &[T],
|
|
|
|
) -> Result<(), Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
let len = buffer.size as usize / std::mem::size_of::<T>();
|
|
|
|
buffer.resource.write_resource(len, contents.as_ptr())?;
|
|
|
|
Ok(())
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn create_semaphore(&self) -> Result<Self::Semaphore, Error> {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn create_fence(&self, signaled: bool) -> Result<Self::Fence, Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
let fence = self.device.create_fence(0)?;
|
|
|
|
let event = wrappers::Event::create(false, signaled)?;
|
|
|
|
let val = Cell::new(0);
|
|
|
|
Ok(Fence { fence, event, val })
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn wait_and_reset(&self, fences: &[Self::Fence]) -> Result<(), Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
for fence in fences {
|
|
|
|
// TODO: probably handle errors here.
|
|
|
|
let _status = fence.event.wait(winapi::um::winbase::INFINITE);
|
|
|
|
}
|
|
|
|
Ok(())
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn get_fence_status(&self, fence: Self::Fence) -> Result<bool, Error> {
|
2021-04-30 15:06:13 +10:00
|
|
|
let fence_val = fence.fence.get_value();
|
|
|
|
Ok(fence_val == fence.val.get())
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl crate::CmdBuf<Dx12Device> for CmdBuf {
|
2021-04-30 15:06:13 +10:00
|
|
|
unsafe fn begin(&mut self) {}
|
2020-11-20 04:29:57 +11:00
|
|
|
|
|
|
|
unsafe fn finish(&mut self) {
|
2021-04-30 15:06:13 +10:00
|
|
|
let _ = self.0.close();
|
2020-11-20 04:29:57 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn dispatch(
|
|
|
|
&mut self,
|
|
|
|
pipeline: &Pipeline,
|
|
|
|
descriptor_set: &DescriptorSet,
|
|
|
|
size: (u32, u32, u32),
|
|
|
|
) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn memory_barrier(&mut self) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn host_barrier(&mut self) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn image_barrier(
|
|
|
|
&mut self,
|
|
|
|
image: &Image,
|
|
|
|
src_layout: crate::ImageLayout,
|
|
|
|
dst_layout: crate::ImageLayout,
|
|
|
|
) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn clear_buffer(&self, buffer: &Buffer) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn copy_buffer(&self, src: &Buffer, dst: &Buffer) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn copy_image_to_buffer(&self, src: &Image, dst: &Buffer) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn copy_buffer_to_image(&self, src: &Buffer, dst: &Image) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn blit_image(&self, src: &Image, dst: &Image) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn reset_query_pool(&mut self, pool: &QueryPool) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn write_timestamp(&mut self, pool: &QueryPool, query: u32) {
|
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl crate::MemFlags for MemFlags {
|
|
|
|
fn device_local() -> Self {
|
|
|
|
MemFlags::DeviceLocal
|
|
|
|
}
|
|
|
|
|
|
|
|
fn host_coherent() -> Self {
|
|
|
|
MemFlags::HostCoherent
|
|
|
|
}
|
|
|
|
}
|