mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 23:41:19 +01:00
3ea2a1387c
Might or might not work for you, yet. Anyway, read the soon to be created forum thread on this plugin before asking any questions. Huge thanks to rodolfoosvaldobogado for helping me out in various areas. Also, thanks to everyone on IRC who supported me during development ;) git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@5680 8ced0084-cf51-0410-be5f-012b33b47a6e
315 lines
13 KiB
C++
315 lines
13 KiB
C++
// Copyright (C) 2003 Dolphin Project.
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, version 2.0.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
#include "D3DBase.h"
|
|
#include "D3DTexture.h"
|
|
#include "D3DUtil.h"
|
|
#include "Render.h"
|
|
#include "FBManager.h"
|
|
#include "VideoConfig.h"
|
|
#include "PixelShaderCache.h"
|
|
#include "VertexShaderCache.h"
|
|
|
|
#undef CHECK
|
|
#define CHECK(cond, Message) if (!(cond)) { PanicAlert(__FUNCTION__ "Failed in %s at line %d: %s" , __FILE__, __LINE__, Message); }
|
|
|
|
FramebufferManager FBManager;
|
|
ID3D11SamplerState* copytoVirtualXFBsampler = NULL;
|
|
|
|
D3DTexture2D* &FramebufferManager::GetEFBColorTexture() { return m_efb.color_tex; }
|
|
ID3D11Texture2D* &FramebufferManager::GetEFBColorStagingBuffer() { return m_efb.color_staging_buf; }
|
|
|
|
D3DTexture2D* &FramebufferManager::GetEFBDepthTexture() { return m_efb.depth_tex; }
|
|
D3DTexture2D* &FramebufferManager::GetEFBDepthReadTexture() { return m_efb.depth_read_texture; }
|
|
ID3D11Texture2D* &FramebufferManager::GetEFBDepthStagingBuffer() { return m_efb.depth_staging_buf; }
|
|
|
|
void FramebufferManager::Create()
|
|
{
|
|
unsigned int target_width = Renderer::GetFullTargetWidth();
|
|
unsigned int target_height = Renderer::GetFullTargetHeight();
|
|
ID3D11Texture2D* buf;
|
|
D3D11_TEXTURE2D_DESC texdesc;
|
|
HRESULT hr;
|
|
|
|
// create framebuffer color texture
|
|
m_efb.color_tex = D3DTexture2D::Create(target_width, target_height, (D3D11_BIND_FLAG)(D3D11_BIND_RENDER_TARGET|D3D11_BIND_SHADER_RESOURCE), D3D11_USAGE_DEFAULT, DXGI_FORMAT_B8G8R8A8_UNORM);
|
|
CHECK(m_efb.color_tex != NULL, "create EFB color texture");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.color_tex->GetTex(), "EFB color texture");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.color_tex->GetRTV(), "EFB color texture render target view");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.color_tex->GetSRV(), "EFB color texture shader resource view");
|
|
|
|
// create a staging texture for Renderer::AccessEFB
|
|
texdesc = CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_B8G8R8A8_UNORM, 1, 1, 1, 1, 0, D3D11_USAGE_STAGING, D3D11_CPU_ACCESS_WRITE|D3D11_CPU_ACCESS_READ);
|
|
hr = D3D::device->CreateTexture2D(&texdesc, NULL, &m_efb.color_staging_buf);
|
|
CHECK(hr==S_OK, "create EFB color staging buffer");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.color_staging_buf, "EFB color staging texture (used for Renderer::AccessEFB)");
|
|
|
|
// EFB depth buffer
|
|
// TODO: Only bind as shader resource if EFB access enabled
|
|
texdesc = CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R24G8_TYPELESS, target_width, target_height, 1, 1, D3D11_BIND_DEPTH_STENCIL|D3D11_BIND_SHADER_RESOURCE);
|
|
hr = D3D::device->CreateTexture2D(&texdesc, NULL, &buf);
|
|
CHECK(hr==S_OK, "create EFB depth texture");
|
|
m_efb.depth_tex = new D3DTexture2D(buf, (D3D11_BIND_FLAG)(D3D11_BIND_DEPTH_STENCIL|D3D11_BIND_SHADER_RESOURCE), DXGI_FORMAT_R24_UNORM_X8_TYPELESS, DXGI_FORMAT_D24_UNORM_S8_UINT);
|
|
buf->Release();
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.depth_tex->GetTex(), "EFB depth texture");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.depth_tex->GetDSV(), "EFB depth texture depth stencil view");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.depth_tex->GetSRV(), "EFB depth texture shader resource view");
|
|
|
|
// render target for depth buffer access in Renderer::AccessEFB
|
|
texdesc = CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R32_FLOAT, 4, 4, 1, 1, D3D11_BIND_RENDER_TARGET);
|
|
hr = D3D::device->CreateTexture2D(&texdesc, NULL, &buf);
|
|
CHECK(hr==S_OK, "create EFB depth read texture");
|
|
m_efb.depth_read_texture = new D3DTexture2D(buf, D3D11_BIND_RENDER_TARGET);
|
|
buf->Release();
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.depth_read_texture->GetTex(), "EFB depth read texture (used in Renderer::AccessEFB)");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.depth_read_texture->GetRTV(), "EFB depth read texture render target view (used in Renderer::AccessEFB)");
|
|
|
|
// staging texture to which we copy the data from m_efb.depth_read_texture
|
|
texdesc = CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R32_FLOAT, 4, 4, 1, 1, 0, D3D11_USAGE_STAGING, D3D11_CPU_ACCESS_READ|D3D11_CPU_ACCESS_WRITE);
|
|
hr = D3D::device->CreateTexture2D(&texdesc, NULL, &m_efb.depth_staging_buf);
|
|
CHECK(hr==S_OK, "create EFB depth staging buffer");
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_efb.depth_staging_buf, "EFB depth staging texture (used for Renderer::AccessEFB)");
|
|
|
|
// sampler state for FramebufferManager::copyToVirtualXFB
|
|
float border[4] = {0.f, 0.f, 0.f, 0.f};
|
|
D3D11_SAMPLER_DESC samplerdesc = CD3D11_SAMPLER_DESC(D3D11_FILTER_MIN_MAG_MIP_LINEAR, D3D11_TEXTURE_ADDRESS_CLAMP, D3D11_TEXTURE_ADDRESS_CLAMP, D3D11_TEXTURE_ADDRESS_CLAMP, 0.f, 1, D3D11_COMPARISON_ALWAYS, border, -D3D11_FLOAT32_MAX, D3D11_FLOAT32_MAX);
|
|
D3D::device->CreateSamplerState(&samplerdesc, ©toVirtualXFBsampler);
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)copytoVirtualXFBsampler, "sampler state used for FramebufferManager::copyToVirtualXFB");
|
|
}
|
|
|
|
void FramebufferManager::Destroy()
|
|
{
|
|
SAFE_RELEASE(m_efb.color_tex);
|
|
SAFE_RELEASE(m_efb.color_staging_buf);
|
|
SAFE_RELEASE(m_efb.depth_tex);
|
|
SAFE_RELEASE(m_efb.depth_staging_buf);
|
|
SAFE_RELEASE(m_efb.depth_read_texture);
|
|
|
|
SAFE_RELEASE(copytoVirtualXFBsampler);
|
|
|
|
for (VirtualXFBListType::iterator it = m_virtualXFBList.begin(); it != m_virtualXFBList.end(); ++it)
|
|
SAFE_RELEASE(it->xfbSource.tex);
|
|
|
|
m_virtualXFBList.clear();
|
|
SAFE_RELEASE(m_realXFBSource.tex);
|
|
}
|
|
|
|
void FramebufferManager::CopyToXFB(u32 xfbAddr, u32 fbWidth, u32 fbHeight, const EFBRectangle& sourceRc)
|
|
{
|
|
copyToVirtualXFB(xfbAddr, fbWidth, fbHeight, sourceRc);
|
|
}
|
|
|
|
const XFBSource** FramebufferManager::GetXFBSource(u32 xfbAddr, u32 fbWidth, u32 fbHeight, u32 &xfbCount)
|
|
{
|
|
return getVirtualXFBSource(xfbAddr, fbWidth, fbHeight, xfbCount);
|
|
}
|
|
|
|
FramebufferManager::VirtualXFBListType::iterator FramebufferManager::findVirtualXFB(u32 xfbAddr, u32 width, u32 height)
|
|
{
|
|
u32 srcLower = xfbAddr;
|
|
u32 srcUpper = xfbAddr + 2 * width * height;
|
|
|
|
VirtualXFBListType::iterator it;
|
|
for (it = m_virtualXFBList.begin(); it != m_virtualXFBList.end(); ++it)
|
|
{
|
|
u32 dstLower = it->xfbAddr;
|
|
u32 dstUpper = it->xfbAddr + 2 * it->xfbWidth * it->xfbHeight;
|
|
|
|
if (dstLower >= srcLower && dstUpper <= srcUpper)
|
|
return it;
|
|
}
|
|
|
|
// that address is not in the Virtual XFB list.
|
|
return m_virtualXFBList.end();
|
|
}
|
|
|
|
void FramebufferManager::replaceVirtualXFB()
|
|
{
|
|
VirtualXFBListType::iterator it = m_virtualXFBList.begin();
|
|
|
|
s32 srcLower = it->xfbAddr;
|
|
s32 srcUpper = it->xfbAddr + 2 * it->xfbWidth * it->xfbHeight;
|
|
s32 lineSize = 2 * it->xfbWidth;
|
|
|
|
it++;
|
|
|
|
while (it != m_virtualXFBList.end())
|
|
{
|
|
s32 dstLower = it->xfbAddr;
|
|
s32 dstUpper = it->xfbAddr + 2 * it->xfbWidth * it->xfbHeight;
|
|
|
|
if (dstLower >= srcLower && dstUpper <= srcUpper)
|
|
{
|
|
// invalidate the data
|
|
it->xfbAddr = 0;
|
|
it->xfbHeight = 0;
|
|
it->xfbWidth = 0;
|
|
}
|
|
else if (addrRangesOverlap(srcLower, srcUpper, dstLower, dstUpper))
|
|
{
|
|
s32 upperOverlap = (srcUpper - dstLower) / lineSize;
|
|
s32 lowerOverlap = (dstUpper - srcLower) / lineSize;
|
|
|
|
if (upperOverlap > 0 && lowerOverlap < 0)
|
|
{
|
|
it->xfbAddr += lineSize * upperOverlap;
|
|
it->xfbHeight -= upperOverlap;
|
|
}
|
|
else if (lowerOverlap > 0)
|
|
{
|
|
it->xfbHeight -= lowerOverlap;
|
|
}
|
|
}
|
|
|
|
it++;
|
|
}
|
|
}
|
|
|
|
void FramebufferManager::copyToRealXFB(u32 xfbAddr, u32 fbWidth, u32 fbHeight, const EFBRectangle& sourceRc)
|
|
{
|
|
// TODO
|
|
PanicAlert("copyToRealXFB not implemented, yet\n");
|
|
}
|
|
|
|
void FramebufferManager::copyToVirtualXFB(u32 xfbAddr, u32 fbWidth, u32 fbHeight, const EFBRectangle& sourceRc)
|
|
{
|
|
D3DTexture2D* xfbTex;
|
|
HRESULT hr = 0;
|
|
|
|
VirtualXFBListType::iterator it = findVirtualXFB(xfbAddr, fbWidth, fbHeight);
|
|
|
|
if (it == m_virtualXFBList.end() && (int)m_virtualXFBList.size() >= MAX_VIRTUAL_XFB)
|
|
{
|
|
PanicAlert("Requested creating a new virtual XFB although the maximum number has already been reached! Report this to the devs");
|
|
return;
|
|
}
|
|
|
|
float scaleX = Renderer::GetTargetScaleX();
|
|
float scaleY = Renderer::GetTargetScaleY();
|
|
TargetRectangle targetSource,efbSource;
|
|
efbSource = Renderer::ConvertEFBRectangle(sourceRc);
|
|
targetSource.top = (int)(sourceRc.top * scaleY);
|
|
targetSource.bottom = (int)(sourceRc.bottom * scaleY);
|
|
targetSource.left = (int)(sourceRc.left * scaleX);
|
|
targetSource.right = (int)(sourceRc.right * scaleX);
|
|
unsigned int target_width = targetSource.right - targetSource.left;
|
|
unsigned int target_height = targetSource.bottom - targetSource.top;
|
|
if (it != m_virtualXFBList.end()) // overwrite an existing Virtual XFB
|
|
{
|
|
it->xfbAddr = xfbAddr;
|
|
it->xfbWidth = fbWidth;
|
|
it->xfbHeight = fbHeight;
|
|
|
|
it->xfbSource.srcAddr = xfbAddr;
|
|
it->xfbSource.srcWidth = fbWidth;
|
|
it->xfbSource.srcHeight = fbHeight;
|
|
|
|
if (it->xfbSource.texWidth != target_width || it->xfbSource.texHeight != target_height || !(it->xfbSource.tex))
|
|
{
|
|
SAFE_RELEASE(it->xfbSource.tex);
|
|
it->xfbSource.tex = D3DTexture2D::Create(target_width, target_height, (D3D11_BIND_FLAG)(D3D11_BIND_RENDER_TARGET|D3D11_BIND_SHADER_RESOURCE), D3D11_USAGE_DEFAULT, DXGI_FORMAT_B8G8R8A8_UNORM);
|
|
if (it->xfbSource.tex == NULL) PanicAlert("Failed to create XFB texture\n");
|
|
}
|
|
xfbTex = it->xfbSource.tex;
|
|
|
|
it->xfbSource.texWidth = target_width;
|
|
it->xfbSource.texHeight = target_height;
|
|
|
|
// move this Virtual XFB to the front of the list.
|
|
m_virtualXFBList.splice(m_virtualXFBList.begin(), m_virtualXFBList, it);
|
|
|
|
// keep stale XFB data from being used
|
|
replaceVirtualXFB();
|
|
}
|
|
else // create a new Virtual XFB and place it at the front of the list
|
|
{
|
|
VirtualXFB newVirt;
|
|
|
|
newVirt.xfbSource.tex = D3DTexture2D::Create(target_width, target_height, (D3D11_BIND_FLAG)(D3D11_BIND_RENDER_TARGET|D3D11_BIND_SHADER_RESOURCE), D3D11_USAGE_DEFAULT, DXGI_FORMAT_B8G8R8A8_UNORM);
|
|
if (newVirt.xfbSource.tex == NULL) PanicAlert("Failed to create a new virtual XFB");
|
|
|
|
newVirt.xfbAddr = xfbAddr;
|
|
newVirt.xfbWidth = fbWidth;
|
|
newVirt.xfbHeight = fbHeight;
|
|
|
|
xfbTex = newVirt.xfbSource.tex;
|
|
newVirt.xfbSource.texWidth = target_width;
|
|
newVirt.xfbSource.texHeight = target_height;
|
|
|
|
// Add the new Virtual XFB to the list
|
|
if (m_virtualXFBList.size() >= MAX_VIRTUAL_XFB)
|
|
{
|
|
PanicAlert("Requested creating a new virtual XFB although the maximum number has already been reached! Report this to the devs");
|
|
newVirt.xfbSource.tex->Release();
|
|
return;
|
|
// TODO, possible alternative to failing: just delete the oldest virtual XFB:
|
|
// m_virtualXFBList.back().xfbSource.tex->Release();
|
|
// m_virtualXFBList.pop_back();
|
|
}
|
|
m_virtualXFBList.push_front(newVirt);
|
|
}
|
|
if (!xfbTex->GetRTV()) return;
|
|
|
|
Renderer::ResetAPIState(); // reset any game specific settings
|
|
|
|
// copy EFB data to XFB and restore render target again
|
|
D3D11_RECT sourcerect = CD3D11_RECT(efbSource.left, efbSource.top, efbSource.right, efbSource.bottom);
|
|
D3D11_VIEWPORT vp = CD3D11_VIEWPORT(0.f, 0.f, (float)target_width, (float)target_height);
|
|
D3D::context->RSSetViewports(1, &vp);
|
|
D3D::context->PSSetSamplers(0, 1, ©toVirtualXFBsampler);
|
|
D3D::context->OMSetRenderTargets(1, &xfbTex->GetRTV(), NULL);
|
|
D3D::drawShadedTexQuad(GetEFBColorTexture()->GetSRV(), &sourcerect,
|
|
Renderer::GetFullTargetWidth(), Renderer::GetFullTargetHeight(),
|
|
PixelShaderCache::GetColorCopyProgram(), VertexShaderCache::GetSimpleVertexShader(),
|
|
VertexShaderCache::GetSimpleInputLayout());
|
|
D3D::context->OMSetRenderTargets(1, &GetEFBColorTexture()->GetRTV(), GetEFBDepthTexture()->GetDSV());
|
|
Renderer::RestoreAPIState();
|
|
}
|
|
|
|
const XFBSource** FramebufferManager::getRealXFBSource(u32 xfbAddr, u32 fbWidth, u32 fbHeight, u32 &xfbCount)
|
|
{
|
|
PanicAlert("getRealXFBSource not implemented, yet\n");
|
|
return NULL;
|
|
}
|
|
|
|
const XFBSource** FramebufferManager::getVirtualXFBSource(u32 xfbAddr, u32 fbWidth, u32 fbHeight, u32 &xfbCount)
|
|
{
|
|
xfbCount = 0;
|
|
|
|
if (m_virtualXFBList.size() == 0) // no Virtual XFBs available
|
|
return NULL;
|
|
|
|
u32 srcLower = xfbAddr;
|
|
u32 srcUpper = xfbAddr + 2 * fbWidth * fbHeight;
|
|
|
|
VirtualXFBListType::iterator it;
|
|
for (it = m_virtualXFBList.end(); it != m_virtualXFBList.begin();)
|
|
{
|
|
--it;
|
|
|
|
u32 dstLower = it->xfbAddr;
|
|
u32 dstUpper = it->xfbAddr + 2 * it->xfbWidth * it->xfbHeight;
|
|
|
|
if (addrRangesOverlap(srcLower, srcUpper, dstLower, dstUpper))
|
|
{
|
|
m_overlappingXFBArray[xfbCount] = &(it->xfbSource);
|
|
xfbCount++;
|
|
}
|
|
}
|
|
return &m_overlappingXFBArray[0];
|
|
}
|