From a1b6f0b76a026e6239d19cfefb7cdc70173fa188 Mon Sep 17 00:00:00 2001 From: sonicfind <52436993+sonicfind@users.noreply.github.com> Date: Tue, 15 Dec 2020 11:15:51 -0600 Subject: [PATCH] recording: Log all PadData and draw all VirtualPads altogether once per frame Fixes improper virtualPad behavior for games that do not check for button pressure values (ex: Sonic Mega Collection Plus) --- pcsx2/Recording/InputRecording.cpp | 113 ++++++++++++--------- pcsx2/Recording/InputRecording.h | 2 + pcsx2/Recording/InputRecordingControls.cpp | 3 +- pcsx2/Recording/PadData.h | 1 - 4 files changed, 69 insertions(+), 50 deletions(-) diff --git a/pcsx2/Recording/InputRecording.cpp b/pcsx2/Recording/InputRecording.cpp index 0bd6dfd1ac..fc1486cedf 100644 --- a/pcsx2/Recording/InputRecording.cpp +++ b/pcsx2/Recording/InputRecording.cpp @@ -97,58 +97,58 @@ void InputRecording::ControllerInterrupt(u8& data, u8& port, u16& bufCount, u8 b if (bufCount == 1) fInterruptFrame = data == READ_DATA_AND_VIBRATE_FIRST_BYTE; - else if (bufCount == 2 && buf[bufCount] != READ_DATA_AND_VIBRATE_SECOND_BYTE) - fInterruptFrame = false; - - // We do not want to record or save the first two bytes in the data returned from the PAD plugin - else if (fInterruptFrame && bufCount >= 3 && frameCounter >= 0 && frameCounter < INT_MAX) + else if (bufCount == 2) + { + if (buf[bufCount] != READ_DATA_AND_VIBRATE_SECOND_BYTE) + fInterruptFrame = false; + } + else if (fInterruptFrame) { u8& bufVal = buf[bufCount]; const u16 bufIndex = bufCount - 3; - if (state == InputRecordingMode::Replaying) { - u8 tmp = 0; - if (inputRecordingData.ReadKeyBuffer(tmp, frameCounter, port, bufIndex)) + if (frameCounter >= 0 && frameCounter < INT_MAX) { - // Overwrite value originally provided by the PAD plugin - bufVal = tmp; + if (!inputRecordingData.ReadKeyBuffer(bufVal, frameCounter, port, bufIndex)) + inputRec::consoleLog(fmt::format("[REC]: Failed to read input data at frame {}", frameCounter)); + + // Update controller data state for future VirtualPad / logging usage. + padData[port]->UpdateControllerData(bufIndex, bufVal); + + if (virtualPads[port]->IsShown()) + virtualPads[port]->UpdateControllerData(bufIndex, padData[port]); } } - - // Update controller data state for future VirtualPad / logging usage. - padData[port]->UpdateControllerData(bufIndex, bufVal); - - if (virtualPads[port] && - virtualPads[port]->IsShown() && - virtualPads[port]->UpdateControllerData(bufIndex, padData[port]) && - state != InputRecordingMode::Replaying) + else { + // Update controller data state for future VirtualPad / logging usage. + padData[port]->UpdateControllerData(bufIndex, bufVal); + + // Commit the byte to the movie file if we are recording + if (state == InputRecordingMode::Recording) + { + if (frameCounter >= 0) + { + // If the VirtualPad updated the PadData, we have to update the buffer + // before committing it to the recording / sending it to the game + if (virtualPads[port]->IsShown() && virtualPads[port]->UpdateControllerData(bufIndex, padData[port])) + bufVal = padData[port]->PollControllerData(bufIndex); + + if (incrementUndo) + { + inputRecordingData.IncrementUndoCount(); + incrementUndo = false; + } + + if (frameCounter < INT_MAX && !inputRecordingData.WriteKeyBuffer(frameCounter, port, bufIndex, bufVal)) + inputRec::consoleLog(fmt::format("[REC]: Failed to write input data at frame {}", frameCounter)); + } + } // If the VirtualPad updated the PadData, we have to update the buffer - // before committing it to the recording / sending it to the game - // - Do not do this if we are in replay mode! - bufVal = padData[port]->PollControllerData(bufIndex); - } - - // If we have reached the end of the pad data, log it out - if (bufIndex == PadData::END_INDEX_CONTROLLER_BUFFER) - { - padData[port]->LogPadData(port); - // As well as re-render the virtual pad UI, if applicable - // - Don't render if it's minimized - if (virtualPads[port] && virtualPads[port]->IsShown() && !virtualPads[port]->IsIconized()) - virtualPads[port]->Redraw(); - } - - // Finally, commit the byte to the movie file if we are recording - if (state == InputRecordingMode::Recording) - { - if (incrementUndo) - { - inputRecordingData.IncrementUndoCount(); - incrementUndo = false; - } - inputRecordingData.WriteKeyBuffer(frameCounter, port, bufIndex, bufVal); + // before sending it to the game + else if (virtualPads[port]->IsShown() && virtualPads[port]->UpdateControllerData(bufIndex, padData[port])) + bufVal = padData[port]->PollControllerData(bufIndex); } } } @@ -170,12 +170,31 @@ u32 InputRecording::GetStartingFrame() void InputRecording::IncrementFrameCounter() { - frameCounter++; - if (state == InputRecordingMode::Recording) + if (frameCounter < INT_MAX) { - GetInputRecordingData().SetTotalFrames(frameCounter); - if (frameCounter == inputRecordingData.GetTotalFrames()) - incrementUndo = false; + frameCounter++; + switch (state) + { + case InputRecordingMode::Recording: + GetInputRecordingData().SetTotalFrames(frameCounter); + [[fallthrough]]; + case InputRecordingMode::Replaying: + if (frameCounter == inputRecordingData.GetTotalFrames()) + incrementUndo = false; + } + } + g_InputRecording.LogAndRedraw(); +} + +void InputRecording::LogAndRedraw() +{ + for (u8 port = 0; port < 2; port++) + { + padData[port]->LogPadData(port); + // As well as re-render the virtual pad UI, if applicable + // - Don't render if it's minimized + if (virtualPads[port]->IsShown() && !virtualPads[port]->IsIconized()) + virtualPads[port]->Redraw(); } } diff --git a/pcsx2/Recording/InputRecording.h b/pcsx2/Recording/InputRecording.h index b881a342bf..385ffc97fa 100644 --- a/pcsx2/Recording/InputRecording.h +++ b/pcsx2/Recording/InputRecording.h @@ -89,6 +89,8 @@ public: void Stop(); // Initialze VirtualPad window void setVirtualPadPtr(VirtualPad* ptr, int const port); + // Logs the padData and redraws the virtualPad windows of active pads + void LogAndRedraw(); // Resets a recording if the base savestate could not be loaded at the start void FailedSavestate(); diff --git a/pcsx2/Recording/InputRecordingControls.cpp b/pcsx2/Recording/InputRecordingControls.cpp index f91a0a38d0..e14ebb1d18 100644 --- a/pcsx2/Recording/InputRecordingControls.cpp +++ b/pcsx2/Recording/InputRecordingControls.cpp @@ -60,8 +60,7 @@ void InputRecordingControls::HandleFrameAdvanceAndPausing() else if (frameCountTracker != g_FrameCount) { frameCountTracker = g_FrameCount; - if (g_InputRecording.GetFrameCounter() < INT_MAX) - g_InputRecording.IncrementFrameCounter(); + g_InputRecording.IncrementFrameCounter(); } else { diff --git a/pcsx2/Recording/PadData.h b/pcsx2/Recording/PadData.h index 6290a94afb..1f9e79eacc 100644 --- a/pcsx2/Recording/PadData.h +++ b/pcsx2/Recording/PadData.h @@ -22,7 +22,6 @@ class PadData public: /// Constants static const u8 ANALOG_VECTOR_NEUTRAL = 127; - static const u16 END_INDEX_CONTROLLER_BUFFER = 17; enum class BufferIndex {