646 lines
21 KiB
Rust
646 lines
21 KiB
Rust
|
use std::sync::mpsc::Receiver;
|
||
|
|
||
|
const WIDTH: i32 = 900;
|
||
|
const HEIGHT: i32 = 700;
|
||
|
const TITLE: &str = "librashader DirectX 11";
|
||
|
|
||
|
use windows::{
|
||
|
core::*, Win32::Foundation::*, Win32::Graphics::Direct3D::Fxc::*, Win32::Graphics::Direct3D::*,
|
||
|
Win32::Graphics::Direct3D11::*, Win32::Graphics::Dxgi::Common::*, Win32::Graphics::Dxgi::*,
|
||
|
Win32::System::LibraryLoader::*, Win32::System::Threading::*,
|
||
|
Win32::System::WindowsProgramming::*, Win32::UI::WindowsAndMessaging::*,
|
||
|
};
|
||
|
|
||
|
static VERTEX_SHADER: &'static [u8] = b"
|
||
|
cbuffer cb : register(b0)
|
||
|
{
|
||
|
row_major float4x4 projectionMatrix : packoffset(c0);
|
||
|
row_major float4x4 modelMatrix : packoffset(c4);
|
||
|
row_major float4x4 viewMatrix : packoffset(c8);
|
||
|
};
|
||
|
|
||
|
struct VertexInput
|
||
|
{
|
||
|
float3 inPos : POSITION;
|
||
|
float3 inColor : COLOR;
|
||
|
};
|
||
|
|
||
|
struct VertexOutput
|
||
|
{
|
||
|
float3 color : COLOR;
|
||
|
float4 position : SV_Position;
|
||
|
};
|
||
|
|
||
|
VertexOutput main(VertexInput vertexInput)
|
||
|
{
|
||
|
float3 inColor = vertexInput.inColor;
|
||
|
float3 inPos = vertexInput.inPos;
|
||
|
float3 outColor = inColor;
|
||
|
float4 position = mul(float4(inPos, 1.0), mul(modelMatrix, mul(viewMatrix, projectionMatrix)));
|
||
|
|
||
|
VertexOutput output;
|
||
|
output.position = position;
|
||
|
output.color = outColor;
|
||
|
return output;
|
||
|
}\0";
|
||
|
|
||
|
static PIXEL_SHADER: &'static [u8] = b"
|
||
|
struct PixelInput
|
||
|
{
|
||
|
float3 color : COLOR;
|
||
|
};
|
||
|
|
||
|
struct PixelOutput
|
||
|
{
|
||
|
float4 attachment0 : SV_Target0;
|
||
|
};
|
||
|
|
||
|
PixelOutput main(PixelInput pixelInput)
|
||
|
{
|
||
|
float3 inColor = pixelInput.color;
|
||
|
PixelOutput output;
|
||
|
output.attachment0 = float4(inColor, 1.0f);
|
||
|
return output;
|
||
|
}\0";
|
||
|
|
||
|
|
||
|
use std::mem::transmute;
|
||
|
use gfx_maths::Mat4;
|
||
|
|
||
|
trait DXSample {
|
||
|
fn new() -> Result<Self>
|
||
|
where
|
||
|
Self: Sized;
|
||
|
|
||
|
fn bind_to_window(&mut self, hwnd: &HWND) -> Result<()>;
|
||
|
|
||
|
fn update(&mut self) {}
|
||
|
fn render(&mut self) -> Result<()> { Ok(()) }
|
||
|
fn on_key_up(&mut self, _key: u8) {}
|
||
|
fn on_key_down(&mut self, _key: u8) {}
|
||
|
|
||
|
fn title(&self) -> String {
|
||
|
TITLE.into()
|
||
|
}
|
||
|
|
||
|
fn window_size(&self) -> (i32, i32) {
|
||
|
(WIDTH, HEIGHT)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn run_sample<S>() -> Result<()>
|
||
|
where
|
||
|
S: DXSample,
|
||
|
{
|
||
|
let instance = unsafe { GetModuleHandleA(None)? };
|
||
|
|
||
|
let wc = WNDCLASSEXA {
|
||
|
cbSize: std::mem::size_of::<WNDCLASSEXA>() as u32,
|
||
|
style: CS_HREDRAW | CS_VREDRAW,
|
||
|
lpfnWndProc: Some(wndproc::<S>),
|
||
|
hInstance: instance,
|
||
|
hCursor: unsafe { LoadCursorW(None, IDC_ARROW)? },
|
||
|
lpszClassName: s!("RustWindowClass"),
|
||
|
..Default::default()
|
||
|
};
|
||
|
|
||
|
let mut sample = S::new()?;
|
||
|
|
||
|
let size = sample.window_size();
|
||
|
|
||
|
let atom = unsafe { RegisterClassExA(&wc) };
|
||
|
debug_assert_ne!(atom, 0);
|
||
|
|
||
|
let mut window_rect = RECT {
|
||
|
left: 0,
|
||
|
top: 0,
|
||
|
right: size.0,
|
||
|
bottom: size.1,
|
||
|
};
|
||
|
unsafe { AdjustWindowRect(&mut window_rect, WS_OVERLAPPEDWINDOW, false) };
|
||
|
|
||
|
let mut title = sample.title();
|
||
|
|
||
|
|
||
|
|
||
|
title.push('\0');
|
||
|
|
||
|
let hwnd = unsafe {
|
||
|
CreateWindowExA(
|
||
|
WINDOW_EX_STYLE::default(),
|
||
|
s!("RustWindowClass"),
|
||
|
PCSTR(title.as_ptr()),
|
||
|
WS_OVERLAPPEDWINDOW,
|
||
|
CW_USEDEFAULT,
|
||
|
CW_USEDEFAULT,
|
||
|
window_rect.right - window_rect.left,
|
||
|
window_rect.bottom - window_rect.top,
|
||
|
None, // no parent window
|
||
|
None, // no menus
|
||
|
instance,
|
||
|
Some(&mut sample as *mut _ as _),
|
||
|
)
|
||
|
};
|
||
|
|
||
|
sample.bind_to_window(&hwnd)?;
|
||
|
unsafe { ShowWindow(hwnd, SW_SHOW) };
|
||
|
|
||
|
loop {
|
||
|
let mut message = MSG::default();
|
||
|
|
||
|
if unsafe { PeekMessageA(&mut message, None, 0, 0, PM_REMOVE) }.into() {
|
||
|
unsafe {
|
||
|
TranslateMessage(&message);
|
||
|
DispatchMessageA(&message);
|
||
|
}
|
||
|
|
||
|
if message.message == WM_QUIT {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn sample_wndproc<S: DXSample>(sample: &mut S, message: u32, wparam: WPARAM) -> bool {
|
||
|
match message {
|
||
|
WM_KEYDOWN => {
|
||
|
sample.on_key_down(wparam.0 as u8);
|
||
|
true
|
||
|
}
|
||
|
WM_KEYUP => {
|
||
|
sample.on_key_up(wparam.0 as u8);
|
||
|
true
|
||
|
}
|
||
|
WM_PAINT => {
|
||
|
sample.update();
|
||
|
sample.render().unwrap();
|
||
|
true
|
||
|
}
|
||
|
_ => false,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
extern "system" fn wndproc<S: DXSample>(
|
||
|
window: HWND,
|
||
|
message: u32,
|
||
|
wparam: WPARAM,
|
||
|
lparam: LPARAM,
|
||
|
) -> LRESULT {
|
||
|
match message {
|
||
|
WM_CREATE => {
|
||
|
unsafe {
|
||
|
let create_struct: &CREATESTRUCTA = transmute(lparam);
|
||
|
SetWindowLongPtrA(window, GWLP_USERDATA, create_struct.lpCreateParams as _);
|
||
|
}
|
||
|
LRESULT::default()
|
||
|
}
|
||
|
WM_DESTROY => {
|
||
|
unsafe { PostQuitMessage(0) };
|
||
|
LRESULT::default()
|
||
|
}
|
||
|
_ => {
|
||
|
let user_data = unsafe { GetWindowLongPtrA(window, GWLP_USERDATA) };
|
||
|
let sample = std::ptr::NonNull::<S>::new(user_data as _);
|
||
|
let handled = sample.map_or(false, |mut s| {
|
||
|
sample_wndproc(unsafe { s.as_mut() }, message, wparam)
|
||
|
});
|
||
|
|
||
|
if handled {
|
||
|
LRESULT::default()
|
||
|
} else {
|
||
|
unsafe { DefWindowProcA(window, message, wparam, lparam) }
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
#[repr(C)]
|
||
|
struct Vertex {
|
||
|
position: [f32; 3],
|
||
|
color: [f32; 3],
|
||
|
}
|
||
|
|
||
|
#[repr(C)]
|
||
|
#[derive(Default)]
|
||
|
struct TriangleUniforms {
|
||
|
projection_matrix: Mat4,
|
||
|
model_matrix: Mat4,
|
||
|
view_matrix: Mat4
|
||
|
}
|
||
|
|
||
|
mod d3d11_hello_triangle {
|
||
|
use std::slice;
|
||
|
use std::time::Instant;
|
||
|
use gfx_maths::{Quaternion, Vec3};
|
||
|
use super::*;
|
||
|
|
||
|
const FRAME_COUNT: u32 = 2;
|
||
|
|
||
|
pub struct Sample {
|
||
|
dxgi_factory: IDXGIFactory4,
|
||
|
device: ID3D11Device,
|
||
|
context: ID3D11DeviceContext,
|
||
|
resources: Option<Resources>
|
||
|
}
|
||
|
|
||
|
pub struct Resources {
|
||
|
swapchain: IDXGISwapChain,
|
||
|
depth_buffer: ID3D11Texture2D,
|
||
|
depth_stencil_view: ID3D11DepthStencilView,
|
||
|
triangle_vertices: ID3D11Buffer,
|
||
|
triangle_indices: ID3D11Buffer,
|
||
|
triangle_uniforms: ID3D11Buffer,
|
||
|
vs: ID3D11VertexShader,
|
||
|
ps: ID3D11PixelShader,
|
||
|
input_layout: ID3D11InputLayout,
|
||
|
frame_start: Instant,
|
||
|
frame_end: Instant,
|
||
|
elapsed: f32,
|
||
|
triangle_uniform_values: TriangleUniforms,
|
||
|
pub backbuffer: ID3D11Texture2D,
|
||
|
pub rtv: ID3D11RenderTargetView,
|
||
|
pub viewport: D3D11_VIEWPORT,
|
||
|
}
|
||
|
impl DXSample for Sample {
|
||
|
fn new() -> Result<Self> {
|
||
|
let (dxgi_factory, device, context) = create_device()?;
|
||
|
|
||
|
Ok(Sample {
|
||
|
dxgi_factory,
|
||
|
device,
|
||
|
context,
|
||
|
resources: None
|
||
|
})
|
||
|
}
|
||
|
|
||
|
fn bind_to_window(&mut self, hwnd: &HWND) -> Result<()> {
|
||
|
let swapchain = create_swapchain(&self.dxgi_factory, &self.device, *hwnd)?;
|
||
|
let (rtv, backbuffer) = create_rtv(&self.device, &swapchain)?;
|
||
|
let (depth_buffer, depth_stencil_view) = create_depth_buffer(&self.device)?;
|
||
|
let (triangle_vbo, triangle_indices) = create_triangle_buffers(&self.device)?;
|
||
|
let triangle_uniforms = create_triangle_uniforms(&self.device)?;
|
||
|
|
||
|
let vs_blob = compile_shader(VERTEX_SHADER, b"main\0", b"vs_5_0")?;
|
||
|
let ps_blob = compile_shader(PIXEL_SHADER, b"main\0", b"ps_5_0")?;
|
||
|
|
||
|
let vs_compiled = unsafe {
|
||
|
// SAFETY: slice as valid for as long as vs_blob is alive.
|
||
|
slice::from_raw_parts(vs_blob.GetBufferPointer().cast::<u8>(), vs_blob.GetBufferSize())
|
||
|
};
|
||
|
|
||
|
let vs = unsafe {
|
||
|
self.device.CreateVertexShader(vs_compiled, None)
|
||
|
}?;
|
||
|
|
||
|
let ps = unsafe {
|
||
|
let ps = slice::from_raw_parts(ps_blob.GetBufferPointer().cast::<u8>(), ps_blob.GetBufferSize());
|
||
|
self.device.CreatePixelShader(ps, None)
|
||
|
}?;
|
||
|
|
||
|
|
||
|
let (input_layout, stencil_state, raster_state)
|
||
|
= create_pipeline_state(&self.device, vs_compiled)?;
|
||
|
|
||
|
unsafe {
|
||
|
self.context.OMSetDepthStencilState(&stencil_state, 1);
|
||
|
self.context.RSSetState(&raster_state);
|
||
|
}
|
||
|
|
||
|
self.resources = Some(Resources {
|
||
|
swapchain,
|
||
|
rtv,
|
||
|
backbuffer,
|
||
|
depth_buffer,
|
||
|
depth_stencil_view,
|
||
|
triangle_vertices: triangle_vbo,
|
||
|
triangle_indices,
|
||
|
triangle_uniforms,
|
||
|
vs,
|
||
|
ps,
|
||
|
input_layout,
|
||
|
frame_end: Instant::now(),
|
||
|
frame_start: Instant::now(),
|
||
|
elapsed: 0f32,
|
||
|
triangle_uniform_values: Default::default(),
|
||
|
viewport: D3D11_VIEWPORT {
|
||
|
TopLeftX: 0.0,
|
||
|
TopLeftY: 0.0,
|
||
|
Width: WIDTH as f32,
|
||
|
Height: HEIGHT as f32,
|
||
|
MinDepth: D3D11_MIN_DEPTH,
|
||
|
MaxDepth: D3D11_MAX_DEPTH,
|
||
|
}
|
||
|
});
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn render(&mut self) -> Result<()> {
|
||
|
let Some(resources) = &mut self.resources else {
|
||
|
return Ok(());
|
||
|
};
|
||
|
|
||
|
resources.frame_end = Instant::now();
|
||
|
let time = resources.frame_end - resources.frame_start;
|
||
|
let time = time.as_secs() as f32 * 1000.0;
|
||
|
|
||
|
// framelimit set to 60fps
|
||
|
if time < (1000.0f32 / 144.0f32) {
|
||
|
return Ok(());
|
||
|
}
|
||
|
|
||
|
resources.elapsed += 0.0000001 * time;
|
||
|
resources.elapsed %= 6.283185307179586f32;
|
||
|
|
||
|
// resources.triangle_uniform_values.model_matrix = Mat4::rotate(Quaternion::axis_angle(Vec3::new(0.0, 0.0, 1.0), resources.elapsed));
|
||
|
resources.triangle_uniform_values.model_matrix = Mat4::identity();
|
||
|
|
||
|
let buffer_number = 0;
|
||
|
|
||
|
unsafe {
|
||
|
let mapped_resource = self.context.Map(&resources.triangle_uniforms, 0, D3D11_MAP_WRITE_DISCARD, 0)?;
|
||
|
std::ptr::copy_nonoverlapping(&resources.triangle_uniform_values, mapped_resource.pData.cast(), 1);
|
||
|
self.context.Unmap(&resources.triangle_uniforms, 0);
|
||
|
}
|
||
|
|
||
|
unsafe {
|
||
|
self.context.VSSetConstantBuffers(buffer_number, Some(&[Some(resources.triangle_uniforms.clone())]));
|
||
|
self.context.OMSetRenderTargets(Some(&[Some(resources.rtv.clone())]), &resources.depth_stencil_view);
|
||
|
self.context.RSSetViewports(Some(&[resources.viewport]))
|
||
|
}
|
||
|
|
||
|
unsafe {
|
||
|
let color = [0.3, 0.4, 0.6, 1.0];
|
||
|
self.context.ClearRenderTargetView(&resources.rtv, color.as_ptr());
|
||
|
self.context.ClearDepthStencilView(&resources.depth_stencil_view, D3D11_CLEAR_DEPTH.0 as u32, 1.0, 0);
|
||
|
self.context.IASetInputLayout(&resources.input_layout);
|
||
|
}
|
||
|
|
||
|
unsafe {
|
||
|
self.context.VSSetShader(&resources.vs, None);
|
||
|
self.context.PSSetShader(&resources.ps, None);
|
||
|
|
||
|
let stride =std::mem::size_of::<Vertex>() as u32;
|
||
|
let offset = 0;
|
||
|
self.context.IASetVertexBuffers(0, 1, Some(&Some(resources.triangle_vertices.clone())), Some(&stride), Some(&offset));
|
||
|
self.context.IASetIndexBuffer(&resources.triangle_indices, DXGI_FORMAT_R32_UINT, 0);
|
||
|
self.context.IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST);
|
||
|
}
|
||
|
|
||
|
unsafe {
|
||
|
self.context.DrawIndexed(3, 0, 0);
|
||
|
resources.swapchain.Present(0, 0).ok()?;
|
||
|
}
|
||
|
Ok(())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn create_rtv(device: &ID3D11Device, swapchain: &IDXGISwapChain) -> Result<(ID3D11RenderTargetView, ID3D11Texture2D)>{
|
||
|
unsafe {
|
||
|
let backbuffer: ID3D11Texture2D = swapchain.GetBuffer(0)?;
|
||
|
let rtv = device.CreateRenderTargetView(&backbuffer, None)?;
|
||
|
|
||
|
Ok((rtv, backbuffer))
|
||
|
}
|
||
|
}
|
||
|
fn create_pipeline_state(device: &ID3D11Device, vs_blob: &[u8]) -> Result<(ID3D11InputLayout, ID3D11DepthStencilState, ID3D11RasterizerState)> {
|
||
|
unsafe {
|
||
|
let input_layout = device.CreateInputLayout(&[
|
||
|
D3D11_INPUT_ELEMENT_DESC {
|
||
|
SemanticName: s!("POSITION"),
|
||
|
SemanticIndex: 0,
|
||
|
Format: DXGI_FORMAT_R32G32B32_FLOAT,
|
||
|
InputSlot: 0,
|
||
|
AlignedByteOffset: 0,
|
||
|
InputSlotClass: D3D11_INPUT_PER_VERTEX_DATA,
|
||
|
InstanceDataStepRate: 0
|
||
|
},
|
||
|
D3D11_INPUT_ELEMENT_DESC {
|
||
|
SemanticName: s!("COLOR"),
|
||
|
SemanticIndex: 0,
|
||
|
Format: DXGI_FORMAT_R32G32B32_FLOAT,
|
||
|
InputSlot: 0,
|
||
|
AlignedByteOffset: D3D11_APPEND_ALIGNED_ELEMENT,
|
||
|
InputSlotClass: D3D11_INPUT_PER_VERTEX_DATA,
|
||
|
InstanceDataStepRate: 0
|
||
|
},
|
||
|
], vs_blob)?;
|
||
|
|
||
|
let stencil_state = device.CreateDepthStencilState(&D3D11_DEPTH_STENCIL_DESC {
|
||
|
DepthEnable: BOOL::from(true),
|
||
|
DepthWriteMask: D3D11_DEPTH_WRITE_MASK_ALL,
|
||
|
DepthFunc: D3D11_COMPARISON_LESS,
|
||
|
StencilEnable: BOOL::from(true),
|
||
|
StencilReadMask: 0xff,
|
||
|
StencilWriteMask: 0xff,
|
||
|
FrontFace: D3D11_DEPTH_STENCILOP_DESC {
|
||
|
StencilFailOp: D3D11_STENCIL_OP_KEEP,
|
||
|
StencilDepthFailOp: D3D11_STENCIL_OP_INCR,
|
||
|
StencilPassOp: D3D11_STENCIL_OP_KEEP,
|
||
|
StencilFunc: D3D11_COMPARISON_ALWAYS,
|
||
|
},
|
||
|
BackFace: D3D11_DEPTH_STENCILOP_DESC {
|
||
|
StencilFailOp: D3D11_STENCIL_OP_KEEP,
|
||
|
StencilDepthFailOp: D3D11_STENCIL_OP_DECR,
|
||
|
StencilPassOp: D3D11_STENCIL_OP_KEEP,
|
||
|
StencilFunc: D3D11_COMPARISON_ALWAYS,
|
||
|
}
|
||
|
})?;
|
||
|
|
||
|
let rasterizer_state = device.CreateRasterizerState(&D3D11_RASTERIZER_DESC {
|
||
|
AntialiasedLineEnable: BOOL::from(false),
|
||
|
CullMode: D3D11_CULL_NONE,
|
||
|
DepthBias: 0,
|
||
|
DepthBiasClamp: 0.0f32,
|
||
|
DepthClipEnable: BOOL::from(true),
|
||
|
FillMode: D3D11_FILL_SOLID,
|
||
|
FrontCounterClockwise: BOOL::from(false),
|
||
|
MultisampleEnable: BOOL::from(false),
|
||
|
ScissorEnable: BOOL::from(false),
|
||
|
SlopeScaledDepthBias: 0.0f32
|
||
|
})?;
|
||
|
|
||
|
Ok((input_layout, stencil_state, rasterizer_state))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn create_depth_buffer(device: &ID3D11Device) -> Result<(ID3D11Texture2D, ID3D11DepthStencilView)>{
|
||
|
unsafe {
|
||
|
let buffer = device.CreateTexture2D(&D3D11_TEXTURE2D_DESC {
|
||
|
Width: WIDTH as u32,
|
||
|
Height: HEIGHT as u32,
|
||
|
MipLevels: 1,
|
||
|
ArraySize: 1,
|
||
|
Format: DXGI_FORMAT_D24_UNORM_S8_UINT,
|
||
|
SampleDesc: DXGI_SAMPLE_DESC {
|
||
|
Count: 1,
|
||
|
Quality: 0
|
||
|
},
|
||
|
Usage: D3D11_USAGE_DEFAULT,
|
||
|
BindFlags: D3D11_BIND_DEPTH_STENCIL,
|
||
|
CPUAccessFlags: Default::default(),
|
||
|
MiscFlags: Default::default(),
|
||
|
}, None)?;
|
||
|
|
||
|
let view = device.CreateDepthStencilView(&buffer,Some(&D3D11_DEPTH_STENCIL_VIEW_DESC {
|
||
|
Format: DXGI_FORMAT_D24_UNORM_S8_UINT,
|
||
|
ViewDimension: D3D11_DSV_DIMENSION_TEXTURE2D,
|
||
|
Anonymous: D3D11_DEPTH_STENCIL_VIEW_DESC_0 {
|
||
|
Texture2D: D3D11_TEX2D_DSV {
|
||
|
MipSlice: 0,
|
||
|
},
|
||
|
},
|
||
|
..Default::default()
|
||
|
}))?;
|
||
|
|
||
|
Ok((buffer, view))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn create_triangle_uniforms(device: &ID3D11Device) -> Result<ID3D11Buffer> {
|
||
|
unsafe {
|
||
|
device.CreateBuffer(&D3D11_BUFFER_DESC {
|
||
|
ByteWidth: (std::mem::size_of::<TriangleUniforms>()) as u32,
|
||
|
Usage: D3D11_USAGE_DYNAMIC,
|
||
|
BindFlags: D3D11_BIND_CONSTANT_BUFFER,
|
||
|
CPUAccessFlags: D3D11_CPU_ACCESS_WRITE,
|
||
|
MiscFlags: Default::default(),
|
||
|
StructureByteStride: 0,
|
||
|
}, None)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
fn create_triangle_buffers(
|
||
|
device: &ID3D11Device,
|
||
|
) -> Result<(ID3D11Buffer, ID3D11Buffer)> {
|
||
|
let vertices = [
|
||
|
Vertex {
|
||
|
position: [0.5f32, -0.5, 0.0],
|
||
|
color: [1.0, 0.0, 0.0],
|
||
|
},
|
||
|
Vertex {
|
||
|
position: [-0.5, -0.5, 0.0],
|
||
|
color: [0.0, 1.0, 0.0],
|
||
|
},
|
||
|
Vertex {
|
||
|
position: [0.0, 0.5, 0.0],
|
||
|
color: [0.0, 0.0, 1.0],
|
||
|
},
|
||
|
];
|
||
|
|
||
|
let indices = [0, 1, 2];
|
||
|
unsafe {
|
||
|
let vertex_buffer = device.CreateBuffer(&D3D11_BUFFER_DESC {
|
||
|
ByteWidth: (std::mem::size_of::<Vertex>() * vertices.len()) as u32,
|
||
|
Usage: D3D11_USAGE_DEFAULT,
|
||
|
BindFlags: D3D11_BIND_VERTEX_BUFFER,
|
||
|
CPUAccessFlags: Default::default(),
|
||
|
MiscFlags: Default::default(),
|
||
|
StructureByteStride: 0,
|
||
|
}, Some(&D3D11_SUBRESOURCE_DATA {
|
||
|
pSysMem: vertices.as_ptr().cast(),
|
||
|
SysMemPitch: 0,
|
||
|
SysMemSlicePitch: 0,
|
||
|
}))?;
|
||
|
|
||
|
let index_buffer = device.CreateBuffer(&D3D11_BUFFER_DESC {
|
||
|
ByteWidth: (std::mem::size_of::<u32>() * indices.len()) as u32,
|
||
|
Usage: D3D11_USAGE_DEFAULT,
|
||
|
BindFlags: D3D11_BIND_INDEX_BUFFER,
|
||
|
CPUAccessFlags: Default::default(),
|
||
|
MiscFlags: Default::default(),
|
||
|
StructureByteStride: 0,
|
||
|
}, Some(&D3D11_SUBRESOURCE_DATA {
|
||
|
pSysMem: indices.as_ptr().cast(),
|
||
|
SysMemPitch: 0,
|
||
|
SysMemSlicePitch: 0,
|
||
|
}))?;
|
||
|
|
||
|
Ok((vertex_buffer, index_buffer))
|
||
|
}
|
||
|
|
||
|
}
|
||
|
fn create_device() -> Result<(IDXGIFactory4, ID3D11Device, ID3D11DeviceContext)> {
|
||
|
let dxgi_factory_flags = if cfg!(debug_assertions) {
|
||
|
DXGI_CREATE_FACTORY_DEBUG
|
||
|
} else {
|
||
|
0
|
||
|
};
|
||
|
|
||
|
let dxgi_factory: IDXGIFactory4 = unsafe { CreateDXGIFactory2(dxgi_factory_flags) }?;
|
||
|
let feature_levels = vec![D3D_FEATURE_LEVEL_11_0, D3D_FEATURE_LEVEL_11_1];
|
||
|
|
||
|
let mut out_device = None;
|
||
|
let mut out_context = None;
|
||
|
let mut _out_feature_level = D3D_FEATURE_LEVEL_11_0;
|
||
|
|
||
|
unsafe { D3D11CreateDevice(None, D3D_DRIVER_TYPE_HARDWARE,
|
||
|
HINSTANCE::default(),
|
||
|
Default::default(),
|
||
|
Some(&feature_levels),
|
||
|
D3D11_SDK_VERSION,
|
||
|
Some(&mut out_device),
|
||
|
Some(&mut _out_feature_level),
|
||
|
Some(&mut out_context))
|
||
|
}?;
|
||
|
Ok((dxgi_factory, out_device.unwrap(), out_context.unwrap()))
|
||
|
}
|
||
|
|
||
|
fn create_swapchain(fac: &IDXGIFactory4, device: &ID3D11Device, hwnd: HWND) -> Result<IDXGISwapChain>{
|
||
|
let swapchain_desc = DXGI_SWAP_CHAIN_DESC {
|
||
|
BufferDesc: DXGI_MODE_DESC {
|
||
|
Width: WIDTH as u32,
|
||
|
Height: HEIGHT as u32,
|
||
|
RefreshRate: DXGI_RATIONAL {
|
||
|
Numerator: 0,
|
||
|
Denominator: 1,
|
||
|
},
|
||
|
Format: DXGI_FORMAT_R8G8B8A8_UNORM,
|
||
|
ScanlineOrdering: DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED,
|
||
|
Scaling: DXGI_MODE_SCALING_UNSPECIFIED,
|
||
|
},
|
||
|
SampleDesc: DXGI_SAMPLE_DESC {
|
||
|
Count: 1,
|
||
|
Quality: 0,
|
||
|
},
|
||
|
BufferUsage: DXGI_USAGE_RENDER_TARGET_OUTPUT,
|
||
|
BufferCount: 1,
|
||
|
OutputWindow: hwnd,
|
||
|
Windowed: BOOL(1),
|
||
|
SwapEffect: DXGI_SWAP_EFFECT_DISCARD,
|
||
|
Flags: 0,
|
||
|
};
|
||
|
|
||
|
|
||
|
let mut swap_chain = None;
|
||
|
unsafe {
|
||
|
fac.CreateSwapChain(&*device, &swapchain_desc, &mut swap_chain)
|
||
|
.ok()?;
|
||
|
}
|
||
|
|
||
|
Ok(swap_chain.expect("[dx11] swapchain creation failed."))
|
||
|
}
|
||
|
|
||
|
fn compile_shader(source: &[u8], entry: &[u8], version: &[u8]) -> Result<ID3DBlob>{
|
||
|
unsafe {
|
||
|
let mut blob = None;
|
||
|
D3DCompile(source.as_ptr().cast(), source.len(),
|
||
|
None, None, None,
|
||
|
PCSTR(entry.as_ptr()),
|
||
|
PCSTR(version.as_ptr()), D3DCOMPILE_DEBUG | D3DCOMPILE_SKIP_OPTIMIZATION, 0, &mut blob, None)?;
|
||
|
|
||
|
Ok(blob.unwrap())
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn main() -> Result<()> {
|
||
|
run_sample::<d3d11_hello_triangle::Sample>()?;
|
||
|
|
||
|
Ok(())
|
||
|
}
|