2020-06-21 19:31:57 +00:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2+
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2020-06-21 16:36:28 +00:00
|
|
|
#pragma once
|
|
|
|
#include <algorithm>
|
|
|
|
#include <functional>
|
2020-06-21 19:31:57 +00:00
|
|
|
#include <mutex>
|
2020-06-22 00:32:43 +00:00
|
|
|
#include <thread>
|
2020-07-07 10:01:08 +00:00
|
|
|
#include <unordered_map>
|
2020-06-21 19:31:57 +00:00
|
|
|
#include <libusb.h>
|
2020-06-21 16:36:28 +00:00
|
|
|
#include "common/common_types.h"
|
2020-06-21 22:43:01 +00:00
|
|
|
#include "common/threadsafe_queue.h"
|
|
|
|
|
|
|
|
namespace GCAdapter {
|
2020-06-21 16:36:28 +00:00
|
|
|
|
|
|
|
enum {
|
|
|
|
PAD_USE_ORIGIN = 0x0080,
|
|
|
|
PAD_GET_ORIGIN = 0x2000,
|
|
|
|
PAD_ERR_STATUS = 0x8000,
|
|
|
|
};
|
|
|
|
|
2020-06-24 15:39:30 +00:00
|
|
|
enum class PadButton {
|
2020-06-21 16:36:28 +00: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 15:52:07 +00:00
|
|
|
extern const std::array<PadButton, 12> PadButtonArray;
|
2020-06-23 21:37:15 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
enum class PadAxes : u8 {
|
|
|
|
StickX,
|
|
|
|
StickY,
|
|
|
|
SubstickX,
|
|
|
|
SubstickY,
|
|
|
|
TriggerLeft,
|
|
|
|
TriggerRight,
|
|
|
|
Undefined,
|
|
|
|
};
|
2020-06-21 16:36:28 +00:00
|
|
|
|
|
|
|
struct GCPadStatus {
|
2020-06-24 15:39:30 +00: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 22:43:01 +00:00
|
|
|
|
2020-06-22 01:50:58 +00: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 04:40:48 +00: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 15:39:30 +00:00
|
|
|
u8 port{};
|
|
|
|
PadAxes axis{PadAxes::Undefined};
|
|
|
|
u8 axis_value{255};
|
2020-06-21 16:36:28 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct GCState {
|
|
|
|
std::unordered_map<int, bool> buttons;
|
|
|
|
std::unordered_map<int, u16> axes;
|
|
|
|
};
|
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
enum class ControllerTypes { None, Wired, Wireless };
|
2020-06-21 16:36:28 +00:00
|
|
|
|
|
|
|
enum {
|
|
|
|
NO_ADAPTER_DETECTED = 0,
|
|
|
|
ADAPTER_DETECTED = 1,
|
|
|
|
};
|
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
class Adapter {
|
|
|
|
public:
|
2020-06-22 01:50:58 +00:00
|
|
|
/// Initialize the GC Adapter capture and read sequence
|
|
|
|
Adapter();
|
2020-06-21 22:43:01 +00:00
|
|
|
|
2020-06-22 01:50:58 +00:00
|
|
|
/// Close the adapter read thread and release the adapter
|
|
|
|
~Adapter();
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Used for polling
|
|
|
|
void BeginConfiguration();
|
|
|
|
void EndConfiguration();
|
|
|
|
|
|
|
|
std::array<Common::SPSCQueue<GCPadStatus>, 4>& GetPadQueue();
|
2020-06-22 01:50:58 +00:00
|
|
|
const std::array<Common::SPSCQueue<GCPadStatus>, 4>& GetPadQueue() const;
|
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
std::array<GCState, 4>& GetPadState();
|
2020-06-22 01:50:58 +00:00
|
|
|
const std::array<GCState, 4>& GetPadState() const;
|
2020-06-21 22:43:01 +00:00
|
|
|
|
|
|
|
private:
|
2020-07-12 19:36:27 +00:00
|
|
|
GCPadStatus GetPadStatus(std::size_t port, const std::array<u8, 37>& adapter_payload);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-22 22:11:59 +00:00
|
|
|
void PadToState(const GCPadStatus& pad, GCState& state);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
void Read();
|
|
|
|
void ScanThreadFunc();
|
|
|
|
/// Begin scanning for the GC Adapter.
|
|
|
|
void StartScanThread();
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Stop scanning for the adapter
|
|
|
|
void StopScanThread();
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Returns true if there is a device connected to port
|
2020-07-12 19:36:27 +00:00
|
|
|
bool DeviceConnected(std::size_t port);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Resets status of device connected to port
|
2020-07-12 19:36:27 +00:00
|
|
|
void ResetDeviceType(std::size_t port);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Returns true if we successfully gain access to GC Adapter
|
|
|
|
bool CheckDeviceAccess(libusb_device* device);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Captures GC Adapter endpoint address,
|
|
|
|
void GetGCEndpoint(libusb_device* device);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// For shutting down, clear all data, join all threads, release usb
|
|
|
|
void Reset();
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// For use in initialization, querying devices to find the adapter
|
|
|
|
void Setup();
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
int current_status = NO_ADAPTER_DETECTED;
|
|
|
|
libusb_device_handle* usb_adapter_handle = nullptr;
|
2020-06-22 01:15:58 +00:00
|
|
|
std::array<ControllerTypes, 4> adapter_controllers_status{};
|
2020-06-21 22:43:01 +00: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 16:52:50 +00:00
|
|
|
u8 output_endpoint = 0;
|
2020-06-21 22:43:01 +00:00
|
|
|
|
|
|
|
bool configuring = false;
|
|
|
|
|
|
|
|
std::array<Common::SPSCQueue<GCPadStatus>, 4> pad_queue;
|
|
|
|
std::array<GCState, 4> state;
|
|
|
|
};
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-22 22:11:59 +00:00
|
|
|
} // namespace GCAdapter
|