From 97bc28aac44cd8c4393f03bf1684c2b891646a22 Mon Sep 17 00:00:00 2001 From: Thales MG <> Date: Wed, 25 Dec 2024 14:09:00 -0300 Subject: [PATCH] feat(linux): allow configuring real wiimotes with known bluetooth addresses This adds the option to configure real Wiimotes by specifying their Bluetooth addresses in the configuration file. This allows off-brand Wiimotes to work without using the Bluetooth Passthrough option, if you know their Bluetooth addresses beforehand. Despite correctly setting the LAP to `0x9e8b00` in `WiimoteScannerLinux::FindWiimotes` while scanning, which is indeed enough to make off-brand / knock-off Wiimotes respond to a Bluetooth Inquiry, some (several? all?) bluetooth adapters seem to override and ignore this given LAP value when performing the Inquiry, and actually use the `0x9e8b33` value as if a null pointer have been given to `hci_inquiry`, as inspection of USB/Bluetooth packets by Wireshark indicate. Off-brand Wiimotes don't respond to inquiries with this LAP. If one happens to know the Bluetooth address of their Wiimote (for example, by checking `BluetoothPassthrough.LinkKeys` after using Bluetooth Passthrough, or other means such as directly using `libusb` to force the adapter to use the correct LAP in the Inquiry), then it's enough to add those addresses to the vector of found Wiimotes. Since this a niche use case and I only happen to know and have tested in Linux, this change only affects the `WiimoteScannerLinux` backend. It's likely that it could be added to other backends, but I'm unfamiliar with these. If no addresses are given or this config section does not exist, behavior is completely unchanged. --- Source/Core/Core/Config/MainSettings.cpp | 2 ++ Source/Core/Core/Config/MainSettings.h | 1 + Source/Core/Core/HW/WiimoteReal/IOLinux.cpp | 24 +++++++++++++++++++++ Source/Core/Core/HW/WiimoteReal/IOLinux.h | 2 ++ 4 files changed, 29 insertions(+) diff --git a/Source/Core/Core/Config/MainSettings.cpp b/Source/Core/Core/Config/MainSettings.cpp index 0442347fdd..6fdb5b80ae 100644 --- a/Source/Core/Core/Config/MainSettings.cpp +++ b/Source/Core/Core/Config/MainSettings.cpp @@ -187,6 +187,8 @@ const Info MAIN_WII_SD_CARD_FILESIZE{{System::Main, "Core", "WiiSDCardFiles const Info MAIN_WII_KEYBOARD{{System::Main, "Core", "WiiKeyboard"}, false}; const Info MAIN_WIIMOTE_CONTINUOUS_SCANNING{ {System::Main, "Core", "WiimoteContinuousScanning"}, false}; +const Info MAIN_WIIMOTE_AUTO_CONNECT_ADDRESSES{ + {System::Main, "Core", "WiimoteAutoConnectAddresses"}, ""}; const Info MAIN_WIIMOTE_ENABLE_SPEAKER{{System::Main, "Core", "WiimoteEnableSpeaker"}, false}; const Info MAIN_CONNECT_WIIMOTES_FOR_CONTROLLER_INTERFACE{ {System::Main, "Core", "WiimoteControllerInterface"}, false}; diff --git a/Source/Core/Core/Config/MainSettings.h b/Source/Core/Core/Config/MainSettings.h index b6e8f966c7..76a1d864a8 100644 --- a/Source/Core/Core/Config/MainSettings.h +++ b/Source/Core/Core/Config/MainSettings.h @@ -106,6 +106,7 @@ extern const Info MAIN_WII_SD_CARD_ENABLE_FOLDER_SYNC; extern const Info MAIN_WII_SD_CARD_FILESIZE; extern const Info MAIN_WII_KEYBOARD; extern const Info MAIN_WIIMOTE_CONTINUOUS_SCANNING; +extern const Info MAIN_WIIMOTE_AUTO_CONNECT_ADDRESSES; extern const Info MAIN_WIIMOTE_ENABLE_SPEAKER; extern const Info MAIN_CONNECT_WIIMOTES_FOR_CONTROLLER_INTERFACE; extern const Info MAIN_MMU; diff --git a/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp b/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp index 8fe5456146..70627b31fd 100644 --- a/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp +++ b/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp @@ -14,6 +14,7 @@ #include "Common/CommonTypes.h" #include "Common/Logging/Log.h" +#include "Core/Config/MainSettings.h" namespace WiimoteReal { @@ -52,6 +53,8 @@ bool WiimoteScannerLinux::IsReady() const void WiimoteScannerLinux::FindWiimotes(std::vector& found_wiimotes, Wiimote*& found_board) { + WiimoteScannerLinux::AddAutoConnectAddresses(found_wiimotes); + // supposedly 1.28 seconds int const wait_len = 1; @@ -111,6 +114,27 @@ void WiimoteScannerLinux::FindWiimotes(std::vector& found_wiimotes, Wi } } +void WiimoteScannerLinux::AddAutoConnectAddresses(std::vector& found_wiimotes) +{ + std::string entries = Config::Get(Config::MAIN_WIIMOTE_AUTO_CONNECT_ADDRESSES); + if (entries.empty()) + return; + for (const auto& bt_address_str : SplitString(entries, ',')) + { + bdaddr_t bt_addr; + if (str2ba(bt_address_str.c_str(), &bt_addr) < 0) + { + WARN_LOG_FMT(WIIMOTE, "Bad Known Bluetooth Address: {}", bt_address_str); + continue; + } + if (!IsNewWiimote(bt_address_str)) + continue; + Wiimote* wm = new WiimoteLinux(bt_addr); + found_wiimotes.push_back(wm); + NOTICE_LOG_FMT(WIIMOTE, "Added Wiimote with fixed address ({}).", bt_address_str); + } +} + WiimoteLinux::WiimoteLinux(bdaddr_t bdaddr) : m_bdaddr(bdaddr) { m_really_disconnect = true; diff --git a/Source/Core/Core/HW/WiimoteReal/IOLinux.h b/Source/Core/Core/HW/WiimoteReal/IOLinux.h index a27faacee9..0f84659a39 100644 --- a/Source/Core/Core/HW/WiimoteReal/IOLinux.h +++ b/Source/Core/Core/HW/WiimoteReal/IOLinux.h @@ -50,6 +50,8 @@ public: private: int m_device_id; int m_device_sock; + + void AddAutoConnectAddresses(std::vector&); }; } // namespace WiimoteReal