mirror of
https://github.com/libretro/RetroArch.git
synced 2024-12-17 22:29:27 +00:00
11fed40c79
== DETAILS I fixed a similar bug in a past commit, with the same root cause: making assumptions about the length of the array. - Add validation to joypad_connection_init() so that if >MAX_USERS is requested, a warning is logged and only MAX_USERS is allocated. - Rewrote the iteration routines so they strictly use the joypad_is_end_of_list() method to detect the end.
261 lines
7.3 KiB
C
261 lines
7.3 KiB
C
/* RetroArch - A frontend for libretro.
|
|
* Copyright (C) 2013-2014 - Jason Fetters
|
|
* Copyright (C) 2011-2017 - Daniel De Matteis
|
|
*
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
*
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <string/stdstring.h>
|
|
|
|
#include "../input_driver.h"
|
|
#include "../../verbosity.h"
|
|
|
|
#include "joypad_connection.h"
|
|
|
|
static bool joypad_is_end_of_list(joypad_connection_t *pad);
|
|
|
|
int pad_connection_find_vacant_pad(joypad_connection_t *joyconn)
|
|
{
|
|
unsigned i;
|
|
|
|
if (!joyconn)
|
|
return -1;
|
|
|
|
for (i = 0; !joypad_is_end_of_list(&joyconn[i]); i++)
|
|
{
|
|
if(!joyconn[i].connected)
|
|
return i;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
static void set_end_of_list(joypad_connection_t *list, unsigned end)
|
|
{
|
|
joypad_connection_t *entry = (joypad_connection_t *)&list[end];
|
|
entry->connected = false;
|
|
entry->iface = NULL;
|
|
entry->data = (void *)0xdeadbeef;
|
|
}
|
|
|
|
static bool joypad_is_end_of_list(joypad_connection_t *pad) {
|
|
return pad && !pad->connected && !pad->iface && pad->data == (void *)0xdeadbeef;
|
|
}
|
|
|
|
/**
|
|
* Since the pad_connection_destroy() call needs to iterate through this
|
|
* list, we allocate pads+1 entries and use the extra spot to store a
|
|
* marker.
|
|
*/
|
|
joypad_connection_t *pad_connection_init(unsigned pads)
|
|
{
|
|
unsigned i;
|
|
|
|
if(pads > MAX_USERS)
|
|
{
|
|
RARCH_WARN("[joypad] invalid number of pads requested (%d), using default (%d)\n",
|
|
pads, MAX_USERS);
|
|
pads = MAX_USERS;
|
|
}
|
|
|
|
joypad_connection_t *joyconn = (joypad_connection_t*)
|
|
calloc(pads+1, sizeof(joypad_connection_t));
|
|
|
|
if (!joyconn)
|
|
return NULL;
|
|
|
|
for (i = 0; i < pads; i++)
|
|
{
|
|
joypad_connection_t *conn = (joypad_connection_t*)&joyconn[i];
|
|
|
|
conn->connected = false;
|
|
conn->iface = NULL;
|
|
conn->data = NULL;
|
|
}
|
|
|
|
set_end_of_list(joyconn, pads);
|
|
|
|
return joyconn;
|
|
}
|
|
|
|
int32_t pad_connection_pad_init(joypad_connection_t *joyconn,
|
|
const char *name, uint16_t vid, uint16_t pid,
|
|
void *data, hid_driver_t *driver)
|
|
{
|
|
|
|
static const struct
|
|
{
|
|
const char* name;
|
|
uint16_t vid;
|
|
uint16_t pid;
|
|
pad_connection_interface_t *iface;
|
|
} pad_map[] =
|
|
{
|
|
{ "Nintendo RVL-CNT-01", VID_NINTENDO, PID_NINTENDO_PRO, &pad_connection_wii },
|
|
{ "Nintendo RVL-CNT-01-UC", VID_NINTENDO, PID_NINTENDO_PRO, &pad_connection_wiiupro },
|
|
{ "Wireless Controller", VID_SONY, PID_SONY_DS4, &pad_connection_ps4 },
|
|
{ "PLAYSTATION(R)3 Controller", VID_SONY, PID_SONY_DS3, &pad_connection_ps3 },
|
|
{ "PLAYSTATION(R)3 Controller", VID_PS3_CLONE, PID_DS3_CLONE, &pad_connection_ps3 },
|
|
{ "Generic SNES USB Controller", VID_SNES_CLONE, PID_SNES_CLONE, &pad_connection_snesusb },
|
|
{ "Generic NES USB Controller", VID_MICRONTEK, PID_MICRONTEK_NES, &pad_connection_nesusb },
|
|
{ "Wii U GC Controller Adapter", VID_NINTENDO, PID_NINTENDO_GCA, &pad_connection_wiiugca },
|
|
{ "PS2/PSX Controller Adapter", VID_PCS, PID_PCS_PS2PSX, &pad_connection_ps2adapter },
|
|
{ "PSX to PS3 Controller Adapter", VID_PCS, PID_PCS_PSX2PS3, &pad_connection_psxadapter },
|
|
{ "Mayflash DolphinBar", 1406, 774, &pad_connection_wii },
|
|
{ 0, 0}
|
|
};
|
|
joypad_connection_t *s = NULL;
|
|
int pad = pad_connection_find_vacant_pad(joyconn);
|
|
|
|
if (pad == -1)
|
|
return -1;
|
|
|
|
s = &joyconn[pad];
|
|
|
|
if (s)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; name && pad_map[i].name; i++)
|
|
{
|
|
const char *name_match = strstr(pad_map[i].name, name);
|
|
|
|
/* Never change, Nintendo. */
|
|
if(pad_map[i].vid == 1406 && pad_map[i].pid == 816)
|
|
{
|
|
if(!string_is_equal(pad_map[i].name, name))
|
|
continue;
|
|
}
|
|
|
|
#if 0
|
|
RARCH_LOG("[connect] %s\n", pad_map[i].name);
|
|
RARCH_LOG("[connect] VID: Expected: %04x got: %04x\n",
|
|
pad_map[i].vid, vid);
|
|
RARCH_LOG("[connect] PID: Expected: %04x got: %04x\n",
|
|
pad_map[i].pid, pid);
|
|
#endif
|
|
|
|
if (name_match || (pad_map[i].vid == vid && pad_map[i].pid == pid))
|
|
{
|
|
s->iface = pad_map[i].iface;
|
|
s->data = s->iface->init(data, pad, driver);
|
|
s->connected = true;
|
|
#if 0
|
|
RARCH_LOG("%s found \n", pad_map[i].name);
|
|
#endif
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
#if 0
|
|
RARCH_LOG("%s not found \n", pad_map[i].name);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/* We failed to find a matching pad,
|
|
* set up one without an interface */
|
|
if (!s->connected)
|
|
{
|
|
s->iface = NULL;
|
|
s->data = data;
|
|
s->connected = true;
|
|
}
|
|
}
|
|
|
|
return pad;
|
|
}
|
|
|
|
void pad_connection_pad_deinit(joypad_connection_t *joyconn, uint32_t pad)
|
|
{
|
|
if (!joyconn || !joyconn->connected)
|
|
return;
|
|
|
|
if (joyconn->iface)
|
|
{
|
|
joyconn->iface->set_rumble(joyconn->data, RETRO_RUMBLE_STRONG, 0);
|
|
joyconn->iface->set_rumble(joyconn->data, RETRO_RUMBLE_WEAK, 0);
|
|
|
|
if (joyconn->iface->deinit)
|
|
joyconn->iface->deinit(joyconn->data);
|
|
}
|
|
|
|
joyconn->iface = NULL;
|
|
joyconn->connected = false;
|
|
}
|
|
|
|
void pad_connection_packet(joypad_connection_t *joyconn, uint32_t pad,
|
|
uint8_t* data, uint32_t length)
|
|
{
|
|
if (!joyconn || !joyconn->connected)
|
|
return;
|
|
if (joyconn->iface && joyconn->data && joyconn->iface->packet_handler)
|
|
joyconn->iface->packet_handler(joyconn->data, data, length);
|
|
}
|
|
|
|
void pad_connection_get_buttons(joypad_connection_t *joyconn, unsigned pad, retro_bits_t* state)
|
|
{
|
|
if (joyconn->iface)
|
|
joyconn->iface->get_buttons(joyconn->data, state);
|
|
else
|
|
BIT256_CLEAR_ALL_PTR( state );
|
|
}
|
|
|
|
int16_t pad_connection_get_axis(joypad_connection_t *joyconn,
|
|
unsigned idx, unsigned i)
|
|
{
|
|
if (!joyconn->iface)
|
|
return 0;
|
|
return joyconn->iface->get_axis(joyconn->data, i);
|
|
}
|
|
|
|
bool pad_connection_has_interface(joypad_connection_t *joyconn, unsigned pad)
|
|
{
|
|
if ( joyconn && pad < MAX_USERS
|
|
&& joyconn[pad].connected
|
|
&& joyconn[pad].iface)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
void pad_connection_destroy(joypad_connection_t *joyconn)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; !joypad_is_end_of_list(&joyconn[i]); i ++)
|
|
pad_connection_pad_deinit(&joyconn[i], i);
|
|
|
|
free(joyconn);
|
|
}
|
|
|
|
bool pad_connection_rumble(joypad_connection_t *joyconn,
|
|
unsigned pad, enum retro_rumble_effect effect, uint16_t strength)
|
|
{
|
|
if (!joyconn->connected)
|
|
return false;
|
|
if (!joyconn->iface || !joyconn->iface->set_rumble)
|
|
return false;
|
|
|
|
joyconn->iface->set_rumble(joyconn->data, effect, strength);
|
|
return true;
|
|
}
|
|
|
|
const char* pad_connection_get_name(joypad_connection_t *joyconn, unsigned pad)
|
|
{
|
|
if (!joyconn->iface || !joyconn->iface->get_name)
|
|
return NULL;
|
|
return joyconn->iface->get_name(joyconn->data);
|
|
}
|