Merge pull request #9736 from Techjar/bbox-pixel-quads

VideoCommon: Account for pixel quads in bounding box calculation
This commit is contained in:
Mai M 2021-05-22 13:14:07 -04:00 committed by GitHub
commit 77e5d79354
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 59 additions and 29 deletions

View File

@ -264,12 +264,12 @@ void Renderer::UnbindTexture(const AbstractTexture* texture)
D3D::stateman->ApplyTextures(); D3D::stateman->ApplyTextures();
} }
u16 Renderer::BBoxRead(int index) u16 Renderer::BBoxReadImpl(int index)
{ {
return static_cast<u16>(BBox::Get(index)); return static_cast<u16>(BBox::Get(index));
} }
void Renderer::BBoxWrite(int index, u16 value) void Renderer::BBoxWriteImpl(int index, u16 value)
{ {
BBox::Set(index, value); BBox::Set(index, value);
} }

View File

@ -61,8 +61,8 @@ public:
void SetFullscreen(bool enable_fullscreen) override; void SetFullscreen(bool enable_fullscreen) override;
bool IsFullscreen() const override; bool IsFullscreen() const override;
u16 BBoxRead(int index) override; u16 BBoxReadImpl(int index) override;
void BBoxWrite(int index, u16 value) override; void BBoxWriteImpl(int index, u16 value) override;
void Flush() override; void Flush() override;
void WaitForGPUIdle() override; void WaitForGPUIdle() override;

View File

@ -106,17 +106,17 @@ std::unique_ptr<AbstractPipeline> Renderer::CreatePipeline(const AbstractPipelin
return DXPipeline::Create(config, cache_data, cache_data_length); return DXPipeline::Create(config, cache_data, cache_data_length);
} }
u16 Renderer::BBoxRead(int index) u16 Renderer::BBoxReadImpl(int index)
{ {
return static_cast<u16>(m_bounding_box->Get(index)); return static_cast<u16>(m_bounding_box->Get(index));
} }
void Renderer::BBoxWrite(int index, u16 value) void Renderer::BBoxWriteImpl(int index, u16 value)
{ {
m_bounding_box->Set(index, value); m_bounding_box->Set(index, value);
} }
void Renderer::BBoxFlush() void Renderer::BBoxFlushImpl()
{ {
m_bounding_box->Flush(); m_bounding_box->Flush();
m_bounding_box->Invalidate(); m_bounding_box->Invalidate();

View File

@ -47,9 +47,9 @@ public:
const void* cache_data = nullptr, const void* cache_data = nullptr,
size_t cache_data_length = 0) override; size_t cache_data_length = 0) override;
u16 BBoxRead(int index) override; u16 BBoxReadImpl(int index) override;
void BBoxWrite(int index, u16 value) override; void BBoxWriteImpl(int index, u16 value) override;
void BBoxFlush() override; void BBoxFlushImpl() override;
void Flush() override; void Flush() override;
void WaitForGPUIdle() override; void WaitForGPUIdle() override;

View File

@ -34,8 +34,8 @@ public:
u32 AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data) override { return 0; } u32 AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data) override { return 0; }
void PokeEFB(EFBAccessType type, const EfbPokeData* points, size_t num_points) override {} void PokeEFB(EFBAccessType type, const EfbPokeData* points, size_t num_points) override {}
u16 BBoxRead(int index) override { return 0; } u16 BBoxReadImpl(int index) override { return 0; }
void BBoxWrite(int index, u16 value) override {} void BBoxWriteImpl(int index, u16 value) override {}
void ClearScreen(const MathUtil::Rectangle<int>& rc, bool colorEnable, bool alphaEnable, void ClearScreen(const MathUtil::Rectangle<int>& rc, bool colorEnable, bool alphaEnable,
bool zEnable, u32 color, u32 z) override bool zEnable, u32 color, u32 z) override

View File

@ -854,7 +854,7 @@ void Renderer::SetScissorRect(const MathUtil::Rectangle<int>& rc)
glScissor(rc.left, rc.top, rc.GetWidth(), rc.GetHeight()); glScissor(rc.left, rc.top, rc.GetWidth(), rc.GetHeight());
} }
u16 Renderer::BBoxRead(int index) u16 Renderer::BBoxReadImpl(int index)
{ {
// swap 2 and 3 for top/bottom // swap 2 and 3 for top/bottom
if (index >= 2) if (index >= 2)
@ -870,7 +870,7 @@ u16 Renderer::BBoxRead(int index)
return static_cast<u16>(value); return static_cast<u16>(value);
} }
void Renderer::BBoxWrite(int index, u16 value) void Renderer::BBoxWriteImpl(int index, u16 value)
{ {
s32 swapped_value = value; s32 swapped_value = value;
if (index >= 2) if (index >= 2)

View File

@ -126,8 +126,8 @@ public:
void BindBackbuffer(const ClearColor& clear_color = {}) override; void BindBackbuffer(const ClearColor& clear_color = {}) override;
void PresentBackbuffer() override; void PresentBackbuffer() override;
u16 BBoxRead(int index) override; u16 BBoxReadImpl(int index) override;
void BBoxWrite(int index, u16 value) override; void BBoxWriteImpl(int index, u16 value) override;
void BeginUtilityDrawing() override; void BeginUtilityDrawing() override;
void EndUtilityDrawing() override; void EndUtilityDrawing() override;

View File

@ -126,12 +126,12 @@ u32 SWRenderer::AccessEFB(EFBAccessType type, u32 x, u32 y, u32 InputData)
return value; return value;
} }
u16 SWRenderer::BBoxRead(int index) u16 SWRenderer::BBoxReadImpl(int index)
{ {
return BoundingBox::GetCoordinate(static_cast<BoundingBox::Coordinate>(index)); return BoundingBox::GetCoordinate(static_cast<BoundingBox::Coordinate>(index));
} }
void SWRenderer::BBoxWrite(int index, u16 value) void SWRenderer::BBoxWriteImpl(int index, u16 value)
{ {
BoundingBox::SetCoordinate(static_cast<BoundingBox::Coordinate>(index), value); BoundingBox::SetCoordinate(static_cast<BoundingBox::Coordinate>(index), value);
} }

