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>
|
|
|
|
#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,
|
|
|
|
};
|
|
|
|
|
|
|
|
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 22:43:01 +00:00
|
|
|
enum class PadAxes : u8 {
|
|
|
|
StickX,
|
|
|
|
StickY,
|
|
|
|
SubstickX,
|
|
|
|
SubstickY,
|
|
|
|
TriggerLeft,
|
|
|
|
TriggerRight,
|
|
|
|
Undefined,
|
|
|
|
};
|
|
|
|
const struct GCPadConstants {
|
|
|
|
const u8 MAIN_STICK_CENTER_X = 0x80;
|
|
|
|
const u8 MAIN_STICK_CENTER_Y = 0x80;
|
|
|
|
const u8 MAIN_STICK_RADIUS = 0x7f;
|
|
|
|
const u8 C_STICK_CENTER_X = 0x80;
|
|
|
|
const u8 C_STICK_CENTER_Y = 0x80;
|
|
|
|
const u8 C_STICK_RADIUS = 0x7f;
|
|
|
|
|
|
|
|
const u8 TRIGGER_CENTER = 20;
|
|
|
|
const u8 THRESHOLD = 10;
|
|
|
|
} pad_constants;
|
2020-06-21 16:36:28 +00:00
|
|
|
|
|
|
|
struct GCPadStatus {
|
2020-06-21 22:43:01 +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 16:36:28 +00:00
|
|
|
u8 port;
|
2020-06-21 22:43:01 +00:00
|
|
|
PadAxes axis = PadAxes::Undefined;
|
2020-06-21 16:36:28 +00:00
|
|
|
u8 axis_value = 255;
|
|
|
|
};
|
|
|
|
|
|
|
|
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
|
|
|
/// Singleton Adapter class
|
|
|
|
class Adapter {
|
|
|
|
public:
|
|
|
|
/// For retreiving the singleton instance
|
|
|
|
static Adapter* GetInstance();
|
|
|
|
|
|
|
|
/// Used for polling
|
|
|
|
void BeginConfiguration();
|
|
|
|
void EndConfiguration();
|
|
|
|
|
|
|
|
std::array<Common::SPSCQueue<GCPadStatus>, 4>& GetPadQueue();
|
|
|
|
std::array<GCState, 4>& GetPadState();
|
|
|
|
|
|
|
|
private:
|
|
|
|
/// Singleton instance.
|
|
|
|
static Adapter* adapter_instance;
|
|
|
|
|
|
|
|
/// Initialize the GC Adapter capture and read sequence
|
|
|
|
Adapter();
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Close the adapter read thread and release the adapter
|
|
|
|
~Adapter();
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
GCPadStatus CheckStatus(int port, u8 adapter_payload[37]);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
void PadToState(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
|
|
|
|
bool DeviceConnected(int port);
|
2020-06-21 16:36:28 +00:00
|
|
|
|
2020-06-21 22:43:01 +00:00
|
|
|
/// Resets status of device connected to port
|
|
|
|
void ResetDeviceType(int 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;
|
|
|
|
ControllerTypes adapter_controllers_status[4] = {ControllerTypes::None, ControllerTypes::None,
|
|
|
|
ControllerTypes::None, ControllerTypes::None};
|
|
|
|
|
|
|
|
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 16:36:28 +00:00
|
|
|
|
|
|
|
} // end of namespace GCAdapter
|