2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2009 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-17 23:09:55 -04:00
|
|
|
// Refer to the license.txt file included.
|
2009-07-06 02:10:26 +00:00
|
|
|
|
2019-07-16 20:18:48 -04:00
|
|
|
#include "VideoCommon/BPFunctions.h"
|
|
|
|
|
2017-12-26 00:38:44 +01:00
|
|
|
#include <algorithm>
|
2020-11-13 22:33:26 -05:00
|
|
|
#include <string_view>
|
2017-12-26 00:38:44 +01:00
|
|
|
|
2014-09-07 20:06:58 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "Common/Logging/Log.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
#include "VideoCommon/AbstractFramebuffer.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "VideoCommon/BPMemory.h"
|
2019-02-15 11:59:50 +10:00
|
|
|
#include "VideoCommon/FramebufferManager.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/RenderBase.h"
|
2017-04-30 01:00:45 +10:00
|
|
|
#include "VideoCommon/RenderState.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/VertexManagerBase.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2018-01-21 22:04:15 +10:00
|
|
|
#include "VideoCommon/XFMemory.h"
|
2009-07-06 02:10:26 +00:00
|
|
|
|
|
|
|
namespace BPFunctions
|
|
|
|
{
|
2010-09-28 02:15:02 +00:00
|
|
|
// ----------------------------------------------
|
|
|
|
// State translation lookup tables
|
2014-06-07 11:30:39 +09:00
|
|
|
// Reference: Yet Another GameCube Documentation
|
2010-09-28 02:15:02 +00:00
|
|
|
// ----------------------------------------------
|
|
|
|
|
2009-07-06 02:10:26 +00:00
|
|
|
void FlushPipeline()
|
|
|
|
{
|
2016-08-21 23:02:37 -04:00
|
|
|
g_vertex_manager->Flush();
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
|
|
|
|
2012-01-01 12:46:02 -08:00
|
|
|
void SetGenerationMode()
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
2018-02-25 01:15:35 +10:00
|
|
|
g_vertex_manager->SetRasterizationStateChanged();
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
|
|
|
|
2011-09-05 22:04:28 +02:00
|
|
|
void SetScissor()
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
2014-05-03 12:45:10 -04:00
|
|
|
/* NOTE: the minimum value here for the scissor rect and offset is -342.
|
|
|
|
* GX internally adds on an offset of 342 to both the offset and scissor
|
|
|
|
* coords to ensure that the register was always unsigned.
|
|
|
|
*
|
|
|
|
* The code that was here before tried to "undo" this offset, but
|
|
|
|
* since we always take the difference, the +342 added to both
|
|
|
|
* sides cancels out. */
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-05-03 12:45:10 -04:00
|
|
|
/* The scissor offset is always even, so to save space, the scissor offset
|
|
|
|
* register is scaled down by 2. So, if somebody calls
|
|
|
|
* GX_SetScissorBoxOffset(20, 20); the registers will be set to 10, 10. */
|
|
|
|
const int xoff = bpmem.scissorOffset.x * 2;
|
|
|
|
const int yoff = bpmem.scissorOffset.y * 2;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-04-16 00:47:46 +10:00
|
|
|
MathUtil::Rectangle<int> native_rc(bpmem.scissorTL.x - xoff, bpmem.scissorTL.y - yoff,
|
|
|
|
bpmem.scissorBR.x - xoff + 1, bpmem.scissorBR.y - yoff + 1);
|
2018-01-21 22:12:32 +10:00
|
|
|
native_rc.ClampUL(0, 0, EFB_WIDTH, EFB_HEIGHT);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
auto target_rc = g_renderer->ConvertEFBRectangle(native_rc);
|
|
|
|
auto converted_rc =
|
|
|
|
g_renderer->ConvertFramebufferRectangle(target_rc, g_renderer->GetCurrentFramebuffer());
|
|
|
|
g_renderer->SetScissorRect(converted_rc);
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
2009-09-02 18:55:36 +00:00
|
|
|
|
2018-01-21 22:04:15 +10:00
|
|
|
void SetViewport()
|
|
|
|
{
|
|
|
|
int scissor_x_off = bpmem.scissorOffset.x * 2;
|
|
|
|
int scissor_y_off = bpmem.scissorOffset.y * 2;
|
|
|
|
float x = g_renderer->EFBToScaledXf(xfmem.viewport.xOrig - xfmem.viewport.wd - scissor_x_off);
|
|
|
|
float y = g_renderer->EFBToScaledYf(xfmem.viewport.yOrig + xfmem.viewport.ht - scissor_y_off);
|
|
|
|
|
|
|
|
float width = g_renderer->EFBToScaledXf(2.0f * xfmem.viewport.wd);
|
|
|
|
float height = g_renderer->EFBToScaledYf(-2.0f * xfmem.viewport.ht);
|
|
|
|
float min_depth = (xfmem.viewport.farZ - xfmem.viewport.zRange) / 16777216.0f;
|
|
|
|
float max_depth = xfmem.viewport.farZ / 16777216.0f;
|
|
|
|
if (width < 0.f)
|
|
|
|
{
|
|
|
|
x += width;
|
|
|
|
width *= -1;
|
|
|
|
}
|
|
|
|
if (height < 0.f)
|
|
|
|
{
|
|
|
|
y += height;
|
|
|
|
height *= -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The maximum depth that is written to the depth buffer should never exceed this value.
|
|
|
|
// This is necessary because we use a 2^24 divisor for all our depth values to prevent
|
|
|
|
// floating-point round-trip errors. However the console GPU doesn't ever write a value
|
|
|
|
// to the depth buffer that exceeds 2^24 - 1.
|
|
|
|
constexpr float GX_MAX_DEPTH = 16777215.0f / 16777216.0f;
|
|
|
|
if (!g_ActiveConfig.backend_info.bSupportsDepthClamp)
|
|
|
|
{
|
|
|
|
// There's no way to support oversized depth ranges in this situation. Let's just clamp the
|
|
|
|
// range to the maximum value supported by the console GPU and hope for the best.
|
2017-12-26 00:38:44 +01:00
|
|
|
min_depth = std::clamp(min_depth, 0.0f, GX_MAX_DEPTH);
|
|
|
|
max_depth = std::clamp(max_depth, 0.0f, GX_MAX_DEPTH);
|
2018-01-21 22:04:15 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (g_renderer->UseVertexDepthRange())
|
|
|
|
{
|
|
|
|
// We need to ensure depth values are clamped the maximum value supported by the console GPU.
|
|
|
|
// Taking into account whether the depth range is inverted or not.
|
|
|
|
if (xfmem.viewport.zRange < 0.0f && g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
|
|
|
{
|
|
|
|
min_depth = GX_MAX_DEPTH;
|
|
|
|
max_depth = 0.0f;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
min_depth = 0.0f;
|
|
|
|
max_depth = GX_MAX_DEPTH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
float near_depth, far_depth;
|
|
|
|
if (g_ActiveConfig.backend_info.bSupportsReversedDepthRange)
|
|
|
|
{
|
|
|
|
// Set the reversed depth range.
|
|
|
|
near_depth = max_depth;
|
|
|
|
far_depth = min_depth;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We use an inverted depth range here to apply the Reverse Z trick.
|
|
|
|
// This trick makes sure we match the precision provided by the 1:0
|
|
|
|
// clipping depth range on the hardware.
|
|
|
|
near_depth = 1.0f - max_depth;
|
|
|
|
far_depth = 1.0f - min_depth;
|
|
|
|
}
|
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
// Clamp to size if oversized not supported. Required for D3D.
|
|
|
|
if (!g_ActiveConfig.backend_info.bSupportsOversizedViewports)
|
|
|
|
{
|
|
|
|
const float max_width = static_cast<float>(g_renderer->GetCurrentFramebuffer()->GetWidth());
|
|
|
|
const float max_height = static_cast<float>(g_renderer->GetCurrentFramebuffer()->GetHeight());
|
2017-12-26 00:38:44 +01:00
|
|
|
x = std::clamp(x, 0.0f, max_width - 1.0f);
|
|
|
|
y = std::clamp(y, 0.0f, max_height - 1.0f);
|
|
|
|
width = std::clamp(width, 1.0f, max_width - x);
|
|
|
|
height = std::clamp(height, 1.0f, max_height - y);
|
2019-02-15 11:59:50 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
// Lower-left flip.
|
|
|
|
if (g_ActiveConfig.backend_info.bUsesLowerLeftOrigin)
|
|
|
|
y = static_cast<float>(g_renderer->GetCurrentFramebuffer()->GetHeight()) - y - height;
|
|
|
|
|
2018-01-21 22:04:15 +10:00
|
|
|
g_renderer->SetViewport(x, y, width, height, near_depth, far_depth);
|
|
|
|
}
|
|
|
|
|
2012-01-01 12:46:02 -08:00
|
|
|
void SetDepthMode()
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
2018-02-25 01:15:35 +10:00
|
|
|
g_vertex_manager->SetDepthStateChanged();
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
2009-09-15 19:53:22 +00:00
|
|
|
|
2012-01-01 12:46:02 -08:00
|
|
|
void SetBlendMode()
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
2018-02-25 01:15:35 +10:00
|
|
|
g_vertex_manager->SetBlendingStateChanged();
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
2017-04-18 21:42:14 +10:00
|
|
|
|
2010-12-20 16:57:29 +00:00
|
|
|
/* Explanation of the magic behind ClearScreen:
|
|
|
|
There's numerous possible formats for the pixel data in the EFB.
|
2011-02-14 02:18:03 +00:00
|
|
|
However, in the HW accelerated backends we're always using RGBA8
|
2010-12-21 22:18:40 +00:00
|
|
|
for the EFB format, which causes some problems:
|
2010-12-27 18:09:03 +00:00
|
|
|
- We're using an alpha channel although the game doesn't
|
|
|
|
- If the actual EFB format is RGBA6_Z24 or R5G6B5_Z16, we are using more bits per channel than the
|
|
|
|
native HW
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-12-20 16:57:29 +00:00
|
|
|
To properly emulate the above points, we're doing the following:
|
|
|
|
(1)
|
2010-12-21 22:18:40 +00:00
|
|
|
- disable alpha channel writing of any kind of rendering if the actual EFB format doesn't use an
|
|
|
|
alpha channel
|
|
|
|
- NOTE: Always make sure that the EFB has been cleared to an alpha value of 0xFF in this case!
|
|
|
|
- Same for color channels, these need to be cleared to 0x00 though.
|
2010-12-20 16:57:29 +00:00
|
|
|
(2)
|
2010-12-27 18:09:03 +00:00
|
|
|
- convert the RGBA8 color to RGBA6/RGB8/RGB565 and convert it to RGBA8 again
|
|
|
|
- convert the Z24 depth value to Z16 and back to Z24
|
2010-12-20 16:57:29 +00:00
|
|
|
*/
|
2019-04-16 00:47:46 +10:00
|
|
|
void ClearScreen(const MathUtil::Rectangle<int>& rc)
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
2014-08-16 13:23:22 -05:00
|
|
|
bool colorEnable = (bpmem.blendmode.colorupdate != 0);
|
|
|
|
bool alphaEnable = (bpmem.blendmode.alphaupdate != 0);
|
|
|
|
bool zEnable = (bpmem.zmode.updateenable != 0);
|
2014-03-23 21:44:23 +01:00
|
|
|
auto pixel_format = bpmem.zcontrol.pixel_format;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-12-21 22:18:40 +00:00
|
|
|
// (1): Disable unused color channels
|
2021-02-10 18:11:31 -08:00
|
|
|
if (pixel_format == PixelFormat::RGB8_Z24 || pixel_format == PixelFormat::RGB565_Z16 ||
|
|
|
|
pixel_format == PixelFormat::Z24)
|
2010-12-21 22:18:40 +00:00
|
|
|
{
|
2010-12-27 18:09:03 +00:00
|
|
|
alphaEnable = false;
|
2010-12-21 22:18:40 +00:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2009-10-02 21:23:56 +00:00
|
|
|
if (colorEnable || alphaEnable || zEnable)
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
2010-12-27 18:09:03 +00:00
|
|
|
u32 color = (bpmem.clearcolorAR << 16) | bpmem.clearcolorGB;
|
|
|
|
u32 z = bpmem.clearZValue;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-12-27 18:09:03 +00:00
|
|
|
// (2) drop additional accuracy
|
2021-02-10 18:11:31 -08:00
|
|
|
if (pixel_format == PixelFormat::RGBA6_Z24)
|
2010-12-20 16:57:29 +00:00
|
|
|
{
|
2010-12-27 03:09:11 +00:00
|
|
|
color = RGBA8ToRGBA6ToRGBA8(color);
|
2010-12-20 16:57:29 +00:00
|
|
|
}
|
2021-02-10 18:11:31 -08:00
|
|
|
else if (pixel_format == PixelFormat::RGB565_Z16)
|
2010-12-20 16:57:29 +00:00
|
|
|
{
|
2010-12-27 18:09:03 +00:00
|
|
|
color = RGBA8ToRGB565ToRGBA8(color);
|
|
|
|
z = Z24ToZ16ToZ24(z);
|
2010-12-20 16:57:29 +00:00
|
|
|
}
|
2010-12-19 22:00:25 +00:00
|
|
|
g_renderer->ClearScreen(rc, colorEnable, alphaEnable, zEnable, color, z);
|
2009-10-02 21:23:56 +00:00
|
|
|
}
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
|
|
|
|
2012-01-01 12:46:02 -08:00
|
|
|
void OnPixelFormatChange()
|
2010-12-27 21:56:20 +00:00
|
|
|
{
|
2011-03-04 23:23:59 +00:00
|
|
|
// TODO : Check for Z compression format change
|
2011-03-04 23:14:21 +00:00
|
|
|
// When using 16bit Z, the game may enable a special compression format which we need to handle
|
|
|
|
// If we don't, Z values will be completely screwed up, currently only Star Wars:RS2 uses that.
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-12-27 21:56:20 +00:00
|
|
|
/*
|
|
|
|
* When changing the EFB format, the pixel data won't get converted to the new format but stays
|
|
|
|
* the same.
|
|
|
|
* Since we are always using an RGBA8 buffer though, this causes issues in some games.
|
|
|
|
* Thus, we reinterpret the old EFB data with the new format here.
|
|
|
|
*/
|
2014-04-04 16:17:14 -07:00
|
|
|
if (!g_ActiveConfig.bEFBEmulateFormatChanges)
|
2010-12-27 21:56:20 +00:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2020-11-13 22:33:26 -05:00
|
|
|
const auto old_format = g_renderer->GetPrevPixelFormat();
|
|
|
|
const auto new_format = bpmem.zcontrol.pixel_format;
|
2019-02-15 11:59:50 +10:00
|
|
|
g_renderer->StorePixelFormat(new_format);
|
|
|
|
|
2021-02-10 18:11:31 -08:00
|
|
|
DEBUG_LOG_FMT(VIDEO, "pixelfmt: pixel={}, zc={}", new_format, bpmem.zcontrol.zformat);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-03-04 23:23:59 +00:00
|
|
|
// no need to reinterpret pixel data in these cases
|
2021-02-10 18:11:31 -08:00
|
|
|
if (new_format == old_format || old_format == PixelFormat::INVALID_FMT)
|
2019-02-15 11:59:50 +10:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-03-04 23:14:21 +00:00
|
|
|
// Check for pixel format changes
|
2011-03-04 23:23:59 +00:00
|
|
|
switch (old_format)
|
2010-12-27 21:56:20 +00:00
|
|
|
{
|
2021-02-10 18:11:31 -08:00
|
|
|
case PixelFormat::RGB8_Z24:
|
|
|
|
case PixelFormat::Z24:
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
2010-12-27 21:56:20 +00:00
|
|
|
// Z24 and RGB8_Z24 are treated equal, so just return in this case
|
2021-02-10 18:11:31 -08:00
|
|
|
if (new_format == PixelFormat::RGB8_Z24 || new_format == PixelFormat::Z24)
|
2019-02-15 11:59:50 +10:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-02-10 18:11:31 -08:00
|
|
|
if (new_format == PixelFormat::RGBA6_Z24)
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
|
|
|
g_renderer->ReinterpretPixelData(EFBReinterpretType::RGB8ToRGBA6);
|
|
|
|
return;
|
|
|
|
}
|
2021-02-10 18:11:31 -08:00
|
|
|
else if (new_format == PixelFormat::RGB565_Z16)
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
|
|
|
g_renderer->ReinterpretPixelData(EFBReinterpretType::RGB8ToRGB565);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-02-10 18:11:31 -08:00
|
|
|
case PixelFormat::RGBA6_Z24:
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
2021-02-10 18:11:31 -08:00
|
|
|
if (new_format == PixelFormat::RGB8_Z24 || new_format == PixelFormat::Z24)
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
|
|
|
g_renderer->ReinterpretPixelData(EFBReinterpretType::RGBA6ToRGB8);
|
|
|
|
return;
|
|
|
|
}
|
2021-02-10 18:11:31 -08:00
|
|
|
else if (new_format == PixelFormat::RGB565_Z16)
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
|
|
|
g_renderer->ReinterpretPixelData(EFBReinterpretType::RGBA6ToRGB565);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-02-10 18:11:31 -08:00
|
|
|
case PixelFormat::RGB565_Z16:
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
2021-02-10 18:11:31 -08:00
|
|
|
if (new_format == PixelFormat::RGB8_Z24 || new_format == PixelFormat::Z24)
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
|
|
|
g_renderer->ReinterpretPixelData(EFBReinterpretType::RGB565ToRGB8);
|
|
|
|
return;
|
|
|
|
}
|
2021-02-10 18:11:31 -08:00
|
|
|
else if (new_format == PixelFormat::RGBA6_Z24)
|
2019-02-15 11:59:50 +10:00
|
|
|
{
|
|
|
|
g_renderer->ReinterpretPixelData(EFBReinterpretType::RGB565ToRGBA6);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-12-27 21:56:20 +00:00
|
|
|
default:
|
2010-12-28 01:10:40 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-02-10 18:11:31 -08:00
|
|
|
ERROR_LOG_FMT(VIDEO, "Unhandled EFB format change: {} to {}", old_format, new_format);
|
2010-12-27 21:56:20 +00:00
|
|
|
}
|
2010-12-27 03:09:11 +00:00
|
|
|
|
2009-07-26 09:52:35 +00:00
|
|
|
void SetInterlacingMode(const BPCmd& bp)
|
2009-07-06 02:10:26 +00:00
|
|
|
{
|
|
|
|
// TODO
|
2011-07-18 08:27:30 +00:00
|
|
|
switch (bp.address)
|
|
|
|
{
|
|
|
|
case BPMEM_FIELDMODE:
|
|
|
|
{
|
|
|
|
// SDK always sets bpmem.lineptwidth.lineaspect via BPMEM_LINEPTWIDTH
|
|
|
|
// just before this cmd
|
2021-02-10 18:11:31 -08:00
|
|
|
DEBUG_LOG_FMT(VIDEO, "BPMEM_FIELDMODE texLOD:{} lineaspect:{}", bpmem.fieldmode.texLOD,
|
|
|
|
bpmem.lineptwidth.adjust_for_aspect_ratio);
|
2011-07-18 08:27:30 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BPMEM_FIELDMASK:
|
|
|
|
{
|
|
|
|
// Determines if fields will be written to EFB (always computed)
|
2021-02-10 18:11:31 -08:00
|
|
|
DEBUG_LOG_FMT(VIDEO, "BPMEM_FIELDMASK even:{} odd:{}", bpmem.fieldmask.even,
|
|
|
|
bpmem.fieldmask.odd);
|
2011-07-18 08:27:30 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2020-11-13 22:33:26 -05:00
|
|
|
ERROR_LOG_FMT(VIDEO, "SetInterlacingMode default");
|
2011-07-18 08:27:30 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-07-06 02:10:26 +00:00
|
|
|
}
|
2019-02-15 11:59:50 +10:00
|
|
|
}; // namespace BPFunctions
|