mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-20 04:51:21 +01:00
61c3a0d9e4
This removes OSD support for video software, but it was already broken before. This commit does not try to fix coding style issues, the rewrite of this presentation API is splitted up.
351 lines
8.0 KiB
C++
351 lines
8.0 KiB
C++
// Copyright 2009 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <atomic>
|
|
#include <string>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/FileUtil.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "Common/Logging/LogManager.h"
|
|
|
|
#include "Core/ConfigManager.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/Host.h"
|
|
#include "Core/HW/Memmap.h"
|
|
#include "Core/HW/VideoInterface.h"
|
|
|
|
#include "VideoBackends/Software/BPMemLoader.h"
|
|
#include "VideoBackends/Software/Clipper.h"
|
|
#include "VideoBackends/Software/DebugUtil.h"
|
|
#include "VideoBackends/Software/EfbInterface.h"
|
|
#include "VideoBackends/Software/OpcodeDecoder.h"
|
|
#include "VideoBackends/Software/Rasterizer.h"
|
|
#include "VideoBackends/Software/SWCommandProcessor.h"
|
|
#include "VideoBackends/Software/SWOGLWindow.h"
|
|
#include "VideoBackends/Software/SWRenderer.h"
|
|
#include "VideoBackends/Software/SWStatistics.h"
|
|
#include "VideoBackends/Software/SWVertexLoader.h"
|
|
#include "VideoBackends/Software/SWVideoConfig.h"
|
|
#include "VideoBackends/Software/VideoBackend.h"
|
|
#include "VideoBackends/Software/XFMemLoader.h"
|
|
|
|
#include "VideoCommon/BoundingBox.h"
|
|
#include "VideoCommon/Fifo.h"
|
|
#include "VideoCommon/OnScreenDisplay.h"
|
|
#include "VideoCommon/PixelEngine.h"
|
|
#include "VideoCommon/XFMemory.h"
|
|
|
|
#define VSYNC_ENABLED 0
|
|
|
|
static std::atomic<bool> s_swapRequested;
|
|
|
|
static volatile struct
|
|
{
|
|
u32 xfbAddr;
|
|
u32 fbWidth;
|
|
u32 fbHeight;
|
|
} s_beginFieldArgs;
|
|
|
|
namespace SW
|
|
{
|
|
|
|
static std::atomic<bool> fifoStateRun;
|
|
static std::atomic<bool> emuRunningState;
|
|
static std::mutex m_csSWVidOccupied;
|
|
|
|
std::string VideoSoftware::GetName() const
|
|
{
|
|
return "Software Renderer";
|
|
}
|
|
|
|
std::string VideoSoftware::GetDisplayName() const
|
|
{
|
|
return "Software Renderer";
|
|
}
|
|
|
|
std::string VideoSoftware::GetConfigName() const
|
|
{
|
|
return "gfx_software";
|
|
}
|
|
|
|
void VideoSoftware::ShowConfig(void *hParent)
|
|
{
|
|
Host_ShowVideoConfig(hParent, GetDisplayName(), GetConfigName());
|
|
}
|
|
|
|
bool VideoSoftware::Initialize(void *window_handle)
|
|
{
|
|
g_SWVideoConfig.Load((File::GetUserPath(D_CONFIG_IDX) + GetConfigName() + ".ini").c_str());
|
|
|
|
SWOGLWindow::Init(window_handle);
|
|
|
|
InitBPMemory();
|
|
InitXFMemory();
|
|
SWCommandProcessor::Init();
|
|
PixelEngine::Init();
|
|
OpcodeDecoder::Init();
|
|
Clipper::Init();
|
|
Rasterizer::Init();
|
|
SWRenderer::Init();
|
|
DebugUtil::Init();
|
|
|
|
return true;
|
|
}
|
|
|
|
void VideoSoftware::DoState(PointerWrap& p)
|
|
{
|
|
bool software = true;
|
|
p.Do(software);
|
|
if (p.GetMode() == PointerWrap::MODE_READ && software == false)
|
|
// change mode to abort load of incompatible save state.
|
|
p.SetMode(PointerWrap::MODE_VERIFY);
|
|
|
|
// TODO: incomplete?
|
|
SWCommandProcessor::DoState(p);
|
|
PixelEngine::DoState(p);
|
|
EfbInterface::DoState(p);
|
|
OpcodeDecoder::DoState(p);
|
|
Clipper::DoState(p);
|
|
p.Do(xfmem);
|
|
p.Do(bpmem);
|
|
p.DoPOD(swstats);
|
|
|
|
// CP Memory
|
|
DoCPState(p);
|
|
}
|
|
|
|
void VideoSoftware::CheckInvalidState()
|
|
{
|
|
// there is no state to invalidate
|
|
}
|
|
|
|
void VideoSoftware::PauseAndLock(bool doLock, bool unpauseOnUnlock)
|
|
{
|
|
if (doLock)
|
|
{
|
|
EmuStateChange(EMUSTATE_CHANGE_PAUSE);
|
|
if (!Core::IsGPUThread())
|
|
m_csSWVidOccupied.lock();
|
|
}
|
|
else
|
|
{
|
|
if (unpauseOnUnlock)
|
|
EmuStateChange(EMUSTATE_CHANGE_PLAY);
|
|
if (!Core::IsGPUThread())
|
|
m_csSWVidOccupied.unlock();
|
|
}
|
|
}
|
|
|
|
void VideoSoftware::RunLoop(bool enable)
|
|
{
|
|
emuRunningState.store(enable);
|
|
}
|
|
|
|
void VideoSoftware::EmuStateChange(EMUSTATE_CHANGE newState)
|
|
{
|
|
emuRunningState.store(newState == EMUSTATE_CHANGE_PLAY);
|
|
}
|
|
|
|
void VideoSoftware::Shutdown()
|
|
{
|
|
// TODO: should be in Video_Cleanup
|
|
SWRenderer::Shutdown();
|
|
DebugUtil::Shutdown();
|
|
|
|
// Do our OSD callbacks
|
|
OSD::DoCallbacks(OSD::OSD_SHUTDOWN);
|
|
|
|
SWOGLWindow::Shutdown();
|
|
}
|
|
|
|
void VideoSoftware::Video_Cleanup()
|
|
{
|
|
}
|
|
|
|
// This is called after Video_Initialize() from the Core
|
|
void VideoSoftware::Video_Prepare()
|
|
{
|
|
// Do our OSD callbacks
|
|
OSD::DoCallbacks(OSD::OSD_INIT);
|
|
|
|
SWRenderer::Prepare();
|
|
|
|
INFO_LOG(VIDEO, "Video backend initialized.");
|
|
}
|
|
|
|
// Run from the CPU thread (from VideoInterface.cpp)
|
|
void VideoSoftware::Video_BeginField(u32 xfbAddr, u32 fbWidth, u32 fbStride, u32 fbHeight)
|
|
{
|
|
// XXX: fbStride should be implemented properly here
|
|
// If stride isn't implemented then there are problems with XFB
|
|
// Animal Crossing is a good example for this.
|
|
s_beginFieldArgs.xfbAddr = xfbAddr;
|
|
s_beginFieldArgs.fbWidth = fbWidth;
|
|
s_beginFieldArgs.fbHeight = fbHeight;
|
|
}
|
|
|
|
// Run from the CPU thread (from VideoInterface.cpp)
|
|
void VideoSoftware::Video_EndField()
|
|
{
|
|
// Techincally the XFB is continually rendered out scanline by scanline between
|
|
// BeginField and EndFeild, We could possibly get away with copying out the whole thing
|
|
// at BeginField for less lag, but for the safest emulation we run it here.
|
|
|
|
if (g_bSkipCurrentFrame || s_beginFieldArgs.xfbAddr == 0)
|
|
{
|
|
swstats.frameCount++;
|
|
swstats.ResetFrame();
|
|
Core::Callback_VideoCopiedToXFB(false);
|
|
return;
|
|
}
|
|
if (!g_SWVideoConfig.bBypassXFB)
|
|
{
|
|
EfbInterface::yuv422_packed *xfb = (EfbInterface::yuv422_packed *) Memory::GetPointer(s_beginFieldArgs.xfbAddr);
|
|
|
|
SWRenderer::UpdateColorTexture(xfb, s_beginFieldArgs.fbWidth, s_beginFieldArgs.fbHeight);
|
|
}
|
|
|
|
// Ideally we would just move all the OpenGL context stuff to the CPU thread,
|
|
// but this gets messy when the hardware rasterizer is enabled.
|
|
// And neobrain loves his hardware rasterizer.
|
|
|
|
// If BypassXFB has already done a swap (cf. EfbCopy::CopyToXfb), skip this.
|
|
if (!g_SWVideoConfig.bBypassXFB)
|
|
{
|
|
// Dump frame if needed
|
|
DebugUtil::OnFrameEnd(s_beginFieldArgs.fbWidth, s_beginFieldArgs.fbHeight);
|
|
|
|
// If we are in dual core mode, notify the GPU thread about the new color texture.
|
|
if (SConfig::GetInstance().bCPUThread)
|
|
s_swapRequested.store(true);
|
|
else
|
|
SWRenderer::Swap(s_beginFieldArgs.fbWidth, s_beginFieldArgs.fbHeight);
|
|
}
|
|
}
|
|
|
|
u32 VideoSoftware::Video_AccessEFB(EFBAccessType type, u32 x, u32 y, u32 InputData)
|
|
{
|
|
u32 value = 0;
|
|
|
|
switch (type)
|
|
{
|
|
case PEEK_Z:
|
|
{
|
|
value = EfbInterface::GetDepth(x, y);
|
|
break;
|
|
}
|
|
|
|
case POKE_Z:
|
|
break;
|
|
|
|
case PEEK_COLOR:
|
|
{
|
|
u32 color = 0;
|
|
EfbInterface::GetColor(x, y, (u8*)&color);
|
|
|
|
// rgba to argb
|
|
value = (color >> 8) | (color & 0xff) << 24;
|
|
break;
|
|
}
|
|
|
|
case POKE_COLOR:
|
|
break;
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
u32 VideoSoftware::Video_GetQueryResult(PerfQueryType type)
|
|
{
|
|
return EfbInterface::perf_values[type];
|
|
}
|
|
|
|
u16 VideoSoftware::Video_GetBoundingBox(int index)
|
|
{
|
|
return BoundingBox::coords[index];
|
|
}
|
|
|
|
bool VideoSoftware::Video_Screenshot(const std::string& filename)
|
|
{
|
|
SWRenderer::SetScreenshot(filename.c_str());
|
|
return true;
|
|
}
|
|
|
|
// Run from the graphics thread
|
|
static void VideoFifo_CheckSwapRequest()
|
|
{
|
|
if (s_swapRequested.load())
|
|
{
|
|
SWRenderer::Swap(s_beginFieldArgs.fbWidth, s_beginFieldArgs.fbHeight);
|
|
s_swapRequested.store(false);
|
|
}
|
|
}
|
|
|
|
// -------------------------------
|
|
// Enter and exit the video loop
|
|
// -------------------------------
|
|
void VideoSoftware::Video_EnterLoop()
|
|
{
|
|
std::lock_guard<std::mutex> lk(m_csSWVidOccupied);
|
|
fifoStateRun.store(true);
|
|
|
|
while (fifoStateRun.load())
|
|
{
|
|
VideoFifo_CheckSwapRequest();
|
|
g_video_backend->PeekMessages();
|
|
|
|
if (!SWCommandProcessor::RunBuffer())
|
|
{
|
|
Common::YieldCPU();
|
|
}
|
|
|
|
while (!emuRunningState.load() && fifoStateRun.load())
|
|
{
|
|
g_video_backend->PeekMessages();
|
|
VideoFifo_CheckSwapRequest();
|
|
m_csSWVidOccupied.unlock();
|
|
Common::SleepCurrentThread(1);
|
|
m_csSWVidOccupied.lock();
|
|
}
|
|
}
|
|
}
|
|
|
|
void VideoSoftware::Video_ExitLoop()
|
|
{
|
|
fifoStateRun.store(false);
|
|
}
|
|
|
|
// TODO : could use the OSD class in video common, we would need to implement the Renderer class
|
|
// however most of it is useless for the SW backend so we could as well move it to its own class
|
|
void VideoSoftware::Video_AddMessage(const std::string& msg, u32 milliseconds)
|
|
{
|
|
}
|
|
void VideoSoftware::Video_ClearMessages()
|
|
{
|
|
}
|
|
|
|
void VideoSoftware::Video_SetRendering(bool bEnabled)
|
|
{
|
|
SWCommandProcessor::SetRendering(bEnabled);
|
|
}
|
|
|
|
void VideoSoftware::Video_GatherPipeBursted()
|
|
{
|
|
SWCommandProcessor::GatherPipeBursted();
|
|
}
|
|
|
|
void VideoSoftware::RegisterCPMMIO(MMIO::Mapping* mmio, u32 base)
|
|
{
|
|
SWCommandProcessor::RegisterMMIO(mmio, base);
|
|
}
|
|
|
|
// Draw messages on top of the screen
|
|
unsigned int VideoSoftware::PeekMessages()
|
|
{
|
|
return SWOGLWindow::s_instance->PeekMessages();
|
|
}
|
|
|
|
}
|