#include "bluetooth_core.hpp" #include #include #include #include "../btdrv_mitm_flags.hpp" #include "../controllermanager.hpp" #include "../btdrv_mitm_logging.hpp" namespace ams::bluetooth::core { namespace { const constexpr char* g_licProControllerName = "Lic Pro Controller"; std::atomic g_isInitialized(false); os::Mutex g_eventDataLock(false); u8 g_eventDataBuffer[0x400]; BluetoothEventType g_currentEventType; os::SystemEventType g_btSystemEvent; os::SystemEventType g_btSystemEventFwd; os::SystemEventType g_btSystemEventUser; os::EventType g_dataReadEvent; void _LogEvent(BluetoothEventType type, BluetoothEventData *eventData) { size_t dataSize; switch (type) { case BluetoothEvent_DeviceFound: dataSize = sizeof(eventData->deviceFound); break; case BluetoothEvent_DiscoveryStateChanged: dataSize = sizeof(eventData->discoveryState); break; case BluetoothEvent_PinRequest: dataSize = sizeof(eventData->pinReply); break; case BluetoothEvent_SspRequest: dataSize = sizeof(eventData->sspReply); break; case BluetoothEvent_BondStateChanged: if (hos::GetVersion() < hos::Version_9_0_0) dataSize = sizeof(eventData->bondState); else dataSize = sizeof(eventData->bondState.v2); break; default: dataSize = sizeof(g_eventDataBuffer); break; } //BTDRV_LOG_DATA(eventData, dataSize); BTDRV_LOG_DATA_MSG(eventData, dataSize, "[%02d] Bluetooth core event", type); } } bool IsInitialized(void) { return g_isInitialized; } os::SystemEventType *GetSystemEvent(void) { return &g_btSystemEvent; } os::SystemEventType *GetForwardEvent(void) { return &g_btSystemEventFwd; } os::SystemEventType *GetUserForwardEvent(void) { return &g_btSystemEventUser; } Result Initialize(Handle eventHandle) { //os::AttachReadableHandleToSystemEvent(&g_btSystemEvent, eventHandle, false, os::EventClearMode_AutoClear); os::AttachReadableHandleToSystemEvent(&g_btSystemEvent, eventHandle, false, os::EventClearMode_ManualClear); R_TRY(os::CreateSystemEvent(&g_btSystemEventFwd, os::EventClearMode_AutoClear, true)); R_TRY(os::CreateSystemEvent(&g_btSystemEventUser, os::EventClearMode_AutoClear, true)); os::InitializeEvent(&g_dataReadEvent, false, os::EventClearMode_AutoClear); g_isInitialized = true; return ams::ResultSuccess(); } void Finalize(void) { os::FinalizeEvent(&g_dataReadEvent); os::DestroySystemEvent(&g_btSystemEventUser); os::DestroySystemEvent(&g_btSystemEventFwd); g_isInitialized = false; } void handleDeviceFoundEvent(BluetoothEventData *eventData) { if (ams::mitm::btdrv::IsController(&eventData->deviceFound.cod) && !ams::mitm::btdrv::IsValidSwitchControllerName(eventData->deviceFound.name)) { std::strncpy(eventData->deviceFound.name, g_licProControllerName, sizeof(BluetoothName) - 1); eventData->pinReply.cod = {0x00, 0x25, 0x08}; } else { BTDRV_LOG_FMT("handleDeviceFoundEvent: [%02x%02x%02x] | %s", eventData->deviceFound.cod.cod[0], eventData->deviceFound.cod.cod[1], eventData->deviceFound.cod.cod[2], eventData->deviceFound.name ); } } void handlePinRequesEvent(BluetoothEventData *eventData) { if (ams::mitm::btdrv::IsController(&eventData->pinReply.cod) && !ams::mitm::btdrv::IsValidSwitchControllerName(eventData->pinReply.name)) { std::strncpy(eventData->pinReply.name, g_licProControllerName, sizeof(BluetoothName) - 1); eventData->pinReply.cod = {0x00, 0x25, 0x08}; } else { BTDRV_LOG_FMT("handleDeviceFoundEvent: [%02x%02x%02x] | %s", eventData->pinReply.cod.cod[0], eventData->pinReply.cod.cod[1], eventData->pinReply.cod.cod[2], eventData->pinReply.name ); } } void handleSspRequesEvent(BluetoothEventData *eventData) { if (ams::mitm::btdrv::IsController(&eventData->sspReply.cod) && !ams::mitm::btdrv::IsValidSwitchControllerName(eventData->sspReply.name)) { std::strncpy(eventData->sspReply.name, g_licProControllerName, sizeof(BluetoothName) - 1); eventData->pinReply.cod = {0x00, 0x25, 0x08}; } else { BTDRV_LOG_FMT("handleDeviceFoundEvent: [%02x%02x%02x] | %s", eventData->sspReply.cod.cod[0], eventData->sspReply.cod.cod[1], eventData->sspReply.cod.cod[2], eventData->sspReply.name ); } } Result GetEventInfo(ncm::ProgramId program_id, BluetoothEventType *type, u8* buffer, size_t size) { std::scoped_lock lk(g_eventDataLock); *type = g_currentEventType; std::memcpy(buffer, g_eventDataBuffer, size); BluetoothEventData *eventData = reinterpret_cast(buffer); if (program_id == ncm::SystemProgramId::Btm) { switch (g_currentEventType) { case BluetoothEvent_DeviceFound: handleDeviceFoundEvent(eventData); break; case BluetoothEvent_DiscoveryStateChanged: break; case BluetoothEvent_PinRequest: handlePinRequesEvent(eventData); break; case BluetoothEvent_SspRequest: handleSspRequesEvent(eventData); break; case BluetoothEvent_BondStateChanged: break; default: break; } } //_LogEvent(g_currentEventType, eventData); os::SignalEvent(&g_dataReadEvent); return ams::ResultSuccess(); } void HandleEvent(void) { { std::scoped_lock lk(g_eventDataLock); R_ABORT_UNLESS(btdrvGetEventInfo(&g_currentEventType, g_eventDataBuffer, sizeof(g_eventDataBuffer))); } if (!g_redirectEvents) { os::SignalSystemEvent(&g_btSystemEventFwd); os::WaitEvent(&g_dataReadEvent); } if (g_btSystemEventUser.state) { //os::SignalSystemEvent(&g_btSystemEventUser); //os::TimedWaitEvent(&g_dataReadEvent, TimeSpan::FromMilliSeconds(500)); } BTDRV_LOG_FMT("[%02d] Core Event", g_currentEventType); } }