| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  | // Copyright 2014 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2+
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  | #include <mutex>
 | 
					
						
							| 
									
										
										
										
											2020-06-21 20:32:43 -04:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2020-06-21 15:31:57 -04:00
										 |  |  | #include <libusb.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | #include "common/threadsafe_queue.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace GCAdapter { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     PAD_USE_ORIGIN = 0x0080, | 
					
						
							|  |  |  |     PAD_GET_ORIGIN = 0x2000, | 
					
						
							|  |  |  |     PAD_ERR_STATUS = 0x8000, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum PadButton { | 
					
						
							|  |  |  |     PAD_BUTTON_LEFT = 0x0001, | 
					
						
							|  |  |  |     PAD_BUTTON_RIGHT = 0x0002, | 
					
						
							|  |  |  |     PAD_BUTTON_DOWN = 0x0004, | 
					
						
							|  |  |  |     PAD_BUTTON_UP = 0x0008, | 
					
						
							|  |  |  |     PAD_TRIGGER_Z = 0x0010, | 
					
						
							|  |  |  |     PAD_TRIGGER_R = 0x0020, | 
					
						
							|  |  |  |     PAD_TRIGGER_L = 0x0040, | 
					
						
							|  |  |  |     PAD_BUTTON_A = 0x0100, | 
					
						
							|  |  |  |     PAD_BUTTON_B = 0x0200, | 
					
						
							|  |  |  |     PAD_BUTTON_X = 0x0400, | 
					
						
							|  |  |  |     PAD_BUTTON_Y = 0x0800, | 
					
						
							|  |  |  |     PAD_BUTTON_START = 0x1000, | 
					
						
							|  |  |  |     // Below is for compatibility with "AxisButton" type
 | 
					
						
							|  |  |  |     PAD_STICK = 0x2000, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | enum class PadAxes : u8 { | 
					
						
							|  |  |  |     StickX, | 
					
						
							|  |  |  |     StickY, | 
					
						
							|  |  |  |     SubstickX, | 
					
						
							|  |  |  |     SubstickY, | 
					
						
							|  |  |  |     TriggerLeft, | 
					
						
							|  |  |  |     TriggerRight, | 
					
						
							|  |  |  |     Undefined, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct GCPadStatus { | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     u16 button;       // Or-ed PAD_BUTTON_* and PAD_TRIGGER_* bits
 | 
					
						
							|  |  |  |     u8 stick_x;       // 0 <= stick_x       <= 255
 | 
					
						
							|  |  |  |     u8 stick_y;       // 0 <= stick_y       <= 255
 | 
					
						
							|  |  |  |     u8 substick_x;    // 0 <= substick_x    <= 255
 | 
					
						
							|  |  |  |     u8 substick_y;    // 0 <= substick_y    <= 255
 | 
					
						
							|  |  |  |     u8 trigger_left;  // 0 <= trigger_left  <= 255
 | 
					
						
							|  |  |  |     u8 trigger_right; // 0 <= trigger_right <= 255
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  |     static constexpr u8 MAIN_STICK_CENTER_X = 0x80; | 
					
						
							|  |  |  |     static constexpr u8 MAIN_STICK_CENTER_Y = 0x80; | 
					
						
							|  |  |  |     static constexpr u8 MAIN_STICK_RADIUS = 0x7f; | 
					
						
							|  |  |  |     static constexpr u8 C_STICK_CENTER_X = 0x80; | 
					
						
							|  |  |  |     static constexpr u8 C_STICK_CENTER_Y = 0x80; | 
					
						
							|  |  |  |     static constexpr u8 C_STICK_RADIUS = 0x7f; | 
					
						
							|  |  |  |     static constexpr u8 TRIGGER_CENTER = 20; | 
					
						
							|  |  |  |     static constexpr u8 THRESHOLD = 10; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     u8 port; | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     PadAxes axis = PadAxes::Undefined; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     u8 axis_value = 255; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct GCState { | 
					
						
							|  |  |  |     std::unordered_map<int, bool> buttons; | 
					
						
							|  |  |  |     std::unordered_map<int, u16> axes; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | enum class ControllerTypes { None, Wired, Wireless }; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     NO_ADAPTER_DETECTED = 0, | 
					
						
							|  |  |  |     ADAPTER_DETECTED = 1, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | class Adapter { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  |     /// Initialize the GC Adapter capture and read sequence
 | 
					
						
							|  |  |  |     Adapter(); | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  |     /// Close the adapter read thread and release the adapter
 | 
					
						
							|  |  |  |     ~Adapter(); | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// Used for polling
 | 
					
						
							|  |  |  |     void BeginConfiguration(); | 
					
						
							|  |  |  |     void EndConfiguration(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::array<Common::SPSCQueue<GCPadStatus>, 4>& GetPadQueue(); | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  |     const std::array<Common::SPSCQueue<GCPadStatus>, 4>& GetPadQueue() const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     std::array<GCState, 4>& GetPadState(); | 
					
						
							| 
									
										
										
										
											2020-06-21 21:50:58 -04:00
										 |  |  |     const std::array<GCState, 4>& GetPadState() const; | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  |     GCPadStatus CheckStatus(int port, const std::array<u8, 37>& adapter_payload); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  |     void PadToState(const GCPadStatus& pad, GCState& state); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     void Read(); | 
					
						
							|  |  |  |     void ScanThreadFunc(); | 
					
						
							|  |  |  |     /// Begin scanning for the GC Adapter.
 | 
					
						
							|  |  |  |     void StartScanThread(); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// Stop scanning for the adapter
 | 
					
						
							|  |  |  |     void StopScanThread(); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// Returns true if there is a device connected to port
 | 
					
						
							|  |  |  |     bool DeviceConnected(int port); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// Resets status of device connected to port
 | 
					
						
							|  |  |  |     void ResetDeviceType(int port); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// Returns true if we successfully gain access to GC Adapter
 | 
					
						
							|  |  |  |     bool CheckDeviceAccess(libusb_device* device); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// Captures GC Adapter endpoint address,
 | 
					
						
							|  |  |  |     void GetGCEndpoint(libusb_device* device); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// For shutting down, clear all data, join all threads, release usb
 | 
					
						
							|  |  |  |     void Reset(); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     /// For use in initialization, querying devices to find the adapter
 | 
					
						
							|  |  |  |     void Setup(); | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     int current_status = NO_ADAPTER_DETECTED; | 
					
						
							|  |  |  |     libusb_device_handle* usb_adapter_handle = nullptr; | 
					
						
							| 
									
										
										
										
											2020-06-21 21:15:58 -04:00
										 |  |  |     std::array<ControllerTypes, 4> adapter_controllers_status{}; | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::mutex s_mutex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::thread adapter_input_thread; | 
					
						
							|  |  |  |     bool adapter_thread_running; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::mutex initialization_mutex; | 
					
						
							|  |  |  |     std::thread detect_thread; | 
					
						
							|  |  |  |     bool detect_thread_running = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     libusb_context* libusb_ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u8 input_endpoint = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool configuring = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::array<Common::SPSCQueue<GCPadStatus>, 4> pad_queue; | 
					
						
							|  |  |  |     std::array<GCState, 4> state; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  | } // namespace GCAdapter
 |