mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 15:31:17 +01:00
56afebeb44
ScaleRectangleFromTexture, which does a draw, and CopyRectangleFromTexture, which where possible, does a bit-for-bit copy.
364 lines
13 KiB
C++
364 lines
13 KiB
C++
// Copyright 2017 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <algorithm>
|
|
#include <cstddef>
|
|
|
|
#include "Common/Assert.h"
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/Logging/Log.h"
|
|
|
|
#include "VideoBackends/D3D/D3DBase.h"
|
|
#include "VideoBackends/D3D/D3DState.h"
|
|
#include "VideoBackends/D3D/D3DTexture.h"
|
|
#include "VideoBackends/D3D/D3DUtil.h"
|
|
#include "VideoBackends/D3D/DXTexture.h"
|
|
#include "VideoBackends/D3D/FramebufferManager.h"
|
|
#include "VideoBackends/D3D/GeometryShaderCache.h"
|
|
#include "VideoBackends/D3D/PixelShaderCache.h"
|
|
#include "VideoBackends/D3D/TextureCache.h"
|
|
#include "VideoBackends/D3D/VertexShaderCache.h"
|
|
|
|
#include "VideoCommon/ImageWrite.h"
|
|
#include "VideoCommon/TextureConfig.h"
|
|
|
|
namespace DX11
|
|
{
|
|
namespace
|
|
{
|
|
DXGI_FORMAT GetDXGIFormatForHostFormat(AbstractTextureFormat format)
|
|
{
|
|
switch (format)
|
|
{
|
|
case AbstractTextureFormat::DXT1:
|
|
return DXGI_FORMAT_BC1_UNORM;
|
|
case AbstractTextureFormat::DXT3:
|
|
return DXGI_FORMAT_BC2_UNORM;
|
|
case AbstractTextureFormat::DXT5:
|
|
return DXGI_FORMAT_BC3_UNORM;
|
|
case AbstractTextureFormat::BPTC:
|
|
return DXGI_FORMAT_BC7_UNORM;
|
|
case AbstractTextureFormat::RGBA8:
|
|
return DXGI_FORMAT_R8G8B8A8_UNORM;
|
|
case AbstractTextureFormat::BGRA8:
|
|
return DXGI_FORMAT_B8G8R8A8_UNORM;
|
|
default:
|
|
PanicAlert("Unhandled texture format.");
|
|
return DXGI_FORMAT_R8G8B8A8_UNORM;
|
|
}
|
|
}
|
|
} // Anonymous namespace
|
|
|
|
DXTexture::DXTexture(const TextureConfig& tex_config) : AbstractTexture(tex_config)
|
|
{
|
|
DXGI_FORMAT dxgi_format = GetDXGIFormatForHostFormat(m_config.format);
|
|
if (m_config.rendertarget)
|
|
{
|
|
m_texture = D3DTexture2D::Create(
|
|
m_config.width, m_config.height,
|
|
(D3D11_BIND_FLAG)((int)D3D11_BIND_RENDER_TARGET | (int)D3D11_BIND_SHADER_RESOURCE),
|
|
D3D11_USAGE_DEFAULT, dxgi_format, 1, m_config.layers);
|
|
}
|
|
else
|
|
{
|
|
const D3D11_TEXTURE2D_DESC texdesc =
|
|
CD3D11_TEXTURE2D_DESC(dxgi_format, m_config.width, m_config.height, 1, m_config.levels,
|
|
D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_DEFAULT, 0);
|
|
|
|
ID3D11Texture2D* pTexture;
|
|
const HRESULT hr = D3D::device->CreateTexture2D(&texdesc, nullptr, &pTexture);
|
|
CHECK(SUCCEEDED(hr), "Create texture of the TextureCache");
|
|
|
|
m_texture = new D3DTexture2D(pTexture, D3D11_BIND_SHADER_RESOURCE);
|
|
|
|
// TODO: better debug names
|
|
D3D::SetDebugObjectName(m_texture->GetTex(), "a texture of the TextureCache");
|
|
D3D::SetDebugObjectName(m_texture->GetSRV(),
|
|
"shader resource view of a texture of the TextureCache");
|
|
|
|
SAFE_RELEASE(pTexture);
|
|
}
|
|
}
|
|
|
|
DXTexture::~DXTexture()
|
|
{
|
|
m_texture->Release();
|
|
SAFE_RELEASE(m_staging_texture);
|
|
}
|
|
|
|
D3DTexture2D* DXTexture::GetRawTexIdentifier() const
|
|
{
|
|
return m_texture;
|
|
}
|
|
|
|
void DXTexture::Bind(unsigned int stage)
|
|
{
|
|
D3D::stateman->SetTexture(stage, m_texture->GetSRV());
|
|
}
|
|
|
|
std::optional<AbstractTexture::RawTextureInfo> DXTexture::MapFullImpl()
|
|
{
|
|
CD3D11_TEXTURE2D_DESC staging_texture_desc(DXGI_FORMAT_R8G8B8A8_UNORM, m_config.width,
|
|
m_config.height, 1, 1, 0, D3D11_USAGE_STAGING,
|
|
D3D11_CPU_ACCESS_READ);
|
|
|
|
HRESULT hr = D3D::device->CreateTexture2D(&staging_texture_desc, nullptr, &m_staging_texture);
|
|
if (FAILED(hr))
|
|
{
|
|
WARN_LOG(VIDEO, "Failed to create texture dumping readback texture: %X", static_cast<u32>(hr));
|
|
return {};
|
|
}
|
|
|
|
// Copy the selected data to the staging texture
|
|
D3D::context->CopyResource(m_staging_texture, m_texture->GetTex());
|
|
|
|
// Map the staging texture to client memory, and encode it as a .png image.
|
|
D3D11_MAPPED_SUBRESOURCE map;
|
|
hr = D3D::context->Map(m_staging_texture, 0, D3D11_MAP_READ, 0, &map);
|
|
if (FAILED(hr))
|
|
{
|
|
WARN_LOG(VIDEO, "Failed to map texture dumping readback texture: %X", static_cast<u32>(hr));
|
|
return {};
|
|
}
|
|
|
|
return AbstractTexture::RawTextureInfo{reinterpret_cast<u8*>(map.pData), map.RowPitch,
|
|
m_config.width, m_config.height};
|
|
}
|
|
|
|
std::optional<AbstractTexture::RawTextureInfo> DXTexture::MapRegionImpl(u32 level, u32 x, u32 y,
|
|
u32 width, u32 height)
|
|
{
|
|
CD3D11_TEXTURE2D_DESC staging_texture_desc(DXGI_FORMAT_R8G8B8A8_UNORM, width, height, 1, 1, 0,
|
|
D3D11_USAGE_STAGING, D3D11_CPU_ACCESS_READ);
|
|
|
|
HRESULT hr = D3D::device->CreateTexture2D(&staging_texture_desc, nullptr, &m_staging_texture);
|
|
if (FAILED(hr))
|
|
{
|
|
WARN_LOG(VIDEO, "Failed to create texture dumping readback texture: %X", static_cast<u32>(hr));
|
|
return {};
|
|
}
|
|
|
|
// Copy the selected data to the staging texture
|
|
CD3D11_BOX src_box(x, y, 0, width, height, 1);
|
|
D3D::context->CopySubresourceRegion(m_staging_texture, 0, 0, 0, 0, m_texture->GetTex(),
|
|
D3D11CalcSubresource(level, 0, m_config.levels), &src_box);
|
|
|
|
// Map the staging texture to client memory, and encode it as a .png image.
|
|
D3D11_MAPPED_SUBRESOURCE map;
|
|
hr = D3D::context->Map(m_staging_texture, 0, D3D11_MAP_READ, 0, &map);
|
|
if (FAILED(hr))
|
|
{
|
|
WARN_LOG(VIDEO, "Failed to map texture dumping readback texture: %X", static_cast<u32>(hr));
|
|
return {};
|
|
}
|
|
|
|
return AbstractTexture::RawTextureInfo{reinterpret_cast<u8*>(map.pData), map.RowPitch,
|
|
m_config.width, m_config.height};
|
|
}
|
|
|
|
void DXTexture::Unmap()
|
|
{
|
|
if (!m_staging_texture)
|
|
return;
|
|
|
|
D3D::context->Unmap(m_staging_texture, 0);
|
|
}
|
|
|
|
void DXTexture::CopyRectangleFromTexture(const AbstractTexture* src,
|
|
const MathUtil::Rectangle<int>& src_rect, u32 src_layer,
|
|
u32 src_level, const MathUtil::Rectangle<int>& dst_rect,
|
|
u32 dst_layer, u32 dst_level)
|
|
{
|
|
const DXTexture* srcentry = static_cast<const DXTexture*>(src);
|
|
_assert_(src_rect.GetWidth() == dst_rect.GetWidth() &&
|
|
src_rect.GetHeight() == dst_rect.GetHeight());
|
|
|
|
D3D11_BOX src_box;
|
|
src_box.left = src_rect.left;
|
|
src_box.top = src_rect.top;
|
|
src_box.right = src_rect.right;
|
|
src_box.bottom = src_rect.bottom;
|
|
src_box.front = 0;
|
|
src_box.back = 1;
|
|
|
|
D3D::context->CopySubresourceRegion(
|
|
m_texture->GetTex(), D3D11CalcSubresource(dst_level, dst_layer, m_config.levels),
|
|
dst_rect.left, dst_rect.top, 0, srcentry->m_texture->GetTex(),
|
|
D3D11CalcSubresource(src_level, src_layer, srcentry->m_config.levels), &src_box);
|
|
}
|
|
|
|
void DXTexture::ScaleRectangleFromTexture(const AbstractTexture* source,
|
|
const MathUtil::Rectangle<int>& srcrect,
|
|
const MathUtil::Rectangle<int>& dstrect)
|
|
{
|
|
const DXTexture* srcentry = static_cast<const DXTexture*>(source);
|
|
_assert_(m_config.rendertarget);
|
|
|
|
g_renderer->ResetAPIState(); // reset any game specific settings
|
|
|
|
const D3D11_VIEWPORT vp = CD3D11_VIEWPORT(float(dstrect.left), float(dstrect.top),
|
|
float(dstrect.GetWidth()), float(dstrect.GetHeight()));
|
|
|
|
D3D::stateman->UnsetTexture(m_texture->GetSRV());
|
|
D3D::stateman->Apply();
|
|
|
|
D3D::context->OMSetRenderTargets(1, &m_texture->GetRTV(), nullptr);
|
|
D3D::context->RSSetViewports(1, &vp);
|
|
D3D::SetLinearCopySampler();
|
|
D3D11_RECT srcRC;
|
|
srcRC.left = srcrect.left;
|
|
srcRC.right = srcrect.right;
|
|
srcRC.top = srcrect.top;
|
|
srcRC.bottom = srcrect.bottom;
|
|
D3D::drawShadedTexQuad(srcentry->m_texture->GetSRV(), &srcRC, srcentry->m_config.width,
|
|
srcentry->m_config.height, PixelShaderCache::GetColorCopyProgram(false),
|
|
VertexShaderCache::GetSimpleVertexShader(),
|
|
VertexShaderCache::GetSimpleInputLayout(),
|
|
GeometryShaderCache::GetCopyGeometryShader(), 1.0, 0);
|
|
|
|
FramebufferManager::BindEFBRenderTarget();
|
|
g_renderer->RestoreAPIState();
|
|
}
|
|
|
|
void DXTexture::Load(u32 level, u32 width, u32 height, u32 row_length, const u8* buffer,
|
|
size_t buffer_size)
|
|
{
|
|
size_t src_pitch = CalculateStrideForFormat(m_config.format, row_length);
|
|
D3D::context->UpdateSubresource(m_texture->GetTex(), level, nullptr, buffer,
|
|
static_cast<UINT>(src_pitch), 0);
|
|
}
|
|
|
|
DXStagingTexture::DXStagingTexture(StagingTextureType type, const TextureConfig& config,
|
|
ID3D11Texture2D* tex)
|
|
: AbstractStagingTexture(type, config), m_tex(tex)
|
|
{
|
|
}
|
|
|
|
DXStagingTexture::~DXStagingTexture()
|
|
{
|
|
if (IsMapped())
|
|
DXStagingTexture::Unmap();
|
|
SAFE_RELEASE(m_tex);
|
|
}
|
|
|
|
std::unique_ptr<DXStagingTexture> DXStagingTexture::Create(StagingTextureType type,
|
|
const TextureConfig& config)
|
|
{
|
|
D3D11_USAGE usage;
|
|
UINT cpu_flags;
|
|
if (type == StagingTextureType::Readback)
|
|
{
|
|
usage = D3D11_USAGE_STAGING;
|
|
cpu_flags = D3D11_CPU_ACCESS_READ;
|
|
}
|
|
else if (type == StagingTextureType::Upload)
|
|
{
|
|
usage = D3D11_USAGE_DYNAMIC;
|
|
cpu_flags = D3D11_CPU_ACCESS_WRITE;
|
|
}
|
|
else
|
|
{
|
|
usage = D3D11_USAGE_STAGING;
|
|
cpu_flags = D3D11_CPU_ACCESS_READ | D3D11_CPU_ACCESS_WRITE;
|
|
}
|
|
|
|
CD3D11_TEXTURE2D_DESC desc(GetDXGIFormatForHostFormat(config.format), config.width, config.height,
|
|
1, 1, 0, usage, cpu_flags);
|
|
|
|
ID3D11Texture2D* texture;
|
|
HRESULT hr = D3D::device->CreateTexture2D(&desc, nullptr, &texture);
|
|
CHECK(SUCCEEDED(hr), "Create staging texture");
|
|
if (FAILED(hr))
|
|
return nullptr;
|
|
|
|
return std::unique_ptr<DXStagingTexture>(new DXStagingTexture(type, config, texture));
|
|
}
|
|
|
|
void DXStagingTexture::CopyFromTexture(const AbstractTexture* src,
|
|
const MathUtil::Rectangle<int>& src_rect, u32 src_layer,
|
|
u32 src_level, const MathUtil::Rectangle<int>& dst_rect)
|
|
{
|
|
_assert_(m_type == StagingTextureType::Readback);
|
|
_assert_(src_rect.GetWidth() == dst_rect.GetWidth() &&
|
|
src_rect.GetHeight() == dst_rect.GetHeight());
|
|
_assert_(src_rect.left >= 0 && static_cast<u32>(src_rect.right) <= src->GetConfig().width &&
|
|
src_rect.top >= 0 && static_cast<u32>(src_rect.bottom) <= src->GetConfig().height);
|
|
_assert_(dst_rect.left >= 0 && static_cast<u32>(dst_rect.right) <= m_config.width &&
|
|
dst_rect.top >= 0 && static_cast<u32>(dst_rect.bottom) <= m_config.height);
|
|
|
|
if (IsMapped())
|
|
DXStagingTexture::Unmap();
|
|
|
|
CD3D11_BOX src_box(src_rect.left, src_rect.top, 0, src_rect.right, src_rect.bottom, 1);
|
|
D3D::context->CopySubresourceRegion(
|
|
m_tex, 0, static_cast<u32>(dst_rect.left), static_cast<u32>(dst_rect.top), 0,
|
|
static_cast<const DXTexture*>(src)->GetRawTexIdentifier()->GetTex(),
|
|
D3D11CalcSubresource(src_level, src_layer, src->GetConfig().levels), &src_box);
|
|
|
|
m_needs_flush = true;
|
|
}
|
|
|
|
void DXStagingTexture::CopyToTexture(const MathUtil::Rectangle<int>& src_rect, AbstractTexture* dst,
|
|
const MathUtil::Rectangle<int>& dst_rect, u32 dst_layer,
|
|
u32 dst_level)
|
|
{
|
|
_assert_(m_type == StagingTextureType::Upload);
|
|
_assert_(src_rect.GetWidth() == dst_rect.GetWidth() &&
|
|
src_rect.GetHeight() == dst_rect.GetHeight());
|
|
_assert_(src_rect.left >= 0 && static_cast<u32>(src_rect.right) <= m_config.width &&
|
|
src_rect.top >= 0 && static_cast<u32>(src_rect.bottom) <= m_config.height);
|
|
_assert_(dst_rect.left >= 0 && static_cast<u32>(dst_rect.right) <= dst->GetConfig().width &&
|
|
dst_rect.top >= 0 && static_cast<u32>(dst_rect.bottom) <= dst->GetConfig().height);
|
|
|
|
if (IsMapped())
|
|
DXStagingTexture::Unmap();
|
|
|
|
CD3D11_BOX src_box(src_rect.left, src_rect.top, 0, src_rect.right, src_rect.bottom, 1);
|
|
D3D::context->CopySubresourceRegion(
|
|
static_cast<const DXTexture*>(dst)->GetRawTexIdentifier()->GetTex(),
|
|
D3D11CalcSubresource(dst_level, dst_layer, dst->GetConfig().levels),
|
|
static_cast<u32>(dst_rect.left), static_cast<u32>(dst_rect.top), 0, m_tex, 0, &src_box);
|
|
}
|
|
|
|
bool DXStagingTexture::Map()
|
|
{
|
|
if (m_map_pointer)
|
|
return true;
|
|
|
|
D3D11_MAP map_type;
|
|
if (m_type == StagingTextureType::Readback)
|
|
map_type = D3D11_MAP_READ;
|
|
else if (m_type == StagingTextureType::Upload)
|
|
map_type = D3D11_MAP_WRITE;
|
|
else
|
|
map_type = D3D11_MAP_READ_WRITE;
|
|
|
|
D3D11_MAPPED_SUBRESOURCE sr;
|
|
HRESULT hr = D3D::context->Map(m_tex, 0, map_type, 0, &sr);
|
|
CHECK(SUCCEEDED(hr), "Map readback texture");
|
|
if (FAILED(hr))
|
|
return false;
|
|
|
|
m_map_pointer = reinterpret_cast<char*>(sr.pData);
|
|
m_map_stride = sr.RowPitch;
|
|
return true;
|
|
}
|
|
|
|
void DXStagingTexture::Unmap()
|
|
{
|
|
if (!m_map_pointer)
|
|
return;
|
|
|
|
D3D::context->Unmap(m_tex, 0);
|
|
m_map_pointer = nullptr;
|
|
}
|
|
|
|
void DXStagingTexture::Flush()
|
|
{
|
|
// Flushing is handled by the API.
|
|
m_needs_flush = false;
|
|
}
|
|
|
|
} // namespace DX11
|