| 
									
										
										
										
											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-07-07 10:01:08 +00:00
										 |  |  | #include <unordered_map>
 | 
					
						
							| 
									
										
										
										
											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"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-13 18:48:19 +00:00
										 |  |  | struct libusb_context; | 
					
						
							|  |  |  | struct libusb_device; | 
					
						
							|  |  |  | struct libusb_device_handle; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | namespace GCAdapter { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-24 11:39:30 -04:00
										 |  |  | enum class PadButton { | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  |     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-07-03 11:52:07 -04:00
										 |  |  | extern const std::array<PadButton, 12> PadButtonArray; | 
					
						
							| 
									
										
										
										
											2020-06-23 17:37:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-24 11:39:30 -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 18:43:01 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 THRESHOLD = 10; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-04 00:40:48 -04:00
										 |  |  |     // 256/4, at least a quarter press to count as a press. For polling mostly
 | 
					
						
							|  |  |  |     static constexpr u8 TRIGGER_THRESHOLD = 64; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-24 11:39:30 -04:00
										 |  |  |     u8 port{}; | 
					
						
							|  |  |  |     PadAxes axis{PadAxes::Undefined}; | 
					
						
							|  |  |  |     u8 axis_value{255}; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-14 11:23:10 -04:00
										 |  |  |     /// Returns true if there is a device connected to port
 | 
					
						
							|  |  |  |     bool DeviceConnected(std::size_t port); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  |     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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-07 12:20:59 -04:00
										 |  |  |     int GetOriginValue(int port, int axis) const; | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-07-12 15:36:27 -04:00
										 |  |  |     GCPadStatus GetPadStatus(std::size_t 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
										 |  |  |     /// Resets status of device connected to port
 | 
					
						
							| 
									
										
										
										
											2020-07-12 15:36:27 -04:00
										 |  |  |     void ResetDeviceType(std::size_t 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; | 
					
						
							| 
									
										
										
										
											2020-07-01 12:52:50 -04:00
										 |  |  |     u8 output_endpoint = 0; | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool configuring = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::array<Common::SPSCQueue<GCPadStatus>, 4> pad_queue; | 
					
						
							|  |  |  |     std::array<GCState, 4> state; | 
					
						
							| 
									
										
										
										
											2020-07-06 21:58:31 -04:00
										 |  |  |     std::array<bool, 4> get_origin; | 
					
						
							|  |  |  |     std::array<GCPadStatus, 4> origin_status; | 
					
						
							| 
									
										
										
										
											2020-06-21 18:43:01 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-21 12:36:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 18:11:59 -04:00
										 |  |  | } // namespace GCAdapter
 |