2015-05-24 04:55:12 +00:00
|
|
|
// Copyright 2010 Dolphin Emulator Project
|
2021-07-05 01:22:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
#include "VideoBackends/D3D/D3DGfx.h"
|
2017-01-23 16:20:20 +00:00
|
|
|
|
2017-12-25 23:38:44 +00:00
|
|
|
#include <algorithm>
|
2017-04-09 20:15:57 +00:00
|
|
|
#include <array>
|
2013-10-26 09:55:41 +00:00
|
|
|
#include <cmath>
|
2017-04-09 21:27:20 +00:00
|
|
|
#include <cstring>
|
2015-12-22 23:47:20 +00:00
|
|
|
#include <memory>
|
2014-05-26 00:52:58 +00:00
|
|
|
#include <string>
|
2014-02-17 10:18:15 +00:00
|
|
|
#include <strsafe.h>
|
2017-04-09 19:05:24 +00:00
|
|
|
#include <tuple>
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2017-09-08 09:42:56 +00:00
|
|
|
#include "Common/Assert.h"
|
2016-01-17 21:54:31 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2017-02-01 15:56:13 +00:00
|
|
|
#include "Common/Logging/Log.h"
|
2015-05-06 22:37:58 +00:00
|
|
|
#include "Common/MathUtil.h"
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2014-02-19 11:14:09 +00:00
|
|
|
#include "Core/Core.h"
|
2011-01-25 16:43:08 +00:00
|
|
|
|
2014-02-17 10:18:15 +00:00
|
|
|
#include "VideoBackends/D3D/D3DBase.h"
|
2020-09-15 12:50:34 +00:00
|
|
|
#include "VideoBackends/D3D/D3DBoundingBox.h"
|
2014-06-23 06:11:07 +00:00
|
|
|
#include "VideoBackends/D3D/D3DState.h"
|
2020-09-15 12:50:34 +00:00
|
|
|
#include "VideoBackends/D3D/D3DSwapChain.h"
|
2017-09-08 09:42:56 +00:00
|
|
|
#include "VideoBackends/D3D/DXPipeline.h"
|
|
|
|
#include "VideoBackends/D3D/DXShader.h"
|
2017-05-29 22:02:09 +00:00
|
|
|
#include "VideoBackends/D3D/DXTexture.h"
|
2014-02-17 10:18:15 +00:00
|
|
|
|
|
|
|
#include "VideoCommon/BPFunctions.h"
|
2019-02-15 01:59:50 +00:00
|
|
|
#include "VideoCommon/FramebufferManager.h"
|
|
|
|
#include "VideoCommon/PostProcessing.h"
|
2023-01-27 04:03:15 +00:00
|
|
|
#include "VideoCommon/Present.h"
|
2017-04-29 14:54:22 +00:00
|
|
|
#include "VideoCommon/RenderState.h"
|
2017-09-03 02:30:34 +00:00
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2017-02-01 15:56:13 +00:00
|
|
|
#include "VideoCommon/XFMemory.h"
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2011-01-29 20:16:51 +00:00
|
|
|
namespace DX11
|
|
|
|
{
|
2023-01-29 10:58:32 +00:00
|
|
|
Gfx::Gfx(std::unique_ptr<SwapChain> swap_chain, float backbuffer_scale)
|
|
|
|
: m_backbuffer_scale(backbuffer_scale), m_swap_chain(std::move(swap_chain))
|
2017-11-21 09:53:38 +00:00
|
|
|
{
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
Gfx::~Gfx() = default;
|
2019-02-15 01:59:50 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
bool Gfx::IsHeadless() const
|
2018-10-03 13:03:13 +00:00
|
|
|
{
|
2019-03-09 13:31:36 +00:00
|
|
|
return !m_swap_chain;
|
2018-10-03 13:03:13 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
std::unique_ptr<AbstractTexture> Gfx::CreateTexture(const TextureConfig& config,
|
2023-01-31 04:29:16 +00:00
|
|
|
std::string_view name)
|
2017-09-30 06:25:36 +00:00
|
|
|
{
|
2021-08-28 05:30:05 +00:00
|
|
|
return DXTexture::Create(config, name);
|
2017-09-30 06:25:36 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
std::unique_ptr<AbstractStagingTexture> Gfx::CreateStagingTexture(StagingTextureType type,
|
2023-01-31 04:29:16 +00:00
|
|
|
const TextureConfig& config)
|
2017-10-21 14:49:40 +00:00
|
|
|
{
|
|
|
|
return DXStagingTexture::Create(type, config);
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
std::unique_ptr<AbstractFramebuffer> Gfx::CreateFramebuffer(AbstractTexture* color_attachment,
|
2023-01-31 04:29:16 +00:00
|
|
|
AbstractTexture* depth_attachment)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
return DXFramebuffer::Create(static_cast<DXTexture*>(color_attachment),
|
|
|
|
static_cast<DXTexture*>(depth_attachment));
|
2018-01-21 10:22:45 +00:00
|
|
|
}
|
|
|
|
|
2021-08-28 05:30:05 +00:00
|
|
|
std::unique_ptr<AbstractShader>
|
2023-01-29 10:58:32 +00:00
|
|
|
Gfx::CreateShaderFromSource(ShaderStage stage, std::string_view source, std::string_view name)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
2019-07-26 23:34:27 +00:00
|
|
|
auto bytecode = DXShader::CompileShader(D3D::feature_level, stage, source);
|
|
|
|
if (!bytecode)
|
2019-03-09 13:31:36 +00:00
|
|
|
return nullptr;
|
|
|
|
|
2021-08-28 05:30:05 +00:00
|
|
|
return DXShader::CreateFromBytecode(stage, std::move(*bytecode), name);
|
2017-09-08 09:42:56 +00:00
|
|
|
}
|
|
|
|
|
2023-01-31 04:29:16 +00:00
|
|
|
std::unique_ptr<AbstractShader> Gfx::CreateShaderFromBinary(ShaderStage stage, const void* data,
|
|
|
|
size_t length, std::string_view name)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
2021-08-28 05:30:05 +00:00
|
|
|
return DXShader::CreateFromBytecode(stage, DXShader::CreateByteCode(data, length), name);
|
2017-09-08 09:42:56 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
std::unique_ptr<AbstractPipeline> Gfx::CreatePipeline(const AbstractPipelineConfig& config,
|
2023-01-31 04:29:16 +00:00
|
|
|
const void* cache_data,
|
|
|
|
size_t cache_data_length)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
|
|
|
return DXPipeline::Create(config);
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetPipeline(const AbstractPipeline* pipeline)
|
2017-09-08 09:42:56 +00:00
|
|
|
{
|
|
|
|
const DXPipeline* dx_pipeline = static_cast<const DXPipeline*>(pipeline);
|
2019-02-15 01:59:50 +00:00
|
|
|
if (m_current_pipeline == dx_pipeline)
|
2018-02-24 13:13:00 +00:00
|
|
|
return;
|
2017-09-08 09:42:56 +00:00
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
if (dx_pipeline)
|
2016-05-01 14:29:36 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
D3D::stateman->SetRasterizerState(dx_pipeline->GetRasterizerState());
|
|
|
|
D3D::stateman->SetDepthState(dx_pipeline->GetDepthState());
|
|
|
|
D3D::stateman->SetBlendState(dx_pipeline->GetBlendState());
|
|
|
|
D3D::stateman->SetPrimitiveTopology(dx_pipeline->GetPrimitiveTopology());
|
|
|
|
D3D::stateman->SetInputLayout(dx_pipeline->GetInputLayout());
|
|
|
|
D3D::stateman->SetVertexShader(dx_pipeline->GetVertexShader());
|
|
|
|
D3D::stateman->SetGeometryShader(dx_pipeline->GetGeometryShader());
|
|
|
|
D3D::stateman->SetPixelShader(dx_pipeline->GetPixelShader());
|
|
|
|
D3D::stateman->SetIntegerRTV(dx_pipeline->UseLogicOp());
|
2016-05-01 14:29:36 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
// These will be destroyed at pipeline destruction.
|
|
|
|
D3D::stateman->SetInputLayout(nullptr);
|
|
|
|
D3D::stateman->SetVertexShader(nullptr);
|
|
|
|
D3D::stateman->SetGeometryShader(nullptr);
|
|
|
|
D3D::stateman->SetPixelShader(nullptr);
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
2015-12-19 05:45:21 +00:00
|
|
|
}
|
2010-10-22 19:40:05 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetScissorRect(const MathUtil::Rectangle<int>& rc)
|
2015-12-19 05:45:21 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
// TODO: Move to stateman
|
|
|
|
const CD3D11_RECT rect(rc.left, rc.top, std::max(rc.right, rc.left + 1),
|
|
|
|
std::max(rc.bottom, rc.top + 1));
|
|
|
|
D3D::context->RSSetScissorRects(1, &rect);
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetViewport(float x, float y, float width, float height, float near_depth,
|
2023-01-31 04:29:16 +00:00
|
|
|
float far_depth)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
// TODO: Move to stateman
|
|
|
|
const CD3D11_VIEWPORT vp(x, y, width, height, near_depth, far_depth);
|
2011-06-11 19:37:21 +00:00
|
|
|
D3D::context->RSSetViewports(1, &vp);
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
2010-09-28 02:15:02 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::Draw(u32 base_vertex, u32 num_vertices)
|
2018-11-27 07:16:53 +00:00
|
|
|
{
|
|
|
|
D3D::stateman->Apply();
|
|
|
|
D3D::context->Draw(num_vertices, base_vertex);
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::DrawIndexed(u32 base_index, u32 num_indices, u32 base_vertex)
|
2018-11-27 07:16:53 +00:00
|
|
|
{
|
|
|
|
D3D::stateman->Apply();
|
|
|
|
D3D::context->DrawIndexed(num_indices, base_index, base_vertex);
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::DispatchComputeShader(const AbstractShader* shader, u32 groupsize_x, u32 groupsize_y,
|
2023-01-31 04:29:16 +00:00
|
|
|
u32 groupsize_z, u32 groups_x, u32 groups_y, u32 groups_z)
|
2010-12-27 21:56:20 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
D3D::stateman->SetComputeShader(static_cast<const DXShader*>(shader)->GetD3DComputeShader());
|
|
|
|
D3D::stateman->SyncComputeBindings();
|
|
|
|
D3D::context->Dispatch(groups_x, groups_y, groups_z);
|
2010-12-27 21:56:20 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::BindBackbuffer(const ClearColor& clear_color)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2019-03-09 13:31:36 +00:00
|
|
|
CheckForSwapChainChanges();
|
|
|
|
SetAndClearFramebuffer(m_swap_chain->GetFramebuffer(), clear_color);
|
2018-11-28 04:30:47 +00:00
|
|
|
}
|
2016-06-24 08:43:46 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::PresentBackbuffer()
|
2018-11-28 04:30:47 +00:00
|
|
|
{
|
2019-03-09 13:31:36 +00:00
|
|
|
m_swap_chain->Present();
|
2018-11-28 04:30:47 +00:00
|
|
|
}
|
2016-06-24 08:43:46 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::OnConfigChanged(u32 bits)
|
2018-11-28 04:30:47 +00:00
|
|
|
{
|
2023-01-30 11:49:23 +00:00
|
|
|
AbstractGfx::OnConfigChanged(bits);
|
|
|
|
|
2019-03-09 13:31:36 +00:00
|
|
|
// Quad-buffer changes require swap chain recreation.
|
|
|
|
if (bits & CONFIG_CHANGE_BIT_STEREO_MODE && m_swap_chain)
|
|
|
|
m_swap_chain->SetStereo(SwapChain::WantsStereo());
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::CheckForSwapChainChanges()
|
2018-01-26 06:23:24 +00:00
|
|
|
{
|
2023-01-27 04:03:15 +00:00
|
|
|
const bool surface_changed = g_presenter->SurfaceChangedTestAndClear();
|
2019-03-09 13:31:36 +00:00
|
|
|
const bool surface_resized =
|
2023-01-27 04:03:15 +00:00
|
|
|
g_presenter->SurfaceResizedTestAndClear() || m_swap_chain->CheckForFullscreenChange();
|
2019-03-09 13:31:36 +00:00
|
|
|
if (!surface_changed && !surface_resized)
|
2018-01-26 06:23:24 +00:00
|
|
|
return;
|
|
|
|
|
2019-03-09 13:31:36 +00:00
|
|
|
if (surface_changed)
|
2018-01-26 06:23:24 +00:00
|
|
|
{
|
2023-01-27 04:03:15 +00:00
|
|
|
m_swap_chain->ChangeSurface(g_presenter->GetNewSurfaceHandle());
|
2018-01-26 06:23:24 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-03-09 13:31:36 +00:00
|
|
|
m_swap_chain->ResizeSwapChain();
|
2018-01-26 06:23:24 +00:00
|
|
|
}
|
2019-03-09 13:31:36 +00:00
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
g_presenter->SetBackbuffer(m_swap_chain->GetWidth(), m_swap_chain->GetHeight());
|
2018-01-26 06:23:24 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetFramebuffer(AbstractFramebuffer* framebuffer)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
if (m_current_framebuffer == framebuffer)
|
|
|
|
return;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
// We can't leave the framebuffer bound as a texture and a render target.
|
|
|
|
DXFramebuffer* fb = static_cast<DXFramebuffer*>(framebuffer);
|
|
|
|
if ((fb->GetColorAttachment() &&
|
|
|
|
D3D::stateman->UnsetTexture(
|
|
|
|
static_cast<DXTexture*>(fb->GetColorAttachment())->GetD3DSRV()) != 0) ||
|
|
|
|
(fb->GetDepthAttachment() &&
|
|
|
|
D3D::stateman->UnsetTexture(
|
|
|
|
static_cast<DXTexture*>(fb->GetDepthAttachment())->GetD3DSRV()) != 0))
|
|
|
|
{
|
|
|
|
D3D::stateman->ApplyTextures();
|
|
|
|
}
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2019-02-15 01:59:50 +00:00
|
|
|
D3D::stateman->SetFramebuffer(fb);
|
2018-01-21 10:22:45 +00:00
|
|
|
m_current_framebuffer = fb;
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetAndDiscardFramebuffer(AbstractFramebuffer* framebuffer)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
|
|
|
SetFramebuffer(framebuffer);
|
|
|
|
}
|
|
|
|
|
2023-01-31 04:29:16 +00:00
|
|
|
void Gfx::SetAndClearFramebuffer(AbstractFramebuffer* framebuffer, const ClearColor& color_value,
|
|
|
|
float depth_value)
|
2018-01-21 10:22:45 +00:00
|
|
|
{
|
|
|
|
SetFramebuffer(framebuffer);
|
2019-02-15 01:59:50 +00:00
|
|
|
D3D::stateman->Apply();
|
|
|
|
|
2018-01-21 10:22:45 +00:00
|
|
|
if (framebuffer->GetColorFormat() != AbstractTextureFormat::Undefined)
|
|
|
|
{
|
|
|
|
D3D::context->ClearRenderTargetView(
|
|
|
|
static_cast<const DXFramebuffer*>(framebuffer)->GetRTVArray()[0], color_value.data());
|
|
|
|
}
|
|
|
|
if (framebuffer->GetDepthFormat() != AbstractTextureFormat::Undefined)
|
|
|
|
{
|
|
|
|
D3D::context->ClearDepthStencilView(static_cast<const DXFramebuffer*>(framebuffer)->GetDSV(),
|
|
|
|
D3D11_CLEAR_DEPTH, depth_value, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetTexture(u32 index, const AbstractTexture* texture)
|
2018-01-21 13:13:25 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
D3D::stateman->SetTexture(index, texture ? static_cast<const DXTexture*>(texture)->GetD3DSRV() :
|
|
|
|
nullptr);
|
2018-01-21 13:13:25 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetSamplerState(u32 index, const SamplerState& state)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2018-02-24 15:15:35 +00:00
|
|
|
D3D::stateman->SetSampler(index, m_state_cache.Get(state));
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetComputeImageTexture(AbstractTexture* texture, bool read, bool write)
|
2018-01-21 13:13:25 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
D3D::stateman->SetComputeUAV(texture ? static_cast<DXTexture*>(texture)->GetD3DUAV() : nullptr);
|
2018-01-21 13:13:25 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::UnbindTexture(const AbstractTexture* texture)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2019-02-15 01:59:50 +00:00
|
|
|
if (D3D::stateman->UnsetTexture(static_cast<const DXTexture*>(texture)->GetD3DSRV()) != 0)
|
|
|
|
D3D::stateman->ApplyTextures();
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::Flush()
|
2019-02-15 01:59:50 +00:00
|
|
|
{
|
|
|
|
D3D::context->Flush();
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::WaitForGPUIdle()
|
2019-02-15 01:59:50 +00:00
|
|
|
{
|
|
|
|
// There is no glFinish() equivalent in D3D.
|
|
|
|
D3D::context->Flush();
|
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
void Gfx::SetFullscreen(bool enable_fullscreen)
|
2016-11-09 00:07:56 +00:00
|
|
|
{
|
2019-03-09 13:31:36 +00:00
|
|
|
if (m_swap_chain)
|
|
|
|
m_swap_chain->SetFullscreen(enable_fullscreen);
|
2016-11-09 00:07:56 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
bool Gfx::IsFullscreen() const
|
2016-11-09 00:41:38 +00:00
|
|
|
{
|
2019-03-09 13:31:36 +00:00
|
|
|
return m_swap_chain && m_swap_chain->GetFullscreen();
|
2016-11-09 00:41:38 +00:00
|
|
|
}
|
|
|
|
|
2023-01-29 10:58:32 +00:00
|
|
|
SurfaceInfo Gfx::GetSurfaceInfo() const
|
|
|
|
{
|
2023-01-31 04:29:16 +00:00
|
|
|
return {m_swap_chain ? static_cast<u32>(m_swap_chain->GetWidth()) : 0,
|
|
|
|
m_swap_chain ? static_cast<u32>(m_swap_chain->GetHeight()) : 0, m_backbuffer_scale,
|
|
|
|
m_swap_chain ? m_swap_chain->GetFormat() : AbstractTextureFormat::Undefined};
|
2023-01-29 10:58:32 +00:00
|
|
|
}
|
|
|
|
|
2012-01-06 12:45:51 +00:00
|
|
|
} // namespace DX11
|