2001-04-26 16:45:43 +00:00
|
|
|
/*
|
2011-04-08 13:03:26 -07:00
|
|
|
Simple DirectMedia Layer
|
2011-12-31 09:28:07 -05:00
|
|
|
Copyright (C) 1997-2012 Sam Lantinga <slouken@libsdl.org>
|
2011-04-08 13:03:26 -07:00
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
|
|
|
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
2001-04-26 16:45:43 +00:00
|
|
|
*/
|
2006-02-21 08:46:50 +00:00
|
|
|
#include "SDL_config.h"
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2006-04-14 04:46:47 +00:00
|
|
|
#ifdef SDL_JOYSTICK_LINUX
|
|
|
|
|
2012-12-11 11:07:48 -05:00
|
|
|
#ifndef SDL_INPUT_LINUXEV
|
|
|
|
#error SDL now requires a Linux 2.4+ kernel with /dev/input/event support.
|
|
|
|
#endif
|
|
|
|
|
2001-04-26 16:45:43 +00:00
|
|
|
/* This is the system specific header for the SDL joystick API */
|
|
|
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/ioctl.h>
|
2006-07-10 21:04:37 +00:00
|
|
|
#include <limits.h> /* For the definition of PATH_MAX */
|
2001-04-26 16:45:43 +00:00
|
|
|
#include <linux/joystick.h>
|
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
#include "SDL_assert.h"
|
2001-04-26 16:45:43 +00:00
|
|
|
#include "SDL_joystick.h"
|
2012-12-11 12:07:06 -05:00
|
|
|
#include "SDL_endian.h"
|
2006-02-16 10:11:48 +00:00
|
|
|
#include "../SDL_sysjoystick.h"
|
|
|
|
#include "../SDL_joystick_c.h"
|
2008-08-25 09:55:03 +00:00
|
|
|
#include "SDL_sysjoystick_c.h"
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
/* !!! FIXME: move this somewhere else. */
|
|
|
|
#if !SDL_EVENTS_DISABLED
|
|
|
|
#include "../../events/SDL_events_c.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* !!! FIXME: move all the udev stuff to src/core/linux, so I can reuse it
|
|
|
|
* !!! FIXME: for audio hardware disconnects.
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_LIBUDEV_H
|
|
|
|
#define SDL_USE_LIBUDEV 1
|
|
|
|
#include "SDL_loadso.h"
|
|
|
|
#include <libudev.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
/* we never link directly to libudev. */
|
|
|
|
/* !!! FIXME: can we generalize this? ALSA, etc, do the same things. */
|
2012-12-14 18:50:07 +00:00
|
|
|
static const char *udev_library = "libudev.so.0";
|
2012-12-11 12:07:06 -05:00
|
|
|
static void *udev_handle = NULL;
|
|
|
|
|
|
|
|
/* !!! FIXME: this is kinda ugly. */
|
|
|
|
static SDL_bool
|
|
|
|
load_udev_sym(const char *fn, void **addr)
|
|
|
|
{
|
|
|
|
*addr = SDL_LoadFunction(udev_handle, fn);
|
|
|
|
if (*addr == NULL) {
|
|
|
|
/* Don't call SDL_SetError(): SDL_LoadFunction already did. */
|
|
|
|
return SDL_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SDL_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* libudev entry points... */
|
|
|
|
static const char *(*UDEV_udev_device_get_action)(struct udev_device *) = NULL;
|
|
|
|
static const char *(*UDEV_udev_device_get_devnode)(struct udev_device *) = NULL;
|
|
|
|
static const char *(*UDEV_udev_device_get_property_value)(struct udev_device *, const char *) = NULL;
|
|
|
|
static struct udev_device *(*UDEV_udev_device_new_from_syspath)(struct udev *, const char *) = NULL;
|
|
|
|
static void (*UDEV_udev_device_unref)(struct udev_device *) = NULL;
|
|
|
|
static int (*UDEV_udev_enumerate_add_match_property)(struct udev_enumerate *, const char *, const char *) = NULL;
|
|
|
|
static int (*UDEV_udev_enumerate_add_match_subsystem)(struct udev_enumerate *, const char *) = NULL;
|
|
|
|
static struct udev_list_entry *(*UDEV_udev_enumerate_get_list_entry)(struct udev_enumerate *) = NULL;
|
|
|
|
static struct udev_enumerate *(*UDEV_udev_enumerate_new)(struct udev *) = NULL;
|
|
|
|
static int (*UDEV_udev_enumerate_scan_devices)(struct udev_enumerate *) = NULL;
|
|
|
|
static void (*UDEV_udev_enumerate_unref)(struct udev_enumerate *) = NULL;
|
|
|
|
static const char *(*UDEV_udev_list_entry_get_name)(struct udev_list_entry *) = NULL;
|
|
|
|
static struct udev_list_entry *(*UDEV_udev_list_entry_get_next)(struct udev_list_entry *) = NULL;
|
|
|
|
static int (*UDEV_udev_monitor_enable_receiving)(struct udev_monitor *) = NULL;
|
|
|
|
static int (*UDEV_udev_monitor_filter_add_match_subsystem_devtype)(struct udev_monitor *, const char *, const char *) = NULL;
|
|
|
|
static int (*UDEV_udev_monitor_get_fd)(struct udev_monitor *) = NULL;
|
|
|
|
static struct udev_monitor *(*UDEV_udev_monitor_new_from_netlink)(struct udev *, const char *) = NULL;
|
|
|
|
static struct udev_device *(*UDEV_udev_monitor_receive_device)(struct udev_monitor *) = NULL;
|
|
|
|
static void (*UDEV_udev_monitor_unref)(struct udev_monitor *) = NULL;
|
|
|
|
static struct udev *(*UDEV_udev_new)(void) = NULL;
|
|
|
|
static void (*UDEV_udev_unref)(struct udev *) = NULL;
|
|
|
|
|
|
|
|
static int
|
|
|
|
load_udev_syms(void)
|
|
|
|
{
|
|
|
|
/* cast funcs to char* first, to please GCC's strict aliasing rules. */
|
|
|
|
#define SDL_UDEV_SYM(x) \
|
|
|
|
if (!load_udev_sym(#x, (void **) (char *) &UDEV_##x)) return -1
|
|
|
|
|
|
|
|
SDL_UDEV_SYM(udev_device_get_action);
|
|
|
|
SDL_UDEV_SYM(udev_device_get_devnode);
|
|
|
|
SDL_UDEV_SYM(udev_device_get_property_value);
|
|
|
|
SDL_UDEV_SYM(udev_device_new_from_syspath);
|
|
|
|
SDL_UDEV_SYM(udev_device_unref);
|
|
|
|
SDL_UDEV_SYM(udev_enumerate_add_match_property);
|
|
|
|
SDL_UDEV_SYM(udev_enumerate_add_match_subsystem);
|
|
|
|
SDL_UDEV_SYM(udev_enumerate_get_list_entry);
|
|
|
|
SDL_UDEV_SYM(udev_enumerate_new);
|
|
|
|
SDL_UDEV_SYM(udev_enumerate_scan_devices);
|
|
|
|
SDL_UDEV_SYM(udev_enumerate_unref);
|
|
|
|
SDL_UDEV_SYM(udev_list_entry_get_name);
|
|
|
|
SDL_UDEV_SYM(udev_list_entry_get_next);
|
|
|
|
SDL_UDEV_SYM(udev_monitor_enable_receiving);
|
|
|
|
SDL_UDEV_SYM(udev_monitor_filter_add_match_subsystem_devtype);
|
|
|
|
SDL_UDEV_SYM(udev_monitor_get_fd);
|
|
|
|
SDL_UDEV_SYM(udev_monitor_new_from_netlink);
|
|
|
|
SDL_UDEV_SYM(udev_monitor_receive_device);
|
|
|
|
SDL_UDEV_SYM(udev_monitor_unref);
|
|
|
|
SDL_UDEV_SYM(udev_new);
|
|
|
|
SDL_UDEV_SYM(udev_unref);
|
|
|
|
|
|
|
|
#undef SDL_UDEV_SYM
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
UnloadUDEVLibrary(void)
|
|
|
|
{
|
|
|
|
if (udev_handle != NULL) {
|
|
|
|
SDL_UnloadObject(udev_handle);
|
|
|
|
udev_handle = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
LoadUDEVLibrary(void)
|
|
|
|
{
|
|
|
|
int retval = 0;
|
|
|
|
if (udev_handle == NULL) {
|
|
|
|
udev_handle = SDL_LoadObject(udev_library);
|
|
|
|
if (udev_handle == NULL) {
|
|
|
|
retval = -1;
|
|
|
|
/* Don't call SDL_SetError(): SDL_LoadObject already did. */
|
|
|
|
} else {
|
|
|
|
retval = load_udev_syms();
|
|
|
|
if (retval < 0) {
|
|
|
|
UnloadUDEVLibrary();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct udev *udev = NULL;
|
|
|
|
static struct udev_monitor *udev_mon = NULL;
|
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
|
|
|
|
/* A linked list of available joysticks */
|
|
|
|
typedef struct SDL_joylist_item
|
Date: Sun, 18 Apr 2004 16:09:53 -0400 (EDT)
From: David MacCormack
Subject: [SDL] Linux joystick patch
I recently got myself a PS2 -> USB converter (a super joybox 5). It
accepts 4 PSX/PS2 controllers. It's implemented as a HID, which is nice
because it doesn't require its own driver, but the problem is that it's
implemented as a *single* HID -- that is, it shows up as a single
joystick with 19 axes, 4 hats, and 48 buttons. This poses a problem for a
number of apps which use SDL (stella, fce ultra, zsnes, to name a few) and
see only a single (physical) joystick even though there are really 4
(logical) joysticks. There are a number of these types of devices on the
market, and I've seen others post messages (in the zsnes forum, for
example) with the same problem, so I came up with what I think is a pretty
generic solution.
I patched src/joystick/linux/SDL_sysjoystic.c to include support for
logical joysticks; basically, it's a static array and supporting functions
that map a single physical joystick to multiple logical joysticks. The
attached patch has the new code. It's wrapped inside #ifndef
statements so that you can get the old behavior if you want.
--HG--
extra : convert_revision : svn%3Ac70aab31-4412-0410-b14c-859654838e24/trunk%40893
2004-05-16 18:46:24 +00:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
int device_instance;
|
|
|
|
char *path; /* "/dev/input/event2" or whatever */
|
|
|
|
char *name; /* "SideWinder 3D Pro" or whatever */
|
2012-12-11 12:08:36 -08:00
|
|
|
SDL_JoystickGUID guid;
|
2012-12-11 12:07:06 -05:00
|
|
|
dev_t devnum;
|
|
|
|
struct joystick_hwdata *hwdata;
|
|
|
|
struct SDL_joylist_item *next;
|
|
|
|
} SDL_joylist_item;
|
Date: Sun, 18 Apr 2004 16:09:53 -0400 (EDT)
From: David MacCormack
Subject: [SDL] Linux joystick patch
I recently got myself a PS2 -> USB converter (a super joybox 5). It
accepts 4 PSX/PS2 controllers. It's implemented as a HID, which is nice
because it doesn't require its own driver, but the problem is that it's
implemented as a *single* HID -- that is, it shows up as a single
joystick with 19 axes, 4 hats, and 48 buttons. This poses a problem for a
number of apps which use SDL (stella, fce ultra, zsnes, to name a few) and
see only a single (physical) joystick even though there are really 4
(logical) joysticks. There are a number of these types of devices on the
market, and I've seen others post messages (in the zsnes forum, for
example) with the same problem, so I came up with what I think is a pretty
generic solution.
I patched src/joystick/linux/SDL_sysjoystic.c to include support for
logical joysticks; basically, it's a static array and supporting functions
that map a single physical joystick to multiple logical joysticks. The
attached patch has the new code. It's wrapped inside #ifndef
statements so that you can get the old behavior if you want.
--HG--
extra : convert_revision : svn%3Ac70aab31-4412-0410-b14c-859654838e24/trunk%40893
2004-05-16 18:46:24 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
static SDL_joylist_item *SDL_joylist = NULL;
|
|
|
|
static SDL_joylist_item *SDL_joylist_tail = NULL;
|
|
|
|
static int numjoysticks = 0;
|
|
|
|
static int instance_counter = 0;
|
2001-04-26 16:45:43 +00:00
|
|
|
|
|
|
|
#define test_bit(nr, addr) \
|
2012-12-11 12:07:06 -05:00
|
|
|
(((1UL << ((nr) % (sizeof(long) * 8))) & ((addr)[(nr) / (sizeof(long) * 8)])) != 0)
|
2009-10-18 16:14:35 +00:00
|
|
|
#define NBITS(x) ((((x)-1)/(sizeof(long) * 8))+1)
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static int
|
2012-12-11 12:08:36 -08:00
|
|
|
IsJoystick(int fd, char *namebuf, const size_t namebuflen, SDL_JoystickGUID *guid)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2009-10-18 16:14:35 +00:00
|
|
|
unsigned long evbit[NBITS(EV_MAX)] = { 0 };
|
|
|
|
unsigned long keybit[NBITS(KEY_MAX)] = { 0 };
|
|
|
|
unsigned long absbit[NBITS(ABS_MAX)] = { 0 };
|
2012-12-11 12:07:06 -05:00
|
|
|
struct input_id inpid;
|
|
|
|
Uint16 *guid16 = (Uint16 *) ((char *) &guid->data);
|
2006-07-10 21:04:37 +00:00
|
|
|
|
|
|
|
if ((ioctl(fd, EVIOCGBIT(0, sizeof(evbit)), evbit) < 0) ||
|
|
|
|
(ioctl(fd, EVIOCGBIT(EV_KEY, sizeof(keybit)), keybit) < 0) ||
|
|
|
|
(ioctl(fd, EVIOCGBIT(EV_ABS, sizeof(absbit)), absbit) < 0)) {
|
|
|
|
return (0);
|
|
|
|
}
|
2009-02-21 18:02:55 +00:00
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
if (!(test_bit(EV_KEY, evbit) && test_bit(EV_ABS, evbit) &&
|
2009-02-21 18:02:55 +00:00
|
|
|
test_bit(ABS_X, absbit) && test_bit(ABS_Y, absbit))) {
|
2006-07-10 21:04:37 +00:00
|
|
|
return 0;
|
2009-02-21 18:02:55 +00:00
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
|
|
|
|
if (ioctl(fd, EVIOCGNAME(namebuflen), namebuf) < 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ioctl(fd, EVIOCGID, &inpid) < 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-02-01 17:09:01 -08:00
|
|
|
#ifdef DEBUG_JOYSTICK
|
|
|
|
printf("Joystick: %s, bustype = %d, vendor = 0x%x, product = 0x%x, version = %d\n", namebuf, inpid.bustype, inpid.vendor, inpid.product, inpid.version);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
SDL_memset(guid->data, 0, sizeof(guid->data));
|
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
/* We only need 16 bits for each of these; space them out to fill 128. */
|
|
|
|
/* Byteswap so devices get same GUID on little/big endian platforms. */
|
|
|
|
*(guid16++) = SDL_SwapLE16(inpid.bustype);
|
|
|
|
*(guid16++) = 0;
|
2013-02-01 17:09:01 -08:00
|
|
|
|
|
|
|
if (inpid.vendor && inpid.product && inpid.version) {
|
|
|
|
*(guid16++) = SDL_SwapLE16(inpid.vendor);
|
|
|
|
*(guid16++) = 0;
|
|
|
|
*(guid16++) = SDL_SwapLE16(inpid.product);
|
|
|
|
*(guid16++) = 0;
|
|
|
|
*(guid16++) = SDL_SwapLE16(inpid.version);
|
|
|
|
*(guid16++) = 0;
|
|
|
|
} else {
|
|
|
|
SDL_strlcpy((char*)guid16, namebuf, sizeof(guid->data) - 4);
|
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
|
|
|
|
return 1;
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
/* !!! FIXME: I would love to dump this code and use libudev instead. */
|
|
|
|
static int
|
|
|
|
MaybeAddDevice(const char *path)
|
|
|
|
{
|
|
|
|
struct stat sb;
|
|
|
|
int fd = -1;
|
|
|
|
int isstick = 0;
|
|
|
|
char namebuf[128];
|
2012-12-11 12:08:36 -08:00
|
|
|
SDL_JoystickGUID guid;
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_joylist_item *item;
|
2012-11-26 16:37:54 -08:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
if (path == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stat(path, &sb) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check to make sure it's not already in list. */
|
|
|
|
for (item = SDL_joylist; item != NULL; item = item->next) {
|
|
|
|
if (sb.st_rdev == item->devnum) {
|
|
|
|
return -1; /* already have this one */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fd = open(path, O_RDONLY, 0);
|
|
|
|
if (fd < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
|
|
|
printf("Checking %s\n", path);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
isstick = IsJoystick(fd, namebuf, sizeof (namebuf), &guid);
|
|
|
|
close(fd);
|
|
|
|
if (!isstick) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
item = (SDL_joylist_item *) SDL_malloc(sizeof (SDL_joylist_item));
|
|
|
|
if (item == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_zerop(item);
|
|
|
|
item->devnum = sb.st_rdev;
|
|
|
|
item->path = SDL_strdup(path);
|
|
|
|
item->name = SDL_strdup(namebuf);
|
|
|
|
item->guid = guid;
|
|
|
|
|
|
|
|
if ( (item->path == NULL) || (item->name == NULL) ) {
|
|
|
|
SDL_free(item->path);
|
|
|
|
SDL_free(item->name);
|
|
|
|
SDL_free(item);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
item->device_instance = instance_counter++;
|
|
|
|
if (SDL_joylist_tail == NULL) {
|
|
|
|
SDL_joylist = SDL_joylist_tail = item;
|
|
|
|
} else {
|
|
|
|
SDL_joylist_tail->next = item;
|
|
|
|
SDL_joylist_tail = item;
|
|
|
|
}
|
|
|
|
|
|
|
|
return numjoysticks++;
|
|
|
|
}
|
|
|
|
|
2012-12-11 19:25:35 -05:00
|
|
|
#if SDL_USE_LIBUDEV
|
2012-12-11 12:07:06 -05:00
|
|
|
/* !!! FIXME: I would love to dump this code and use libudev instead. */
|
|
|
|
static int
|
|
|
|
MaybeRemoveDevice(const char *path)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_joylist_item *item;
|
|
|
|
SDL_joylist_item *prev = NULL;
|
|
|
|
|
|
|
|
if (path == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (item = SDL_joylist; item != NULL; item = item->next) {
|
|
|
|
/* found it, remove it. */
|
|
|
|
if (SDL_strcmp(path, item->path) == 0) {
|
|
|
|
const int retval = item->device_instance;
|
|
|
|
if (item->hwdata) {
|
2012-12-13 22:26:30 -05:00
|
|
|
item->hwdata->item = NULL;
|
2012-12-11 12:07:06 -05:00
|
|
|
}
|
|
|
|
if (prev != NULL) {
|
|
|
|
prev->next = item->next;
|
|
|
|
if (item == SDL_joylist_tail) {
|
|
|
|
SDL_joylist_tail = prev;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SDL_assert(!SDL_joylist);
|
|
|
|
SDL_assert(!SDL_joylist_tail);
|
|
|
|
SDL_joylist = SDL_joylist_tail = NULL;
|
|
|
|
}
|
|
|
|
SDL_free(item->path);
|
|
|
|
SDL_free(item->name);
|
|
|
|
SDL_free(item);
|
|
|
|
numjoysticks--;
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
prev = item;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
2012-12-11 19:25:35 -05:00
|
|
|
#endif
|
2012-12-11 12:07:06 -05:00
|
|
|
|
|
|
|
static int
|
|
|
|
JoystickInitWithoutUdev(void)
|
|
|
|
{
|
|
|
|
int i;
|
2006-07-10 21:04:37 +00:00
|
|
|
char path[PATH_MAX];
|
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
/* !!! FIXME: only finds sticks if they're called /dev/input/event[0..31] */
|
|
|
|
/* !!! FIXME: we could at least readdir() through /dev/input...? */
|
|
|
|
/* !!! FIXME: (or delete this and rely on libudev?) */
|
|
|
|
for (i = 0; i < 32; i++) {
|
|
|
|
SDL_snprintf(path, SDL_arraysize(path), "/dev/input/event%d", i);
|
|
|
|
MaybeAddDevice(path);
|
|
|
|
}
|
2006-07-10 21:04:37 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
return numjoysticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if SDL_USE_LIBUDEV
|
|
|
|
static int
|
|
|
|
JoystickInitWithUdev(void)
|
|
|
|
{
|
|
|
|
struct udev_enumerate *enumerate = NULL;
|
|
|
|
struct udev_list_entry *devs = NULL;
|
|
|
|
struct udev_list_entry *item = NULL;
|
|
|
|
|
|
|
|
SDL_assert(udev == NULL);
|
|
|
|
udev = UDEV_udev_new();
|
|
|
|
if (udev == NULL) {
|
|
|
|
SDL_SetError("udev_new() failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
udev_mon = UDEV_udev_monitor_new_from_netlink(udev, "udev");
|
|
|
|
if (udev_mon != NULL) { /* okay if it's NULL, we just lose hotplugging. */
|
|
|
|
UDEV_udev_monitor_filter_add_match_subsystem_devtype(udev_mon,
|
|
|
|
"input", NULL);
|
|
|
|
UDEV_udev_monitor_enable_receiving(udev_mon);
|
|
|
|
}
|
|
|
|
|
|
|
|
enumerate = UDEV_udev_enumerate_new(udev);
|
|
|
|
if (enumerate == NULL) {
|
|
|
|
SDL_SetError("udev_enumerate_new() failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
UDEV_udev_enumerate_add_match_subsystem(enumerate, "input");
|
|
|
|
UDEV_udev_enumerate_add_match_property(enumerate, "ID_INPUT_JOYSTICK", "1");
|
|
|
|
UDEV_udev_enumerate_scan_devices(enumerate);
|
|
|
|
devs = UDEV_udev_enumerate_get_list_entry(enumerate);
|
|
|
|
for (item = devs; item; item = UDEV_udev_list_entry_get_next(item)) {
|
|
|
|
const char *path = UDEV_udev_list_entry_get_name(item);
|
|
|
|
struct udev_device *dev = UDEV_udev_device_new_from_syspath(udev, path);
|
|
|
|
MaybeAddDevice(UDEV_udev_device_get_devnode(dev));
|
|
|
|
UDEV_udev_device_unref(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
UDEV_udev_enumerate_unref(enumerate);
|
|
|
|
|
|
|
|
return numjoysticks;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int
|
|
|
|
SDL_SYS_JoystickInit(void)
|
|
|
|
{
|
2011-02-16 04:51:13 -08:00
|
|
|
/* First see if the user specified one or more joysticks to use */
|
2006-07-10 21:04:37 +00:00
|
|
|
if (SDL_getenv("SDL_JOYSTICK_DEVICE") != NULL) {
|
2011-02-16 04:51:13 -08:00
|
|
|
char *envcopy, *envpath, *delim;
|
|
|
|
envcopy = SDL_strdup(SDL_getenv("SDL_JOYSTICK_DEVICE"));
|
|
|
|
envpath = envcopy;
|
|
|
|
while (envpath != NULL) {
|
|
|
|
delim = SDL_strchr(envpath, ':');
|
|
|
|
if (delim != NULL) {
|
|
|
|
*delim++ = '\0';
|
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
MaybeAddDevice(envpath);
|
2011-02-16 04:51:13 -08:00
|
|
|
envpath = delim;
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
2011-02-16 04:51:13 -08:00
|
|
|
SDL_free(envcopy);
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
#if SDL_USE_LIBUDEV
|
|
|
|
if (LoadUDEVLibrary() == 0) { /* okay if this fails, FOR NOW. */
|
|
|
|
return JoystickInitWithUdev();
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
#endif
|
2002-12-02 03:11:36 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
return JoystickInitWithoutUdev();
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2012-11-27 00:58:12 -08:00
|
|
|
int SDL_SYS_NumJoysticks()
|
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
return numjoysticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SDL_bool
|
|
|
|
HotplugUpdateAvailable(void)
|
|
|
|
{
|
|
|
|
#if SDL_USE_LIBUDEV
|
|
|
|
if (udev_mon != NULL) {
|
|
|
|
const int fd = UDEV_udev_monitor_get_fd(udev_mon);
|
|
|
|
fd_set fds;
|
|
|
|
struct timeval tv;
|
|
|
|
|
|
|
|
FD_ZERO(&fds);
|
|
|
|
FD_SET(fd, &fds);
|
|
|
|
tv.tv_sec = 0;
|
|
|
|
tv.tv_usec = 0;
|
|
|
|
if ((select(fd+1, &fds, NULL, NULL, &tv) > 0) && (FD_ISSET(fd, &fds))) {
|
|
|
|
return SDL_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return SDL_FALSE;
|
2012-11-27 00:58:12 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDL_SYS_JoystickDetect()
|
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
#if SDL_USE_LIBUDEV
|
|
|
|
struct udev_device *dev = NULL;
|
|
|
|
const char *devnode = NULL;
|
|
|
|
const char *action = NULL;
|
|
|
|
const char *val = NULL;
|
|
|
|
|
|
|
|
while (HotplugUpdateAvailable()) {
|
|
|
|
dev = UDEV_udev_monitor_receive_device(udev_mon);
|
|
|
|
if (dev == NULL) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
val = UDEV_udev_device_get_property_value(dev, "ID_INPUT_JOYSTICK");
|
|
|
|
if ((!val) || (SDL_strcmp(val, "1") != 0)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
action = UDEV_udev_device_get_action(dev);
|
|
|
|
devnode = UDEV_udev_device_get_devnode(dev);
|
|
|
|
|
|
|
|
if (SDL_strcmp(action, "add") == 0) {
|
|
|
|
const int device_index = MaybeAddDevice(devnode);
|
|
|
|
if (device_index != -1) {
|
|
|
|
/* !!! FIXME: Move this to an SDL_PrivateJoyDeviceAdded() function? */
|
|
|
|
#if !SDL_EVENTS_DISABLED
|
|
|
|
SDL_Event event;
|
|
|
|
event.type = SDL_JOYDEVICEADDED;
|
|
|
|
|
|
|
|
if (SDL_GetEventState(event.type) == SDL_ENABLE) {
|
|
|
|
event.jdevice.which = device_index;
|
|
|
|
if ( (SDL_EventOK == NULL) ||
|
|
|
|
(*SDL_EventOK) (SDL_EventOKParam, &event) ) {
|
|
|
|
SDL_PushEvent(&event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // !SDL_EVENTS_DISABLED
|
|
|
|
}
|
|
|
|
} else if (SDL_strcmp(action, "remove") == 0) {
|
|
|
|
const int inst = MaybeRemoveDevice(devnode);
|
|
|
|
if (inst != -1) {
|
|
|
|
/* !!! FIXME: Move this to an SDL_PrivateJoyDeviceRemoved() function? */
|
|
|
|
#if !SDL_EVENTS_DISABLED
|
|
|
|
SDL_Event event;
|
|
|
|
event.type = SDL_JOYDEVICEREMOVED;
|
|
|
|
|
|
|
|
if (SDL_GetEventState(event.type) == SDL_ENABLE) {
|
|
|
|
event.jdevice.which = inst;
|
|
|
|
if ( (SDL_EventOK == NULL) ||
|
|
|
|
(*SDL_EventOK) (SDL_EventOKParam, &event) ) {
|
|
|
|
SDL_PushEvent(&event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // !SDL_EVENTS_DISABLED
|
|
|
|
}
|
|
|
|
}
|
|
|
|
UDEV_udev_device_unref(dev);
|
|
|
|
}
|
|
|
|
#endif
|
2012-11-27 00:58:12 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
SDL_bool SDL_SYS_JoystickNeedsPolling()
|
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
/*
|
|
|
|
* This results in a select() call, so technically we're polling to
|
|
|
|
* decide if we should poll, but I think this function is here because
|
|
|
|
* Windows has to do an enormous amount of work to detect new sticks,
|
|
|
|
* whereas libudev just needs to see if there's more data available on
|
|
|
|
* a socket...so this should be acceptable, I hope.
|
|
|
|
*/
|
|
|
|
return HotplugUpdateAvailable();
|
|
|
|
}
|
|
|
|
|
|
|
|
static SDL_joylist_item *
|
|
|
|
JoystickByDevIndex(int device_index)
|
|
|
|
{
|
|
|
|
SDL_joylist_item *item = SDL_joylist;
|
|
|
|
|
|
|
|
if ((device_index < 0) || (device_index >= numjoysticks)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (device_index > 0) {
|
|
|
|
SDL_assert(item != NULL);
|
|
|
|
device_index--;
|
|
|
|
item = item->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return item;
|
2012-11-27 00:58:12 -08:00
|
|
|
}
|
|
|
|
|
2001-04-26 16:45:43 +00:00
|
|
|
/* Function to get the device-dependent name of a joystick */
|
2006-07-10 21:04:37 +00:00
|
|
|
const char *
|
2012-11-27 00:58:12 -08:00
|
|
|
SDL_SYS_JoystickNameForDeviceIndex(int device_index)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
return JoystickByDevIndex(device_index)->name;
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2012-11-27 00:58:12 -08:00
|
|
|
/* Function to perform the mapping from device index to the instance id for this index */
|
|
|
|
SDL_JoystickID SDL_SYS_GetInstanceIdOfDeviceIndex(int device_index)
|
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
return JoystickByDevIndex(device_index)->device_instance;
|
2012-11-27 00:58:12 -08:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static int
|
|
|
|
allocate_hatdata(SDL_Joystick * joystick)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
joystick->hwdata->hats =
|
|
|
|
(struct hwdata_hat *) SDL_malloc(joystick->nhats *
|
|
|
|
sizeof(struct hwdata_hat));
|
|
|
|
if (joystick->hwdata->hats == NULL) {
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
for (i = 0; i < joystick->nhats; ++i) {
|
|
|
|
joystick->hwdata->hats[i].axis[0] = 1;
|
|
|
|
joystick->hwdata->hats[i].axis[1] = 1;
|
|
|
|
}
|
|
|
|
return (0);
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static int
|
|
|
|
allocate_balldata(SDL_Joystick * joystick)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
joystick->hwdata->balls =
|
|
|
|
(struct hwdata_ball *) SDL_malloc(joystick->nballs *
|
|
|
|
sizeof(struct hwdata_ball));
|
|
|
|
if (joystick->hwdata->balls == NULL) {
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
for (i = 0; i < joystick->nballs; ++i) {
|
|
|
|
joystick->hwdata->balls[i].axis[0] = 0;
|
|
|
|
joystick->hwdata->balls[i].axis[1] = 0;
|
|
|
|
}
|
|
|
|
return (0);
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2012-12-11 11:07:48 -05:00
|
|
|
static void
|
|
|
|
ConfigJoystick(SDL_Joystick * joystick, int fd)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
int i, t;
|
2009-10-18 16:14:35 +00:00
|
|
|
unsigned long keybit[NBITS(KEY_MAX)] = { 0 };
|
|
|
|
unsigned long absbit[NBITS(ABS_MAX)] = { 0 };
|
|
|
|
unsigned long relbit[NBITS(REL_MAX)] = { 0 };
|
2006-07-10 21:04:37 +00:00
|
|
|
|
|
|
|
/* See if this device uses the new unified event API */
|
|
|
|
if ((ioctl(fd, EVIOCGBIT(EV_KEY, sizeof(keybit)), keybit) >= 0) &&
|
|
|
|
(ioctl(fd, EVIOCGBIT(EV_ABS, sizeof(absbit)), absbit) >= 0) &&
|
|
|
|
(ioctl(fd, EVIOCGBIT(EV_REL, sizeof(relbit)), relbit) >= 0)) {
|
|
|
|
|
|
|
|
/* Get the number of buttons, axes, and other thingamajigs */
|
|
|
|
for (i = BTN_JOYSTICK; i < KEY_MAX; ++i) {
|
|
|
|
if (test_bit(i, keybit)) {
|
2001-04-26 16:45:43 +00:00
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
2006-07-10 21:04:37 +00:00
|
|
|
printf("Joystick has button: 0x%x\n", i);
|
2001-04-26 16:45:43 +00:00
|
|
|
#endif
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata->key_map[i - BTN_MISC] = joystick->nbuttons;
|
|
|
|
++joystick->nbuttons;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = BTN_MISC; i < BTN_JOYSTICK; ++i) {
|
|
|
|
if (test_bit(i, keybit)) {
|
2001-04-26 16:45:43 +00:00
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
2006-07-10 21:04:37 +00:00
|
|
|
printf("Joystick has button: 0x%x\n", i);
|
2001-04-26 16:45:43 +00:00
|
|
|
#endif
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata->key_map[i - BTN_MISC] = joystick->nbuttons;
|
|
|
|
++joystick->nbuttons;
|
|
|
|
}
|
|
|
|
}
|
2011-07-13 17:38:09 -07:00
|
|
|
for (i = 0; i < ABS_MISC; ++i) {
|
2006-07-10 21:04:37 +00:00
|
|
|
/* Skip hats */
|
|
|
|
if (i == ABS_HAT0X) {
|
|
|
|
i = ABS_HAT3Y;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (test_bit(i, absbit)) {
|
2011-01-24 14:36:12 -08:00
|
|
|
struct input_absinfo absinfo;
|
2006-07-10 21:04:37 +00:00
|
|
|
|
2011-01-24 15:10:16 -08:00
|
|
|
if (ioctl(fd, EVIOCGABS(i), &absinfo) < 0)
|
2006-07-10 21:04:37 +00:00
|
|
|
continue;
|
2001-04-26 16:45:43 +00:00
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
2006-07-10 21:04:37 +00:00
|
|
|
printf("Joystick has absolute axis: %x\n", i);
|
|
|
|
printf("Values = { %d, %d, %d, %d, %d }\n",
|
2011-01-24 14:36:12 -08:00
|
|
|
absinfo.value, absinfo.minimum, absinfo.maximum,
|
|
|
|
absinfo.fuzz, absinfo.flat);
|
2001-04-26 16:45:43 +00:00
|
|
|
#endif /* DEBUG_INPUT_EVENTS */
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata->abs_map[i] = joystick->naxes;
|
2011-01-24 14:36:12 -08:00
|
|
|
if (absinfo.minimum == absinfo.maximum) {
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata->abs_correct[i].used = 0;
|
|
|
|
} else {
|
|
|
|
joystick->hwdata->abs_correct[i].used = 1;
|
|
|
|
joystick->hwdata->abs_correct[i].coef[0] =
|
2011-01-24 14:36:12 -08:00
|
|
|
(absinfo.maximum + absinfo.minimum) / 2 - absinfo.flat;
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata->abs_correct[i].coef[1] =
|
2011-01-24 14:36:12 -08:00
|
|
|
(absinfo.maximum + absinfo.minimum) / 2 + absinfo.flat;
|
|
|
|
t = ((absinfo.maximum - absinfo.minimum) / 2 - 2 * absinfo.flat);
|
2006-07-10 21:04:37 +00:00
|
|
|
if (t != 0) {
|
|
|
|
joystick->hwdata->abs_correct[i].coef[2] =
|
|
|
|
(1 << 29) / t;
|
|
|
|
} else {
|
|
|
|
joystick->hwdata->abs_correct[i].coef[2] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++joystick->naxes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = ABS_HAT0X; i <= ABS_HAT3Y; i += 2) {
|
|
|
|
if (test_bit(i, absbit) || test_bit(i + 1, absbit)) {
|
2001-04-26 16:45:43 +00:00
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
2006-07-10 21:04:37 +00:00
|
|
|
printf("Joystick has hat %d\n", (i - ABS_HAT0X) / 2);
|
2001-04-26 16:45:43 +00:00
|
|
|
#endif
|
2006-07-10 21:04:37 +00:00
|
|
|
++joystick->nhats;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (test_bit(REL_X, relbit) || test_bit(REL_Y, relbit)) {
|
|
|
|
++joystick->nballs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate data to keep track of these thingamajigs */
|
|
|
|
if (joystick->nhats > 0) {
|
|
|
|
if (allocate_hatdata(joystick) < 0) {
|
|
|
|
joystick->nhats = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (joystick->nballs > 0) {
|
|
|
|
if (allocate_balldata(joystick) < 0) {
|
|
|
|
joystick->nballs = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
Date: Sun, 18 Apr 2004 16:09:53 -0400 (EDT)
From: David MacCormack
Subject: [SDL] Linux joystick patch
I recently got myself a PS2 -> USB converter (a super joybox 5). It
accepts 4 PSX/PS2 controllers. It's implemented as a HID, which is nice
because it doesn't require its own driver, but the problem is that it's
implemented as a *single* HID -- that is, it shows up as a single
joystick with 19 axes, 4 hats, and 48 buttons. This poses a problem for a
number of apps which use SDL (stella, fce ultra, zsnes, to name a few) and
see only a single (physical) joystick even though there are really 4
(logical) joysticks. There are a number of these types of devices on the
market, and I've seen others post messages (in the zsnes forum, for
example) with the same problem, so I came up with what I think is a pretty
generic solution.
I patched src/joystick/linux/SDL_sysjoystic.c to include support for
logical joysticks; basically, it's a static array and supporting functions
that map a single physical joystick to multiple logical joysticks. The
attached patch has the new code. It's wrapped inside #ifndef
statements so that you can get the old behavior if you want.
--HG--
extra : convert_revision : svn%3Ac70aab31-4412-0410-b14c-859654838e24/trunk%40893
2004-05-16 18:46:24 +00:00
|
|
|
|
2001-04-26 16:45:43 +00:00
|
|
|
/* Function to open a joystick for use.
|
|
|
|
The joystick to open is specified by the index field of the joystick.
|
|
|
|
This should fill the nbuttons and naxes fields of the joystick structure.
|
|
|
|
It returns 0, or -1 if there is an error.
|
|
|
|
*/
|
2006-07-10 21:04:37 +00:00
|
|
|
int
|
2012-11-26 16:37:54 -08:00
|
|
|
SDL_SYS_JoystickOpen(SDL_Joystick * joystick, int device_index)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_joylist_item *item = JoystickByDevIndex(device_index);
|
|
|
|
char *fname = NULL;
|
|
|
|
int fd = -1;
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
if (item == NULL) {
|
|
|
|
SDL_SetError("No such device");
|
|
|
|
return -1;
|
|
|
|
}
|
Date: Sun, 18 Apr 2004 16:09:53 -0400 (EDT)
From: David MacCormack
Subject: [SDL] Linux joystick patch
I recently got myself a PS2 -> USB converter (a super joybox 5). It
accepts 4 PSX/PS2 controllers. It's implemented as a HID, which is nice
because it doesn't require its own driver, but the problem is that it's
implemented as a *single* HID -- that is, it shows up as a single
joystick with 19 axes, 4 hats, and 48 buttons. This poses a problem for a
number of apps which use SDL (stella, fce ultra, zsnes, to name a few) and
see only a single (physical) joystick even though there are really 4
(logical) joysticks. There are a number of these types of devices on the
market, and I've seen others post messages (in the zsnes forum, for
example) with the same problem, so I came up with what I think is a pretty
generic solution.
I patched src/joystick/linux/SDL_sysjoystic.c to include support for
logical joysticks; basically, it's a static array and supporting functions
that map a single physical joystick to multiple logical joysticks. The
attached patch has the new code. It's wrapped inside #ifndef
statements so that you can get the old behavior if you want.
--HG--
extra : convert_revision : svn%3Ac70aab31-4412-0410-b14c-859654838e24/trunk%40893
2004-05-16 18:46:24 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
fname = item->path;
|
|
|
|
fd = open(fname, O_RDONLY, 0);
|
2006-07-10 21:04:37 +00:00
|
|
|
if (fd < 0) {
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_SetError("Unable to open %s", fname);
|
|
|
|
return -1;
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
|
2012-12-13 22:18:32 -05:00
|
|
|
joystick->instance_id = item->device_instance;
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata = (struct joystick_hwdata *)
|
|
|
|
SDL_malloc(sizeof(*joystick->hwdata));
|
|
|
|
if (joystick->hwdata == NULL) {
|
|
|
|
close(fd);
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_OutOfMemory();
|
2006-07-10 21:04:37 +00:00
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
SDL_memset(joystick->hwdata, 0, sizeof(*joystick->hwdata));
|
2012-12-13 22:26:30 -05:00
|
|
|
joystick->hwdata->item = item;
|
2012-12-11 12:07:06 -05:00
|
|
|
joystick->hwdata->guid = item->guid;
|
2006-07-10 21:04:37 +00:00
|
|
|
joystick->hwdata->fd = fd;
|
2012-12-11 12:07:06 -05:00
|
|
|
joystick->hwdata->fname = SDL_strdup(item->path);
|
|
|
|
if (joystick->hwdata->fname == NULL) {
|
|
|
|
SDL_free(joystick->hwdata);
|
|
|
|
joystick->hwdata = NULL;
|
|
|
|
close(fd);
|
|
|
|
SDL_OutOfMemory();
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_assert(item->hwdata == NULL);
|
|
|
|
item->hwdata = joystick->hwdata;
|
2006-07-10 21:04:37 +00:00
|
|
|
|
|
|
|
/* Set the joystick to non-blocking read mode */
|
|
|
|
fcntl(fd, F_SETFL, O_NONBLOCK);
|
|
|
|
|
|
|
|
/* Get the number of buttons and axes on the joystick */
|
2012-12-11 11:07:48 -05:00
|
|
|
ConfigJoystick(joystick, fd);
|
2002-12-02 03:11:36 +00:00
|
|
|
|
2013-02-11 16:45:24 -08:00
|
|
|
// mark joystick as fresh and ready
|
|
|
|
joystick->hwdata->fresh = 1;
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
return (0);
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2012-11-27 00:58:12 -08:00
|
|
|
/* Function to determine is this joystick is attached to the system right now */
|
|
|
|
SDL_bool SDL_SYS_JoystickAttached(SDL_Joystick *joystick)
|
|
|
|
{
|
2012-12-13 22:26:30 -05:00
|
|
|
return !joystick->closed && (joystick->hwdata->item != NULL);
|
2012-11-27 00:58:12 -08:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static __inline__ void
|
|
|
|
HandleHat(SDL_Joystick * stick, Uint8 hat, int axis, int value)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
struct hwdata_hat *the_hat;
|
|
|
|
const Uint8 position_map[3][3] = {
|
|
|
|
{SDL_HAT_LEFTUP, SDL_HAT_UP, SDL_HAT_RIGHTUP},
|
|
|
|
{SDL_HAT_LEFT, SDL_HAT_CENTERED, SDL_HAT_RIGHT},
|
|
|
|
{SDL_HAT_LEFTDOWN, SDL_HAT_DOWN, SDL_HAT_RIGHTDOWN}
|
|
|
|
};
|
|
|
|
|
|
|
|
the_hat = &stick->hwdata->hats[hat];
|
|
|
|
if (value < 0) {
|
|
|
|
value = 0;
|
|
|
|
} else if (value == 0) {
|
|
|
|
value = 1;
|
|
|
|
} else if (value > 0) {
|
|
|
|
value = 2;
|
|
|
|
}
|
|
|
|
if (value != the_hat->axis[axis]) {
|
|
|
|
the_hat->axis[axis] = value;
|
|
|
|
SDL_PrivateJoystickHat(stick, hat,
|
2009-01-10 21:50:26 +00:00
|
|
|
position_map[the_hat->
|
|
|
|
axis[1]][the_hat->axis[0]]);
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static __inline__ void
|
|
|
|
HandleBall(SDL_Joystick * stick, Uint8 ball, int axis, int value)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
stick->hwdata->balls[ball].axis[axis] += value;
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
|
|
|
|
static __inline__ int
|
2012-12-11 11:07:48 -05:00
|
|
|
AxisCorrect(SDL_Joystick * joystick, int which, int value)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
struct axis_correct *correct;
|
|
|
|
|
|
|
|
correct = &joystick->hwdata->abs_correct[which];
|
|
|
|
if (correct->used) {
|
|
|
|
if (value > correct->coef[0]) {
|
|
|
|
if (value < correct->coef[1]) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
value -= correct->coef[1];
|
|
|
|
} else {
|
|
|
|
value -= correct->coef[0];
|
|
|
|
}
|
|
|
|
value *= correct->coef[2];
|
|
|
|
value >>= 14;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clamp and return */
|
|
|
|
if (value < -32768)
|
|
|
|
return -32768;
|
|
|
|
if (value > 32767)
|
|
|
|
return 32767;
|
|
|
|
|
|
|
|
return value;
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2013-02-11 16:45:24 -08:00
|
|
|
static __inline__ void
|
|
|
|
PollAllValues(SDL_Joystick * joystick)
|
|
|
|
{
|
|
|
|
struct input_absinfo absinfo;
|
|
|
|
int a, b = 0;
|
|
|
|
|
|
|
|
// Poll all axis
|
|
|
|
for (a = ABS_X; b < ABS_MAX; a++) {
|
|
|
|
switch (a) {
|
|
|
|
case ABS_HAT0X:
|
|
|
|
case ABS_HAT0Y:
|
|
|
|
case ABS_HAT1X:
|
|
|
|
case ABS_HAT1Y:
|
|
|
|
case ABS_HAT2X:
|
|
|
|
case ABS_HAT2Y:
|
|
|
|
case ABS_HAT3X:
|
|
|
|
case ABS_HAT3Y:
|
|
|
|
// ingore hats
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (joystick->hwdata->abs_correct[b].used) {
|
|
|
|
if (ioctl(joystick->hwdata->fd, EVIOCGABS(a), &absinfo) >= 0) {
|
|
|
|
absinfo.value = AxisCorrect(joystick, b, absinfo.value);
|
|
|
|
|
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
|
|
|
printf("Joystick : Re-read Axis %d (%d) val= %d\n",
|
|
|
|
joystick->hwdata->abs_map[b], a, absinfo.value);
|
|
|
|
#endif
|
|
|
|
SDL_PrivateJoystickAxis(joystick,
|
|
|
|
joystick->hwdata->abs_map[b],
|
|
|
|
absinfo.value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static __inline__ void
|
2012-12-11 11:07:48 -05:00
|
|
|
HandleInputEvents(SDL_Joystick * joystick)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
struct input_event events[32];
|
|
|
|
int i, len;
|
|
|
|
int code;
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2013-02-11 16:45:24 -08:00
|
|
|
if (joystick->hwdata->fresh) {
|
|
|
|
PollAllValues(joystick);
|
|
|
|
joystick->hwdata->fresh = 0;
|
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
while ((len = read(joystick->hwdata->fd, events, (sizeof events))) > 0) {
|
|
|
|
len /= sizeof(events[0]);
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
|
|
code = events[i].code;
|
|
|
|
switch (events[i].type) {
|
|
|
|
case EV_KEY:
|
|
|
|
if (code >= BTN_MISC) {
|
|
|
|
code -= BTN_MISC;
|
2012-12-10 15:50:42 -05:00
|
|
|
SDL_PrivateJoystickButton(joystick,
|
|
|
|
joystick->hwdata->key_map[code],
|
|
|
|
events[i].value);
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case EV_ABS:
|
2012-06-03 05:05:34 -04:00
|
|
|
if (code >= ABS_MISC) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
switch (code) {
|
|
|
|
case ABS_HAT0X:
|
|
|
|
case ABS_HAT0Y:
|
|
|
|
case ABS_HAT1X:
|
|
|
|
case ABS_HAT1Y:
|
|
|
|
case ABS_HAT2X:
|
|
|
|
case ABS_HAT2Y:
|
|
|
|
case ABS_HAT3X:
|
|
|
|
case ABS_HAT3Y:
|
|
|
|
code -= ABS_HAT0X;
|
|
|
|
HandleHat(joystick, code / 2, code % 2, events[i].value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
events[i].value =
|
2012-12-11 11:07:48 -05:00
|
|
|
AxisCorrect(joystick, code, events[i].value);
|
2012-12-10 15:50:42 -05:00
|
|
|
SDL_PrivateJoystickAxis(joystick,
|
|
|
|
joystick->hwdata->abs_map[code],
|
|
|
|
events[i].value);
|
2006-07-10 21:04:37 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case EV_REL:
|
|
|
|
switch (code) {
|
|
|
|
case REL_X:
|
|
|
|
case REL_Y:
|
|
|
|
code -= REL_X;
|
|
|
|
HandleBall(joystick, code / 2, code % 2, events[i].value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2013-02-11 16:45:24 -08:00
|
|
|
case EV_SYN:
|
|
|
|
switch (code) {
|
|
|
|
case SYN_DROPPED :
|
|
|
|
#ifdef DEBUG_INPUT_EVENTS
|
|
|
|
printf("Event SYN_DROPPED dectected\n");
|
|
|
|
#endif
|
|
|
|
PollAllValues(joystick);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2006-07-10 21:04:37 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
void
|
|
|
|
SDL_SYS_JoystickUpdate(SDL_Joystick * joystick)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
int i;
|
|
|
|
|
2012-12-11 11:07:48 -05:00
|
|
|
HandleInputEvents(joystick);
|
2006-07-10 21:04:37 +00:00
|
|
|
|
|
|
|
/* Deliver ball motion updates */
|
|
|
|
for (i = 0; i < joystick->nballs; ++i) {
|
|
|
|
int xrel, yrel;
|
|
|
|
|
|
|
|
xrel = joystick->hwdata->balls[i].axis[0];
|
|
|
|
yrel = joystick->hwdata->balls[i].axis[1];
|
|
|
|
if (xrel || yrel) {
|
|
|
|
joystick->hwdata->balls[i].axis[0] = 0;
|
|
|
|
joystick->hwdata->balls[i].axis[1] = 0;
|
|
|
|
SDL_PrivateJoystickBall(joystick, (Uint8) i, xrel, yrel);
|
|
|
|
}
|
|
|
|
}
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Function to close a joystick after use */
|
2006-07-10 21:04:37 +00:00
|
|
|
void
|
|
|
|
SDL_SYS_JoystickClose(SDL_Joystick * joystick)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
if (joystick->hwdata) {
|
2012-12-10 15:50:42 -05:00
|
|
|
close(joystick->hwdata->fd);
|
2012-12-13 22:26:30 -05:00
|
|
|
if (joystick->hwdata->item) {
|
|
|
|
joystick->hwdata->item->hwdata = NULL;
|
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_free(joystick->hwdata->hats);
|
|
|
|
SDL_free(joystick->hwdata->balls);
|
|
|
|
SDL_free(joystick->hwdata->fname);
|
2006-07-10 21:04:37 +00:00
|
|
|
SDL_free(joystick->hwdata);
|
|
|
|
joystick->hwdata = NULL;
|
|
|
|
}
|
2012-11-26 16:37:54 -08:00
|
|
|
joystick->closed = 1;
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Function to perform any system-specific joystick related cleanup */
|
2006-07-10 21:04:37 +00:00
|
|
|
void
|
|
|
|
SDL_SYS_JoystickQuit(void)
|
2001-04-26 16:45:43 +00:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_joylist_item *item = NULL;
|
|
|
|
SDL_joylist_item *next = NULL;
|
|
|
|
|
|
|
|
for (item = SDL_joylist; item; item = next) {
|
|
|
|
next = item->next;
|
|
|
|
SDL_free(item->path);
|
|
|
|
SDL_free(item->name);
|
|
|
|
SDL_free(item);
|
|
|
|
}
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2012-12-11 12:07:06 -05:00
|
|
|
SDL_joylist = SDL_joylist_tail = NULL;
|
|
|
|
|
|
|
|
numjoysticks = 0;
|
|
|
|
instance_counter = 0;
|
|
|
|
|
|
|
|
#if SDL_USE_LIBUDEV
|
|
|
|
if (udev_mon != NULL) {
|
|
|
|
UDEV_udev_monitor_unref(udev_mon);
|
|
|
|
udev_mon = NULL;
|
|
|
|
}
|
|
|
|
if (udev != NULL) {
|
|
|
|
UDEV_udev_unref(udev);
|
|
|
|
udev = NULL;
|
2006-07-10 21:04:37 +00:00
|
|
|
}
|
2012-12-11 12:07:06 -05:00
|
|
|
UnloadUDEVLibrary();
|
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
}
|
|
|
|
|
2012-12-11 11:54:32 -08:00
|
|
|
SDL_JoystickGUID SDL_SYS_JoystickGetDeviceGUID( int device_index )
|
2012-11-26 16:37:54 -08:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
return JoystickByDevIndex(device_index)->guid;
|
2012-11-26 16:37:54 -08:00
|
|
|
}
|
|
|
|
|
2012-12-11 11:54:32 -08:00
|
|
|
SDL_JoystickGUID SDL_SYS_JoystickGetGUID(SDL_Joystick * joystick)
|
2012-11-26 16:37:54 -08:00
|
|
|
{
|
2012-12-11 12:07:06 -05:00
|
|
|
return joystick->hwdata->guid;
|
2012-11-26 16:37:54 -08:00
|
|
|
}
|
|
|
|
|
2006-04-14 04:46:47 +00:00
|
|
|
#endif /* SDL_JOYSTICK_LINUX */
|
2012-11-26 21:11:28 -08:00
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
/* vi: set ts=4 sw=4 expandtab: */
|