GPU: Defer frame done if not running events
Prevents interruption/execution exiting if the frame was ticked over by a MMIO access, e.g. reading GPUSTAT.
This commit is contained in:
parent
88bc3a2278
commit
f169b892c1
|
@ -49,6 +49,9 @@ static TimingEvent s_crtc_tick_event(
|
||||||
static TimingEvent s_command_tick_event(
|
static TimingEvent s_command_tick_event(
|
||||||
"GPU Command Tick", 1, 1, [](void* param, TickCount ticks, TickCount ticks_late) { g_gpu->CommandTickEvent(ticks); },
|
"GPU Command Tick", 1, 1, [](void* param, TickCount ticks, TickCount ticks_late) { g_gpu->CommandTickEvent(ticks); },
|
||||||
nullptr);
|
nullptr);
|
||||||
|
static TimingEvent s_frame_done_event(
|
||||||
|
"Frame Done", 1, 1, [](void* param, TickCount ticks, TickCount ticks_late) { g_gpu->FrameDoneEvent(ticks); },
|
||||||
|
nullptr);
|
||||||
|
|
||||||
static std::deque<std::thread> s_screenshot_threads;
|
static std::deque<std::thread> s_screenshot_threads;
|
||||||
static std::mutex s_screenshot_threads_mutex;
|
static std::mutex s_screenshot_threads_mutex;
|
||||||
|
@ -76,6 +79,7 @@ GPU::~GPU()
|
||||||
{
|
{
|
||||||
s_command_tick_event.Deactivate();
|
s_command_tick_event.Deactivate();
|
||||||
s_crtc_tick_event.Deactivate();
|
s_crtc_tick_event.Deactivate();
|
||||||
|
s_frame_done_event.Deactivate();
|
||||||
|
|
||||||
JoinScreenshotThreads();
|
JoinScreenshotThreads();
|
||||||
DestroyDeinterlaceTextures();
|
DestroyDeinterlaceTextures();
|
||||||
|
@ -1089,7 +1093,19 @@ void GPU::CRTCTickEvent(TickCount ticks)
|
||||||
UpdateCRTCTickEvent();
|
UpdateCRTCTickEvent();
|
||||||
|
|
||||||
if (frame_done)
|
if (frame_done)
|
||||||
System::FrameDone();
|
{
|
||||||
|
// we can't issue frame done if we're in the middle of executing a rec block, e.g. from reading GPUSTAT
|
||||||
|
// defer it until the end of the block in this case.
|
||||||
|
if (!TimingEvents::IsRunningEvents()) [[unlikely]]
|
||||||
|
{
|
||||||
|
DEBUG_LOG("Deferring frame done call");
|
||||||
|
s_frame_done_event.Schedule(0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
System::FrameDone();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void GPU::CommandTickEvent(TickCount ticks)
|
void GPU::CommandTickEvent(TickCount ticks)
|
||||||
|
@ -1102,6 +1118,13 @@ void GPU::CommandTickEvent(TickCount ticks)
|
||||||
m_executing_commands = false;
|
m_executing_commands = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GPU::FrameDoneEvent(TickCount ticks)
|
||||||
|
{
|
||||||
|
DebugAssert(TimingEvents::IsRunningEvents());
|
||||||
|
s_frame_done_event.Deactivate();
|
||||||
|
System::FrameDone();
|
||||||
|
}
|
||||||
|
|
||||||
void GPU::UpdateCommandTickEvent()
|
void GPU::UpdateCommandTickEvent()
|
||||||
{
|
{
|
||||||
if (m_pending_command_ticks <= 0)
|
if (m_pending_command_ticks <= 0)
|
||||||
|
|
|
@ -206,6 +206,7 @@ public:
|
||||||
// Ticks for hblank/vblank.
|
// Ticks for hblank/vblank.
|
||||||
void CRTCTickEvent(TickCount ticks);
|
void CRTCTickEvent(TickCount ticks);
|
||||||
void CommandTickEvent(TickCount ticks);
|
void CommandTickEvent(TickCount ticks);
|
||||||
|
void FrameDoneEvent(TickCount ticks);
|
||||||
|
|
||||||
// Dumps raw VRAM to a file.
|
// Dumps raw VRAM to a file.
|
||||||
bool DumpVRAMToFile(const char* filename);
|
bool DumpVRAMToFile(const char* filename);
|
||||||
|
|
|
@ -1479,6 +1479,7 @@ void System::ResetSystem()
|
||||||
Host::AddIconOSDMessage("SystemReset", ICON_FA_POWER_OFF, TRANSLATE_STR("OSDMessage", "System reset."),
|
Host::AddIconOSDMessage("SystemReset", ICON_FA_POWER_OFF, TRANSLATE_STR("OSDMessage", "System reset."),
|
||||||
Host::OSD_QUICK_DURATION);
|
Host::OSD_QUICK_DURATION);
|
||||||
|
|
||||||
|
ResetPerformanceCounters();
|
||||||
InterruptExecution();
|
InterruptExecution();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue