#include #include #include "log.h" #include "SwitchUSBDevice.h" #include "ControllerHelpers.h" #include "SwitchHDLHandler.h" #include "SwitchAbstractedPadHandler.h" #include "configFile.h" struct VendorEvent { uint16_t vendor; Event event; }; Result mainLoop() { WriteToLog("\n\nNew sysmodule session started"); Result rc = 0; bool useAbstractedPad = hosversionBetween(5, 7); VendorEvent events[2]; std::vector> controllerInterfaces; UTimer filecheckTimer; Waiter filecheckTimerWaiter = waiterForUTimer(&filecheckTimer); utimerCreate(&filecheckTimer, 1e+9L, TimerType_Repeating); utimerStart(&filecheckTimer); CheckForFileChanges(); LoadAllConfigs(); { UsbHsInterfaceFilter filter; filter.Flags = UsbHsInterfaceFilterFlags_idVendor; filter.idVendor = VENDOR_MICROSOFT; events[0] = {VENDOR_MICROSOFT, Event()}; rc = usbHsCreateInterfaceAvailableEvent(&events[0].event, true, 0, &filter); if (R_FAILED(rc)) WriteToLog("Failed to open event for microsoft"); else WriteToLog("Successfully created event for microsoft"); filter.Flags = UsbHsInterfaceFilterFlags_idVendor | UsbHsInterfaceFilterFlags_idProduct; filter.idVendor = VENDOR_SONY; filter.idProduct = PRODUCT_DUALSHOCK3; events[1] = {VENDOR_SONY, Event()}; rc = usbHsCreateInterfaceAvailableEvent(&events[1].event, true, 1, &filter); if (R_FAILED(rc)) WriteToLog("Failed to open event for sony dualshock 3"); else WriteToLog("Successfully created event for sony dualshock 3"); } controllerInterfaces.reserve(8); while (appletMainLoop()) { #ifdef __APPLET__ hidScanInput(); u64 kDown = hidKeysDown(CONTROLLER_P1_AUTO); /* if (kDown & KEY_Y) { for (int i = 0; i != 8; ++i) { bool isConnected = hidIsControllerConnected(static_cast(i)); WriteToLog("Is controller ", i, " connected: ", isConnected); if (isConnected) { HidControllerType type = hidGetControllerType(static_cast(i)); WriteToLog("Controller ", i, " type: ", type); } } } if (kDown & KEY_X) { WriteToLog("Sending rumble to all controllers\n"); for (auto &&con : controllerInterfaces) { u32 *vibrationHandle = con->GetVibrationHandle(); bool flag; Result rc = hidIsVibrationDeviceMounted(vibrationHandle, &flag); if (R_SUCCEEDED(rc)) { WriteToLog("Is vibration device mounted: ", flag); HidVibrationValue value; value.amp_high = 0.5f; value.amp_low = 0.5f; value.freq_high = 0.0f; value.freq_low = 0.0f; rc = hidSendVibrationValue(vibrationHandle, &value); WriteToLog("SendVirationValue result: ", rc); rc = hidGetActualVibrationValue(vibrationHandle, &value); WriteToLog("hidGetActualVibrationValue result: ", rc); if (R_SUCCEEDED(rc)) { WriteToLog("Amp high: ", value.amp_high, " Amp low: ", value.amp_low, " Freq high: ", value.freq_high, " Freq low: ", value.freq_low); } } else WriteToLog("failed to check for vibration device"); } } */ if (kDown & KEY_B) break; #endif //Iterate over each event and check if it went off, then iterate over each vendor product to see which one fits for (auto &&event : events) { rc = eventWait(&event.event, 0); if (R_SUCCEEDED(rc)) { WriteToLog("Succeeded event ", event.vendor); WriteToLog("Interfaces size: ", controllerInterfaces.size(), "; capacity: ", controllerInterfaces.capacity()); for (auto &&product : GetVendorProducts(event.vendor)) { if (controllerInterfaces.size() == 8) { WriteToLog("Reached controller limit! skipping initialization"); break; } UsbHsInterface interfaces[4]; s32 total_entries; UsbHsInterfaceFilter tempFilter; tempFilter.Flags = UsbHsInterfaceFilterFlags_idVendor | UsbHsInterfaceFilterFlags_idProduct; tempFilter.idVendor = event.vendor; tempFilter.idProduct = product; rc = usbHsQueryAvailableInterfaces(&tempFilter, interfaces, sizeof(interfaces), &total_entries); if (R_FAILED(rc)) continue; if (total_entries == 0) continue; std::unique_ptr switchHandler; if (useAbstractedPad) switchHandler = std::make_unique(ConstructControllerFromType(GetControllerTypeFromIds(event.vendor, product), std::make_unique(interfaces, total_entries))); else switchHandler = std::make_unique(ConstructControllerFromType(GetControllerTypeFromIds(event.vendor, product), std::make_unique(interfaces, total_entries))); rc = switchHandler->Initialize(); if (R_SUCCEEDED(rc)) { controllerInterfaces.push_back(std::move(switchHandler)); WriteToLog("Interface created successfully on product ", product); } else { WriteToLog("Error creating interface for product ", product, " with error ", rc); } } } } //On interface change event, check if any devices were removed, and erase them from memory appropriately rc = eventWait(usbHsGetInterfaceStateChangeEvent(), 0); if (R_SUCCEEDED(rc)) { WriteToLog("Interface state was changed"); eventClear(usbHsGetInterfaceStateChangeEvent()); UsbHsInterface interfaces[4]; s32 total_entries; rc = usbHsQueryAcquiredInterfaces(interfaces, sizeof(interfaces), &total_entries); if (R_SUCCEEDED(rc)) { for (auto it = controllerInterfaces.begin(); it != controllerInterfaces.end(); ++it) { bool found_flag = false; for (auto &&ptr : (*it)->GetController()->GetDevice()->GetInterfaces()) { //We check if a device was removed by comparing the controller's interfaces and the currently acquired interfaces //If we didn't find a single matching interface ID, we consider a controller removed for (int i = 0; i != total_entries; ++i) { if (interfaces[i].inf.ID == static_cast(ptr.get())->GetID()) { found_flag = true; break; } } } if (!found_flag) { WriteToLog("Erasing controller! ", (*it)->GetController()->GetType()); controllerInterfaces.erase(it--); WriteToLog("Controller erased!"); } } } } rc = waitSingle(filecheckTimerWaiter, 0); if (R_SUCCEEDED(rc)) { if (CheckForFileChanges()) { WriteToLog("File check succeeded! Loading configs..."); LoadAllConfigs(); } } #ifdef __APPLET__ consoleUpdate(nullptr); #else svcSleepThread(1e+7L); #endif } //After we break out of the loop, close all events and exit for (auto &&event : events) { WriteToLog("Destroying event " + event.vendor); usbHsDestroyInterfaceAvailableEvent(&event.event, 0); } //controllerInterfaces.clear(); return rc; }