4 * Copyright 1997 Andreas Mohr
6 * nearly all joystick functions can be regarded as obsolete,
7 * as Linux (2.1.x) now supports extended joysticks
8 * with a completely new joystick driver interface
9 * new driver's docu says:
10 * "For backward compatibility the old interface is still included,
11 * but will be dropped in the future."
12 * Thus we should implement the new interface and at most keep the old
13 * routines for backward compatibility.
21 #include <sys/ioctl.h>
30 static int count_use
[4] = {0, 0, 0, 0};
32 static int joy_nr_open
= 0;
33 static BOOL16 joyCaptured
= FALSE
;
34 static HWND16 CaptureWnd
[2] = {0, 0};
35 static int joy_dev
[2] = {-1, -1};
36 static JOYINFO16 joyCapData
[2];
37 static unsigned int joy_threshold
[2] = {0, 0};
47 /**************************************************************************
48 * joyOpenDriver [internal]
50 BOOL16
joyOpenDriver(WORD wID
)
52 char dev_name
[] = "/dev/jsx";
54 if (joy_dev
[wID
] >= 0) return TRUE
;
55 dev_name
[strlen(dev_name
)-1]=(char) wID
+0x30;
56 if ((joy_dev
[wID
] = open(dev_name
, O_RDONLY
)) >= 0) {
63 /**************************************************************************
64 * joyCloseDriver [internal]
66 void joyCloseDriver(WORD wID
)
68 if (joy_dev
[wID
] >= 0) {
75 /**************************************************************************
76 * joySendMessages [internal]
78 void joySendMessages(void)
84 for (joy
=0; joy
< 4; joy
++)
85 if (joy_dev
[joy
] >= 0) {
86 if (count_use
[joy
] > 250) {
93 if (joyCaptured
== FALSE
) return;
94 dprintf_mmsys(stddeb
, "JoySendMessages()\n");
95 for (joy
=0; joy
< 4; joy
++) {
96 if (joyOpenDriver(joy
) == FALSE
) continue;
97 dev_stat
= read(joy_dev
[joy
], &js
, sizeof(js
));
98 if (dev_stat
== sizeof(js
)) {
101 if ((joyCapData
[joy
].wXpos
!= js
.x
) || (joyCapData
[joy
].wYpos
!= js
.y
)) {
102 SendMessage32A(CaptureWnd
[joy
], MM_JOY1MOVE
+ joy
, js
.buttons
, MAKELONG(js
.x
, js
.y
));
103 joyCapData
[joy
].wXpos
= js
.x
;
104 joyCapData
[joy
].wYpos
= js
.y
;
106 if (joyCapData
[joy
].wButtons
!= js
.buttons
) {
107 unsigned int ButtonChanged
= (WORD
)(joyCapData
[joy
].wButtons
^ js
.buttons
)<<8;
108 if (joyCapData
[joy
].wButtons
< js
.buttons
)
109 SendMessage32A(CaptureWnd
[joy
], MM_JOY1BUTTONDOWN
+ joy
, ButtonChanged
, MAKELONG(js
.x
, js
.y
));
111 if (joyCapData
[joy
].wButtons
> js
.buttons
)
112 SendMessage32A(CaptureWnd
[joy
], MM_JOY1BUTTONUP
113 + joy
, ButtonChanged
, MAKELONG(js
.x
, js
.y
));
114 joyCapData
[joy
].wButtons
= js
.buttons
;
121 /**************************************************************************
122 * JoyGetNumDevs [MMSYSTEM.101]
124 UINT32 WINAPI
joyGetNumDevs32(void)
126 return joyGetNumDevs16();
129 /**************************************************************************
130 * JoyGetNumDevs [MMSYSTEM.101]
132 UINT16 WINAPI
joyGetNumDevs16(void)
137 dprintf_mmsys(stddeb
, "JoyGetNumDevs: ");
138 for (joy
=0; joy
<4; joy
++)
139 if (joyOpenDriver(joy
) == TRUE
) {
143 dprintf_mmsys(stddeb
, "returning %d\n", joy_cnt
);
144 if (!joy_cnt
) fprintf(stderr
, "No joystick found - perhaps get joystick-0.8.0.tar.gz and load it as module or use Linux >= 2.1.45 to be able to use joysticks.\n");
148 /**************************************************************************
149 * JoyGetDevCaps [WINMM.27]
151 MMRESULT32 WINAPI
joyGetDevCaps32A(UINT32 wID
, LPJOYCAPS32A lpCaps
,UINT32 wSize
)
154 MMRESULT16 ret
= joyGetDevCaps16(wID
,&jc16
,sizeof(jc16
));
156 lpCaps
->wMid
= jc16
.wMid
;
157 lpCaps
->wPid
= jc16
.wPid
;
158 lstrcpy32A(lpCaps
->szPname
,jc16
.szPname
);
159 lpCaps
->wXmin
= jc16
.wXmin
;
160 lpCaps
->wXmax
= jc16
.wXmax
;
161 lpCaps
->wYmin
= jc16
.wYmin
;
162 lpCaps
->wYmax
= jc16
.wYmax
;
163 lpCaps
->wZmin
= jc16
.wZmin
;
164 lpCaps
->wZmax
= jc16
.wZmax
;
165 lpCaps
->wNumButtons
= jc16
.wNumButtons
;
166 lpCaps
->wPeriodMin
= jc16
.wPeriodMin
;
167 lpCaps
->wPeriodMax
= jc16
.wPeriodMax
;
169 lpCaps
->wRmin
= jc16
.wRmin
;
170 lpCaps
->wRmax
= jc16
.wRmax
;
171 lpCaps
->wUmin
= jc16
.wUmin
;
172 lpCaps
->wUmax
= jc16
.wUmax
;
173 lpCaps
->wVmin
= jc16
.wVmin
;
174 lpCaps
->wVmax
= jc16
.wVmax
;
175 lpCaps
->wCaps
= jc16
.wCaps
;
176 lpCaps
->wMaxAxes
= jc16
.wMaxAxes
;
177 lpCaps
->wNumAxes
= jc16
.wNumAxes
;
178 lpCaps
->wMaxButtons
= jc16
.wMaxButtons
;
179 lstrcpy32A(lpCaps
->szRegKey
,jc16
.szRegKey
);
180 lstrcpy32A(lpCaps
->szOEMVxD
,jc16
.szOEMVxD
);
184 /**************************************************************************
185 * JoyGetDevCaps [WINMM.28]
187 MMRESULT32 WINAPI
joyGetDevCaps32W(UINT32 wID
, LPJOYCAPS32W lpCaps
,UINT32 wSize
)
190 MMRESULT16 ret
= joyGetDevCaps16(wID
,&jc16
,sizeof(jc16
));
192 lpCaps
->wMid
= jc16
.wMid
;
193 lpCaps
->wPid
= jc16
.wPid
;
194 lstrcpyAtoW(lpCaps
->szPname
,jc16
.szPname
);
195 lpCaps
->wXmin
= jc16
.wXmin
;
196 lpCaps
->wXmax
= jc16
.wXmax
;
197 lpCaps
->wYmin
= jc16
.wYmin
;
198 lpCaps
->wYmax
= jc16
.wYmax
;
199 lpCaps
->wZmin
= jc16
.wZmin
;
200 lpCaps
->wZmax
= jc16
.wZmax
;
201 lpCaps
->wNumButtons
= jc16
.wNumButtons
;
202 lpCaps
->wPeriodMin
= jc16
.wPeriodMin
;
203 lpCaps
->wPeriodMax
= jc16
.wPeriodMax
;
205 lpCaps
->wRmin
= jc16
.wRmin
;
206 lpCaps
->wRmax
= jc16
.wRmax
;
207 lpCaps
->wUmin
= jc16
.wUmin
;
208 lpCaps
->wUmax
= jc16
.wUmax
;
209 lpCaps
->wVmin
= jc16
.wVmin
;
210 lpCaps
->wVmax
= jc16
.wVmax
;
211 lpCaps
->wCaps
= jc16
.wCaps
;
212 lpCaps
->wMaxAxes
= jc16
.wMaxAxes
;
213 lpCaps
->wNumAxes
= jc16
.wNumAxes
;
214 lpCaps
->wMaxButtons
= jc16
.wMaxButtons
;
215 lstrcpyAtoW(lpCaps
->szRegKey
,jc16
.szRegKey
);
216 lstrcpyAtoW(lpCaps
->szOEMVxD
,jc16
.szOEMVxD
);
219 /**************************************************************************
220 * JoyGetDevCaps [MMSYSTEM.102]
222 MMRESULT16 WINAPI
joyGetDevCaps16(UINT16 wID
, LPJOYCAPS16 lpCaps
, UINT16 wSize
)
224 dprintf_mmsys(stderr
, "JoyGetDevCaps(%04X, %p, %d);\n",
226 if (joyOpenDriver(wID
) == TRUE
) {
227 lpCaps
->wMid
= MM_MICROSOFT
;
228 lpCaps
->wPid
= MM_PC_JOYSTICK
;
229 strcpy(lpCaps
->szPname
, "WineJoy"); /* joystick product name */
230 lpCaps
->wXmin
= 0; /* FIXME */
231 lpCaps
->wXmax
= 0xffff;
233 lpCaps
->wYmax
= 0xffff;
235 lpCaps
->wZmax
= 0xffff;
236 lpCaps
->wNumButtons
= 2;
237 lpCaps
->wPeriodMin
= 0;
238 lpCaps
->wPeriodMax
= 50; /* FIXME end */
239 if (wSize
== sizeof(JOYCAPS16
)) {
240 /* complete 95 structure */
242 lpCaps
->wRmax
= 0xffff;
244 lpCaps
->wUmax
= 0xffff;
246 lpCaps
->wVmax
= 0xffff;
248 lpCaps
->wMaxAxes
= 6;
249 lpCaps
->wNumAxes
= 2;
250 lpCaps
->wMaxButtons
= 3;
251 strcpy(lpCaps
->szRegKey
,"");
252 strcpy(lpCaps
->szOEMVxD
,"");
255 return JOYERR_NOERROR
;
258 return MMSYSERR_NODRIVER
;
261 /**************************************************************************
262 * JoyGetPosEx [WINMM.31]
264 MMRESULT32 WINAPI
joyGetPosEx(UINT32 wID
, LPJOYINFO32 lpInfo
)
266 /* FIXME: implement it */
267 return MMSYSERR_NODRIVER
;
270 /**************************************************************************
271 * JoyGetPos [WINMM.30]
273 MMRESULT32 WINAPI
joyGetPos32(UINT32 wID
, LPJOYINFO32 lpInfo
)
276 MMRESULT16 ret
= joyGetPos16(wID
,&ji
);
278 lpInfo
->wXpos
= ji
.wXpos
;
279 lpInfo
->wYpos
= ji
.wYpos
;
280 lpInfo
->wZpos
= ji
.wZpos
;
281 lpInfo
->wButtons
= ji
.wButtons
;
285 /**************************************************************************
286 * JoyGetPos [MMSYSTEM.103]
288 MMRESULT16 WINAPI
joyGetPos16(UINT16 wID
, LPJOYINFO16 lpInfo
)
292 dprintf_mmsys(stderr
, "JoyGetPos(%04X, %p):", wID
, lpInfo
);
293 if (joyOpenDriver(wID
) == FALSE
) return MMSYSERR_NODRIVER
;
294 dev_stat
= read(joy_dev
[wID
], &js
, sizeof(js
));
295 if (dev_stat
!= sizeof(js
)) {
297 return JOYERR_UNPLUGGED
; /* FIXME: perhaps wrong, but what should I return else ? */
302 lpInfo
->wXpos
= js
.x
; /* FIXME: perhaps multiply it somehow ? */
303 lpInfo
->wYpos
= js
.y
;
304 lpInfo
->wZpos
= 0; /* FIXME: Don't know what to do with this value as joystick driver doesn't provide a Z value */
305 lpInfo
->wButtons
= js
.buttons
;
306 dprintf_mmsys(stderr
, "x: %d, y: %d, buttons: %d\n", js
.x
, js
.y
, js
.buttons
);
307 return JOYERR_NOERROR
;
310 /**************************************************************************
311 * JoyGetThreshold [WINMM.32]
313 MMRESULT32 WINAPI
joyGetThreshold32(UINT32 wID
, LPUINT32 lpThreshold
)
316 MMRESULT16 ret
= joyGetThreshold16(wID
,&thresh
);
318 *lpThreshold
= thresh
;
322 /**************************************************************************
323 * JoyGetThreshold [MMSYSTEM.104]
325 MMRESULT16 WINAPI
joyGetThreshold16(UINT16 wID
, LPUINT16 lpThreshold
)
327 dprintf_mmsys(stderr
, "JoyGetThreshold(%04X, %p);\n", wID
, lpThreshold
);
328 if (wID
> 3) return JOYERR_PARMS
;
329 *lpThreshold
= joy_threshold
[wID
];
330 return JOYERR_NOERROR
;
333 /**************************************************************************
334 * JoyReleaseCapture [WINMM.33]
336 MMRESULT32 WINAPI
joyReleaseCapture32(UINT32 wID
)
338 return joyReleaseCapture16(wID
);
341 /**************************************************************************
342 * JoyReleaseCapture [MMSYSTEM.105]
344 MMRESULT16 WINAPI
joyReleaseCapture16(UINT16 wID
)
346 dprintf_mmsys(stderr
, "JoyReleaseCapture(%04X);\n", wID
);
351 return JOYERR_NOERROR
;
354 /**************************************************************************
355 * JoySetCapture [MMSYSTEM.106]
357 MMRESULT32 WINAPI
joySetCapture32(HWND32 hWnd
,UINT32 wID
,UINT32 wPeriod
,BOOL32 bChanged
)
359 return joySetCapture16(hWnd
,wID
,wPeriod
,bChanged
);
362 /**************************************************************************
363 * JoySetCapture [MMSYSTEM.106]
365 MMRESULT16 WINAPI
joySetCapture16(HWND16 hWnd
,UINT16 wID
,UINT16 wPeriod
,BOOL16 bChanged
)
368 dprintf_mmsys(stderr
, "JoySetCapture(%04X, %04X, %d, %d);\n",
369 hWnd
, wID
, wPeriod
, bChanged
);
371 if (!CaptureWnd
[wID
]) {
372 if (joyOpenDriver(wID
) == FALSE
) return MMSYSERR_NODRIVER
;
374 CaptureWnd
[wID
] = hWnd
;
375 return JOYERR_NOERROR
;
378 return JOYERR_NOCANDO
; /* FIXME: what should be returned ? */
381 /**************************************************************************
382 * JoySetThreshold [WINMM.35]
384 MMRESULT32 WINAPI
joySetThreshold32(UINT32 wID
, UINT32 wThreshold
)
386 return joySetThreshold16(wID
,wThreshold
);
388 /**************************************************************************
389 * JoySetThreshold [MMSYSTEM.107]
391 MMRESULT16 WINAPI
joySetThreshold16(UINT16 wID
, UINT16 wThreshold
)
393 dprintf_mmsys(stderr
, "JoySetThreshold(%04X, %d);\n", wID
, wThreshold
);
395 if (wID
> 3) return JOYERR_PARMS
;
396 joy_threshold
[wID
] = wThreshold
;
397 return JOYERR_NOERROR
;
400 /**************************************************************************
401 * JoySetCalibration [MMSYSTEM.109]
403 MMRESULT16 WINAPI
joySetCalibration16(UINT16 wID
)
405 fprintf(stderr
, "EMPTY STUB !!! joySetCalibration(%04X);\n", wID
);
406 return JOYERR_NOCANDO
;