2017-12-24 01:26:58 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "../../config.h"
|
|
|
|
#endif
|
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
#ifdef HAVE_LIBNX
|
|
|
|
#include <switch.h>
|
|
|
|
#else
|
|
|
|
#include <libtransistor/nx.h>
|
|
|
|
#endif
|
2017-12-28 23:44:41 +00:00
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
#include "../configuration.h"
|
2017-12-24 01:26:58 +00:00
|
|
|
#include "../input_driver.h"
|
|
|
|
|
|
|
|
#include "../../tasks/tasks_internal.h"
|
|
|
|
|
|
|
|
#include "../../retroarch.h"
|
|
|
|
#include "../../command.h"
|
|
|
|
#include "string.h"
|
|
|
|
|
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
#ifdef HAVE_LIBNX
|
|
|
|
|
|
|
|
#ifndef MAX_PADS
|
|
|
|
#define MAX_PADS 8
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2017-12-24 01:26:58 +00:00
|
|
|
#ifndef MAX_PADS
|
|
|
|
#define MAX_PADS 10
|
|
|
|
#endif
|
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
#endif
|
|
|
|
|
2017-12-24 01:26:58 +00:00
|
|
|
static uint16_t pad_state[MAX_PADS];
|
|
|
|
static int16_t analog_state[MAX_PADS][2][2];
|
|
|
|
extern uint64_t lifecycle_state;
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static const char *switch_joypad_name(unsigned pad)
|
|
|
|
{
|
|
|
|
return "Switch Controller";
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static void switch_joypad_autodetect_add(unsigned autoconf_pad)
|
|
|
|
{
|
|
|
|
if(!input_autoconfigure_connect(
|
|
|
|
switch_joypad_name(autoconf_pad), /* name */
|
|
|
|
NULL, /* display name */
|
|
|
|
switch_joypad.ident, /* driver */
|
|
|
|
autoconf_pad, /* idx */
|
|
|
|
0, /* vid */
|
|
|
|
0)) /* pid */
|
|
|
|
input_config_set_device_name(autoconf_pad, switch_joypad_name(autoconf_pad));
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static bool switch_joypad_init(void *data)
|
|
|
|
{
|
2018-09-12 17:54:51 +00:00
|
|
|
#ifdef HAVE_LIBNX
|
|
|
|
unsigned i;
|
|
|
|
hidScanInput();
|
|
|
|
for (i = 0; i < MAX_PADS; i++)
|
|
|
|
switch_joypad_autodetect_add(i);
|
|
|
|
#else
|
2017-12-28 23:44:41 +00:00
|
|
|
hid_init();
|
|
|
|
switch_joypad_autodetect_add(0);
|
|
|
|
switch_joypad_autodetect_add(1);
|
2018-09-12 17:54:51 +00:00
|
|
|
#endif
|
2017-12-28 23:44:41 +00:00
|
|
|
|
|
|
|
return true;
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static bool switch_joypad_button(unsigned port_num, uint16_t key)
|
|
|
|
{
|
|
|
|
if(port_num >= MAX_PADS)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
RARCH_LOG("button(%d, %d)\n", port_num, key);
|
|
|
|
#endif
|
2017-12-24 01:26:58 +00:00
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
return (pad_state[port_num] & (1 << key));
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2018-04-08 18:21:12 +00:00
|
|
|
static void switch_joypad_get_buttons(unsigned port_num, input_bits_t *state)
|
2017-12-28 23:44:41 +00:00
|
|
|
{
|
|
|
|
if(port_num < MAX_PADS)
|
|
|
|
{
|
|
|
|
BITS_COPY16_PTR(state, pad_state[port_num]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
BIT256_CLEAR_ALL_PTR(state);
|
|
|
|
}
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static int16_t switch_joypad_axis(unsigned port_num, uint32_t joyaxis)
|
|
|
|
{
|
|
|
|
int val = 0;
|
|
|
|
int axis = -1;
|
|
|
|
bool is_neg = false;
|
|
|
|
bool is_pos = false;
|
|
|
|
|
|
|
|
if(joyaxis == AXIS_NONE || port_num >= MAX_PADS)
|
|
|
|
{
|
|
|
|
/* TODO/FIXME - implement */
|
|
|
|
}
|
|
|
|
|
|
|
|
if(AXIS_NEG_GET(joyaxis) < 4)
|
|
|
|
{
|
|
|
|
axis = AXIS_NEG_GET(joyaxis);
|
|
|
|
is_neg = true;
|
|
|
|
}
|
|
|
|
else if(AXIS_POS_GET(joyaxis) < 4)
|
|
|
|
{
|
|
|
|
axis = AXIS_POS_GET(joyaxis);
|
|
|
|
is_pos = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(axis)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
val = analog_state[port_num][0][0];
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
val = analog_state[port_num][0][1];
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
val = analog_state[port_num][1][0];
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
val = analog_state[port_num][1][1];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(is_neg && val > 0)
|
|
|
|
val = 0;
|
|
|
|
else if(is_pos && val < 0)
|
|
|
|
val = 0;
|
|
|
|
|
|
|
|
return val;
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static bool switch_joypad_query_pad(unsigned pad)
|
|
|
|
{
|
|
|
|
return pad < MAX_PADS && pad_state[pad];
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2017-12-28 23:44:41 +00:00
|
|
|
static void switch_joypad_destroy(void)
|
|
|
|
{
|
2018-09-12 17:54:51 +00:00
|
|
|
#ifndef HAVE_LIBNX
|
2017-12-28 23:44:41 +00:00
|
|
|
hid_finalize();
|
2018-09-12 17:54:51 +00:00
|
|
|
#endif
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
#ifdef HAVE_LIBNX
|
|
|
|
int lastMode = 0; // 0 = handheld, 1 = whatever
|
2017-12-28 23:44:41 +00:00
|
|
|
static void switch_joypad_poll(void)
|
|
|
|
{
|
2018-09-12 17:54:51 +00:00
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
|
|
|
|
hidScanInput();
|
|
|
|
|
2018-09-30 22:03:57 +00:00
|
|
|
if (!hidGetHandheldMode())
|
2018-09-12 17:54:51 +00:00
|
|
|
{
|
|
|
|
if (lastMode != 1)
|
|
|
|
{
|
2018-09-30 22:03:57 +00:00
|
|
|
int i = 0;
|
|
|
|
for(i = 0; i < MAX_USERS; i += 2){
|
|
|
|
if(settings->uints.input_split_joycon[i]) // CONTROLLER_PLAYER_X, X == i++
|
|
|
|
{
|
|
|
|
hidSetNpadJoyAssignmentModeSingleByDefault(i);
|
|
|
|
hidSetNpadJoyAssignmentModeSingleByDefault(i + 1);
|
2018-12-06 00:12:55 +00:00
|
|
|
hidSetNpadJoyHoldType(HidJoyHoldType_Horizontal);
|
2018-09-30 22:03:57 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-12 17:54:51 +00:00
|
|
|
lastMode = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (lastMode != 0)
|
|
|
|
{
|
2018-09-30 22:03:57 +00:00
|
|
|
int i = 0;
|
|
|
|
for(i = 0; i < MAX_USERS; i += 2){
|
|
|
|
if(settings->uints.input_split_joycon[i]) // CONTROLLER_PLAYER_X, X == i++
|
|
|
|
{
|
|
|
|
hidSetNpadJoyAssignmentModeDual(i);
|
|
|
|
hidSetNpadJoyAssignmentModeDual(i + 1);
|
2018-12-06 00:29:28 +00:00
|
|
|
hidMergeSingleJoyAsDualJoy(i, i + 1);
|
2018-09-30 22:03:57 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-12 17:54:51 +00:00
|
|
|
lastMode = 0;
|
|
|
|
}
|
2018-09-30 22:03:57 +00:00
|
|
|
}
|
2018-09-14 18:38:42 +00:00
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
for (int i = 0; i < MAX_PADS; i++)
|
|
|
|
{
|
|
|
|
HidControllerID target = (i == 0) ? CONTROLLER_P1_AUTO : i;
|
|
|
|
pad_state[i] = hidKeysDown(target) | hidKeysHeld(target);
|
|
|
|
JoystickPosition joyPositionLeft, joyPositionRight;
|
|
|
|
hidJoystickRead(&joyPositionLeft, target, JOYSTICK_LEFT);
|
|
|
|
hidJoystickRead(&joyPositionRight, target, JOYSTICK_RIGHT);
|
|
|
|
analog_state[i][RETRO_DEVICE_INDEX_ANALOG_LEFT][RETRO_DEVICE_ID_ANALOG_X] = joyPositionLeft.dx;
|
|
|
|
analog_state[i][RETRO_DEVICE_INDEX_ANALOG_LEFT][RETRO_DEVICE_ID_ANALOG_Y] = -joyPositionLeft.dy;
|
|
|
|
analog_state[i][RETRO_DEVICE_INDEX_ANALOG_RIGHT][RETRO_DEVICE_ID_ANALOG_X] = joyPositionRight.dx;
|
|
|
|
analog_state[i][RETRO_DEVICE_INDEX_ANALOG_RIGHT][RETRO_DEVICE_ID_ANALOG_Y] = -joyPositionRight.dy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static void switch_joypad_poll(void)
|
|
|
|
{
|
|
|
|
int16_t lsx, lsy, rsx, rsy;
|
2018-01-05 21:31:18 +00:00
|
|
|
hid_controller_t *controllers = hid_get_shared_memory()->controllers;
|
|
|
|
hid_controller_t *cont = &controllers[0];
|
|
|
|
hid_controller_state_entry_t ent = cont->main.entries[cont->main.latest_idx];
|
|
|
|
hid_controller_state_entry_t ent8 = (cont+8)->main.entries[(cont+8)->main.latest_idx];
|
2018-09-12 17:54:51 +00:00
|
|
|
pad_state[0] = ent.button_state | ent8.button_state;
|
2018-05-24 07:18:38 +00:00
|
|
|
|
2018-09-12 17:54:51 +00:00
|
|
|
lsx = ent.left_stick_x;
|
|
|
|
lsy = ent.left_stick_y;
|
|
|
|
rsx = ent.right_stick_x;
|
|
|
|
rsy = ent.right_stick_y;
|
|
|
|
|
|
|
|
if (ent8.left_stick_x != 0 || ent8.left_stick_y != 0)
|
|
|
|
{
|
|
|
|
/* handheld overrides player 1 */
|
2018-05-24 07:18:38 +00:00
|
|
|
lsx = ent8.left_stick_x;
|
|
|
|
lsy = ent8.left_stick_y;
|
|
|
|
}
|
2018-09-12 17:54:51 +00:00
|
|
|
|
|
|
|
if (ent8.right_stick_x != 0 || ent8.right_stick_y != 0)
|
|
|
|
{
|
|
|
|
/* handheld overrides player 1 */
|
2018-05-24 07:18:38 +00:00
|
|
|
rsx = ent8.right_stick_x;
|
|
|
|
rsy = ent8.right_stick_y;
|
|
|
|
}
|
2018-01-05 21:31:18 +00:00
|
|
|
|
2018-05-24 07:18:38 +00:00
|
|
|
analog_state[0][RETRO_DEVICE_INDEX_ANALOG_LEFT][RETRO_DEVICE_ID_ANALOG_X] = lsx;
|
|
|
|
analog_state[0][RETRO_DEVICE_INDEX_ANALOG_LEFT][RETRO_DEVICE_ID_ANALOG_Y] = -lsy;
|
|
|
|
analog_state[0][RETRO_DEVICE_INDEX_ANALOG_RIGHT][RETRO_DEVICE_ID_ANALOG_X] = rsx;
|
|
|
|
analog_state[0][RETRO_DEVICE_INDEX_ANALOG_RIGHT][RETRO_DEVICE_ID_ANALOG_Y] = -rsy;
|
2017-12-24 01:26:58 +00:00
|
|
|
}
|
2018-09-12 17:54:51 +00:00
|
|
|
#endif
|
2017-12-24 01:26:58 +00:00
|
|
|
|
|
|
|
input_device_driver_t switch_joypad = {
|
|
|
|
switch_joypad_init,
|
|
|
|
switch_joypad_query_pad,
|
|
|
|
switch_joypad_destroy,
|
|
|
|
switch_joypad_button,
|
|
|
|
switch_joypad_get_buttons,
|
|
|
|
switch_joypad_axis,
|
|
|
|
switch_joypad_poll,
|
2017-12-28 23:44:41 +00:00
|
|
|
NULL, /* set_rumble */
|
2017-12-24 01:26:58 +00:00
|
|
|
switch_joypad_name,
|
|
|
|
"switch"
|
|
|
|
};
|