blob: 9a6902e62b31cab3bf83a5ff315a9acd199554f2 [file] [log] [blame]
Eric Pouech2a3b0a12000-02-26 13:14:04 +00001/*
2 * joystick functions
3 *
4 * Copyright 1997 Andreas Mohr
David D. Hagoodd6b65092002-09-04 18:43:05 +00005 * Copyright 2000 Wolfgang Schwotzer
6 * Copyright 2002 David Hagood
Eric Pouech2a3b0a12000-02-26 13:14:04 +00007 *
Alexandre Julliard0799c1a2002-03-09 23:29:33 +00008 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
12 *
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
Jonathan Ernst360a3f92006-05-18 14:49:52 +020020 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000021 *
22 * NOTES:
23 *
Alexandre Julliardb4e49552004-12-06 20:55:25 +000024 * - nearly all joystick functions can be regarded as obsolete,
25 * as Linux (2.1.x) now supports extended joysticks with a completely
26 * new joystick driver interface
Austin English65569202008-01-09 11:59:19 -060027 * New driver's documentation says:
Alexandre Julliardb4e49552004-12-06 20:55:25 +000028 * "For backward compatibility the old interface is still included,
29 * but will be dropped in the future."
30 * Thus we should implement the new interface and at most keep the old
31 * routines for backward compatibility.
32 * - better support of enhanced joysticks (Linux 2.2 interface is available)
33 * - support more joystick drivers (like the XInput extension)
34 * - should load joystick DLL as any other driver (instead of hardcoding)
35 * the driver's name, and load it as any low lever driver.
Eric Pouech2a3b0a12000-02-26 13:14:04 +000036 */
37
Eric Pouech2a3b0a12000-02-26 13:14:04 +000038#include "config.h"
Alexandre Julliard49b7fdc2005-08-03 21:25:10 +000039#include "wine/port.h"
Eric Pouech2a3b0a12000-02-26 13:14:04 +000040
Patrik Stridvalld016f812002-08-17 00:43:16 +000041#ifdef HAVE_UNISTD_H
42# include <unistd.h>
43#endif
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000044#include <stdarg.h>
Eric Pouech2a3b0a12000-02-26 13:14:04 +000045#include <stdio.h>
46#include <stdlib.h>
47#include <string.h>
48#include <fcntl.h>
Vincent Béron9a624912002-05-31 23:06:46 +000049#ifdef HAVE_SYS_IOCTL_H
Eric Pouech2a3b0a12000-02-26 13:14:04 +000050#include <sys/ioctl.h>
Steven Edwards48ac89b2002-05-19 22:25:02 +000051#endif
Alexandre Julliarde8a339c2004-03-01 21:32:02 +000052#ifdef HAVE_LINUX_IOCTL_H
53#include <linux/ioctl.h>
54#endif
Eric Pouech2a3b0a12000-02-26 13:14:04 +000055#ifdef HAVE_LINUX_JOYSTICK_H
56#include <linux/joystick.h>
Mike McCormack25f521f2006-06-13 16:58:38 +090057#ifdef SW_MAX
58#undef SW_MAX
59#endif
Mike Frysingerdd174c52006-04-18 18:58:44 -040060#define JOYDEV_NEW "/dev/input/js%d"
61#define JOYDEV_OLD "/dev/js%d"
Eric Pouech2a3b0a12000-02-26 13:14:04 +000062#endif
63#ifdef HAVE_SYS_ERRNO_H
64#include <sys/errno.h>
65#endif
François Gouget44a18222000-12-19 04:53:20 +000066
Eric Pouech2a3b0a12000-02-26 13:14:04 +000067#include "windef.h"
François Gouget44a18222000-12-19 04:53:20 +000068#include "winbase.h"
Eric Pouech2a3b0a12000-02-26 13:14:04 +000069#include "wingdi.h"
70#include "winuser.h"
Alexandre Julliardb4e49552004-12-06 20:55:25 +000071#include "winnls.h"
Eric Pouech2a3b0a12000-02-26 13:14:04 +000072#include "mmddk.h"
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000073#include "wine/debug.h"
Eric Pouech2a3b0a12000-02-26 13:14:04 +000074
Edgar Hucek2a3c30a2005-11-14 11:23:34 +000075#include "wine/unicode.h"
76
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000077WINE_DEFAULT_DEBUG_CHANNEL(joystick);
Eric Pouech2a3b0a12000-02-26 13:14:04 +000078
79#ifdef HAVE_LINUX_JOYSTICK_H
80
Vitaliy Margolen9fbe2492008-10-26 12:01:28 -060081#define MAXJOYSTICK (JOYSTICKID2 + 30)
Eric Pouech2a3b0a12000-02-26 13:14:04 +000082
Eric Pouech9aa678c2000-03-19 12:41:41 +000083typedef struct tagWINE_JSTCK {
84 int joyIntf;
85 int in_use;
Austin English65569202008-01-09 11:59:19 -060086 /* Some extra info we need to make this actually work under the
David D. Hagoodd6b65092002-09-04 18:43:05 +000087 Linux 2.2 event api.
88 First of all, we cannot keep closing and reopening the device file -
89 that blows away the state of the stick device, and we lose events. So, we
90 need to open the low-level device once, and close it when we are done.
91
92 Secondly, the event API only gives us what's changed. However, Windows apps
93 want the whole state every time, so we have to cache the data.
94 */
95
96 int dev; /* Linux level device file descriptor */
97 int x;
98 int y;
99 int z;
100 int r;
101 int u;
102 int v;
Vincent Pelletiera4b201a2009-01-17 12:54:07 +0100103 int pov_x;
104 int pov_y;
David D. Hagoodd6b65092002-09-04 18:43:05 +0000105 int buttons;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100106 char axesMap[ABS_MAX + 1];
Eric Pouech9aa678c2000-03-19 12:41:41 +0000107} WINE_JSTCK;
108
109static WINE_JSTCK JSTCK_Data[MAXJOYSTICK];
110
111/**************************************************************************
Vincent Béron9a624912002-05-31 23:06:46 +0000112 * JSTCK_drvGet [internal]
Eric Pouech9aa678c2000-03-19 12:41:41 +0000113 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100114static WINE_JSTCK *JSTCK_drvGet(DWORD_PTR dwDevID)
Eric Pouech9aa678c2000-03-19 12:41:41 +0000115{
116 int p;
117
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100118 if ((dwDevID - (DWORD_PTR)JSTCK_Data) % sizeof(JSTCK_Data[0]) != 0)
Eric Pouech9aa678c2000-03-19 12:41:41 +0000119 return NULL;
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100120 p = (dwDevID - (DWORD_PTR)JSTCK_Data) / sizeof(JSTCK_Data[0]);
Eric Pouech9aa678c2000-03-19 12:41:41 +0000121 if (p < 0 || p >= MAXJOYSTICK || !((WINE_JSTCK*)dwDevID)->in_use)
122 return NULL;
123
124 return (WINE_JSTCK*)dwDevID;
125}
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000126
127/**************************************************************************
Vincent Béron9a624912002-05-31 23:06:46 +0000128 * JSTCK_drvOpen [internal]
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000129 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100130static LRESULT JSTCK_drvOpen(LPSTR str, DWORD dwIntf)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000131{
Eric Pouech9aa678c2000-03-19 12:41:41 +0000132 if (dwIntf >= MAXJOYSTICK || JSTCK_Data[dwIntf].in_use)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000133 return 0;
Eric Pouech9aa678c2000-03-19 12:41:41 +0000134
135 JSTCK_Data[dwIntf].joyIntf = dwIntf;
136 JSTCK_Data[dwIntf].in_use = 1;
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100137 return (LRESULT)&JSTCK_Data[dwIntf];
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000138}
139
140/**************************************************************************
Vincent Béron9a624912002-05-31 23:06:46 +0000141 * JSTCK_drvClose [internal]
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000142 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100143static LRESULT JSTCK_drvClose(DWORD_PTR dwDevID)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000144{
Eric Pouech9aa678c2000-03-19 12:41:41 +0000145 WINE_JSTCK* jstck = JSTCK_drvGet(dwDevID);
146
147 if (jstck == NULL)
148 return 0;
149 jstck->in_use = 0;
David D. Hagoodd6b65092002-09-04 18:43:05 +0000150 if (jstck->dev > 0)
151 {
152 close(jstck->dev);
153 jstck->dev = 0;
154 }
Eric Pouech9aa678c2000-03-19 12:41:41 +0000155 return 1;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000156}
157
158struct js_status
159{
160 int buttons;
161 int x;
162 int y;
163};
164
165/**************************************************************************
166 * JSTCK_OpenDevice [internal]
167 */
Eric Pouech9aa678c2000-03-19 12:41:41 +0000168static int JSTCK_OpenDevice(WINE_JSTCK* jstick)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000169{
170 char buf[20];
171 int flags;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000172
David D. Hagoodd6b65092002-09-04 18:43:05 +0000173 if (jstick->dev > 0)
174 return jstick->dev;
175
Mike Frysingerdd174c52006-04-18 18:58:44 -0400176 sprintf(buf, JOYDEV_NEW, jstick->joyIntf);
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000177#ifdef HAVE_LINUX_22_JOYSTICK_API
178 flags = O_RDONLY | O_NONBLOCK;
179#else
180 flags = O_RDONLY;
181#endif
Mike Frysingerdd174c52006-04-18 18:58:44 -0400182 if ((jstick->dev = open(buf, flags)) < 0) {
183 sprintf(buf, JOYDEV_OLD, jstick->joyIntf);
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100184 if ((jstick->dev = open(buf, flags)) < 0)
185 return jstick->dev;
Mike Frysingerdd174c52006-04-18 18:58:44 -0400186 }
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100187#ifdef HAVE_LINUX_22_JOYSTICK_API
188 ioctl(jstick->dev, JSIOCGAXMAP, jstick->axesMap);
189#endif
Vincent Pelletier279690e2009-01-17 12:31:39 +0100190 return jstick->dev;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000191}
192
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000193
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000194/**************************************************************************
195 * JoyGetDevCaps [MMSYSTEM.102]
196 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100197static LRESULT JSTCK_GetDevCaps(DWORD_PTR dwDevID, LPJOYCAPSW lpCaps, DWORD dwSize)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000198{
Eric Pouech9aa678c2000-03-19 12:41:41 +0000199 WINE_JSTCK* jstck;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000200#ifdef HAVE_LINUX_22_JOYSTICK_API
201 int dev;
202 char nrOfAxes;
203 char nrOfButtons;
204 char identString[MAXPNAMELEN];
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100205 int i;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000206 int driverVersion;
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000207#else
208static const WCHAR ini[] = {'W','i','n','e',' ','J','o','y','s','t','i','c','k',' ','D','r','i','v','e','r',0};
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000209#endif
210
Eric Pouech9aa678c2000-03-19 12:41:41 +0000211 if ((jstck = JSTCK_drvGet(dwDevID)) == NULL)
212 return MMSYSERR_NODRIVER;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000213
214#ifdef HAVE_LINUX_22_JOYSTICK_API
Eric Pouech9aa678c2000-03-19 12:41:41 +0000215 if ((dev = JSTCK_OpenDevice(jstck)) < 0) return JOYERR_PARMS;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000216 ioctl(dev, JSIOCGAXES, &nrOfAxes);
217 ioctl(dev, JSIOCGBUTTONS, &nrOfButtons);
218 ioctl(dev, JSIOCGVERSION, &driverVersion);
Andrew Talbot1fd750c2008-07-12 20:37:24 +0100219 ioctl(dev, JSIOCGNAME(sizeof(identString)), identString);
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000220 TRACE("Driver: 0x%06x, Name: %s, #Axes: %d, #Buttons: %d\n",
221 driverVersion, identString, nrOfAxes, nrOfButtons);
222 lpCaps->wMid = MM_MICROSOFT;
223 lpCaps->wPid = MM_PC_JOYSTICK;
Rob Shearman466b6822009-02-18 20:32:27 +0000224 MultiByteToWideChar(CP_UNIXCP, 0, identString, -1, lpCaps->szPname, MAXPNAMELEN);
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000225 lpCaps->szPname[MAXPNAMELEN-1] = '\0';
226 lpCaps->wXmin = 0;
227 lpCaps->wXmax = 0xFFFF;
228 lpCaps->wYmin = 0;
229 lpCaps->wYmax = 0xFFFF;
230 lpCaps->wZmin = 0;
231 lpCaps->wZmax = (nrOfAxes >= 3) ? 0xFFFF : 0;
David D. Hagoodd6b65092002-09-04 18:43:05 +0000232#ifdef BODGE_THE_HAT
233 /* HalfLife won't allow you to map an axis event to things like
234 "next weapon" and "use". Linux reports the hat on my stick as
235 axis U and V. So, IFF BODGE_THE_HAT is defined, lie through our
236 teeth and say we have 32 buttons, and we will map the axises to
237 the high buttons. Really, perhaps this should be a registry entry,
238 or even a parameter to the Linux joystick driver (which would completely
239 remove the need for this.)
240 */
241 lpCaps->wNumButtons = 32;
242#else
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000243 lpCaps->wNumButtons = nrOfButtons;
David D. Hagoodd6b65092002-09-04 18:43:05 +0000244#endif
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000245 if (dwSize == sizeof(JOYCAPSW)) {
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000246 /* complete 95 structure */
247 lpCaps->wRmin = 0;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100248 lpCaps->wRmax = 0xFFFF;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000249 lpCaps->wUmin = 0;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100250 lpCaps->wUmax = 0xFFFF;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000251 lpCaps->wVmin = 0;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100252 lpCaps->wVmax = 0xFFFF;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000253 lpCaps->wMaxAxes = 6; /* same as MS Joystick Driver */
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100254 lpCaps->wNumAxes = 0; /* nr of axes in use */
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000255 lpCaps->wMaxButtons = 32; /* same as MS Joystick Driver */
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000256 lpCaps->szRegKey[0] = 0;
257 lpCaps->szOEMVxD[0] = 0;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000258 lpCaps->wCaps = 0;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100259 for (i = 0; i < nrOfAxes; i++) {
260 switch (jstck->axesMap[i]) {
261 case 0: /* X */
262 case 1: /* Y */
263 lpCaps->wNumAxes++;
264 break;
265 case 2: /* Z */
266 case 6: /* Throttle */
267 lpCaps->wNumAxes++;
268 lpCaps->wCaps |= JOYCAPS_HASZ;
269 break;
270 case 5: /* Rz */
271 case 7: /* Rudder */
272 lpCaps->wNumAxes++;
273 lpCaps->wCaps |= JOYCAPS_HASR;
274 break;
275 case 3: /* Rx */
276 lpCaps->wNumAxes++;
277 lpCaps->wCaps |= JOYCAPS_HASU;
278 break;
279 case 4: /* Ry */
280 lpCaps->wNumAxes++;
281 lpCaps->wCaps |= JOYCAPS_HASV;
282 break;
Vincent Pelletiera4b201a2009-01-17 12:54:07 +0100283 case 16: /* Hat 0 X */
284 case 17: /* Hat 0 Y */
285 lpCaps->wCaps |= JOYCAPS_HASPOV | JOYCAPS_POV4DIR;
286 /* TODO: JOYCAPS_POVCTS handling */
287 break;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100288 default:
289 WARN("Unknown axis %hhu(%hhu). Skipped.\n", jstck->axesMap[i], i);
290 }
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000291 }
292 }
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000293#else
294 lpCaps->wMid = MM_MICROSOFT;
295 lpCaps->wPid = MM_PC_JOYSTICK;
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000296 strcpyW(lpCaps->szPname, ini); /* joystick product name */
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000297 lpCaps->wXmin = 0;
298 lpCaps->wXmax = 0xFFFF;
299 lpCaps->wYmin = 0;
300 lpCaps->wYmax = 0xFFFF;
301 lpCaps->wZmin = 0;
302 lpCaps->wZmax = 0;
303 lpCaps->wNumButtons = 2;
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000304 if (dwSize == sizeof(JOYCAPSW)) {
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000305 /* complete 95 structure */
306 lpCaps->wRmin = 0;
307 lpCaps->wRmax = 0;
308 lpCaps->wUmin = 0;
309 lpCaps->wUmax = 0;
310 lpCaps->wVmin = 0;
311 lpCaps->wVmax = 0;
312 lpCaps->wCaps = 0;
313 lpCaps->wMaxAxes = 2;
314 lpCaps->wNumAxes = 2;
315 lpCaps->wMaxButtons = 4;
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000316 lpCaps->szRegKey[0] = 0;
317 lpCaps->szOEMVxD[0] = 0;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000318 }
319#endif
320
321 return JOYERR_NOERROR;
322}
323
324/**************************************************************************
325 * JSTCK_GetPos [internal]
326 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100327static LRESULT JSTCK_GetPosEx(DWORD_PTR dwDevID, LPJOYINFOEX lpInfo)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000328{
Eric Pouech9aa678c2000-03-19 12:41:41 +0000329 WINE_JSTCK* jstck;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000330 int dev;
331#ifdef HAVE_LINUX_22_JOYSTICK_API
332 struct js_event ev;
333#else
334 struct js_status js;
335 int dev_stat;
336#endif
Vincent Béron9a624912002-05-31 23:06:46 +0000337
Eric Pouech9aa678c2000-03-19 12:41:41 +0000338 if ((jstck = JSTCK_drvGet(dwDevID)) == NULL)
339 return MMSYSERR_NODRIVER;
340
341 if ((dev = JSTCK_OpenDevice(jstck)) < 0) return JOYERR_PARMS;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000342
343#ifdef HAVE_LINUX_22_JOYSTICK_API
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000344 while ((read(dev, &ev, sizeof(struct js_event))) > 0) {
David D. Hagoodd6b65092002-09-04 18:43:05 +0000345 if (ev.type == (JS_EVENT_AXIS)) {
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100346 switch (jstck->axesMap[ev.number]) {
347 case 0: /* X */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000348 jstck->x = ev.value;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000349 break;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100350 case 1: /* Y */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000351 jstck->y = ev.value;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000352 break;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100353 case 2: /* Z */
354 case 6: /* Throttle */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000355 jstck->z = ev.value;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000356 break;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100357 case 5: /* Rz */
358 case 7: /* Rudder */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000359 jstck->r = ev.value;
360 break;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100361 case 3: /* Rx */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000362 jstck->u = ev.value;
363 break;
Vincent Pelletierb04e8222009-01-17 12:35:05 +0100364 case 4: /* Ry */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000365 jstck->v = ev.value;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000366 break;
Vincent Pelletiera4b201a2009-01-17 12:54:07 +0100367 case 16: /* Hat 0 X */
368 jstck->pov_x = ev.value;
369 break;
370 case 17: /* Hat 0 Y */
371 jstck->pov_y = ev.value;
372 break;
Vincent Béron9a624912002-05-31 23:06:46 +0000373 default:
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000374 FIXME("Unknown joystick event '%d'\n", ev.number);
375 }
David D. Hagoodd6b65092002-09-04 18:43:05 +0000376 } else if (ev.type == (JS_EVENT_BUTTON)) {
377 if (ev.value) {
378 jstck->buttons |= (1 << ev.number);
Vincent Béron9a624912002-05-31 23:06:46 +0000379 /* FIXME: what to do for this field when
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000380 * multiple buttons are depressed ?
381 */
David D. Hagoodd6b65092002-09-04 18:43:05 +0000382 if (lpInfo->dwFlags & JOY_RETURNBUTTONS)
383 lpInfo->dwButtonNumber = ev.number + 1;
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000384 }
David D. Hagoodd6b65092002-09-04 18:43:05 +0000385 else
386 jstck->buttons &= ~(1 << ev.number);
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000387 }
388 }
389 /* EAGAIN is returned when the queue is empty */
390 if (errno != EAGAIN) {
391 /* FIXME: error should not be ignored */
Andreas Mohr34965562000-08-26 20:31:48 +0000392 ERR("Error while reading joystick state (%s)\n", strerror(errno));
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000393 }
David D. Hagoodd6b65092002-09-04 18:43:05 +0000394 /* Now, copy the cached values into Window's structure... */
395 if (lpInfo->dwFlags & JOY_RETURNBUTTONS)
396 lpInfo->dwButtons = jstck->buttons;
397 if (lpInfo->dwFlags & JOY_RETURNX)
398 lpInfo->dwXpos = jstck->x + 32767;
399 if (lpInfo->dwFlags & JOY_RETURNY)
400 lpInfo->dwYpos = jstck->y + 32767;
401 if (lpInfo->dwFlags & JOY_RETURNZ)
402 lpInfo->dwZpos = jstck->z + 32767;
403 if (lpInfo->dwFlags & JOY_RETURNR)
404 lpInfo->dwRpos = jstck->r + 32767;
Alexandre Julliardd0ee9f92005-03-02 12:23:20 +0000405# ifdef BODGE_THE_HAT
David D. Hagoodd6b65092002-09-04 18:43:05 +0000406 else if (lpInfo->dwFlags & JOY_RETURNBUTTONS)
407 {
408 if (jstck->r > 0)
409 lpInfo->dwButtons |= 1<<7;
410 else if (jstck->r < 0)
411 lpInfo->dwButtons |= 1<<8;
412 }
Alexandre Julliardd0ee9f92005-03-02 12:23:20 +0000413# endif
David D. Hagoodd6b65092002-09-04 18:43:05 +0000414 if (lpInfo->dwFlags & JOY_RETURNU)
415 lpInfo->dwUpos = jstck->u + 32767;
Alexandre Julliardd0ee9f92005-03-02 12:23:20 +0000416# ifdef BODGE_THE_HAT
David D. Hagoodd6b65092002-09-04 18:43:05 +0000417 else if (lpInfo->dwFlags & JOY_RETURNBUTTONS)
418 {
419 if (jstck->u > 0)
420 lpInfo->dwButtons |= 1<<9;
421 else if (jstck->u < 0)
422 lpInfo->dwButtons |= 1<<10;
423 }
Alexandre Julliardd0ee9f92005-03-02 12:23:20 +0000424# endif
David D. Hagoodd6b65092002-09-04 18:43:05 +0000425 if (lpInfo->dwFlags & JOY_RETURNV)
426 lpInfo->dwVpos = jstck->v + 32767;
Vincent Pelletiera4b201a2009-01-17 12:54:07 +0100427 if (lpInfo->dwFlags & JOY_RETURNPOV) {
428 if (jstck->pov_y > 0) {
429 if (jstck->pov_x < 0)
430 lpInfo->dwPOV = 22500; /* SW */
431 else if (jstck->pov_x > 0)
432 lpInfo->dwPOV = 13500; /* SE */
433 else
434 lpInfo->dwPOV = 18000; /* S, JOY_POVBACKWARD */
435 } else if (jstck->pov_y < 0) {
436 if (jstck->pov_x < 0)
437 lpInfo->dwPOV = 31500; /* NW */
438 else if (jstck->pov_x > 0)
439 lpInfo->dwPOV = 4500; /* NE */
440 else
441 lpInfo->dwPOV = 0; /* N, JOY_POVFORWARD */
442 } else if (jstck->pov_x < 0)
443 lpInfo->dwPOV = 27000; /* W, JOY_POVLEFT */
444 else if (jstck->pov_x > 0)
445 lpInfo->dwPOV = 9000; /* E, JOY_POVRIGHT */
446 else
447 lpInfo->dwPOV = JOY_POVCENTERED; /* Center */
448 }
David D. Hagoodd6b65092002-09-04 18:43:05 +0000449
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000450#else
451 dev_stat = read(dev, &js, sizeof(js));
452 if (dev_stat != sizeof(js)) {
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000453 return JOYERR_UNPLUGGED; /* FIXME: perhaps wrong, but what should I return else ? */
454 }
455 js.x = js.x<<8;
456 js.y = js.y<<8;
457 if (lpInfo->dwFlags & JOY_RETURNX)
458 lpInfo->dwXpos = js.x; /* FIXME: perhaps multiply it somehow ? */
459 if (lpInfo->dwFlags & JOY_RETURNY)
460 lpInfo->dwYpos = js.y;
461 if (lpInfo->dwFlags & JOY_RETURNBUTTONS)
462 lpInfo->dwButtons = js.buttons;
463#endif
464
Michael Stefaniuc41dd56c2006-09-29 01:19:00 +0200465 TRACE("x: %d, y: %d, z: %d, r: %d, u: %d, v: %d, buttons: 0x%04x, flags: 0x%04x (fd %d)\n",
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000466 lpInfo->dwXpos, lpInfo->dwYpos, lpInfo->dwZpos,
467 lpInfo->dwRpos, lpInfo->dwUpos, lpInfo->dwVpos,
Andrew Talbot2d90d0d2008-02-08 21:58:40 +0000468 lpInfo->dwButtons, lpInfo->dwFlags, dev
David D. Hagoodd6b65092002-09-04 18:43:05 +0000469 );
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000470
471 return JOYERR_NOERROR;
472}
473
474/**************************************************************************
475 * JSTCK_GetPos [internal]
476 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100477static LRESULT JSTCK_GetPos(DWORD_PTR dwDevID, LPJOYINFO lpInfo)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000478{
479 JOYINFOEX ji;
480 LONG ret;
481
482 memset(&ji, 0, sizeof(ji));
483
484 ji.dwSize = sizeof(ji);
485 ji.dwFlags = JOY_RETURNX | JOY_RETURNY | JOY_RETURNZ | JOY_RETURNBUTTONS;
486 ret = JSTCK_GetPosEx(dwDevID, &ji);
487 if (ret == JOYERR_NOERROR) {
488 lpInfo->wXpos = ji.dwXpos;
489 lpInfo->wYpos = ji.dwYpos;
490 lpInfo->wZpos = ji.dwZpos;
491 lpInfo->wButtons = ji.dwButtons;
492 }
493
494 return ret;
495}
496
497/**************************************************************************
Patrik Stridvall044855c2001-07-11 18:56:41 +0000498 * DriverProc (JOYSTICK.@)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000499 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100500LRESULT CALLBACK JSTCK_DriverProc(DWORD_PTR dwDevID, HDRVR hDriv, UINT wMsg,
501 LPARAM dwParam1, LPARAM dwParam2)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000502{
503 /* EPP TRACE("(%08lX, %04X, %08lX, %08lX, %08lX)\n", */
504 /* EPP dwDevID, hDriv, wMsg, dwParam1, dwParam2); */
Vincent Béron9a624912002-05-31 23:06:46 +0000505
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000506 switch(wMsg) {
507 case DRV_LOAD: return 1;
508 case DRV_FREE: return 1;
Eric Pouech9aa678c2000-03-19 12:41:41 +0000509 case DRV_OPEN: return JSTCK_drvOpen((LPSTR)dwParam1, dwParam2);
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000510 case DRV_CLOSE: return JSTCK_drvClose(dwDevID);
511 case DRV_ENABLE: return 1;
512 case DRV_DISABLE: return 1;
513 case DRV_QUERYCONFIGURE: return 1;
514 case DRV_CONFIGURE: MessageBoxA(0, "JoyStick MultiMedia Driver !", "JoyStick Driver", MB_OK); return 1;
515 case DRV_INSTALL: return DRVCNF_RESTART;
516 case DRV_REMOVE: return DRVCNF_RESTART;
517
Eric Pouech9aa678c2000-03-19 12:41:41 +0000518 case JDD_GETNUMDEVS: return 1;
Alexandre Julliardb4e49552004-12-06 20:55:25 +0000519 case JDD_GETDEVCAPS: return JSTCK_GetDevCaps(dwDevID, (LPJOYCAPSW)dwParam1, dwParam2);
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000520 case JDD_GETPOS: return JSTCK_GetPos(dwDevID, (LPJOYINFO)dwParam1);
Vincent Béron9a624912002-05-31 23:06:46 +0000521 case JDD_SETCALIBRATION:
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000522 case JDD_CONFIGCHANGED: return JOYERR_NOCANDO;
523 case JDD_GETPOSEX: return JSTCK_GetPosEx(dwDevID, (LPJOYINFOEX)dwParam1);
524 default:
525 return DefDriverProc(dwDevID, hDriv, wMsg, dwParam1, dwParam2);
526 }
527}
528
529#else
530
531/**************************************************************************
Patrik Stridvall044855c2001-07-11 18:56:41 +0000532 * DriverProc (JOYSTICK.@)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000533 */
Dmitry Timoshkov8140d662005-11-28 20:53:22 +0100534LRESULT CALLBACK JSTCK_DriverProc(DWORD_PTR dwDevID, HDRVR hDriv, UINT wMsg,
535 LPARAM dwParam1, LPARAM dwParam2)
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000536{
537 /* EPP TRACE("(%08lX, %04X, %08lX, %08lX, %08lX)\n", */
538 /* EPP dwDevID, hDriv, wMsg, dwParam1, dwParam2); */
Vincent Béron9a624912002-05-31 23:06:46 +0000539
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000540 switch(wMsg) {
Vincent Béron9a624912002-05-31 23:06:46 +0000541 case DRV_LOAD:
542 case DRV_FREE:
543 case DRV_OPEN:
544 case DRV_CLOSE:
545 case DRV_ENABLE:
546 case DRV_DISABLE:
Eric Pouech2a3b0a12000-02-26 13:14:04 +0000547 case DRV_QUERYCONFIGURE: return 0;
548 case DRV_CONFIGURE: MessageBoxA(0, "JoyStick MultiMedia Driver !", "JoyStick Driver", MB_OK); return 1;
549 case DRV_INSTALL: return DRVCNF_RESTART;
550 case DRV_REMOVE: return DRVCNF_RESTART;
551 default:
552 return DefDriverProc(dwDevID, hDriv, wMsg, dwParam1, dwParam2);
553 }
554}
555
556#endif