/* * Copyright (c) 2020-2021 ndeadly * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "controller_management.hpp" #include #include #include #include #include namespace ams::controller { namespace { const std::string official_npad_names[] = { "Joy-Con", "Pro Controller", "Lic Pro Controller", "NES Controller", "HVC Controller", "SNES Controller", "NintendoGamepad", }; constexpr auto cod_major_peripheral = 0x05; constexpr auto cod_minor_gamepad = 0x08; constexpr auto cod_minor_joystick = 0x04; constexpr auto cod_minor_keyboard = 0x40; os::Mutex g_controller_lock(false); std::vector> g_controllers; inline bool bdcmp(const bluetooth::Address *addr1, const bluetooth::Address *addr2) { return std::memcmp(addr1, addr2, sizeof(bluetooth::Address)) == 0; } } ControllerType Identify(const bluetooth::DevicesSettings *device) { if (IsOfficialSwitchControllerName(device->name.name)) return ControllerType_Switch; for (auto hwId : WiiController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Wii; } } for (auto hwId : Dualshock4Controller::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Dualshock4; } } for (auto hwId : DualsenseController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Dualsense; } } for (auto hwId : XboxOneController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_XboxOne; } } for (auto hwId : OuyaController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Ouya; } } for (auto hwId : GamestickController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Gamestick; } } for (auto hwId : GemboxController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Gembox; } } for (auto hwId : IpegaController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Ipega; } } for (auto hwId : XiaomiController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Xiaomi; } } for (auto hwId : GamesirController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Gamesir; } } for (auto hwId : SteelseriesController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Steelseries; } } for (auto hwId : NvidiaShieldController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_NvidiaShield; } } for (auto hwId : EightBitDoController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_8BitDo; } } for (auto hwId : PowerAController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_PowerA; } } for (auto hwId : MadCatzController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_MadCatz; } } for (auto hwId : MocuteController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Mocute; } } for (auto hwId : RazerController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_Razer; } } for (auto hwId : ICadeController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_ICade; } } for (auto hwId : LanShenController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_LanShen; } } for (auto hwId : AtGamesController::hardware_ids) { if ( (device->vid == hwId.vid) && (device->pid == hwId.pid) ) { return ControllerType_AtGames; } } return ControllerType_Unknown; } bool IsAllowedDeviceClass(const bluetooth::DeviceClass *cod) { return ((cod->class_of_device[1] & 0x0f) == cod_major_peripheral) && (((cod->class_of_device[2] & 0x0f) == cod_minor_gamepad) || ((cod->class_of_device[2] & 0x0f) == cod_minor_joystick) || ((cod->class_of_device[2] & 0x40) == cod_minor_keyboard)); } bool IsOfficialSwitchControllerName(const std::string& name) { for (auto n : official_npad_names) { if (name.rfind(n, 0) == 0) return true; } return false; } void AttachHandler(const bluetooth::Address *address) { std::scoped_lock lk(g_controller_lock); bluetooth::DevicesSettings device_settings; R_ABORT_UNLESS(btdrvGetPairedDeviceInfo(*address, &device_settings)); HardwareID id = { device_settings.vid, device_settings.pid }; switch (Identify(&device_settings)) { case ControllerType_Switch: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Wii: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Dualshock4: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Dualsense: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_XboxOne: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Ouya: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Gamestick: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Gembox: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Ipega: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Xiaomi: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Gamesir: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Steelseries: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_NvidiaShield: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_8BitDo: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_PowerA: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_MadCatz: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Mocute: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_Razer: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_ICade: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_LanShen: g_controllers.push_back(std::make_unique(address, id)); break; case ControllerType_AtGames: g_controllers.push_back(std::make_unique(address, id)); break; default: g_controllers.push_back(std::make_unique(address, id)); break; } R_ABORT_UNLESS(g_controllers.back()->Initialize()); } void RemoveHandler(const bluetooth::Address *address) { std::scoped_lock lk(g_controller_lock); for (auto it = g_controllers.begin(); it < g_controllers.end(); ++it) { if (bdcmp(&(*it)->Address(), address)) { g_controllers.erase(it); return; } } } SwitchController *LocateHandler(const bluetooth::Address *address) { std::scoped_lock lk(g_controller_lock); for (auto it = g_controllers.begin(); it < g_controllers.end(); ++it) { if (bdcmp(&(*it)->Address(), address)) { return (*it).get(); } } return nullptr; } }