2019-03-28 10:35:46 +00:00
|
|
|
// Copyright 2019 Dolphin Emulator Project
|
2021-07-05 01:22:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2019-03-28 10:35:46 +00:00
|
|
|
|
|
|
|
#pragma once
|
2019-05-30 07:07:40 +00:00
|
|
|
|
2019-03-28 10:35:46 +00:00
|
|
|
#include <d3d12.h>
|
2020-09-15 13:07:22 +00:00
|
|
|
#include "VideoBackends/D3D12/DescriptorAllocator.h"
|
2019-03-28 10:35:46 +00:00
|
|
|
#include "VideoBackends/D3D12/DescriptorHeapManager.h"
|
|
|
|
#include "VideoCommon/RenderBase.h"
|
|
|
|
|
2021-06-09 11:42:21 +00:00
|
|
|
class BoundingBox;
|
|
|
|
|
2019-03-28 10:35:46 +00:00
|
|
|
namespace DX12
|
|
|
|
{
|
|
|
|
class DXFramebuffer;
|
|
|
|
class DXTexture;
|
|
|
|
class DXShader;
|
|
|
|
class DXPipeline;
|
|
|
|
class SwapChain;
|
|
|
|
|
|
|
|
class Renderer final : public ::Renderer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Renderer(std::unique_ptr<SwapChain> swap_chain, float backbuffer_scale);
|
|
|
|
~Renderer() override;
|
|
|
|
|
|
|
|
static Renderer* GetInstance() { return static_cast<Renderer*>(g_renderer.get()); }
|
|
|
|
|
|
|
|
bool IsHeadless() const override;
|
|
|
|
|
|
|
|
bool Initialize() override;
|
|
|
|
void Shutdown() override;
|
|
|
|
|
2021-08-28 05:30:05 +00:00
|
|
|
std::unique_ptr<AbstractTexture> CreateTexture(const TextureConfig& config,
|
|
|
|
std::string_view name) override;
|
2019-03-28 10:35:46 +00:00
|
|
|
std::unique_ptr<AbstractStagingTexture>
|
|
|
|
CreateStagingTexture(StagingTextureType type, const TextureConfig& config) override;
|
|
|
|
std::unique_ptr<AbstractFramebuffer>
|
|
|
|
CreateFramebuffer(AbstractTexture* color_attachment, AbstractTexture* depth_attachment) override;
|
|
|
|
|
2021-08-28 05:30:05 +00:00
|
|
|
std::unique_ptr<AbstractShader> CreateShaderFromSource(ShaderStage stage, std::string_view source,
|
|
|
|
std::string_view name) override;
|
2019-03-28 10:35:46 +00:00
|
|
|
std::unique_ptr<AbstractShader> CreateShaderFromBinary(ShaderStage stage, const void* data,
|
2021-08-28 05:30:05 +00:00
|
|
|
size_t length,
|
|
|
|
std::string_view name) override;
|
2019-03-28 10:35:46 +00:00
|
|
|
std::unique_ptr<NativeVertexFormat>
|
|
|
|
CreateNativeVertexFormat(const PortableVertexDeclaration& vtx_decl) override;
|
2019-04-15 11:55:26 +00:00
|
|
|
std::unique_ptr<AbstractPipeline> CreatePipeline(const AbstractPipelineConfig& config,
|
|
|
|
const void* cache_data = nullptr,
|
|
|
|
size_t cache_data_length = 0) override;
|
2019-03-28 10:35:46 +00:00
|
|
|
|
|
|
|
void Flush() override;
|
|
|
|
void WaitForGPUIdle() override;
|
|
|
|
|
2019-04-15 14:47:46 +00:00
|
|
|
void ClearScreen(const MathUtil::Rectangle<int>& rc, bool color_enable, bool alpha_enable,
|
|
|
|
bool z_enable, u32 color, u32 z) override;
|
2019-03-28 10:35:46 +00:00
|
|
|
|
|
|
|
void SetPipeline(const AbstractPipeline* pipeline) override;
|
|
|
|
void SetFramebuffer(AbstractFramebuffer* framebuffer) override;
|
|
|
|
void SetAndDiscardFramebuffer(AbstractFramebuffer* framebuffer) override;
|
|
|
|
void SetAndClearFramebuffer(AbstractFramebuffer* framebuffer, const ClearColor& color_value = {},
|
|
|
|
float depth_value = 0.0f) override;
|
|
|
|
void SetScissorRect(const MathUtil::Rectangle<int>& rc) override;
|
|
|
|
void SetTexture(u32 index, const AbstractTexture* texture) override;
|
|
|
|
void SetSamplerState(u32 index, const SamplerState& state) override;
|
|
|
|
void SetComputeImageTexture(AbstractTexture* texture, bool read, bool write) override;
|
|
|
|
void UnbindTexture(const AbstractTexture* texture) override;
|
|
|
|
void SetViewport(float x, float y, float width, float height, float near_depth,
|
|
|
|
float far_depth) override;
|
|
|
|
void Draw(u32 base_vertex, u32 num_vertices) override;
|
|
|
|
void DrawIndexed(u32 base_index, u32 num_indices, u32 base_vertex) override;
|
2022-06-01 09:58:13 +00:00
|
|
|
void DispatchComputeShader(const AbstractShader* shader, u32 groupsize_x, u32 groupsize_y,
|
|
|
|
u32 groupsize_z, u32 groups_x, u32 groups_y, u32 groups_z) override;
|
2019-03-28 10:35:46 +00:00
|
|
|
void BindBackbuffer(const ClearColor& clear_color = {}) override;
|
|
|
|
void PresentBackbuffer() override;
|
|
|
|
|
|
|
|
// Completes the current render pass, executes the command buffer, and restores state ready for
|
|
|
|
// next render. Use when you want to kick the current buffer to make room for new data.
|
|
|
|
void ExecuteCommandList(bool wait_for_completion);
|
|
|
|
|
|
|
|
// Setting constant buffer handles.
|
|
|
|
void SetConstantBuffer(u32 index, D3D12_GPU_VIRTUAL_ADDRESS address);
|
|
|
|
|
|
|
|
// Setting textures via descriptor handles. This is assumed to be in the shadow heap.
|
|
|
|
void SetTextureDescriptor(u32 index, D3D12_CPU_DESCRIPTOR_HANDLE handle);
|
|
|
|
|
|
|
|
// Pixel shader UAV.
|
|
|
|
void SetPixelShaderUAV(D3D12_CPU_DESCRIPTOR_HANDLE handle);
|
|
|
|
|
|
|
|
// Graphics vertex/index buffer binding.
|
2022-06-20 07:55:49 +00:00
|
|
|
void SetVertexBuffer(D3D12_GPU_VIRTUAL_ADDRESS address, D3D12_CPU_DESCRIPTOR_HANDLE srv,
|
|
|
|
u32 stride, u32 size);
|
2019-03-28 10:35:46 +00:00
|
|
|
void SetIndexBuffer(D3D12_GPU_VIRTUAL_ADDRESS address, u32 size, DXGI_FORMAT format);
|
|
|
|
|
2019-04-01 10:49:24 +00:00
|
|
|
// Binds all dirty state
|
|
|
|
bool ApplyState();
|
|
|
|
|
2019-03-28 10:35:46 +00:00
|
|
|
protected:
|
|
|
|
void OnConfigChanged(u32 bits) override;
|
|
|
|
|
2021-06-09 11:42:21 +00:00
|
|
|
std::unique_ptr<BoundingBox> CreateBoundingBox() const override;
|
|
|
|
|
2019-03-28 10:35:46 +00:00
|
|
|
private:
|
|
|
|
static const u32 MAX_TEXTURES = 8;
|
|
|
|
static const u32 NUM_CONSTANT_BUFFERS = 3;
|
|
|
|
|
|
|
|
// Dirty bits
|
|
|
|
enum DirtyStates
|
|
|
|
{
|
|
|
|
DirtyState_Framebuffer = (1 << 0),
|
|
|
|
DirtyState_Pipeline = (1 << 1),
|
|
|
|
DirtyState_Textures = (1 << 2),
|
|
|
|
DirtyState_Samplers = (1 << 3),
|
|
|
|
DirtyState_Viewport = (1 << 4),
|
|
|
|
DirtyState_ScissorRect = (1 << 5),
|
|
|
|
DirtyState_ComputeImageTexture = (1 << 6),
|
|
|
|
DirtyState_PS_UAV = (1 << 7),
|
|
|
|
DirtyState_PS_CBV = (1 << 8),
|
|
|
|
DirtyState_VS_CBV = (1 << 9),
|
|
|
|
DirtyState_GS_CBV = (1 << 10),
|
|
|
|
DirtyState_SRV_Descriptor = (1 << 11),
|
|
|
|
DirtyState_Sampler_Descriptor = (1 << 12),
|
|
|
|
DirtyState_UAV_Descriptor = (1 << 13),
|
|
|
|
DirtyState_VertexBuffer = (1 << 14),
|
|
|
|
DirtyState_IndexBuffer = (1 << 15),
|
|
|
|
DirtyState_PrimitiveTopology = (1 << 16),
|
|
|
|
DirtyState_RootSignature = (1 << 17),
|
|
|
|
DirtyState_ComputeRootSignature = (1 << 18),
|
|
|
|
DirtyState_DescriptorHeaps = (1 << 19),
|
2022-06-20 07:55:49 +00:00
|
|
|
DirtyState_VS_SRV = (1 << 20),
|
|
|
|
DirtyState_VS_SRV_Descriptor = (1 << 21),
|
2019-03-28 10:35:46 +00:00
|
|
|
|
|
|
|
DirtyState_All =
|
|
|
|
DirtyState_Framebuffer | DirtyState_Pipeline | DirtyState_Textures | DirtyState_Samplers |
|
|
|
|
DirtyState_Viewport | DirtyState_ScissorRect | DirtyState_ComputeImageTexture |
|
|
|
|
DirtyState_PS_UAV | DirtyState_PS_CBV | DirtyState_VS_CBV | DirtyState_GS_CBV |
|
|
|
|
DirtyState_SRV_Descriptor | DirtyState_Sampler_Descriptor | DirtyState_UAV_Descriptor |
|
|
|
|
DirtyState_VertexBuffer | DirtyState_IndexBuffer | DirtyState_PrimitiveTopology |
|
2022-06-20 07:55:49 +00:00
|
|
|
DirtyState_RootSignature | DirtyState_ComputeRootSignature | DirtyState_DescriptorHeaps |
|
|
|
|
DirtyState_VS_SRV | DirtyState_VS_SRV_Descriptor
|
2019-03-28 10:35:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void CheckForSwapChainChanges();
|
|
|
|
|
|
|
|
void BindFramebuffer(DXFramebuffer* fb);
|
|
|
|
void SetRootSignatures();
|
|
|
|
void SetDescriptorHeaps();
|
|
|
|
void UpdateDescriptorTables();
|
|
|
|
bool UpdateSRVDescriptorTable();
|
|
|
|
bool UpdateUAVDescriptorTable();
|
2022-06-20 07:55:49 +00:00
|
|
|
bool UpdateVSSRVDescriptorTable();
|
2019-03-28 10:35:46 +00:00
|
|
|
bool UpdateComputeUAVDescriptorTable();
|
|
|
|
bool UpdateSamplerDescriptorTable();
|
|
|
|
|
|
|
|
// Owned objects
|
|
|
|
std::unique_ptr<SwapChain> m_swap_chain;
|
|
|
|
|
|
|
|
// Current state
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
ID3D12RootSignature* root_signature = nullptr;
|
|
|
|
DXShader* compute_shader = nullptr;
|
|
|
|
std::array<D3D12_GPU_VIRTUAL_ADDRESS, 3> constant_buffers = {};
|
|
|
|
std::array<D3D12_CPU_DESCRIPTOR_HANDLE, MAX_TEXTURES> textures = {};
|
2022-06-20 07:55:49 +00:00
|
|
|
D3D12_CPU_DESCRIPTOR_HANDLE vs_srv = {};
|
2019-03-28 10:35:46 +00:00
|
|
|
D3D12_CPU_DESCRIPTOR_HANDLE ps_uav = {};
|
|
|
|
SamplerStateSet samplers = {};
|
|
|
|
const DXTexture* compute_image_texture = nullptr;
|
|
|
|
D3D12_VIEWPORT viewport = {};
|
|
|
|
D3D12_RECT scissor = {};
|
2022-06-20 07:55:49 +00:00
|
|
|
D3D12_GPU_DESCRIPTOR_HANDLE vertex_srv_descriptor_base = {};
|
2019-03-28 10:35:46 +00:00
|
|
|
D3D12_GPU_DESCRIPTOR_HANDLE srv_descriptor_base = {};
|
|
|
|
D3D12_GPU_DESCRIPTOR_HANDLE sampler_descriptor_base = {};
|
|
|
|
D3D12_GPU_DESCRIPTOR_HANDLE uav_descriptor_base = {};
|
|
|
|
D3D12_GPU_DESCRIPTOR_HANDLE compute_uav_descriptor_base = {};
|
|
|
|
D3D12_VERTEX_BUFFER_VIEW vertex_buffer = {};
|
|
|
|
D3D12_INDEX_BUFFER_VIEW index_buffer = {};
|
|
|
|
D3D12_PRIMITIVE_TOPOLOGY primitive_topology = D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST;
|
|
|
|
bool using_integer_rtv = false;
|
|
|
|
} m_state;
|
|
|
|
u32 m_dirty_bits = DirtyState_All;
|
|
|
|
};
|
|
|
|
} // namespace DX12
|