test(d3d11): allow hello_triangle to take image as input

This commit is contained in:
chyyran 2023-01-29 21:19:18 -05:00
parent 2d6a967c7f
commit 3c13dc8277
5 changed files with 46 additions and 62 deletions

View file

@ -16,7 +16,8 @@ impl From<FilterMode> for Direct3D11::D3D11_FILTER {
fn from(value: FilterMode) -> Self { fn from(value: FilterMode) -> Self {
match value { match value {
FilterMode::Linear => Direct3D11::D3D11_FILTER_MIN_MAG_MIP_LINEAR, FilterMode::Linear => Direct3D11::D3D11_FILTER_MIN_MAG_MIP_LINEAR,
_ => Direct3D11::D3D11_FILTER_MIN_MAG_MIP_POINT, FilterMode::Nearest => Direct3D11::D3D11_FILTER_MIN_MAG_MIP_POINT,
_ => Direct3D11::D3D11_FILTER_MIN_MAG_MIP_LINEAR,
} }
} }
} }

View file

@ -243,11 +243,12 @@ pub mod d3d11_hello_triangle {
use crate::filter_chain::FilterChainD3D11; use crate::filter_chain::FilterChainD3D11;
use crate::options::{FilterChainOptionsD3D11, FrameOptionsD3D11}; use crate::options::{FilterChainOptionsD3D11, FrameOptionsD3D11};
use crate::texture::D3D11InputView; use crate::texture::{D3D11InputView, LutTexture};
use crate::D3D11OutputView; use crate::D3D11OutputView;
use librashader_common::{Size, Viewport}; use librashader_common::{FilterMode, ImageFormat, Size, Viewport, WrapMode};
use std::slice; use std::slice;
use std::time::Instant; use std::time::Instant;
use librashader_runtime::image::Image;
pub struct Sample { pub struct Sample {
pub dxgi_factory: IDXGIFactory4, pub dxgi_factory: IDXGIFactory4,
@ -255,6 +256,7 @@ pub mod d3d11_hello_triangle {
pub context: ID3D11DeviceContext, pub context: ID3D11DeviceContext,
pub resources: Option<Resources>, pub resources: Option<Resources>,
pub filter: FilterChainD3D11, pub filter: FilterChainD3D11,
pub lut: Option<ID3D11Texture2D>
} }
pub struct Resources { pub struct Resources {
@ -284,15 +286,37 @@ pub mod d3d11_hello_triangle {
pub(crate) fn new( pub(crate) fn new(
filter: impl AsRef<Path>, filter: impl AsRef<Path>,
filter_options: Option<&FilterChainOptionsD3D11>, filter_options: Option<&FilterChainOptionsD3D11>,
image: Option<Image>,
) -> Result<Self> { ) -> Result<Self> {
let (dxgi_factory, device, context) = create_device()?; let (dxgi_factory, device, context) = create_device()?;
let filter = FilterChainD3D11::load_from_path(&device, filter, filter_options).unwrap(); let filter = FilterChainD3D11::load_from_path(&device, filter, filter_options).unwrap();
let lut = if let Some(image) = image {
let lut = LutTexture::new(&device, &context, &image, D3D11_TEXTURE2D_DESC {
Width: image.size.width,
Height: image.size.height,
MipLevels: 1,
ArraySize: 0,
Format: ImageFormat::R8G8B8A8Unorm.into(),
SampleDesc: DXGI_SAMPLE_DESC {
Count: 1,
Quality: 0,
},
Usage: D3D11_USAGE_DYNAMIC,
BindFlags: D3D11_BIND_SHADER_RESOURCE,
CPUAccessFlags: Default::default(),
MiscFlags: Default::default(),
}, FilterMode::Linear, WrapMode::ClampToEdge).unwrap();
Some(lut.handle)
} else {
None
};
Ok(Sample { Ok(Sample {
filter, filter,
dxgi_factory, dxgi_factory,
device, device,
context, context,
resources: None, resources: None,
lut,
}) })
} }
} }
@ -488,32 +512,24 @@ pub mod d3d11_hello_triangle {
unsafe { unsafe {
self.context.DrawIndexed(3, 0, 0); self.context.DrawIndexed(3, 0, 0);
self.context.OMSetRenderTargets(None, None);
} }
unsafe { unsafe {
let input = if let Some(lut) = &self.lut {
lut.clone()
} else {
resources.renderbuffer.clone()
};
let mut tex2d_desc = Default::default(); let mut tex2d_desc = Default::default();
resources.renderbuffer.GetDesc(&mut tex2d_desc); input.GetDesc(&mut tex2d_desc);
let mut backbuffer_desc = Default::default(); let mut backbuffer_desc = Default::default();
resources.backbuffer.as_ref().unwrap().GetDesc(&mut backbuffer_desc); resources.backbuffer.as_ref().unwrap().GetDesc(&mut backbuffer_desc);
let mut renderbuffer_copy = None; let mut input_srv = None;
self.device.CreateTexture2D(
&D3D11_TEXTURE2D_DESC {
BindFlags: D3D11_BIND_SHADER_RESOURCE | D3D11_BIND_RENDER_TARGET,
CPUAccessFlags: D3D11_CPU_ACCESS_READ | D3D11_CPU_ACCESS_WRITE,
..tex2d_desc
},
None,
Some(&mut renderbuffer_copy),
)?;
let backup = renderbuffer_copy.unwrap();
self.context.CopyResource(&backup, &resources.renderbuffer);
let mut copy_srv = None;
self.device.CreateShaderResourceView( self.device.CreateShaderResourceView(
&backup, &input,
Some(&D3D11_SHADER_RESOURCE_VIEW_DESC { Some(&D3D11_SHADER_RESOURCE_VIEW_DESC {
Format: tex2d_desc.Format, Format: tex2d_desc.Format,
ViewDimension: D3D_SRV_DIMENSION_TEXTURE2D, ViewDimension: D3D_SRV_DIMENSION_TEXTURE2D,
@ -524,27 +540,9 @@ pub mod d3d11_hello_triangle {
}, },
}, },
}), }),
Some(&mut copy_srv), Some(&mut input_srv),
)?;
let srv = copy_srv.unwrap();
let mut shader_out = None;
self.device
.CreateTexture2D(&tex2d_desc, None, Some(&mut shader_out))?;
let shader_out = shader_out.unwrap();
let mut rtv = None;
self.device.CreateRenderTargetView(
&shader_out,
Some(&D3D11_RENDER_TARGET_VIEW_DESC {
Format: tex2d_desc.Format,
ViewDimension: D3D11_RTV_DIMENSION_TEXTURE2D,
Anonymous: D3D11_RENDER_TARGET_VIEW_DESC_0 {
Texture2D: D3D11_TEX2D_RTV { MipSlice: 0 },
},
}),
Some(&mut rtv),
)?; )?;
let srv = input_srv.unwrap();
// eprintln!("w: {} h: {}", backbuffer_desc.Width, backbuffer_desc.Height); // eprintln!("w: {} h: {}", backbuffer_desc.Width, backbuffer_desc.Height);
self.filter self.filter
@ -557,8 +555,8 @@ pub mod d3d11_hello_triangle {
}, },
}, },
&Viewport { &Viewport {
x: resources.viewport.TopLeftX, x: 0f32,
y: resources.viewport.TopLeftY, y: 0f32,
output: D3D11OutputView { output: D3D11OutputView {
size: Size { size: Size {
width: backbuffer_desc.Width, width: backbuffer_desc.Width,

View file

@ -27,6 +27,7 @@ pub use texture::D3D11OutputView;
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use librashader_runtime::image::{Image, UVDirection};
use super::*; use super::*;
use crate::options::FilterChainOptionsD3D11; use crate::options::FilterChainOptionsD3D11;
@ -41,6 +42,7 @@ mod tests {
use_deferred_context: false, use_deferred_context: false,
force_no_mipmaps: false, force_no_mipmaps: false,
}), }),
Some(Image::load("../test/sf2.png", UVDirection::TopLeft).unwrap())
) )
.unwrap(); .unwrap();
// let sample = hello_triangle_old::d3d11_hello_triangle::Sample::new( // let sample = hello_triangle_old::d3d11_hello_triangle::Sample::new(

View file

@ -10,6 +10,7 @@ use windows::Win32::Graphics::Direct3D11::{
D3D11_TEXTURE2D_DESC, D3D11_USAGE_DYNAMIC, D3D11_USAGE_STAGING, D3D11_TEXTURE2D_DESC, D3D11_USAGE_DYNAMIC, D3D11_USAGE_STAGING,
}; };
use windows::Win32::Graphics::Dxgi::Common::DXGI_SAMPLE_DESC; use windows::Win32::Graphics::Dxgi::Common::DXGI_SAMPLE_DESC;
use librashader_runtime::scaling::MipmapSize;
use crate::error::{assume_d3d11_init, Result}; use crate::error::{assume_d3d11_init, Result};
use crate::framebuffer::OwnedFramebuffer; use crate::framebuffer::OwnedFramebuffer;
@ -116,15 +117,8 @@ impl LutTexture {
// determine number of mipmaps required // determine number of mipmaps required
if (desc.MiscFlags & D3D11_RESOURCE_MISC_GENERATE_MIPS).0 != 0 { if (desc.MiscFlags & D3D11_RESOURCE_MISC_GENERATE_MIPS).0 != 0 {
let mut width = desc.Width >> 5;
let mut height = desc.Height >> 5;
desc.BindFlags |= D3D11_BIND_RENDER_TARGET; desc.BindFlags |= D3D11_BIND_RENDER_TARGET;
desc.MipLevels = source.size.calculate_miplevels();
while width != 0 && height != 0 {
width >>= 1;
height >>= 1;
desc.MipLevels += 1;
}
} }
// Don't need to determine format support because LUTs are always DXGI_FORMAT_R8G8B8A8_UNORM // Don't need to determine format support because LUTs are always DXGI_FORMAT_R8G8B8A8_UNORM

View file

@ -93,14 +93,3 @@ where
height: height.round().as_(), height: height.round().as_(),
} }
} }
pub fn calculate_miplevels(size: Size<u32>) -> u32 {
let mut size = std::cmp::max(size.width, size.height);
let mut levels = 0;
while size != 0 {
levels += 1;
size >>= 1;
}
levels
}