From ddbc14317840d7d0afe5876c89d952d035917bb6 Mon Sep 17 00:00:00 2001 From: Connor McLaughlin Date: Sun, 12 Jun 2022 00:41:21 +1000 Subject: [PATCH] GS/DX12: Fix creating display textures mid-frame crashing --- common/D3D12/Texture.cpp | 24 +++++++++++++----------- common/D3D12/Texture.h | 8 ++++---- pcsx2/Frontend/D3D12HostDisplay.cpp | 13 +++++++------ pcsx2/Frontend/imgui_impl_dx12.cpp | 2 +- 4 files changed, 25 insertions(+), 22 deletions(-) diff --git a/common/D3D12/Texture.cpp b/common/D3D12/Texture.cpp index befb552c10..322eedf1cb 100644 --- a/common/D3D12/Texture.cpp +++ b/common/D3D12/Texture.cpp @@ -284,7 +284,7 @@ void Texture::TransitionSubresourceToState(ID3D12GraphicsCommandList* cmdlist, u cmdlist->ResourceBarrier(1, &barrier); } -bool Texture::BeginStreamUpdate(u32 level, u32 x, u32 y, u32 width, u32 height, void** out_data, u32* out_data_pitch) +ID3D12GraphicsCommandList* Texture::BeginStreamUpdate(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height, void** out_data, u32* out_data_pitch) { const u32 copy_pitch = Common::AlignUpPow2(width * GetTexelSize(m_format), D3D12_TEXTURE_DATA_PITCH_ALIGNMENT); const u32 upload_size = copy_pitch * height; @@ -297,16 +297,19 @@ bool Texture::BeginStreamUpdate(u32 level, u32 x, u32 y, u32 width, u32 height, if (!g_d3d12_context->GetTextureStreamBuffer().ReserveMemory(upload_size, D3D12_TEXTURE_DATA_PLACEMENT_ALIGNMENT)) { Console.Error("Failed to reserve %u bytes for %ux%u upload", upload_size, width, height); - return false; + return nullptr; } + + // cmdlist change + return g_d3d12_context->GetInitCommandList(); } *out_data = g_d3d12_context->GetTextureStreamBuffer().GetCurrentHostPointer(); *out_data_pitch = copy_pitch; - return true; + return cmdlist; } -void Texture::EndStreamUpdate(u32 level, u32 x, u32 y, u32 width, u32 height) +void Texture::EndStreamUpdate(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height) { const u32 copy_pitch = Common::AlignUpPow2(width * GetTexelSize(m_format), D3D12_TEXTURE_DATA_PITCH_ALIGNMENT); const u32 upload_size = copy_pitch * height; @@ -315,10 +318,10 @@ void Texture::EndStreamUpdate(u32 level, u32 x, u32 y, u32 width, u32 height) const u32 sb_offset = sb.GetCurrentOffset(); sb.CommitMemory(upload_size); - CopyFromBuffer(level, x, y, width, height, copy_pitch, sb.GetBuffer(), sb_offset); + CopyFromBuffer(cmdlist, level, x, y, width, height, copy_pitch, sb.GetBuffer(), sb_offset); } -void Texture::CopyFromBuffer(u32 level, u32 x, u32 y, u32 width, u32 height, u32 pitch, ID3D12Resource* buffer, u32 buffer_offset) +void Texture::CopyFromBuffer(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height, u32 pitch, ID3D12Resource* buffer, u32 buffer_offset) { D3D12_TEXTURE_COPY_LOCATION src; src.pResource = buffer; @@ -337,7 +340,6 @@ void Texture::CopyFromBuffer(u32 level, u32 x, u32 y, u32 width, u32 height, u32 const D3D12_BOX src_box{0u, 0u, 0u, width, height, 1u}; const D3D12_RESOURCE_STATES old_state = m_state; - ID3D12GraphicsCommandList* cmdlist = g_d3d12_context->GetCommandList(); TransitionToState(cmdlist, D3D12_RESOURCE_STATE_COPY_DEST); cmdlist->CopyTextureRegion(&dst, x, y, 0, &src, &src_box); TransitionToState(cmdlist, old_state); @@ -381,7 +383,7 @@ static ID3D12Resource* CreateStagingBuffer(u32 height, const void* data, u32 pit return resource.get(); } -bool Texture::LoadData(u32 level, u32 x, u32 y, u32 width, u32 height, const void* data, u32 pitch) +bool Texture::LoadData(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height, const void* data, u32 pitch) { const u32 texel_size = GetTexelSize(m_format); const u32 upload_pitch = Common::AlignUpPow2(width * texel_size, D3D12_TEXTURE_DATA_PITCH_ALIGNMENT); @@ -392,17 +394,17 @@ bool Texture::LoadData(u32 level, u32 x, u32 y, u32 width, u32 height, const voi if (!staging_buffer) return false; - CopyFromBuffer(level, x, y, width, height, upload_pitch, staging_buffer, 0); + CopyFromBuffer(cmdlist, level, x, y, width, height, upload_pitch, staging_buffer, 0); return true; } void* write_ptr; u32 write_pitch; - if (!BeginStreamUpdate(level, x, y, width, height, &write_ptr, &write_pitch)) + if (!(cmdlist = BeginStreamUpdate(cmdlist, level, x, y, width, height, &write_ptr, &write_pitch))) return false; StringUtil::StrideMemCpy(write_ptr, write_pitch, data, pitch, std::min(pitch, upload_pitch), height); - EndStreamUpdate(level, x, y, width, height); + EndStreamUpdate(cmdlist, level, x, y, width, height); return true; } diff --git a/common/D3D12/Texture.h b/common/D3D12/Texture.h index 9d0f05eb36..1e1a608110 100644 --- a/common/D3D12/Texture.h +++ b/common/D3D12/Texture.h @@ -74,10 +74,10 @@ namespace D3D12 Texture& operator=(Texture&& texture); // NOTE: Does not handle compressed formats. - bool BeginStreamUpdate(u32 level, u32 x, u32 y, u32 width, u32 height, void** out_data, u32* out_data_pitch); - void EndStreamUpdate(u32 level, u32 x, u32 y, u32 width, u32 height); - bool LoadData(u32 level, u32 x, u32 y, u32 width, u32 height, const void* data, u32 pitch); - void CopyFromBuffer(u32 level, u32 x, u32 y, u32 width, u32 height, u32 pitch, ID3D12Resource* buffer, u32 buffer_offset); + ID3D12GraphicsCommandList* BeginStreamUpdate(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height, void** out_data, u32* out_data_pitch); + void EndStreamUpdate(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height); + bool LoadData(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height, const void* data, u32 pitch); + void CopyFromBuffer(ID3D12GraphicsCommandList* cmdlist, u32 level, u32 x, u32 y, u32 width, u32 height, u32 pitch, ID3D12Resource* buffer, u32 buffer_offset); private: static bool CreateSRVDescriptor(ID3D12Resource* resource, u32 levels, DXGI_FORMAT format, DescriptorHandle* dh); diff --git a/pcsx2/Frontend/D3D12HostDisplay.cpp b/pcsx2/Frontend/D3D12HostDisplay.cpp index 31134b3d58..7aef99871d 100644 --- a/pcsx2/Frontend/D3D12HostDisplay.cpp +++ b/pcsx2/Frontend/D3D12HostDisplay.cpp @@ -85,7 +85,8 @@ bool D3D12HostDisplay::HasRenderSurface() const return static_cast(m_swap_chain); } -std::unique_ptr D3D12HostDisplay::CreateTexture(u32 width, u32 height, const void* data, u32 data_stride, bool dynamic /* = false */) +std::unique_ptr D3D12HostDisplay::CreateTexture( + u32 width, u32 height, const void* data, u32 data_stride, bool dynamic /* = false */) { D3D12::Texture tex; if (!tex.Create(width, height, 1, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_FORMAT_UNKNOWN, DXGI_FORMAT_UNKNOWN, @@ -94,17 +95,17 @@ std::unique_ptr D3D12HostDisplay::CreateTexture(u32 width, u return {}; } - if (data && !tex.LoadData(0, 0, 0, width, height, data, data_stride)) + if (data && !tex.LoadData(g_d3d12_context->GetInitCommandList(), 0, 0, 0, width, height, data, data_stride)) return {}; return std::make_unique(std::move(tex)); } -void D3D12HostDisplay::UpdateTexture(HostDisplayTexture* texture, u32 x, u32 y, u32 width, u32 height, - const void* texture_data, u32 texture_data_stride) +void D3D12HostDisplay::UpdateTexture( + HostDisplayTexture* texture, u32 x, u32 y, u32 width, u32 height, const void* texture_data, u32 texture_data_stride) { - static_cast(texture)->GetTexture().LoadData(0, x, y, width, height, texture_data, - texture_data_stride); + static_cast(texture)->GetTexture().LoadData( + g_d3d12_context->GetCommandList(), 0, x, y, width, height, texture_data, texture_data_stride); } bool D3D12HostDisplay::GetHostRefreshRate(float* refresh_rate) diff --git a/pcsx2/Frontend/imgui_impl_dx12.cpp b/pcsx2/Frontend/imgui_impl_dx12.cpp index 553d8f70bb..512900f39f 100644 --- a/pcsx2/Frontend/imgui_impl_dx12.cpp +++ b/pcsx2/Frontend/imgui_impl_dx12.cpp @@ -271,7 +271,7 @@ bool ImGui_ImplDX12_CreateFontsTexture() } } - if (!bd->FontTexture.LoadData(0, 0, 0, width, height, pixels, width * sizeof(u32))) + if (!bd->FontTexture.LoadData(g_d3d12_context->GetInitCommandList(), 0, 0, 0, width, height, pixels, width * sizeof(u32))) return false; io.Fonts->SetTexID((ImTextureID)&bd->FontTexture);