| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | // Copyright 2014 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2+
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 22:58:53 -04:00
										 |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2020-07-13 18:48:19 +00:00
										 |  |  | #include <libusb.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "input_common/gcadapter/gc_adapter.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace GCAdapter { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  | /// Used to loop through and assign button in poller
 | 
					
						
							|  |  |  | constexpr std::array<PadButton, 12> PadButtonArray{ | 
					
						
							|  |  |  |     PadButton::PAD_BUTTON_LEFT, PadButton::PAD_BUTTON_RIGHT, PadButton::PAD_BUTTON_DOWN, | 
					
						
							|  |  |  |     PadButton::PAD_BUTTON_UP,   PadButton::PAD_TRIGGER_Z,    PadButton::PAD_TRIGGER_R, | 
					
						
							|  |  |  |     PadButton::PAD_TRIGGER_L,   PadButton::PAD_BUTTON_A,     PadButton::PAD_BUTTON_B, | 
					
						
							|  |  |  |     PadButton::PAD_BUTTON_X,    PadButton::PAD_BUTTON_Y,     PadButton::PAD_BUTTON_START, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | Adapter::Adapter() { | 
					
						
							|  |  |  |     if (usb_adapter_handle != nullptr) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     LOG_INFO(Input, "GC Adapter Initialization started"); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 13:19:33 -04:00
										 |  |  |     const int init_res = libusb_init(&libusb_ctx); | 
					
						
							|  |  |  |     if (init_res == LIBUSB_SUCCESS) { | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |         Setup(); | 
					
						
							| 
									
										
										
										
											2020-07-08 13:19:33 -04:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "libusb could not be initialized. failed with error = {}", init_res); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-12 15:36:27 -04:00
										 |  |  | GCPadStatus Adapter::GetPadStatus(std::size_t port, const std::array<u8, 37>& adapter_payload) { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     GCPadStatus pad = {}; | 
					
						
							| 
									
										
										
										
											2020-07-17 12:10:32 -04:00
										 |  |  |     const std::size_t offset = 1 + (9 * port); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     adapter_controllers_status[port] = static_cast<ControllerTypes>(adapter_payload[offset] >> 4); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |     static constexpr std::array<PadButton, 8> b1_buttons{ | 
					
						
							| 
									
										
										
										
											2020-06-24 11:39:30 -04:00
										 |  |  |         PadButton::PAD_BUTTON_A,    PadButton::PAD_BUTTON_B,    PadButton::PAD_BUTTON_X, | 
					
						
							|  |  |  |         PadButton::PAD_BUTTON_Y,    PadButton::PAD_BUTTON_LEFT, PadButton::PAD_BUTTON_RIGHT, | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |         PadButton::PAD_BUTTON_DOWN, PadButton::PAD_BUTTON_UP, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |     static constexpr std::array<PadButton, 4> b2_buttons{ | 
					
						
							|  |  |  |         PadButton::PAD_BUTTON_START, | 
					
						
							|  |  |  |         PadButton::PAD_TRIGGER_Z, | 
					
						
							|  |  |  |         PadButton::PAD_TRIGGER_R, | 
					
						
							|  |  |  |         PadButton::PAD_TRIGGER_L, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     static constexpr std::array<PadAxes, 6> axes{ | 
					
						
							|  |  |  |         PadAxes::StickX,    PadAxes::StickY,      PadAxes::SubstickX, | 
					
						
							|  |  |  |         PadAxes::SubstickY, PadAxes::TriggerLeft, PadAxes::TriggerRight, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-06 22:09:07 -04:00
										 |  |  |     if (adapter_controllers_status[port] == ControllerTypes::None && !get_origin[port]) { | 
					
						
							|  |  |  |         // Controller may have been disconnected, recalibrate if reconnected.
 | 
					
						
							|  |  |  |         get_origin[port] = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     if (adapter_controllers_status[port] != ControllerTypes::None) { | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |         const u8 b1 = adapter_payload[offset + 1]; | 
					
						
							|  |  |  |         const u8 b2 = adapter_payload[offset + 2]; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |         for (std::size_t i = 0; i < b1_buttons.size(); ++i) { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |             if ((b1 & (1U << i)) != 0) { | 
					
						
							| 
									
										
										
										
											2020-06-24 11:39:30 -04:00
										 |  |  |                 pad.button |= static_cast<u16>(b1_buttons[i]); | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |         for (std::size_t j = 0; j < b2_buttons.size(); ++j) { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |             if ((b2 & (1U << j)) != 0) { | 
					
						
							| 
									
										
										
										
											2020-06-24 11:39:30 -04:00
										 |  |  |                 pad.button |= static_cast<u16>(b2_buttons[j]); | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |         for (PadAxes axis : axes) { | 
					
						
							| 
									
										
										
										
											2020-07-17 12:10:32 -04:00
										 |  |  |             const std::size_t index = static_cast<std::size_t>(axis); | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |             pad.axis_values[index] = adapter_payload[offset + 3 + index]; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (get_origin[port]) { | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |             origin_status[port].axis_values = pad.axis_values; | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  |             get_origin[port] = false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     return pad; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  | void Adapter::PadToState(const GCPadStatus& pad, GCState& state) { | 
					
						
							| 
									
										
										
										
											2020-06-26 23:46:49 -04:00
										 |  |  |     for (const auto& button : PadButtonArray) { | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |         const u16 button_value = static_cast<u16>(button); | 
					
						
							| 
									
										
										
										
											2020-06-24 11:39:30 -04:00
										 |  |  |         state.buttons.insert_or_assign(button_value, pad.button & button_value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     for (size_t i = 0; i < pad.axis_values.size(); ++i) { | 
					
						
							|  |  |  |         state.axes.insert_or_assign(static_cast<u8>(i), pad.axis_values[i]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | void Adapter::Read() { | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |     LOG_DEBUG(Input, "GC Adapter Read() thread started"); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     int payload_size; | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  |     std::array<u8, 37> adapter_payload; | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     std::array<GCPadStatus, 4> pads; | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     while (adapter_thread_running) { | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  |         libusb_interrupt_transfer(usb_adapter_handle, input_endpoint, adapter_payload.data(), | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |                                   sizeof(adapter_payload), &payload_size, 16); | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |         if (payload_size != sizeof(adapter_payload) || adapter_payload[0] != LIBUSB_DT_HID) { | 
					
						
							| 
									
										
										
										
											2020-07-17 12:10:32 -04:00
										 |  |  |             LOG_ERROR(Input, | 
					
						
							|  |  |  |                       "Error reading payload (size: {}, type: {:02x}) Is the adapter connected?", | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |                       payload_size, adapter_payload[0]); | 
					
						
							| 
									
										
										
										
											2020-06-25 19:31:51 -04:00
										 |  |  |             adapter_thread_running = false; // error reading from adapter, stop reading.
 | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |         for (std::size_t port = 0; port < pads.size(); ++port) { | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |             pads[port] = GetPadStatus(port, adapter_payload); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |             if (DeviceConnected(port) && configuring) { | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  |                 if (pads[port].button != 0) { | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |                     pad_queue[port].Push(pads[port]); | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |                 // Accounting for a threshold here to ensure an intentional press
 | 
					
						
							|  |  |  |                 for (size_t i = 0; i < pads[port].axis_values.size(); ++i) { | 
					
						
							|  |  |  |                     const u8 value = pads[port].axis_values[i]; | 
					
						
							|  |  |  |                     const u8 origin = origin_status[port].axis_values[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (value > origin + pads[port].THRESHOLD || | 
					
						
							|  |  |  |                         value < origin - pads[port].THRESHOLD) { | 
					
						
							|  |  |  |                         pads[port].axis = static_cast<PadAxes>(i); | 
					
						
							|  |  |  |                         pads[port].axis_value = pads[port].axis_values[i]; | 
					
						
							|  |  |  |                         pad_queue[port].Push(pads[port]); | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2020-07-02 15:54:44 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |             PadToState(pads[port], state[port]); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         std::this_thread::yield(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | void Adapter::Setup() { | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     // Initialize all controllers as unplugged
 | 
					
						
							| 
									
										
										
										
											2020-06-21 21:15:58 -04:00
										 |  |  |     adapter_controllers_status.fill(ControllerTypes::None); | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     // Initialize all ports to store axis origin values
 | 
					
						
							|  |  |  |     get_origin.fill(true); | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |     // pointer to list of connected usb devices
 | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |     libusb_device** devices{}; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 17:28:02 -04:00
										 |  |  |     // populate the list of devices, get the count
 | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |     const ssize_t device_count = libusb_get_device_list(libusb_ctx, &devices); | 
					
						
							| 
									
										
										
										
											2020-07-08 13:19:33 -04:00
										 |  |  |     if (device_count < 0) { | 
					
						
							|  |  |  |         LOG_ERROR(Input, "libusb_get_device_list failed with error: {}", device_count); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |     if (devices != nullptr) { | 
					
						
							| 
									
										
										
										
											2020-07-12 15:36:27 -04:00
										 |  |  |         for (std::size_t index = 0; index < static_cast<std::size_t>(device_count); ++index) { | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |             if (CheckDeviceAccess(devices[index])) { | 
					
						
							|  |  |  |                 // GC Adapter found and accessible, registering it
 | 
					
						
							|  |  |  |                 GetGCEndpoint(devices[index]); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |         libusb_free_device_list(devices, 1); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | bool Adapter::CheckDeviceAccess(libusb_device* device) { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     libusb_device_descriptor desc; | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     const int get_descriptor_error = libusb_get_device_descriptor(device, &desc); | 
					
						
							|  |  |  |     if (get_descriptor_error) { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         // could not acquire the descriptor, no point in trying to use it.
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |         LOG_ERROR(Input, "libusb_get_device_descriptor failed with error: {}", | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |                   get_descriptor_error); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (desc.idVendor != 0x057e || desc.idProduct != 0x0337) { | 
					
						
							| 
									
										
										
										
											2020-06-26 23:46:49 -04:00
										 |  |  |         // This isn't the device we are looking for.
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     const int open_error = libusb_open(device, &usb_adapter_handle); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     if (open_error == LIBUSB_ERROR_ACCESS) { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |         LOG_ERROR(Input, "Yuzu can not gain access to this device: ID {:04X}:{:04X}.", | 
					
						
							|  |  |  |                   desc.idVendor, desc.idProduct); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     if (open_error) { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |         LOG_ERROR(Input, "libusb_open failed to open device with error = {}", open_error); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     int kernel_driver_error = libusb_kernel_driver_active(usb_adapter_handle, 0); | 
					
						
							|  |  |  |     if (kernel_driver_error == 1) { | 
					
						
							|  |  |  |         kernel_driver_error = libusb_detach_kernel_driver(usb_adapter_handle, 0); | 
					
						
							|  |  |  |         if (kernel_driver_error != 0 && kernel_driver_error != LIBUSB_ERROR_NOT_SUPPORTED) { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |             LOG_ERROR(Input, "libusb_detach_kernel_driver failed with error = {}", | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |                       kernel_driver_error); | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     if (kernel_driver_error && kernel_driver_error != LIBUSB_ERROR_NOT_SUPPORTED) { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         libusb_close(usb_adapter_handle); | 
					
						
							|  |  |  |         usb_adapter_handle = nullptr; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  |     const int interface_claim_error = libusb_claim_interface(usb_adapter_handle, 0); | 
					
						
							|  |  |  |     if (interface_claim_error) { | 
					
						
							| 
									
										
										
										
											2020-07-03 11:52:07 -04:00
										 |  |  |         LOG_ERROR(Input, "libusb_claim_interface failed with error = {}", interface_claim_error); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         libusb_close(usb_adapter_handle); | 
					
						
							|  |  |  |         usb_adapter_handle = nullptr; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | void Adapter::GetGCEndpoint(libusb_device* device) { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     libusb_config_descriptor* config = nullptr; | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |     const int config_descriptor_return = libusb_get_config_descriptor(device, 0, &config); | 
					
						
							|  |  |  |     if (config_descriptor_return != LIBUSB_SUCCESS) { | 
					
						
							| 
									
										
										
										
											2020-07-08 13:19:33 -04:00
										 |  |  |         LOG_ERROR(Input, "libusb_get_config_descriptor failed with error = {}", | 
					
						
							| 
									
										
										
										
											2020-07-08 14:18:54 -04:00
										 |  |  |                   config_descriptor_return); | 
					
						
							| 
									
										
										
										
											2020-07-08 13:19:33 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     for (u8 ic = 0; ic < config->bNumInterfaces; ic++) { | 
					
						
							|  |  |  |         const libusb_interface* interfaceContainer = &config->interface[ic]; | 
					
						
							|  |  |  |         for (int i = 0; i < interfaceContainer->num_altsetting; i++) { | 
					
						
							|  |  |  |             const libusb_interface_descriptor* interface = &interfaceContainer->altsetting[i]; | 
					
						
							|  |  |  |             for (u8 e = 0; e < interface->bNumEndpoints; e++) { | 
					
						
							|  |  |  |                 const libusb_endpoint_descriptor* endpoint = &interface->endpoint[e]; | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |                 if (endpoint->bEndpointAddress & LIBUSB_ENDPOINT_IN) { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |                     input_endpoint = endpoint->bEndpointAddress; | 
					
						
							| 
									
										
										
										
											2020-07-01 12:52:50 -04:00
										 |  |  |                 } else { | 
					
						
							|  |  |  |                     output_endpoint = endpoint->bEndpointAddress; | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-01 12:52:50 -04:00
										 |  |  |     // This transfer seems to be responsible for clearing the state of the adapter
 | 
					
						
							|  |  |  |     // Used to clear the "busy" state of when the device is unexpectedly unplugged
 | 
					
						
							|  |  |  |     unsigned char clear_payload = 0x13; | 
					
						
							|  |  |  |     libusb_interrupt_transfer(usb_adapter_handle, output_endpoint, &clear_payload, | 
					
						
							|  |  |  |                               sizeof(clear_payload), nullptr, 16); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     adapter_thread_running = true; | 
					
						
							| 
									
										
										
										
											2020-07-25 03:38:22 -04:00
										 |  |  |     adapter_input_thread = std::thread(&Adapter::Read, this); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | Adapter::~Adapter() { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     Reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | void Adapter::Reset() { | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |     if (adapter_thread_running) { | 
					
						
							| 
									
										
										
										
											2020-06-23 12:47:58 -04:00
										 |  |  |         adapter_thread_running = false; | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-26 00:05:57 -07:00
										 |  |  |     if (adapter_input_thread.joinable()) { | 
					
						
							|  |  |  |         adapter_input_thread.join(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 21:15:58 -04:00
										 |  |  |     adapter_controllers_status.fill(ControllerTypes::None); | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  |     get_origin.fill(true); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (usb_adapter_handle) { | 
					
						
							| 
									
										
										
										
											2020-06-25 19:31:51 -04:00
										 |  |  |         libusb_release_interface(usb_adapter_handle, 1); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |         libusb_close(usb_adapter_handle); | 
					
						
							|  |  |  |         usb_adapter_handle = nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-23 12:47:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (libusb_ctx) { | 
					
						
							|  |  |  |         libusb_exit(libusb_ctx); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-12 15:36:27 -04:00
										 |  |  | bool Adapter::DeviceConnected(std::size_t port) { | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     return adapter_controllers_status[port] != ControllerTypes::None; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-12 15:36:27 -04:00
										 |  |  | void Adapter::ResetDeviceType(std::size_t port) { | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     adapter_controllers_status[port] = ControllerTypes::None; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | void Adapter::BeginConfiguration() { | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  |     get_origin.fill(true); | 
					
						
							| 
									
										
										
										
											2020-06-21 21:15:58 -04:00
										 |  |  |     for (auto& pq : pad_queue) { | 
					
						
							|  |  |  |         pq.Clear(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     configuring = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | void Adapter::EndConfiguration() { | 
					
						
							| 
									
										
										
										
											2020-06-21 21:15:58 -04:00
										 |  |  |     for (auto& pq : pad_queue) { | 
					
						
							|  |  |  |         pq.Clear(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     configuring = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | std::array<Common::SPSCQueue<GCPadStatus>, 4>& Adapter::GetPadQueue() { | 
					
						
							|  |  |  |     return pad_queue; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  | const std::array<Common::SPSCQueue<GCPadStatus>, 4>& Adapter::GetPadQueue() const { | 
					
						
							|  |  |  |     return pad_queue; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | std::array<GCState, 4>& Adapter::GetPadState() { | 
					
						
							|  |  |  |     return state; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  | const std::array<GCState, 4>& Adapter::GetPadState() const { | 
					
						
							|  |  |  |     return state; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-07 12:20:59 -04:00
										 |  |  | int Adapter::GetOriginValue(int port, int axis) const { | 
					
						
							| 
									
										
										
										
											2020-07-16 13:00:04 -04:00
										 |  |  |     return origin_status[port].axis_values[axis]; | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  | } // namespace GCAdapter
 |