View File

@ -39,8 +39,8 @@ public:
u32 AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data) override; u32 AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data) override;
void PokeEFB(EFBAccessType type, const EfbPokeData* points, size_t num_points) override {} void PokeEFB(EFBAccessType type, const EfbPokeData* points, size_t num_points) override {}
u16 BBoxRead(int index) override; u16 BBoxReadImpl(int index) override;
void BBoxWrite(int index, u16 value) override; void BBoxWriteImpl(int index, u16 value) override;
void RenderXFBToScreen(const MathUtil::Rectangle<int>& target_rc, void RenderXFBToScreen(const MathUtil::Rectangle<int>& target_rc,
const AbstractTexture* source_texture, const AbstractTexture* source_texture,

View File

@ -131,17 +131,17 @@ void Renderer::SetPipeline(const AbstractPipeline* pipeline)
StateTracker::GetInstance()->SetPipeline(static_cast<const VKPipeline*>(pipeline)); StateTracker::GetInstance()->SetPipeline(static_cast<const VKPipeline*>(pipeline));
} }
u16 Renderer::BBoxRead(int index) u16 Renderer::BBoxReadImpl(int index)
{ {
return static_cast<u16>(m_bounding_box->Get(index)); return static_cast<u16>(m_bounding_box->Get(index));
} }
void Renderer::BBoxWrite(int index, u16 value) void Renderer::BBoxWriteImpl(int index, u16 value)
{ {
m_bounding_box->Set(index, value); m_bounding_box->Set(index, value);
} }
void Renderer::BBoxFlush() void Renderer::BBoxFlushImpl()
{ {
m_bounding_box->Flush(); m_bounding_box->Flush();
m_bounding_box->Invalidate(); m_bounding_box->Invalidate();

View File

@ -54,9 +54,9 @@ public:
SwapChain* GetSwapChain() const { return m_swap_chain.get(); } SwapChain* GetSwapChain() const { return m_swap_chain.get(); }
BoundingBox* GetBoundingBox() const { return m_bounding_box.get(); } BoundingBox* GetBoundingBox() const { return m_bounding_box.get(); }
u16 BBoxRead(int index) override; u16 BBoxReadImpl(int index) override;
void BBoxWrite(int index, u16 value) override; void BBoxWriteImpl(int index, u16 value) override;
void BBoxFlush() override; void BBoxFlushImpl() override;
void Flush() override; void Flush() override;
void WaitForGPUIdle() override; void WaitForGPUIdle() override;

View File

@ -185,6 +185,32 @@ void Renderer::ReinterpretPixelData(EFBReinterpretType convtype)
g_framebuffer_manager->ReinterpretPixelData(convtype); g_framebuffer_manager->ReinterpretPixelData(convtype);
} }
u16 Renderer::BBoxRead(int index)
{
u16 value = BBoxReadImpl(index);
// The GC/Wii GPU rasterizes in 2x2 pixel groups, so bounding box values will be rounded to the
// extents of these groups, rather than the exact pixel.
// This would have been handled in the pixel shader, but all attempts to do so did not work on
// OpenGL/NVIDIA, due to presumably mystical driver behavior with atomics.
if (index == 0 || index == 2)
value &= ~1;
else
value |= 1;
return value;
}
void Renderer::BBoxWrite(int index, u16 value)
{
BBoxWriteImpl(index, value);
}
void Renderer::BBoxFlush()
{
BBoxFlushImpl();
}
u32 Renderer::AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data) u32 Renderer::AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data)
{ {
if (type == EFBAccessType::PeekColor) if (type == EFBAccessType::PeekColor)

View File

@ -211,9 +211,9 @@ public:
virtual u32 AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data); virtual u32 AccessEFB(EFBAccessType type, u32 x, u32 y, u32 poke_data);
virtual void PokeEFB(EFBAccessType type, const EfbPokeData* points, size_t num_points); virtual void PokeEFB(EFBAccessType type, const EfbPokeData* points, size_t num_points);
virtual u16 BBoxRead(int index) = 0; u16 BBoxRead(int index);
virtual void BBoxWrite(int index, u16 value) = 0; void BBoxWrite(int index, u16 value);
virtual void BBoxFlush() {} void BBoxFlush();
virtual void Flush() {} virtual void Flush() {}
virtual void WaitForGPUIdle() {} virtual void WaitForGPUIdle() {}
@ -301,6 +301,10 @@ protected:
// Should be called with the ImGui lock held. // Should be called with the ImGui lock held.
void DrawImGui(); void DrawImGui();
virtual u16 BBoxReadImpl(int index) = 0;
virtual void BBoxWriteImpl(int index, u16 value) = 0;
virtual void BBoxFlushImpl() {}
AbstractFramebuffer* m_current_framebuffer = nullptr; AbstractFramebuffer* m_current_framebuffer = nullptr;
const AbstractPipeline* m_current_pipeline = nullptr; const AbstractPipeline* m_current_pipeline = nullptr;