2013-04-17 23:29:41 -04:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
2010-06-09 01:37:08 +00:00
|
|
|
|
|
|
|
#include "BPMemLoader.h"
|
|
|
|
#include "EfbCopy.h"
|
|
|
|
#include "EfbInterface.h"
|
2011-02-03 19:55:30 +00:00
|
|
|
#include "SWRenderer.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
#include "TextureEncoder.h"
|
2011-02-03 19:55:30 +00:00
|
|
|
#include "SWStatistics.h"
|
2011-01-29 06:26:03 +00:00
|
|
|
#include "SWVideoConfig.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
#include "DebugUtil.h"
|
|
|
|
#include "HwRasterizer.h"
|
2011-02-03 19:55:30 +00:00
|
|
|
#include "SWCommandProcessor.h"
|
2011-01-31 01:28:32 +00:00
|
|
|
#include "HW/Memmap.h"
|
|
|
|
#include "Core.h"
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2013-08-20 23:51:39 +12:00
|
|
|
static const float s_gammaLUT[] =
|
|
|
|
{
|
|
|
|
1.0f,
|
|
|
|
1.7f,
|
|
|
|
2.2f,
|
|
|
|
1.0f
|
|
|
|
};
|
|
|
|
|
2010-06-09 01:37:08 +00:00
|
|
|
namespace EfbCopy
|
|
|
|
{
|
2013-08-20 23:51:39 +12:00
|
|
|
void CopyToXfb(u32 xfbAddr, u32 fbWidth, u32 fbHeight, const EFBRectangle& sourceRc, float Gamma)
|
2013-04-13 23:54:02 -04:00
|
|
|
{
|
2011-01-29 04:52:19 +00:00
|
|
|
if (!g_SWVideoConfig.bHwRasterizer)
|
2013-04-13 23:54:02 -04:00
|
|
|
{
|
2013-08-20 23:51:39 +12:00
|
|
|
INFO_LOG(VIDEO, "xfbaddr: %x, fbwidth: %i, fbheight: %i, source: (%i, %i, %i, %i), Gamma %f",
|
|
|
|
xfbAddr, fbWidth, fbHeight, sourceRc.top, sourceRc.left, sourceRc.bottom, sourceRc.right, Gamma);
|
|
|
|
EfbInterface::yuv422_packed* xfb_in_ram = (EfbInterface::yuv422_packed *) Memory::GetPointer(xfbAddr);
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2013-08-20 23:51:39 +12:00
|
|
|
EfbInterface::CopyToXFB(xfb_in_ram, fbWidth, fbHeight, sourceRc, Gamma);
|
|
|
|
}
|
2013-04-13 23:54:02 -04:00
|
|
|
}
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2013-04-13 23:54:02 -04:00
|
|
|
void CopyToRam()
|
|
|
|
{
|
|
|
|
if (!g_SWVideoConfig.bHwRasterizer)
|
2010-12-02 05:38:48 +00:00
|
|
|
{
|
2011-01-31 01:28:32 +00:00
|
|
|
u8 *dest_ptr = Memory::GetPointer(bpmem.copyTexDest << 5);
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2010-12-02 05:38:48 +00:00
|
|
|
TextureEncoder::Encode(dest_ptr);
|
|
|
|
}
|
2013-04-13 23:54:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void ClearEfb()
|
|
|
|
{
|
|
|
|
u32 clearColor = (bpmem.clearcolorAR & 0xff) << 24 | bpmem.clearcolorGB << 8 | (bpmem.clearcolorAR & 0xff00) >> 8;
|
|
|
|
|
|
|
|
int left = bpmem.copyTexSrcXY.x;
|
|
|
|
int top = bpmem.copyTexSrcXY.y;
|
2013-08-20 23:51:39 +12:00
|
|
|
int right = left + bpmem.copyTexSrcWH.x + 1;
|
|
|
|
int bottom = top + bpmem.copyTexSrcWH.y + 1;
|
2013-04-13 23:54:02 -04:00
|
|
|
|
|
|
|
for (u16 y = top; y <= bottom; y++)
|
|
|
|
{
|
|
|
|
for (u16 x = left; x <= right; x++)
|
|
|
|
{
|
|
|
|
EfbInterface::SetColor(x, y, (u8*)(&clearColor));
|
|
|
|
EfbInterface::SetDepth(x, y, bpmem.clearZValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CopyEfb()
|
|
|
|
{
|
2013-08-20 23:51:39 +12:00
|
|
|
EFBRectangle rc;
|
|
|
|
rc.left = (int)bpmem.copyTexSrcXY.x;
|
|
|
|
rc.top = (int)bpmem.copyTexSrcXY.y;
|
|
|
|
|
|
|
|
// Here Width+1 like Height, otherwise some textures are corrupted already since the native resolution.
|
|
|
|
rc.right = (int)(bpmem.copyTexSrcXY.x + bpmem.copyTexSrcWH.x + 1);
|
|
|
|
rc.bottom = (int)(bpmem.copyTexSrcXY.y + bpmem.copyTexSrcWH.y + 1);
|
|
|
|
|
|
|
|
//if (bpmem.triggerEFBCopy.copy_to_xfb)
|
|
|
|
// DebugUtil::OnFrameEnd(); // FIXME: not actually frame end
|
2013-04-13 23:54:02 -04:00
|
|
|
|
|
|
|
if (!g_bSkipCurrentFrame)
|
|
|
|
{
|
|
|
|
if (bpmem.triggerEFBCopy.copy_to_xfb)
|
|
|
|
{
|
2013-08-20 23:51:39 +12:00
|
|
|
float yScale;
|
|
|
|
if (bpmem.triggerEFBCopy.scale_invert)
|
|
|
|
yScale = 256.0f / (float)bpmem.dispcopyyscale;
|
|
|
|
else
|
|
|
|
yScale = (float)bpmem.dispcopyyscale / 256.0f;
|
2013-04-13 23:54:02 -04:00
|
|
|
|
2013-08-20 23:51:39 +12:00
|
|
|
float xfbLines = ((bpmem.copyTexSrcWH.y + 1.0f) * yScale);
|
|
|
|
|
|
|
|
if (yScale != 1.0)
|
|
|
|
WARN_LOG(VIDEO, "yScale of %f is currently unsupported", yScale);
|
|
|
|
|
|
|
|
if ((u32)xfbLines > MAX_XFB_HEIGHT)
|
|
|
|
{
|
|
|
|
INFO_LOG(VIDEO, "Tried to scale EFB to too many XFB lines (%f)", xfbLines);
|
|
|
|
xfbLines = MAX_XFB_HEIGHT;
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyToXfb(bpmem.copyTexDest << 5,
|
|
|
|
bpmem.copyMipMapStrideChannels << 4,
|
|
|
|
(u32)xfbLines,
|
|
|
|
rc,
|
|
|
|
s_gammaLUT[bpmem.triggerEFBCopy.gamma]);
|
2013-04-13 23:54:02 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-08-20 23:51:39 +12:00
|
|
|
CopyToRam(); // FIXME: should use the rectangle we have already created above
|
2013-04-13 23:54:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bpmem.triggerEFBCopy.clear)
|
|
|
|
{
|
|
|
|
if (g_SWVideoConfig.bHwRasterizer)
|
|
|
|
HwRasterizer::Clear();
|
|
|
|
else
|
|
|
|
ClearEfb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 01:37:08 +00:00
|
|
|
}
|