mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 15:31:17 +01:00
608f9bcd67
Separated out from my gpu-determinism branch by request. It's not a big commit; I just like to write long commit messages. The main reason to kill it is hopefully a slight performance improvement from avoiding the double switch (especially in single core mode); however, this also improves cycle calculation, as described below. - FifoCommandRunnable is removed; in its stead, Decode returns the number of cycles (which only matters for "sync" GPU mode), or 0 if there was not enough data, and is also responsible for unknown opcode alerts. Decode and DecodeSemiNop are almost identical, so the latter is replaced with a skipped_frame parameter to Decode. Doesn't mean we can't improve skipped_frame mode to do less work; if, at such a point, branching on it has too much overhead (it certainly won't now), it can always be changed to a template parameter. - FifoCommandRunnable used a fixed, large cycle count for display lists, regardless of the contents. Presumably the actual hardware's processing time is mostly the processing time of whatever commands are in the list, and with this change InterpretDisplayList can just return the list's cycle count to be added to the total. (Since the calculation for this is part of Decode, it didn't seem easy to split this change up.) To facilitate this, Decode also gains an explicit 'end' parameter in lieu of FifoCommandRunnable's call to GetVideoBufferEndPtr, which can point to there or to the end of a display list (or elsewhere in gpu-determinism, but that's another story). Also, as a small optimization, InterpretDisplayList now calls OpcodeDecoder_Run rather than having its own Decode loop, to allow Decode to be inlined (haven't checked whether this actually happens though). skipped_frame mode still does not traverse display lists and uses the old fake value of 45 cycles. degasus has suggested that this hack is not essential for performance and can be removed, but I want to separate any potential performance impact of that from this commit.
252 lines
6.5 KiB
C++
252 lines
6.5 KiB
C++
// Copyright 2013 Dolphin Emulator Project
|
|
// Licensed under GPLv2
|
|
// Refer to the license.txt file included.
|
|
|
|
#include "Common/Atomic.h"
|
|
#include "Common/ChunkFile.h"
|
|
#include "Common/FPURoundMode.h"
|
|
#include "Common/MemoryUtil.h"
|
|
#include "Common/Thread.h"
|
|
|
|
#include "Core/Core.h"
|
|
#include "Core/CoreTiming.h"
|
|
#include "Core/HW/Memmap.h"
|
|
|
|
#include "VideoCommon/CommandProcessor.h"
|
|
#include "VideoCommon/DataReader.h"
|
|
#include "VideoCommon/Fifo.h"
|
|
#include "VideoCommon/OpcodeDecoding.h"
|
|
#include "VideoCommon/PixelEngine.h"
|
|
#include "VideoCommon/VideoConfig.h"
|
|
|
|
volatile bool g_bSkipCurrentFrame = false;
|
|
|
|
namespace
|
|
{
|
|
static volatile bool GpuRunningState = false;
|
|
static volatile bool EmuRunningState = false;
|
|
static std::mutex m_csHWVidOccupied;
|
|
// STATE_TO_SAVE
|
|
static u8 *videoBuffer;
|
|
static int size = 0;
|
|
} // namespace
|
|
|
|
void Fifo_DoState(PointerWrap &p)
|
|
{
|
|
p.DoArray(videoBuffer, FIFO_SIZE);
|
|
p.Do(size);
|
|
p.DoPointer(g_pVideoData, videoBuffer);
|
|
p.Do(g_bSkipCurrentFrame);
|
|
}
|
|
|
|
void Fifo_PauseAndLock(bool doLock, bool unpauseOnUnlock)
|
|
{
|
|
if (doLock)
|
|
{
|
|
EmulatorState(false);
|
|
if (!Core::IsGPUThread())
|
|
m_csHWVidOccupied.lock();
|
|
_dbg_assert_(COMMON, !CommandProcessor::fifo.isGpuReadingData);
|
|
}
|
|
else
|
|
{
|
|
if (unpauseOnUnlock)
|
|
EmulatorState(true);
|
|
if (!Core::IsGPUThread())
|
|
m_csHWVidOccupied.unlock();
|
|
}
|
|
}
|
|
|
|
|
|
void Fifo_Init()
|
|
{
|
|
videoBuffer = (u8*)AllocateMemoryPages(FIFO_SIZE);
|
|
size = 0;
|
|
GpuRunningState = false;
|
|
Common::AtomicStore(CommandProcessor::VITicks, CommandProcessor::m_cpClockOrigin);
|
|
}
|
|
|
|
void Fifo_Shutdown()
|
|
{
|
|
if (GpuRunningState) PanicAlert("Fifo shutting down while active");
|
|
FreeMemoryPages(videoBuffer, FIFO_SIZE);
|
|
videoBuffer = nullptr;
|
|
}
|
|
|
|
u8* GetVideoBufferStartPtr()
|
|
{
|
|
return videoBuffer;
|
|
}
|
|
|
|
u8* GetVideoBufferEndPtr()
|
|
{
|
|
return &videoBuffer[size];
|
|
}
|
|
|
|
void Fifo_SetRendering(bool enabled)
|
|
{
|
|
g_bSkipCurrentFrame = !enabled;
|
|
}
|
|
|
|
// May be executed from any thread, even the graphics thread.
|
|
// Created to allow for self shutdown.
|
|
void ExitGpuLoop()
|
|
{
|
|
// This should break the wait loop in CPU thread
|
|
CommandProcessor::fifo.bFF_GPReadEnable = false;
|
|
SCPFifoStruct &fifo = CommandProcessor::fifo;
|
|
while (fifo.isGpuReadingData) Common::YieldCPU();
|
|
// Terminate GPU thread loop
|
|
GpuRunningState = false;
|
|
EmuRunningState = true;
|
|
}
|
|
|
|
void EmulatorState(bool running)
|
|
{
|
|
EmuRunningState = running;
|
|
}
|
|
|
|
|
|
// Description: RunGpuLoop() sends data through this function.
|
|
void ReadDataFromFifo(u8* _uData, u32 len)
|
|
{
|
|
if (size + len >= FIFO_SIZE)
|
|
{
|
|
int pos = (int)(g_pVideoData - videoBuffer);
|
|
size -= pos;
|
|
if (size + len > FIFO_SIZE)
|
|
{
|
|
PanicAlert("FIFO out of bounds (size = %i, len = %i at %08x)", size, len, pos);
|
|
}
|
|
memmove(&videoBuffer[0], &videoBuffer[pos], size);
|
|
g_pVideoData = videoBuffer;
|
|
}
|
|
// Copy new video instructions to videoBuffer for future use in rendering the new picture
|
|
memcpy(videoBuffer + size, _uData, len);
|
|
size += len;
|
|
}
|
|
|
|
void ResetVideoBuffer()
|
|
{
|
|
g_pVideoData = videoBuffer;
|
|
size = 0;
|
|
}
|
|
|
|
|
|
// Description: Main FIFO update loop
|
|
// Purpose: Keep the Core HW updated about the CPU-GPU distance
|
|
void RunGpuLoop()
|
|
{
|
|
std::lock_guard<std::mutex> lk(m_csHWVidOccupied);
|
|
GpuRunningState = true;
|
|
SCPFifoStruct &fifo = CommandProcessor::fifo;
|
|
u32 cyclesExecuted = 0;
|
|
|
|
while (GpuRunningState)
|
|
{
|
|
g_video_backend->PeekMessages();
|
|
|
|
VideoFifo_CheckAsyncRequest();
|
|
|
|
CommandProcessor::SetCPStatusFromGPU();
|
|
|
|
Common::AtomicStore(CommandProcessor::VITicks, CommandProcessor::m_cpClockOrigin);
|
|
|
|
// check if we are able to run this buffer
|
|
while (GpuRunningState && !CommandProcessor::interruptWaiting && fifo.bFF_GPReadEnable && fifo.CPReadWriteDistance && !AtBreakpoint())
|
|
{
|
|
fifo.isGpuReadingData = true;
|
|
CommandProcessor::isPossibleWaitingSetDrawDone = fifo.bFF_GPLinkEnable ? true : false;
|
|
|
|
if (!Core::g_CoreStartupParameter.bSyncGPU || Common::AtomicLoad(CommandProcessor::VITicks) > CommandProcessor::m_cpClockOrigin)
|
|
{
|
|
u32 readPtr = fifo.CPReadPointer;
|
|
u8 *uData = Memory::GetPointer(readPtr);
|
|
|
|
if (readPtr == fifo.CPEnd)
|
|
readPtr = fifo.CPBase;
|
|
else
|
|
readPtr += 32;
|
|
|
|
_assert_msg_(COMMANDPROCESSOR, (s32)fifo.CPReadWriteDistance - 32 >= 0 ,
|
|
"Negative fifo.CPReadWriteDistance = %i in FIFO Loop !\nThat can produce instability in the game. Please report it.", fifo.CPReadWriteDistance - 32);
|
|
|
|
ReadDataFromFifo(uData, 32);
|
|
|
|
cyclesExecuted = OpcodeDecoder_Run(g_bSkipCurrentFrame, GetVideoBufferEndPtr());
|
|
|
|
if (Core::g_CoreStartupParameter.bSyncGPU && Common::AtomicLoad(CommandProcessor::VITicks) > cyclesExecuted)
|
|
Common::AtomicAdd(CommandProcessor::VITicks, -(s32)cyclesExecuted);
|
|
|
|
Common::AtomicStore(fifo.CPReadPointer, readPtr);
|
|
Common::AtomicAdd(fifo.CPReadWriteDistance, -32);
|
|
if ((GetVideoBufferEndPtr() - g_pVideoData) == 0)
|
|
Common::AtomicStore(fifo.SafeCPReadPointer, fifo.CPReadPointer);
|
|
}
|
|
|
|
CommandProcessor::SetCPStatusFromGPU();
|
|
|
|
// This call is pretty important in DualCore mode and must be called in the FIFO Loop.
|
|
// If we don't, s_swapRequested or s_efbAccessRequested won't be set to false
|
|
// leading the CPU thread to wait in Video_BeginField or Video_AccessEFB thus slowing things down.
|
|
VideoFifo_CheckAsyncRequest();
|
|
CommandProcessor::isPossibleWaitingSetDrawDone = false;
|
|
}
|
|
|
|
fifo.isGpuReadingData = false;
|
|
|
|
if (EmuRunningState)
|
|
{
|
|
// NOTE(jsd): Calling SwitchToThread() on Windows 7 x64 is a hot spot, according to profiler.
|
|
// See https://docs.google.com/spreadsheet/ccc?key=0Ah4nh0yGtjrgdFpDeF9pS3V6RUotRVE3S3J4TGM1NlE#gid=0
|
|
// for benchmark details.
|
|
#if 0
|
|
Common::YieldCPU();
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
// While the emu is paused, we still handle async requests then sleep.
|
|
while (!EmuRunningState)
|
|
{
|
|
g_video_backend->PeekMessages();
|
|
m_csHWVidOccupied.unlock();
|
|
Common::SleepCurrentThread(1);
|
|
m_csHWVidOccupied.lock();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
bool AtBreakpoint()
|
|
{
|
|
SCPFifoStruct &fifo = CommandProcessor::fifo;
|
|
return fifo.bFF_BPEnable && (fifo.CPReadPointer == fifo.CPBreakpoint);
|
|
}
|
|
|
|
void RunGpu()
|
|
{
|
|
SCPFifoStruct &fifo = CommandProcessor::fifo;
|
|
while (fifo.bFF_GPReadEnable && fifo.CPReadWriteDistance && !AtBreakpoint() )
|
|
{
|
|
u8 *uData = Memory::GetPointer(fifo.CPReadPointer);
|
|
|
|
FPURoundMode::SaveSIMDState();
|
|
FPURoundMode::LoadDefaultSIMDState();
|
|
ReadDataFromFifo(uData, 32);
|
|
OpcodeDecoder_Run(g_bSkipCurrentFrame, GetVideoBufferEndPtr());
|
|
FPURoundMode::LoadSIMDState();
|
|
|
|
//DEBUG_LOG(COMMANDPROCESSOR, "Fifo wraps to base");
|
|
|
|
if (fifo.CPReadPointer == fifo.CPEnd)
|
|
fifo.CPReadPointer = fifo.CPBase;
|
|
else
|
|
fifo.CPReadPointer += 32;
|
|
|
|
fifo.CPReadWriteDistance -= 32;
|
|
}
|
|
CommandProcessor::SetCPStatusFromGPU();
|
|
}
|