From 3273ca36ef72deb271aa818a1e7ba8ccca9bc0fc Mon Sep 17 00:00:00 2001 From: Albert Liu <45282415+ggrtk@users.noreply.github.com> Date: Wed, 14 Apr 2021 01:13:34 -0700 Subject: [PATCH] AnalogController: Simulate status byte toggling --- src/core/analog_controller.cpp | 43 ++++++++++++++++++++-------------- src/core/analog_controller.h | 6 ++--- src/core/save_state_version.h | 2 +- 3 files changed, 28 insertions(+), 23 deletions(-) diff --git a/src/core/analog_controller.cpp b/src/core/analog_controller.cpp index caa35db67..5bc212247 100644 --- a/src/core/analog_controller.cpp +++ b/src/core/analog_controller.cpp @@ -31,9 +31,11 @@ void AnalogController::Reset() m_configuration_mode = false; m_motor_state.fill(0); - m_rumble_unlocked = false; + m_dualshock_enabled = false; ResetRumbleConfig(); + m_status_byte = 0x5A; + if (m_force_analog_on_reset) { if (g_settings.controller_disable_analog_mode_forcing) @@ -56,10 +58,11 @@ bool AnalogController::DoState(StateWrapper& sw, bool apply_input_state) const bool old_analog_mode = m_analog_mode; sw.Do(&m_analog_mode); - sw.Do(&m_rumble_unlocked); + sw.Do(&m_dualshock_enabled); sw.DoEx(&m_legacy_rumble_unlocked, 44, false); sw.Do(&m_configuration_mode); sw.Do(&m_command_param); + sw.DoEx(&m_status_byte, 55, static_cast(0x5A)); u16 button_state = m_button_state; sw.DoEx(&button_state, 44, static_cast(0xFFFF)); @@ -261,7 +264,8 @@ void AnalogController::ProcessAnalogModeToggle() SetAnalogMode(!m_analog_mode); ResetRumbleConfig(); - // TODO: Mode switch detection (0x00 returned on certain commands instead of 0x5A) + if (m_dualshock_enabled) + m_status_byte = 0x00; } } @@ -363,21 +367,21 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::ReadPad; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; } else if (data_in == 0x43) { Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::ConfigModeSetMode; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; } else if (m_configuration_mode && data_in == 0x44) { Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::SetAnalogMode; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; ResetRumbleConfig(); } @@ -386,35 +390,35 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::GetAnalogMode; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x01, 0x02, BoolToUInt8(m_analog_mode), 0x02, 0x01, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x01, 0x02, BoolToUInt8(m_analog_mode), 0x02, 0x01, 0x00}; } else if (m_configuration_mode && data_in == 0x46) { Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::Command46; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; } else if (m_configuration_mode && data_in == 0x47) { Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::Command47; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x02, 0x00, 0x01, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x02, 0x00, 0x01, 0x00}; } else if (m_configuration_mode && data_in == 0x4C) { Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::Command4C; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; } else if (m_configuration_mode && data_in == 0x4D) { Assert(m_command_step == 0); m_response_length = (GetResponseNumHalfwords() + 1) * 2; m_command = Command::GetSetRumble; - m_tx_buffer = {GetIDByte(), GetStatusByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; + m_tx_buffer = {GetIDByte(), m_status_byte, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; m_rumble_config_large_motor_index = -1; m_rumble_config_small_motor_index = -1; @@ -440,7 +444,7 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) { m_tx_buffer[m_command_step] = Truncate8(m_button_state) & GetExtraButtonMaskLSB(); - if (m_rumble_unlocked) + if (m_dualshock_enabled) SetMotorStateForConfigIndex(rumble_index, data_in); } break; @@ -449,7 +453,7 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) { m_tx_buffer[m_command_step] = Truncate8(m_button_state >> 8); - if (m_rumble_unlocked) + if (m_dualshock_enabled) { SetMotorStateForConfigIndex(rumble_index, data_in); } @@ -466,7 +470,7 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) if (m_configuration_mode || m_analog_mode) m_tx_buffer[m_command_step] = m_axis_state[static_cast(Axis::RightX)]; - if (m_rumble_unlocked) + if (m_dualshock_enabled) SetMotorStateForConfigIndex(rumble_index, data_in); } break; @@ -476,7 +480,7 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) if (m_configuration_mode || m_analog_mode) m_tx_buffer[m_command_step] = m_axis_state[static_cast(Axis::RightY)]; - if (m_rumble_unlocked) + if (m_dualshock_enabled) SetMotorStateForConfigIndex(rumble_index, data_in); } break; @@ -486,7 +490,7 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) if (m_configuration_mode || m_analog_mode) m_tx_buffer[m_command_step] = m_axis_state[static_cast(Axis::LeftX)]; - if (m_rumble_unlocked) + if (m_dualshock_enabled) SetMotorStateForConfigIndex(rumble_index, data_in); } break; @@ -496,7 +500,7 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) if (m_configuration_mode || m_analog_mode) m_tx_buffer[m_command_step] = m_axis_state[static_cast(Axis::LeftY)]; - if (m_rumble_unlocked) + if (m_dualshock_enabled) SetMotorStateForConfigIndex(rumble_index, data_in); } break; @@ -567,7 +571,10 @@ bool AnalogController::Transfer(const u8 data_in, u8* data_out) m_configuration_mode = (m_rx_buffer[2] == 1); if (m_configuration_mode) - m_rumble_unlocked = true; + { + m_dualshock_enabled = true; + m_status_byte = 0x5A; + } Log_DevPrintf("0x%02x(%s) config mode", m_rx_buffer[2], m_configuration_mode ? "enter" : "leave"); } diff --git a/src/core/analog_controller.h b/src/core/analog_controller.h index 659516d1e..71060fbc4 100644 --- a/src/core/analog_controller.h +++ b/src/core/analog_controller.h @@ -109,9 +109,6 @@ private: u8 GetModeID() const; u8 GetIDByte() const; - // TODO: Return 0x00 on manual toggles - constexpr u8 GetStatusByte() const { return 0x5A; }; - void SetAnalogMode(bool enabled); void ProcessAnalogModeToggle(); void SetMotorState(u8 motor, u8 value); @@ -128,7 +125,7 @@ private: bool m_analog_mode = false; bool m_analog_locked = false; - bool m_rumble_unlocked = false; + bool m_dualshock_enabled = false; bool m_configuration_mode = false; std::array(Axis::Count)> m_axis_state{}; @@ -144,6 +141,7 @@ private: int m_rumble_config_small_motor_index = -1; bool m_analog_toggle_queued = false; + u8 m_status_byte = 0x5A; // TODO: Set this with command 0x4D and increase response length in digital mode accordingly u8 m_digital_mode_extra_halfwords = 0; diff --git a/src/core/save_state_version.h b/src/core/save_state_version.h index bbed77941..803f853b5 100644 --- a/src/core/save_state_version.h +++ b/src/core/save_state_version.h @@ -2,7 +2,7 @@ #include "types.h" static constexpr u32 SAVE_STATE_MAGIC = 0x43435544; -static constexpr u32 SAVE_STATE_VERSION = 54; +static constexpr u32 SAVE_STATE_VERSION = 55; static constexpr u32 SAVE_STATE_MINIMUM_VERSION = 42; static_assert(SAVE_STATE_VERSION >= SAVE_STATE_MINIMUM_VERSION);