// Copyright 2010 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #include "VideoBackends/D3D/D3DGfx.h" #include #include #include #include #include #include #include #include #include "Common/Assert.h" #include "Common/CommonTypes.h" #include "Common/Logging/Log.h" #include "Common/MathUtil.h" #include "Core/Core.h" #include "VideoBackends/D3D/D3DBase.h" #include "VideoBackends/D3D/D3DBoundingBox.h" #include "VideoBackends/D3D/D3DState.h" #include "VideoBackends/D3D/D3DSwapChain.h" #include "VideoBackends/D3D/DXPipeline.h" #include "VideoBackends/D3D/DXShader.h" #include "VideoBackends/D3D/DXTexture.h" #include "VideoCommon/BPFunctions.h" #include "VideoCommon/FramebufferManager.h" #include "VideoCommon/PostProcessing.h" #include "VideoCommon/Present.h" #include "VideoCommon/RenderState.h" #include "VideoCommon/VideoConfig.h" #include "VideoCommon/XFMemory.h" namespace DX11 { Gfx::Gfx(std::unique_ptr swap_chain, float backbuffer_scale) : m_backbuffer_scale(backbuffer_scale), m_swap_chain(std::move(swap_chain)) { } Gfx::~Gfx() = default; bool Gfx::IsHeadless() const { return !m_swap_chain; } std::unique_ptr Gfx::CreateTexture(const TextureConfig& config, std::string_view name) { return DXTexture::Create(config, name); } std::unique_ptr Gfx::CreateStagingTexture(StagingTextureType type, const TextureConfig& config) { return DXStagingTexture::Create(type, config); } std::unique_ptr Gfx::CreateFramebuffer(AbstractTexture* color_attachment, AbstractTexture* depth_attachment, std::vector additional_color_attachments) { return DXFramebuffer::Create(static_cast(color_attachment), static_cast(depth_attachment), std::move(additional_color_attachments)); } std::unique_ptr Gfx::CreateShaderFromSource(ShaderStage stage, std::string_view source, std::string_view name) { auto bytecode = DXShader::CompileShader(D3D::feature_level, stage, source); if (!bytecode) return nullptr; return DXShader::CreateFromBytecode(stage, std::move(*bytecode), name); } std::unique_ptr Gfx::CreateShaderFromBinary(ShaderStage stage, const void* data, size_t length, std::string_view name) { return DXShader::CreateFromBytecode(stage, DXShader::CreateByteCode(data, length), name); } std::unique_ptr Gfx::CreatePipeline(const AbstractPipelineConfig& config, const void* cache_data, size_t cache_data_length) { return DXPipeline::Create(config); } void Gfx::SetPipeline(const AbstractPipeline* pipeline) { const DXPipeline* dx_pipeline = static_cast(pipeline); if (m_current_pipeline == dx_pipeline) return; if (dx_pipeline) { 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()); } else { // These will be destroyed at pipeline destruction. D3D::stateman->SetInputLayout(nullptr); D3D::stateman->SetVertexShader(nullptr); D3D::stateman->SetGeometryShader(nullptr); D3D::stateman->SetPixelShader(nullptr); } } void Gfx::SetScissorRect(const MathUtil::Rectangle& rc) { // 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); } void Gfx::SetViewport(float x, float y, float width, float height, float near_depth, float far_depth) { // TODO: Move to stateman const CD3D11_VIEWPORT vp(x, y, width, height, near_depth, far_depth); D3D::context->RSSetViewports(1, &vp); } void Gfx::Draw(u32 base_vertex, u32 num_vertices) { D3D::stateman->Apply(); D3D::context->Draw(num_vertices, base_vertex); } void Gfx::DrawIndexed(u32 base_index, u32 num_indices, u32 base_vertex) { D3D::stateman->Apply(); D3D::context->DrawIndexed(num_indices, base_index, base_vertex); } void Gfx::DispatchComputeShader(const AbstractShader* shader, u32 groupsize_x, u32 groupsize_y, u32 groupsize_z, u32 groups_x, u32 groups_y, u32 groups_z) { D3D::stateman->SetComputeShader(static_cast(shader)->GetD3DComputeShader()); D3D::stateman->SyncComputeBindings(); D3D::context->Dispatch(groups_x, groups_y, groups_z); } bool Gfx::BindBackbuffer(const ClearColor& clear_color) { CheckForSwapChainChanges(); SetAndClearFramebuffer(m_swap_chain->GetFramebuffer(), clear_color); return true; } void Gfx::PresentBackbuffer() { m_swap_chain->Present(); } void Gfx::OnConfigChanged(u32 bits) { AbstractGfx::OnConfigChanged(bits); // Quad-buffer changes require swap chain recreation. if (bits & CONFIG_CHANGE_BIT_STEREO_MODE && m_swap_chain) m_swap_chain->SetStereo(SwapChain::WantsStereo()); if (bits & CONFIG_CHANGE_BIT_HDR && m_swap_chain) m_swap_chain->SetHDR(SwapChain::WantsHDR()); } void Gfx::CheckForSwapChainChanges() { const bool surface_changed = g_presenter->SurfaceChangedTestAndClear(); const bool surface_resized = g_presenter->SurfaceResizedTestAndClear() || m_swap_chain->CheckForFullscreenChange(); if (!surface_changed && !surface_resized) return; if (surface_changed) { m_swap_chain->ChangeSurface(g_presenter->GetNewSurfaceHandle()); } else { m_swap_chain->ResizeSwapChain(); } g_presenter->SetBackbuffer(m_swap_chain->GetWidth(), m_swap_chain->GetHeight()); } void Gfx::SetFramebuffer(AbstractFramebuffer* framebuffer) { if (m_current_framebuffer == framebuffer) return; // We can't leave the framebuffer bound as a texture and a render target. DXFramebuffer* fb = static_cast(framebuffer); fb->Unbind(); D3D::stateman->SetFramebuffer(fb); m_current_framebuffer = fb; } void Gfx::SetAndDiscardFramebuffer(AbstractFramebuffer* framebuffer) { SetFramebuffer(framebuffer); } void Gfx::SetAndClearFramebuffer(AbstractFramebuffer* framebuffer, const ClearColor& color_value, float depth_value) { SetFramebuffer(framebuffer); D3D::stateman->Apply(); DXFramebuffer* fb = static_cast(framebuffer); fb->Clear(color_value, depth_value); } void Gfx::SetTexture(u32 index, const AbstractTexture* texture) { D3D::stateman->SetTexture(index, texture ? static_cast(texture)->GetD3DSRV() : nullptr); } void Gfx::SetSamplerState(u32 index, const SamplerState& state) { D3D::stateman->SetSampler(index, m_state_cache.Get(state)); } void Gfx::SetComputeImageTexture(u32 index, AbstractTexture* texture, bool read, bool write) { D3D::stateman->SetComputeUAV(index, texture ? static_cast(texture)->GetD3DUAV() : nullptr); } void Gfx::UnbindTexture(const AbstractTexture* texture) { if (D3D::stateman->UnsetTexture(static_cast(texture)->GetD3DSRV()) != 0) D3D::stateman->ApplyTextures(); } void Gfx::Flush() { D3D::context->Flush(); } void Gfx::WaitForGPUIdle() { // There is no glFinish() equivalent in D3D. D3D::context->Flush(); } void Gfx::SetFullscreen(bool enable_fullscreen) { if (m_swap_chain) m_swap_chain->SetFullscreen(enable_fullscreen); } bool Gfx::IsFullscreen() const { return m_swap_chain && m_swap_chain->GetFullscreen(); } SurfaceInfo Gfx::GetSurfaceInfo() const { return {m_swap_chain ? static_cast(m_swap_chain->GetWidth()) : 0, m_swap_chain ? static_cast(m_swap_chain->GetHeight()) : 0, m_backbuffer_scale, m_swap_chain ? m_swap_chain->GetFormat() : AbstractTextureFormat::Undefined}; } } // namespace DX11