From 4c50f5b55e13283e47cba98e5bf717e62cdd8676 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 | 5 +++++ Source/Core/Core/Config/MainSettings.h | 4 ++++ Source/Core/Core/HW/WiimoteReal/IOLinux.cpp | 24 +++++++++++++++++++++ Source/Core/Core/HW/WiimoteReal/IOLinux.h | 2 ++ 4 files changed, 35 insertions(+) diff --git a/Source/Core/Core/Config/MainSettings.cpp b/Source/Core/Core/Config/MainSettings.cpp index 0442347fdd..a029fcb4d8 100644 --- a/Source/Core/Core/Config/MainSettings.cpp +++ b/Source/Core/Core/Config/MainSettings.cpp @@ -523,6 +523,11 @@ const Info MAIN_BLUETOOTH_PASSTHROUGH_PID{{System::Main, "BluetoothPassthro const Info MAIN_BLUETOOTH_PASSTHROUGH_LINK_KEYS{ {System::Main, "BluetoothPassthrough", "LinkKeys"}, ""}; +// Main.RealWiimote + +const Info MAIN_REAL_WIIMOTE_KNOWN_ADDRESSES{ + {System::Main, "RealWiimote", "KnownAddresses"}, ""}; + // Main.USBPassthrough const Info MAIN_USB_PASSTHROUGH_DEVICES{{System::Main, "USBPassthrough", "Devices"}, diff --git a/Source/Core/Core/Config/MainSettings.h b/Source/Core/Core/Config/MainSettings.h index b6e8f966c7..c120d0fb89 100644 --- a/Source/Core/Core/Config/MainSettings.h +++ b/Source/Core/Core/Config/MainSettings.h @@ -345,6 +345,10 @@ extern const Info MAIN_BLUETOOTH_PASSTHROUGH_VID; extern const Info MAIN_BLUETOOTH_PASSTHROUGH_PID; extern const Info MAIN_BLUETOOTH_PASSTHROUGH_LINK_KEYS; +// Main.RealWiimote + +extern const Info MAIN_REAL_WIIMOTE_KNOWN_ADDRESSES; + // Main.USBPassthrough extern const Info MAIN_USB_PASSTHROUGH_DEVICES; diff --git a/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp b/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp index b8db0b67cc..b5466d47d8 100644 --- a/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp +++ b/Source/Core/Core/HW/WiimoteReal/IOLinux.cpp @@ -12,6 +12,7 @@ #include "Common/CommonTypes.h" #include "Common/Logging/Log.h" +#include "Core/Config/MainSettings.h" namespace WiimoteReal { @@ -50,6 +51,8 @@ bool WiimoteScannerLinux::IsReady() const void WiimoteScannerLinux::FindWiimotes(std::vector& found_wiimotes, Wiimote*& found_board) { + WiimoteScannerLinux::AddKnownAddresses(found_wiimotes); + // supposedly 1.28 seconds int const wait_len = 1; @@ -110,6 +113,27 @@ void WiimoteScannerLinux::FindWiimotes(std::vector& found_wiimotes, Wi } } +void WiimoteScannerLinux::AddKnownAddresses(std::vector& found_wiimotes) +{ + std::string entries = Config::Get(Config::MAIN_REAL_WIIMOTE_KNOWN_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..7e5bdcd236 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 AddKnownAddresses(std::vector&); }; } // namespace WiimoteReal