2018-10-05 16:23:21 +02:00
|
|
|
// Copyright 2018 yuzu emulator team
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
2018-10-06 05:14:42 +02:00
|
|
|
|
2018-10-05 16:23:21 +02:00
|
|
|
#include <array>
|
2018-11-03 17:55:39 +01:00
|
|
|
#include "common/bit_field.h"
|
2018-10-05 16:23:21 +02:00
|
|
|
#include "common/common_types.h"
|
2018-10-06 05:27:06 +02:00
|
|
|
#include "core/frontend/input.h"
|
2018-11-27 00:34:07 +01:00
|
|
|
#include "core/hle/kernel/object.h"
|
2018-11-27 15:18:29 +01:00
|
|
|
#include "core/hle/kernel/writable_event.h"
|
2018-10-05 16:23:21 +02:00
|
|
|
#include "core/hle/service/hid/controllers/controller_base.h"
|
|
|
|
#include "core/settings.h"
|
|
|
|
|
|
|
|
namespace Service::HID {
|
|
|
|
|
2018-11-03 17:55:39 +01:00
|
|
|
constexpr u32 NPAD_HANDHELD = 32;
|
|
|
|
constexpr u32 NPAD_UNKNOWN = 16; // TODO(ogniK): What is this?
|
|
|
|
|
2018-10-05 16:23:21 +02:00
|
|
|
class Controller_NPad final : public ControllerBase {
|
|
|
|
public:
|
2018-10-06 05:14:42 +02:00
|
|
|
Controller_NPad();
|
2018-10-18 02:34:25 +02:00
|
|
|
~Controller_NPad() override;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
// Called when the controller is initialized
|
|
|
|
void OnInit() override;
|
|
|
|
|
|
|
|
// When the controller is released
|
|
|
|
void OnRelease() override;
|
|
|
|
|
|
|
|
// When the controller is requesting an update for the shared memory
|
2019-02-14 18:42:58 +01:00
|
|
|
void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
// Called when input devices should be loaded
|
|
|
|
void OnLoadInputDevices() override;
|
|
|
|
|
|
|
|
struct NPadType {
|
|
|
|
union {
|
|
|
|
u32_le raw{};
|
|
|
|
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<0, 1, u32> pro_controller;
|
|
|
|
BitField<1, 1, u32> handheld;
|
|
|
|
BitField<2, 1, u32> joycon_dual;
|
|
|
|
BitField<3, 1, u32> joycon_left;
|
|
|
|
BitField<4, 1, u32> joycon_right;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<6, 1, u32> pokeball; // TODO(ogniK): Confirm when possible
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
static_assert(sizeof(NPadType) == 4, "NPadType is an invalid size");
|
|
|
|
|
|
|
|
struct Vibration {
|
|
|
|
f32 amp_low;
|
|
|
|
f32 freq_low;
|
|
|
|
f32 amp_high;
|
|
|
|
f32 freq_high;
|
|
|
|
};
|
|
|
|
static_assert(sizeof(Vibration) == 0x10, "Vibration is an invalid size");
|
|
|
|
|
|
|
|
enum class NpadHoldType : u64 {
|
|
|
|
Vertical = 0,
|
|
|
|
Horizontal = 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class NPadAssignments : u32_le {
|
|
|
|
Dual = 0,
|
|
|
|
Single = 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class NPadControllerType {
|
|
|
|
None,
|
|
|
|
ProController,
|
|
|
|
Handheld,
|
2018-10-17 15:11:47 +02:00
|
|
|
JoyDual,
|
2018-10-05 16:23:21 +02:00
|
|
|
JoyLeft,
|
|
|
|
JoyRight,
|
2018-10-06 05:14:42 +02:00
|
|
|
Pokeball,
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
|
|
|
|
2018-10-10 12:38:43 +02:00
|
|
|
struct LedPattern {
|
|
|
|
explicit LedPattern(u64 light1, u64 light2, u64 light3, u64 light4) {
|
|
|
|
position1.Assign(light1);
|
2018-11-27 23:23:44 +01:00
|
|
|
position2.Assign(light2);
|
|
|
|
position3.Assign(light3);
|
|
|
|
position4.Assign(light4);
|
2018-10-18 02:44:11 +02:00
|
|
|
}
|
2018-10-10 12:38:43 +02:00
|
|
|
union {
|
|
|
|
u64 raw{};
|
|
|
|
BitField<0, 1, u64> position1;
|
|
|
|
BitField<1, 1, u64> position2;
|
|
|
|
BitField<2, 1, u64> position3;
|
|
|
|
BitField<3, 1, u64> position4;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2018-10-05 16:23:21 +02:00
|
|
|
void SetSupportedStyleSet(NPadType style_set);
|
|
|
|
NPadType GetSupportedStyleSet() const;
|
|
|
|
|
2018-10-06 05:14:42 +02:00
|
|
|
void SetSupportedNPadIdTypes(u8* data, std::size_t length);
|
2018-10-18 02:37:45 +02:00
|
|
|
void GetSupportedNpadIdTypes(u32* data, std::size_t max_length);
|
2018-10-06 05:14:42 +02:00
|
|
|
std::size_t GetSupportedNPadIdTypesSize() const;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
void SetHoldType(NpadHoldType joy_hold_type);
|
|
|
|
NpadHoldType GetHoldType() const;
|
|
|
|
|
|
|
|
void SetNpadMode(u32 npad_id, NPadAssignments assignment_mode);
|
|
|
|
|
|
|
|
void VibrateController(const std::vector<u32>& controller_ids,
|
|
|
|
const std::vector<Vibration>& vibrations);
|
|
|
|
|
2018-11-27 00:34:07 +01:00
|
|
|
Kernel::SharedPtr<Kernel::ReadableEvent> GetStyleSetChangedEvent() const;
|
2018-10-05 16:23:21 +02:00
|
|
|
Vibration GetLastVibration() const;
|
|
|
|
|
2018-10-17 15:11:47 +02:00
|
|
|
void AddNewController(NPadControllerType controller);
|
2018-10-18 05:11:15 +02:00
|
|
|
void AddNewControllerAt(NPadControllerType controller, u32 npad_id);
|
2018-10-05 16:23:21 +02:00
|
|
|
|
2018-10-07 11:17:04 +02:00
|
|
|
void ConnectNPad(u32 npad_id);
|
|
|
|
void DisconnectNPad(u32 npad_id);
|
2018-10-10 12:38:43 +02:00
|
|
|
LedPattern GetLedPattern(u32 npad_id);
|
2018-10-10 15:58:47 +02:00
|
|
|
void SetVibrationEnabled(bool can_vibrate);
|
2019-09-04 08:43:17 +02:00
|
|
|
bool IsVibrationEnabled();
|
2018-10-18 12:45:10 +02:00
|
|
|
void ClearAllConnectedControllers();
|
|
|
|
void DisconnectAllConnectedControllers();
|
|
|
|
void ConnectAllDisconnectedControllers();
|
|
|
|
void ClearAllControllers();
|
2018-10-07 11:17:04 +02:00
|
|
|
|
2019-07-01 07:12:57 +02:00
|
|
|
void StartLRAssignmentMode();
|
|
|
|
void StopLRAssignmentMode();
|
|
|
|
bool SwapNpadAssignment(u32 npad_id_1, u32 npad_id_2);
|
|
|
|
|
2018-12-24 22:19:16 +01:00
|
|
|
// Logical OR for all buttons presses on all controllers
|
|
|
|
// Specifically for cheat engine and other features.
|
2018-12-29 00:20:29 +01:00
|
|
|
u32 GetAndResetPressState();
|
2018-12-24 22:19:16 +01:00
|
|
|
|
2018-11-03 17:55:39 +01:00
|
|
|
static std::size_t NPadIdToIndex(u32 npad_id);
|
|
|
|
static u32 IndexToNPad(std::size_t index);
|
|
|
|
|
2018-10-05 16:23:21 +02:00
|
|
|
private:
|
|
|
|
struct CommonHeader {
|
|
|
|
s64_le timestamp;
|
|
|
|
s64_le total_entry_count;
|
|
|
|
s64_le last_entry_index;
|
|
|
|
s64_le entry_count;
|
|
|
|
};
|
|
|
|
static_assert(sizeof(CommonHeader) == 0x20, "CommonHeader is an invalid size");
|
|
|
|
|
|
|
|
struct ControllerColor {
|
|
|
|
u32_le body_color;
|
|
|
|
u32_le button_color;
|
|
|
|
};
|
|
|
|
static_assert(sizeof(ControllerColor) == 8, "ControllerColor is an invalid size");
|
|
|
|
|
|
|
|
struct ControllerPadState {
|
|
|
|
union {
|
|
|
|
u64_le raw{};
|
|
|
|
// Button states
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<0, 1, u64> a;
|
|
|
|
BitField<1, 1, u64> b;
|
|
|
|
BitField<2, 1, u64> x;
|
|
|
|
BitField<3, 1, u64> y;
|
|
|
|
BitField<4, 1, u64> l_stick;
|
|
|
|
BitField<5, 1, u64> r_stick;
|
|
|
|
BitField<6, 1, u64> l;
|
|
|
|
BitField<7, 1, u64> r;
|
|
|
|
BitField<8, 1, u64> zl;
|
|
|
|
BitField<9, 1, u64> zr;
|
|
|
|
BitField<10, 1, u64> plus;
|
|
|
|
BitField<11, 1, u64> minus;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
// D-Pad
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<12, 1, u64> d_left;
|
|
|
|
BitField<13, 1, u64> d_up;
|
|
|
|
BitField<14, 1, u64> d_right;
|
|
|
|
BitField<15, 1, u64> d_down;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
// Left JoyStick
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<16, 1, u64> l_stick_left;
|
|
|
|
BitField<17, 1, u64> l_stick_up;
|
|
|
|
BitField<18, 1, u64> l_stick_right;
|
|
|
|
BitField<19, 1, u64> l_stick_down;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
// Right JoyStick
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<20, 1, u64> r_stick_left;
|
|
|
|
BitField<21, 1, u64> r_stick_up;
|
|
|
|
BitField<22, 1, u64> r_stick_right;
|
|
|
|
BitField<23, 1, u64> r_stick_down;
|
2018-10-05 16:23:21 +02:00
|
|
|
|
|
|
|
// Not always active?
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<24, 1, u64> left_sl;
|
|
|
|
BitField<25, 1, u64> left_sr;
|
2018-10-18 10:00:16 +02:00
|
|
|
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<26, 1, u64> right_sl;
|
|
|
|
BitField<27, 1, u64> right_sr;
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
static_assert(sizeof(ControllerPadState) == 8, "ControllerPadState is an invalid size");
|
|
|
|
|
|
|
|
struct AnalogPosition {
|
|
|
|
s32_le x;
|
|
|
|
s32_le y;
|
|
|
|
};
|
|
|
|
static_assert(sizeof(AnalogPosition) == 8, "AnalogPosition is an invalid size");
|
|
|
|
|
|
|
|
struct ConnectionState {
|
|
|
|
union {
|
|
|
|
u32_le raw{};
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<0, 1, u32> IsConnected;
|
|
|
|
BitField<1, 1, u32> IsWired;
|
|
|
|
BitField<2, 1, u32> IsLeftJoyConnected;
|
|
|
|
BitField<3, 1, u32> IsLeftJoyWired;
|
|
|
|
BitField<4, 1, u32> IsRightJoyConnected;
|
|
|
|
BitField<5, 1, u32> IsRightJoyWired;
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
static_assert(sizeof(ConnectionState) == 4, "ConnectionState is an invalid size");
|
|
|
|
|
2018-10-18 05:11:15 +02:00
|
|
|
struct ControllerPad {
|
2018-10-05 16:23:21 +02:00
|
|
|
ControllerPadState pad_states;
|
2018-10-06 05:14:42 +02:00
|
|
|
AnalogPosition l_stick;
|
|
|
|
AnalogPosition r_stick;
|
2018-10-18 05:11:15 +02:00
|
|
|
};
|
|
|
|
static_assert(sizeof(ControllerPad) == 0x18, "ControllerPad is an invalid size");
|
|
|
|
|
|
|
|
struct GenericStates {
|
|
|
|
s64_le timestamp;
|
|
|
|
s64_le timestamp2;
|
|
|
|
ControllerPad pad;
|
2018-10-05 16:23:21 +02:00
|
|
|
ConnectionState connection_status;
|
|
|
|
};
|
|
|
|
static_assert(sizeof(GenericStates) == 0x30, "NPadGenericStates is an invalid size");
|
|
|
|
|
|
|
|
struct NPadGeneric {
|
|
|
|
CommonHeader common;
|
|
|
|
std::array<GenericStates, 17> npad;
|
|
|
|
};
|
|
|
|
static_assert(sizeof(NPadGeneric) == 0x350, "NPadGeneric is an invalid size");
|
|
|
|
|
|
|
|
enum class ColorReadError : u32_le {
|
|
|
|
ReadOk = 0,
|
|
|
|
ColorDoesntExist = 1,
|
|
|
|
NoController = 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NPadProperties {
|
|
|
|
union {
|
|
|
|
s64_le raw{};
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<11, 1, s64> is_vertical;
|
|
|
|
BitField<12, 1, s64> is_horizontal;
|
|
|
|
BitField<13, 1, s64> use_plus;
|
|
|
|
BitField<14, 1, s64> use_minus;
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NPadDevice {
|
|
|
|
union {
|
|
|
|
u32_le raw{};
|
2019-01-25 18:26:47 +01:00
|
|
|
BitField<0, 1, s32> pro_controller;
|
|
|
|
BitField<1, 1, s32> handheld;
|
|
|
|
BitField<2, 1, s32> handheld_left;
|
|
|
|
BitField<3, 1, s32> handheld_right;
|
|
|
|
BitField<4, 1, s32> joycon_left;
|
|
|
|
BitField<5, 1, s32> joycon_right;
|
|
|
|
BitField<6, 1, s32> pokeball;
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NPadEntry {
|
|
|
|
NPadType joy_styles;
|
|
|
|
NPadAssignments pad_assignment;
|
|
|
|
|
|
|
|
ColorReadError single_color_error;
|
|
|
|
ControllerColor single_color;
|
|
|
|
|
|
|
|
ColorReadError dual_color_error;
|
|
|
|
ControllerColor left_color;
|
|
|
|
ControllerColor right_color;
|
|
|
|
|
|
|
|
NPadGeneric main_controller_states;
|
|
|
|
NPadGeneric handheld_states;
|
|
|
|
NPadGeneric dual_states;
|
|
|
|
NPadGeneric left_joy_states;
|
|
|
|
NPadGeneric right_joy_states;
|
|
|
|
NPadGeneric pokeball_states;
|
|
|
|
NPadGeneric libnx; // TODO(ogniK): Find out what this actually is, libnx seems to only be
|
|
|
|
// relying on this for the time being
|
|
|
|
INSERT_PADDING_BYTES(
|
|
|
|
0x708 *
|
2018-10-06 05:14:42 +02:00
|
|
|
6); // TODO(ogniK): SixAxis states, require more information before implementation
|
2018-10-05 16:23:21 +02:00
|
|
|
NPadDevice device_type;
|
|
|
|
NPadProperties properties;
|
2018-10-17 15:11:47 +02:00
|
|
|
INSERT_PADDING_WORDS(1);
|
|
|
|
std::array<u32, 3> battery_level;
|
|
|
|
INSERT_PADDING_BYTES(0x5c);
|
2018-10-05 16:23:21 +02:00
|
|
|
INSERT_PADDING_BYTES(0xdf8);
|
|
|
|
};
|
|
|
|
static_assert(sizeof(NPadEntry) == 0x5000, "NPadEntry is an invalid size");
|
2018-10-07 11:17:04 +02:00
|
|
|
|
|
|
|
struct ControllerHolder {
|
2018-10-18 06:09:55 +02:00
|
|
|
NPadControllerType type;
|
2018-10-07 11:17:04 +02:00
|
|
|
bool is_connected;
|
|
|
|
};
|
|
|
|
|
2018-12-24 22:19:16 +01:00
|
|
|
u32 press_state{};
|
|
|
|
|
2018-10-05 16:23:21 +02:00
|
|
|
NPadType style{};
|
|
|
|
std::array<NPadEntry, 10> shared_memory_entries{};
|
2018-11-02 03:03:17 +01:00
|
|
|
std::array<
|
|
|
|
std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeButton::NUM_BUTTONS_HID>,
|
|
|
|
10>
|
2018-10-05 16:23:21 +02:00
|
|
|
buttons;
|
2018-11-02 03:03:17 +01:00
|
|
|
std::array<
|
|
|
|
std::array<std::unique_ptr<Input::AnalogDevice>, Settings::NativeAnalog::NUM_STICKS_HID>,
|
|
|
|
10>
|
|
|
|
sticks;
|
2018-10-05 16:23:21 +02:00
|
|
|
std::vector<u32> supported_npad_id_types{};
|
|
|
|
NpadHoldType hold_type{NpadHoldType::Vertical};
|
2018-11-27 15:18:29 +01:00
|
|
|
Kernel::EventPair styleset_changed_event;
|
2018-10-05 16:23:21 +02:00
|
|
|
Vibration last_processed_vibration{};
|
2018-10-10 12:38:43 +02:00
|
|
|
std::array<ControllerHolder, 10> connected_controllers{};
|
2018-10-10 15:58:47 +02:00
|
|
|
bool can_controllers_vibrate{true};
|
2018-10-06 05:14:42 +02:00
|
|
|
|
|
|
|
void InitNewlyAddedControler(std::size_t controller_idx);
|
2018-10-20 06:07:18 +02:00
|
|
|
bool IsControllerSupported(NPadControllerType controller) const;
|
|
|
|
NPadControllerType DecideBestController(NPadControllerType priority) const;
|
2018-10-18 05:11:15 +02:00
|
|
|
void RequestPadStateUpdate(u32 npad_id);
|
|
|
|
std::array<ControllerPad, 10> npad_pad_states{};
|
2018-10-18 06:09:55 +02:00
|
|
|
bool IsControllerSupported(NPadControllerType controller);
|
2019-07-01 07:12:57 +02:00
|
|
|
bool is_in_lr_assignment_mode{false};
|
2018-10-05 16:23:21 +02:00
|
|
|
};
|
2018-10-06 05:14:42 +02:00
|
|
|
} // namespace Service::HID
|