mirror of
https://github.com/paparazzi/paparazzi.git
synced 2026-05-09 22:49:53 +08:00
475 lines
14 KiB
C
475 lines
14 KiB
C
/*
|
|
* joystick lib
|
|
*
|
|
* based on Force Feedback: Constant Force Stress Test
|
|
* Copyright (C) 2001 Oliver Hamann
|
|
*
|
|
* This program 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 Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
/* 1/26/2011 - jpeverill added
|
|
support for joystick specific event interface. During testing it was found that many devices do not work properly with the standard event interface (such as the PS3 joystick). The joystick interface seems to work better. The old event interface is used if "event" is found in the name, otherwise the joystick interface is used.
|
|
now kills the process in the event of a read error (joystick unplugs)
|
|
*/
|
|
|
|
//#define STICK_DBG 1
|
|
|
|
#include "usb_stick.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <math.h>
|
|
#include <glib.h>
|
|
#include <sys/ioctl.h>
|
|
#include <Ivy/ivy.h>
|
|
#include <Ivy/ivyglibloop.h>
|
|
|
|
//needed for joystick interface
|
|
#include <linux/input.h>
|
|
#include <linux/joystick.h>
|
|
|
|
#ifdef STICK_DBG
|
|
#define dbgprintf fprintf
|
|
#else
|
|
#define dbgprintf(x ...)
|
|
#endif
|
|
|
|
#define MIN_BUTTON_CODE BTN_JOYSTICK
|
|
#define MAX_BUTTON_CODE BTN_THUMBR+1
|
|
#define MIN_ABS_CODE ABS_X
|
|
#define MAX_ABS_CODE ABS_MAX+1
|
|
|
|
#define ABS_MAX_VALUE 254
|
|
#define ABS_MID_VALUE 127
|
|
|
|
#define BUTTON_COUNT STICK_BUTTON_COUNT
|
|
#define AXIS_COUNT STICK_AXIS_COUNT
|
|
|
|
/* Helper for testing large bit masks */
|
|
#define TEST_BIT(bit,bits) (((bits[bit>>5]>>(bit&0x1f))&1)!=0)
|
|
|
|
/* Global variables about the initialized device */
|
|
int stick_device_handle;
|
|
|
|
int8_t stick_axis_values[AXIS_COUNT] = {0, 0, 0, 0, 0, 0};
|
|
int32_t stick_button_values = 0;
|
|
|
|
int axis_code[AXIS_COUNT];
|
|
int stick_axis_count = 0;
|
|
int button_code[BUTTON_COUNT];
|
|
int stick_button_count = 0;
|
|
int hid_led_count = 0;
|
|
|
|
|
|
int32_t axis_min[AXIS_COUNT], axis_max[AXIS_COUNT];
|
|
|
|
int event_mode = STICK_MODE_UNKNOWN;
|
|
|
|
struct stick_code_param_ stick_init_param = {
|
|
0, {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
0, {0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
|
|
};
|
|
|
|
//struct ff_effect effect;
|
|
|
|
|
|
void led_test( void ) {
|
|
|
|
struct input_event ev; /* the event */
|
|
|
|
int lednum;
|
|
|
|
/* we turn off all the LEDs to start */
|
|
ev.type = EV_LED;
|
|
|
|
while (1) {
|
|
for (lednum=0;lednum<32;lednum++) {
|
|
ev.code = lednum;
|
|
ev.value = 1;
|
|
write(stick_device_handle, &ev, sizeof(struct input_event));
|
|
}
|
|
|
|
usleep(1000000);
|
|
|
|
for (lednum=0;lednum<32;lednum++) {
|
|
ev.code = lednum;
|
|
ev.value = 0;
|
|
write(stick_device_handle, &ev, sizeof(struct input_event));
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
int init_hid_device(char* device_name)
|
|
{
|
|
int cnt;
|
|
unsigned long buttons,axes;
|
|
unsigned long key_bits[32],abs_bits[32],led_bits[32];
|
|
// unsigned long ff_bits[32];
|
|
int valbuf[16];
|
|
char name[MAX_NAME_LENGTH] = "Unknown";
|
|
|
|
stick_button_count = 0;
|
|
stick_axis_count = 0;
|
|
|
|
/* Detect device mode, event interface or joystick */
|
|
if (strstr( device_name, "event" ) == 0 ) {
|
|
dbgprintf(stderr," Using Joystick interface\n");
|
|
event_mode = STICK_MODE_JOYSTICK;
|
|
} else {
|
|
dbgprintf(stderr," Using Event interface\n");
|
|
event_mode = STICK_MODE_EVENT;
|
|
}
|
|
|
|
/* Open event device read only (with write permission for ff) */
|
|
stick_device_handle = open(device_name,O_RDONLY|O_NONBLOCK);
|
|
if (stick_device_handle<0) {
|
|
dbgprintf(stderr,"ERROR: can not open %s (%s) [%s:%d]\n",
|
|
device_name,strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
|
|
/* LED testing */
|
|
|
|
// led_test( );
|
|
|
|
|
|
/* Which buttons has the device? */
|
|
memset(key_bits,0,32*sizeof(unsigned long));
|
|
memset(led_bits,0,32*sizeof(unsigned long));
|
|
if (event_mode == STICK_MODE_EVENT ) {
|
|
if (ioctl(stick_device_handle,EVIOCGBIT(EV_KEY,32*sizeof(unsigned long)),key_bits)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get key bits (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
//read LED count
|
|
if (ioctl(stick_device_handle,EVIOCGLED(32*sizeof(led_bits)),led_bits)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get LED bits (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
dbgprintf(stderr,"LEDs: %X",*led_bits);
|
|
|
|
|
|
} else {
|
|
if (ioctl(stick_device_handle,JSIOCGBUTTONS,&buttons)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get key bits (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
}
|
|
|
|
/* Store buttons */
|
|
if (stick_init_param.button_count > 0) {
|
|
for (cnt = 0; cnt < MIN(stick_init_param.button_count,BUTTON_COUNT); cnt++) {
|
|
if (!TEST_BIT(stick_init_param.button_code[cnt], key_bits)) {
|
|
dbgprintf(stderr,"ERROR: no suitable custom button %d found [%s:%d]\n",cnt,__FILE__,__LINE__);
|
|
return 1;
|
|
}
|
|
}
|
|
stick_button_count = stick_init_param.button_count;
|
|
memcpy(button_code,stick_init_param.button_code,BUTTON_COUNT*sizeof(int));
|
|
}
|
|
else {
|
|
if (event_mode == STICK_MODE_EVENT) {
|
|
for (cnt = MIN_BUTTON_CODE; cnt < MAX_BUTTON_CODE; cnt++) {
|
|
if (TEST_BIT(cnt, key_bits)) {
|
|
button_code[stick_button_count++] = cnt;
|
|
dbgprintf(stderr,"Available button: %d (0x%x)\n",cnt,cnt);
|
|
}
|
|
if (stick_button_count == BUTTON_COUNT) break;
|
|
}
|
|
} else {
|
|
for (cnt = 0; cnt < buttons; cnt++) {
|
|
button_code[stick_button_count++] = cnt;
|
|
dbgprintf(stderr,"Available button: %d (0x%x)\n",cnt,cnt);
|
|
if (stick_button_count == BUTTON_COUNT) break;
|
|
}
|
|
}
|
|
|
|
if (stick_button_count == 0) {
|
|
dbgprintf(stderr,"ERROR: no suitable buttons found [%s:%d]\n",__FILE__,__LINE__);
|
|
}
|
|
}
|
|
|
|
/* Which axis has the device? */
|
|
memset(abs_bits,0,32*sizeof(unsigned long));
|
|
if (event_mode == STICK_MODE_EVENT ) {
|
|
if (ioctl(stick_device_handle,EVIOCGBIT(EV_ABS,32*sizeof(unsigned long)),abs_bits)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get abs bits (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
} else {
|
|
if (ioctl(stick_device_handle,JSIOCGAXES,&axes)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get abs bits (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
}
|
|
|
|
/* Store axis */
|
|
if (stick_init_param.axis_count > 0) {
|
|
for (cnt = 0; cnt < MIN(stick_init_param.axis_count,AXIS_COUNT); cnt++) {
|
|
if (!TEST_BIT(stick_init_param.axis_code[cnt], abs_bits)) {
|
|
dbgprintf(stderr,"ERROR: no suitable custom axis %d found [%s:%d]\n",cnt,__FILE__,__LINE__);
|
|
return 1;
|
|
}
|
|
}
|
|
stick_axis_count = stick_init_param.axis_count;
|
|
memcpy(axis_code,stick_init_param.axis_code,AXIS_COUNT*sizeof(int));
|
|
}
|
|
else {
|
|
if(event_mode == STICK_MODE_EVENT) {
|
|
for (cnt = MIN_ABS_CODE; cnt < MAX_ABS_CODE; cnt++) {
|
|
if (TEST_BIT(cnt, abs_bits)) {
|
|
axis_code[stick_axis_count++] = cnt;
|
|
dbgprintf(stderr,"Available axis: %d (0x%x)\n",cnt,cnt);
|
|
}
|
|
if (stick_axis_count == AXIS_COUNT) break;
|
|
}
|
|
} else {
|
|
for (cnt = 0; cnt < axes; cnt++) {
|
|
axis_code[stick_axis_count++] = cnt;
|
|
dbgprintf(stderr,"Available axis: %d (0x%x)\n",cnt,cnt);
|
|
if (stick_axis_count == AXIS_COUNT) break;
|
|
}
|
|
}
|
|
|
|
// at least 2 axis are needed in auto detection
|
|
if (stick_axis_count < 2) {
|
|
dbgprintf(stderr,"ERROR: no suitable axis found [%s:%d]\n",__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
}
|
|
|
|
/* Axis param */
|
|
for (cnt = 0; cnt < stick_axis_count; cnt++)
|
|
{
|
|
if (event_mode == STICK_MODE_EVENT ) {
|
|
/* get axis value range */
|
|
if (ioctl(stick_device_handle,EVIOCGABS(axis_code[cnt]),valbuf)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get axis %d value range (%s) [%s:%d]\n",
|
|
cnt,strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
axis_min[cnt]=valbuf[1];
|
|
axis_max[cnt]=valbuf[2];
|
|
} else {
|
|
// with joystick interface, all axes are signed 16 bit with full range
|
|
axis_min[cnt]=-32768;
|
|
axis_max[cnt]=32768;
|
|
}
|
|
|
|
if (axis_min[cnt]>=axis_max[cnt]) {
|
|
dbgprintf(stderr,"ERROR: bad axis %d value range (%d,%d) [%s:%d]\n",
|
|
cnt,axis_min[cnt],axis_max[cnt],__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
dbgprintf(stderr,"Axis %d : parameters = [%d,%d]\n",
|
|
cnt,axis_min[cnt],axis_max[cnt]);
|
|
}
|
|
|
|
//--------------------------------------------------
|
|
// force feedback, TBD feature
|
|
//--------------------------------------------------
|
|
#if 0
|
|
/* Now get some information about force feedback */
|
|
memset(ff_bits,0,32*sizeof(unsigned long));
|
|
if (ioctl(device_handle,EVIOCGBIT(EV_FF ,32*sizeof(unsigned long)),ff_bits)<0) {
|
|
dbgprintf(stderr,"ERROR: can not get ff bits (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
|
|
/* force feedback supported? */
|
|
if (!TEST_BIT(FF_CONSTANT,ff_bits)) {
|
|
dbgprintf(stderr,"ERROR: device (or driver) has no force feedback support [%s:%d]\n",
|
|
__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
|
|
/* Switch off auto centering */
|
|
memset(&event,0,sizeof(event));
|
|
event.type=EV_FF;
|
|
event.code=FF_AUTOCENTER;
|
|
event.value=0;
|
|
if (write(device_handle,&event,sizeof(event))!=sizeof(event)) {
|
|
dbgprintf(stderr,"ERROR: failed to disable auto centering (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
|
|
/* Initialize constant force effect */
|
|
memset(&effect,0,sizeof(effect));
|
|
effect.type=FF_CONSTANT;
|
|
effect.id=-1;
|
|
effect.trigger.button=0;
|
|
effect.trigger.interval=0;
|
|
effect.replay.length=0xffff;
|
|
effect.replay.delay=0;
|
|
effect.u.constant.level=0;
|
|
effect.direction=0xC000;
|
|
effect.u.constant.envelope.attack_length=0;
|
|
effect.u.constant.envelope.attack_level=0;
|
|
effect.u.constant.envelope.fade_length=0;
|
|
effect.u.constant.envelope.fade_level=0;
|
|
|
|
/* Upload effect */
|
|
if (ioctl(device_handle,EVIOCSFF,&effect)==-1) {
|
|
dbgprintf(stderr,"ERROR: uploading effect failed (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
|
|
/* Start effect */
|
|
memset(&event,0,sizeof(event));
|
|
event.type=EV_FF;
|
|
event.code=effect.id;
|
|
event.value=1;
|
|
if (write(device_handle,&event,sizeof(event))!=sizeof(event)) {
|
|
dbgprintf(stderr,"ERROR: starting effect failed (%s) [%s:%d]\n",
|
|
strerror(errno),__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
#endif
|
|
|
|
if (event_mode == STICK_MODE_EVENT ) {
|
|
ioctl(stick_device_handle, EVIOCGNAME(sizeof(name)), name);
|
|
} else {
|
|
ioctl(stick_device_handle, JSIOCGNAME(MAX_NAME_LENGTH), name);
|
|
}
|
|
printf("Input device name: \"%s\" on device \"%s\"\n", name, device_name);
|
|
|
|
return(0);
|
|
}
|
|
|
|
|
|
int stick_read( void ) {
|
|
|
|
int cnt;
|
|
struct input_event event;
|
|
struct js_event jsevent;
|
|
|
|
if (event_mode == STICK_MODE_EVENT ) {
|
|
/* Get events */
|
|
while (read(stick_device_handle,&event,sizeof(event))==sizeof(event)) {
|
|
|
|
switch (event.type) {
|
|
case EV_KEY:
|
|
for (cnt = 0; cnt < stick_button_count; cnt++) {
|
|
if (event.code == button_code[cnt]) {
|
|
if (event.value) stick_button_values |= (1 << cnt); // Set bit
|
|
else stick_button_values &= ~(1 << cnt); // Clear bit
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case EV_ABS:
|
|
for (cnt = 0; cnt < stick_axis_count; cnt++) {
|
|
if (event.code == axis_code[cnt]) {
|
|
stick_axis_values[cnt] = (((event.value) - axis_min[cnt]))*ABS_MAX_VALUE / (axis_max[cnt] - axis_min[cnt]) - ABS_MID_VALUE;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Get joystick events */
|
|
while (read(stick_device_handle,&jsevent,sizeof(jsevent))==sizeof(jsevent)) {
|
|
|
|
switch (jsevent.type) {
|
|
case JS_EVENT_BUTTON:
|
|
for (cnt = 0; cnt < stick_button_count; cnt++) {
|
|
if (jsevent.number == button_code[cnt]) {
|
|
if (jsevent.value) stick_button_values |= (1 << cnt); // Set bit
|
|
else stick_button_values &= ~(1 << cnt); // Clear bit
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case JS_EVENT_AXIS:
|
|
for (cnt = 0; cnt < stick_axis_count; cnt++) {
|
|
if (jsevent.number == axis_code[cnt]) {
|
|
stick_axis_values[cnt] = (( (jsevent.value - axis_min[cnt]) * ABS_MAX_VALUE ) / (axis_max[cnt] - axis_min[cnt])) - ABS_MID_VALUE;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (errno != EAGAIN) {
|
|
printf("Joystick read error!\n");
|
|
exit(1);
|
|
}
|
|
|
|
dbgprintf(stderr,"buttons ");
|
|
for (cnt = 0; cnt < stick_button_count; cnt++) {
|
|
dbgprintf(stderr,"%d ",(stick_button_values >> cnt) & 1 );
|
|
}
|
|
|
|
dbgprintf(stderr,"| axes ");
|
|
for (cnt = 0; cnt < stick_axis_count; cnt++) {
|
|
dbgprintf(stderr,"%d ",stick_axis_values[cnt]);
|
|
}
|
|
|
|
dbgprintf(stderr,"\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int stick_init( char * device_name ) {
|
|
|
|
char devname[MAX_NAME_LENGTH];
|
|
int cnt = 0;
|
|
|
|
/* test device_name, else look for a suitable device */
|
|
if (device_name != NULL) {
|
|
if (init_hid_device(device_name) != 0) cnt = STICK_INPUT_DEV_MAX;
|
|
}
|
|
else {
|
|
for (cnt = 0; cnt < STICK_INPUT_DEV_MAX; cnt++) {
|
|
sprintf(devname, STICK_DEVICE_NAME "%d", cnt);
|
|
if (init_hid_device(devname) == 0) break;
|
|
}
|
|
}
|
|
|
|
/* return 1 if no device found */
|
|
if (cnt == STICK_INPUT_DEV_MAX) {
|
|
fprintf(stderr,"ERROR: no suitable joystick found [%s:%d]\n",
|
|
__FILE__,__LINE__);
|
|
return(1);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|