2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2009 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-06-09 01:37:08 +00:00
|
|
|
|
2017-01-23 11:20:20 -05:00
|
|
|
#include "VideoBackends/Software/SWRenderer.h"
|
|
|
|
|
2014-06-03 01:08:54 -04:00
|
|
|
#include <string>
|
2014-05-02 22:47:04 -04:00
|
|
|
|
2014-09-07 20:06:58 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2018-10-03 23:03:13 +10:00
|
|
|
#include "Common/GL/GLContext.h"
|
2015-09-19 04:40:00 +12:00
|
|
|
|
2015-10-09 20:50:36 +02:00
|
|
|
#include "Core/HW/Memmap.h"
|
2022-12-10 16:35:07 +01:00
|
|
|
#include "Core/System.h"
|
2015-09-19 04:40:00 +12:00
|
|
|
|
2015-10-09 20:50:36 +02:00
|
|
|
#include "VideoBackends/Software/EfbCopy.h"
|
2017-11-19 00:16:22 -05:00
|
|
|
#include "VideoBackends/Software/EfbInterface.h"
|
2021-11-09 18:38:24 -08:00
|
|
|
#include "VideoBackends/Software/Rasterizer.h"
|
2021-06-09 07:42:21 -04:00
|
|
|
#include "VideoBackends/Software/SWBoundingBox.h"
|
2015-09-26 10:07:48 +02:00
|
|
|
#include "VideoBackends/Software/SWOGLWindow.h"
|
2017-09-30 16:25:36 +10:00
|
|
|
#include "VideoBackends/Software/SWTexture.h"
|
2015-09-19 04:40:00 +12:00
|
|
|
|
2017-09-08 19:42:56 +10:00
|
|
|
#include "VideoCommon/AbstractPipeline.h"
|
|
|
|
#include "VideoCommon/AbstractShader.h"
|
2019-02-15 11:59:50 +10:00
|
|
|
#include "VideoCommon/AbstractTexture.h"
|
|
|
|
#include "VideoCommon/NativeVertexFormat.h"
|
2022-01-29 13:53:54 -08:00
|
|
|
#include "VideoCommon/PixelEngine.h"
|
2017-01-23 11:20:20 -05:00
|
|
|
#include "VideoCommon/VideoBackendBase.h"
|
2019-07-16 20:18:48 -04:00
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2013-04-13 00:48:53 -05:00
|
|
|
|
2019-02-15 11:59:50 +10:00
|
|
|
namespace SW
|
|
|
|
{
|
2018-10-03 23:03:26 +10:00
|
|
|
SWRenderer::SWRenderer(std::unique_ptr<SWOGLWindow> window)
|
2021-05-22 19:58:27 -07:00
|
|
|
: ::Renderer(static_cast<int>(std::max(window->GetContext()->GetBackBufferWidth(), 1u)),
|
|
|
|
static_cast<int>(std::max(window->GetContext()->GetBackBufferHeight(), 1u)), 1.0f,
|
2017-09-09 16:09:24 +10:00
|
|
|
AbstractTextureFormat::RGBA8),
|
2018-10-03 23:03:26 +10:00
|
|
|
m_window(std::move(window))
|
2017-03-04 16:40:08 +10:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-10-03 23:03:13 +10:00
|
|
|
bool SWRenderer::IsHeadless() const
|
|
|
|
{
|
2018-10-03 23:03:26 +10:00
|
|
|
return m_window->IsHeadless();
|
2018-10-03 23:03:13 +10:00
|
|
|
}
|
|
|
|
|
2021-08-28 00:30:05 -05:00
|
|
|
std::unique_ptr<AbstractTexture> SWRenderer::CreateTexture(const TextureConfig& config,
|
|
|
|
[[maybe_unused]] std::string_view name)
|
2017-09-30 16:25:36 +10:00
|
|
|
{
|
2019-02-15 11:59:50 +10:00
|
|
|
return std::make_unique<SWTexture>(config);
|
2017-09-30 16:25:36 +10:00
|
|
|
}
|
|
|
|
|
2017-10-22 00:49:40 +10:00
|
|
|
std::unique_ptr<AbstractStagingTexture>
|
|
|
|
SWRenderer::CreateStagingTexture(StagingTextureType type, const TextureConfig& config)
|
|
|
|
{
|
2019-02-15 11:59:50 +10:00
|
|
|
return std::make_unique<SWStagingTexture>(type, config);
|
2017-10-22 00:49:40 +10:00
|
|
|
}
|
|
|
|
|
2018-01-21 20:22:45 +10:00
|
|
|
std::unique_ptr<AbstractFramebuffer>
|
2019-02-15 11:59:50 +10:00
|
|
|
SWRenderer::CreateFramebuffer(AbstractTexture* color_attachment, AbstractTexture* depth_attachment)
|
2018-01-21 20:22:45 +10:00
|
|
|
{
|
2019-02-15 11:59:50 +10:00
|
|
|
return SWFramebuffer::Create(static_cast<SWTexture*>(color_attachment),
|
|
|
|
static_cast<SWTexture*>(depth_attachment));
|
2018-01-21 20:22:45 +10:00
|
|
|
}
|
|
|
|
|
2021-05-22 19:58:27 -07:00
|
|
|
void SWRenderer::BindBackbuffer(const ClearColor& clear_color)
|
|
|
|
{
|
|
|
|
// Look for framebuffer resizes
|
|
|
|
if (!m_surface_resized.TestAndClear())
|
|
|
|
return;
|
|
|
|
|
|
|
|
GLContext* context = m_window->GetContext();
|
|
|
|
context->Update();
|
|
|
|
m_backbuffer_width = context->GetBackBufferWidth();
|
|
|
|
m_backbuffer_height = context->GetBackBufferHeight();
|
|
|
|
}
|
|
|
|
|
2017-09-08 19:42:56 +10:00
|
|
|
class SWShader final : public AbstractShader
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit SWShader(ShaderStage stage) : AbstractShader(stage) {}
|
|
|
|
~SWShader() = default;
|
|
|
|
|
|
|
|
BinaryData GetBinary() const override { return {}; }
|
|
|
|
};
|
|
|
|
|
|
|
|
std::unique_ptr<AbstractShader>
|
2021-08-28 00:30:05 -05:00
|
|
|
SWRenderer::CreateShaderFromSource(ShaderStage stage, [[maybe_unused]] std::string_view source,
|
|
|
|
[[maybe_unused]] std::string_view name)
|
2017-09-08 19:42:56 +10:00
|
|
|
{
|
|
|
|
return std::make_unique<SWShader>(stage);
|
|
|
|
}
|
|
|
|
|
2021-08-28 00:30:05 -05:00
|
|
|
std::unique_ptr<AbstractShader>
|
|
|
|
SWRenderer::CreateShaderFromBinary(ShaderStage stage, const void* data, size_t length,
|
|
|
|
[[maybe_unused]] std::string_view name)
|
2017-09-08 19:42:56 +10:00
|
|
|
{
|
|
|
|
return std::make_unique<SWShader>(stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
class SWPipeline final : public AbstractPipeline
|
|
|
|
{
|
|
|
|
public:
|
2019-03-02 19:42:25 +01:00
|
|
|
SWPipeline() = default;
|
2017-09-08 19:42:56 +10:00
|
|
|
~SWPipeline() override = default;
|
|
|
|
};
|
|
|
|
|
2019-04-15 21:55:26 +10:00
|
|
|
std::unique_ptr<AbstractPipeline> SWRenderer::CreatePipeline(const AbstractPipelineConfig& config,
|
|
|
|
const void* cache_data,
|
|
|
|
size_t cache_data_length)
|
2017-09-08 19:42:56 +10:00
|
|
|
{
|
|
|
|
return std::make_unique<SWPipeline>();
|
|
|
|
}
|
|
|
|
|
2013-08-20 23:51:39 +12:00
|
|
|
// Called on the GPU thread
|
2019-04-28 17:00:18 +10:00
|
|
|
void SWRenderer::RenderXFBToScreen(const MathUtil::Rectangle<int>& target_rc,
|
|
|
|
const AbstractTexture* source_texture,
|
|
|
|
const MathUtil::Rectangle<int>& source_rc)
|
2013-08-20 23:51:39 +12:00
|
|
|
{
|
2017-11-23 16:53:44 +10:00
|
|
|
if (!IsHeadless())
|
2019-04-28 17:00:18 +10:00
|
|
|
m_window->ShowImage(source_texture, source_rc);
|
2015-10-09 20:50:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 SWRenderer::AccessEFB(EFBAccessType type, u32 x, u32 y, u32 InputData)
|
|
|
|
{
|
|
|
|
u32 value = 0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-10-09 20:50:36 +02:00
|
|
|
switch (type)
|
|
|
|
{
|
2017-01-23 02:51:46 -05:00
|
|
|
case EFBAccessType::PeekZ:
|
2015-10-09 20:50:36 +02:00
|
|
|
{
|
|
|
|
value = EfbInterface::GetDepth(x, y);
|
|
|
|
break;
|
|
|
|
}
|
2017-01-23 02:51:46 -05:00
|
|
|
case EFBAccessType::PeekColor:
|
2015-10-09 20:50:36 +02:00
|
|
|
{
|
2016-09-21 18:53:13 -04:00
|
|
|
const u32 color = EfbInterface::GetColor(x, y);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-10-09 20:50:36 +02:00
|
|
|
// rgba to argb
|
|
|
|
value = (color >> 8) | (color & 0xff) << 24;
|
2022-01-29 13:53:54 -08:00
|
|
|
|
|
|
|
// check what to do with the alpha channel (GX_PokeAlphaRead)
|
2022-12-10 16:35:07 +01:00
|
|
|
PixelEngine::AlphaReadMode alpha_read_mode =
|
|
|
|
Core::System::GetInstance().GetPixelEngine().GetAlphaReadMode();
|
2022-01-29 13:53:54 -08:00
|
|
|
|
|
|
|
if (alpha_read_mode == PixelEngine::AlphaReadMode::ReadNone)
|
|
|
|
{
|
|
|
|
// value is OK as it is
|
|
|
|
}
|
|
|
|
else if (alpha_read_mode == PixelEngine::AlphaReadMode::ReadFF)
|
|
|
|
{
|
|
|
|
value |= 0xFF000000;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (alpha_read_mode != PixelEngine::AlphaReadMode::Read00)
|
|
|
|
{
|
|
|
|
PanicAlertFmt("Invalid PE alpha read mode: {}", static_cast<u16>(alpha_read_mode));
|
|
|
|
}
|
|
|
|
value &= 0x00FFFFFF;
|
|
|
|
}
|
|
|
|
|
2015-10-09 20:50:36 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-10-09 20:50:36 +02:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2021-06-09 07:42:21 -04:00
|
|
|
std::unique_ptr<BoundingBox> SWRenderer::CreateBoundingBox() const
|
2015-10-09 20:50:36 +02:00
|
|
|
{
|
2021-06-09 07:42:21 -04:00
|
|
|
return std::make_unique<SWBoundingBox>();
|
2015-10-09 20:50:36 +02:00
|
|
|
}
|
|
|
|
|
2019-04-16 00:47:46 +10:00
|
|
|
void SWRenderer::ClearScreen(const MathUtil::Rectangle<int>& rc, bool colorEnable, bool alphaEnable,
|
2015-10-09 20:50:36 +02:00
|
|
|
bool zEnable, u32 color, u32 z)
|
|
|
|
{
|
|
|
|
EfbCopy::ClearEfb();
|
2009-10-12 00:48:24 +00:00
|
|
|
}
|
2019-02-15 11:59:50 +10:00
|
|
|
|
|
|
|
std::unique_ptr<NativeVertexFormat>
|
|
|
|
SWRenderer::CreateNativeVertexFormat(const PortableVertexDeclaration& vtx_decl)
|
|
|
|
{
|
|
|
|
return std::make_unique<NativeVertexFormat>(vtx_decl);
|
|
|
|
}
|
2021-11-09 18:38:24 -08:00
|
|
|
|
|
|
|
void SWRenderer::SetScissorRect(const MathUtil::Rectangle<int>& rc)
|
|
|
|
{
|
|
|
|
// BPFunctions calls SetScissorRect with the "best" scissor rect whenever the viewport or scissor
|
|
|
|
// changes. However, the software renderer is actually able to use multiple scissor rects (which
|
|
|
|
// is necessary in a few renderering edge cases, such as with Major Minor's Majestic March).
|
|
|
|
// Thus, we use this as a signal to update the list of scissor rects, but ignore the parameter.
|
|
|
|
Rasterizer::ScissorChanged();
|
|
|
|
}
|
2019-02-15 11:59:50 +10:00
|
|
|
} // namespace SW
|