This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_joystick.c
586 lines (522 loc) · 13.9 KB
1
/*
2
3
Simple DirectMedia Layer
Copyright (C) 1997-2011 Sam Lantinga <slouken@libsdl.org>
4
5
6
7
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.
8
9
10
11
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:
12
13
14
15
16
17
18
19
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.
20
*/
21
#include "SDL_config.h"
22
23
24
/* This is the joystick API for Simple DirectMedia Layer */
25
#include "SDL_events.h"
26
#include "SDL_sysjoystick.h"
27
28
29
30
#include "SDL_joystick_c.h"
#if !SDL_EVENTS_DISABLED
#include "../events/SDL_events_c.h"
#endif
31
32
33
34
35
Uint8 SDL_numjoysticks = 0;
SDL_Joystick **SDL_joysticks = NULL;
static SDL_Joystick *default_joystick = NULL;
36
37
int
SDL_JoystickInit(void)
38
{
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
int arraylen;
int status;
SDL_numjoysticks = 0;
status = SDL_SYS_JoystickInit();
if (status >= 0) {
arraylen = (status + 1) * sizeof(*SDL_joysticks);
SDL_joysticks = (SDL_Joystick **) SDL_malloc(arraylen);
if (SDL_joysticks == NULL) {
SDL_numjoysticks = 0;
} else {
SDL_memset(SDL_joysticks, 0, arraylen);
SDL_numjoysticks = status;
}
status = 0;
}
default_joystick = NULL;
return (status);
57
58
59
60
61
}
/*
* Count the number of joysticks attached to the system
*/
62
63
int
SDL_NumJoysticks(void)
64
{
65
return SDL_numjoysticks;
66
67
68
69
70
}
/*
* Get the implementation dependent name of a joystick
*/
71
72
const char *
SDL_JoystickName(int device_index)
73
{
74
75
76
77
78
if ((device_index < 0) || (device_index >= SDL_numjoysticks)) {
SDL_SetError("There are %d joysticks available", SDL_numjoysticks);
return (NULL);
}
return (SDL_SYS_JoystickName(device_index));
79
80
81
82
83
84
85
86
87
}
/*
* Open a joystick for use - the index passed as an argument refers to
* the N'th joystick on the system. This index is the value which will
* identify this joystick in future joystick events.
*
* This function returns a joystick identifier, or NULL if an error occurred.
*/
88
89
SDL_Joystick *
SDL_JoystickOpen(int device_index)
90
{
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
int i;
SDL_Joystick *joystick;
if ((device_index < 0) || (device_index >= SDL_numjoysticks)) {
SDL_SetError("There are %d joysticks available", SDL_numjoysticks);
return (NULL);
}
/* If the joystick is already open, return it */
for (i = 0; SDL_joysticks[i]; ++i) {
if (device_index == SDL_joysticks[i]->index) {
joystick = SDL_joysticks[i];
++joystick->ref_count;
return (joystick);
}
}
/* Create and initialize the joystick */
joystick = (SDL_Joystick *) SDL_malloc((sizeof *joystick));
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
if (joystick == NULL) {
SDL_OutOfMemory();
return NULL;
}
SDL_memset(joystick, 0, (sizeof *joystick));
joystick->index = device_index;
if (SDL_SYS_JoystickOpen(joystick) < 0) {
SDL_free(joystick);
return NULL;
}
if (joystick->naxes > 0) {
joystick->axes = (Sint16 *) SDL_malloc
(joystick->naxes * sizeof(Sint16));
}
if (joystick->nhats > 0) {
joystick->hats = (Uint8 *) SDL_malloc
(joystick->nhats * sizeof(Uint8));
}
if (joystick->nballs > 0) {
joystick->balls = (struct balldelta *) SDL_malloc
(joystick->nballs * sizeof(*joystick->balls));
132
}
133
134
135
if (joystick->nbuttons > 0) {
joystick->buttons = (Uint8 *) SDL_malloc
(joystick->nbuttons * sizeof(Uint8));
136
}
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
if (((joystick->naxes > 0) && !joystick->axes)
|| ((joystick->nhats > 0) && !joystick->hats)
|| ((joystick->nballs > 0) && !joystick->balls)
|| ((joystick->nbuttons > 0) && !joystick->buttons)) {
SDL_OutOfMemory();
SDL_JoystickClose(joystick);
return NULL;
}
if (joystick->axes) {
SDL_memset(joystick->axes, 0, joystick->naxes * sizeof(Sint16));
}
if (joystick->hats) {
SDL_memset(joystick->hats, 0, joystick->nhats * sizeof(Uint8));
}
if (joystick->balls) {
SDL_memset(joystick->balls, 0,
joystick->nballs * sizeof(*joystick->balls));
}
if (joystick->buttons) {
SDL_memset(joystick->buttons, 0, joystick->nbuttons * sizeof(Uint8));
}
/* Add joystick to list */
++joystick->ref_count;
for (i = 0; SDL_joysticks[i]; ++i)
/* Skip to next joystick */ ;
SDL_joysticks[i] = joystick;
165
return (joystick);
166
167
168
169
170
}
/*
* Returns 1 if the joystick has been opened, or 0 if it has not.
*/
171
172
int
SDL_JoystickOpened(int device_index)
173
{
174
175
176
177
178
179
180
181
182
183
int i, opened;
opened = 0;
for (i = 0; SDL_joysticks[i]; ++i) {
if (SDL_joysticks[i]->index == (Uint8) device_index) {
opened = 1;
break;
}
}
return (opened);
184
185
}
186
187
188
189
190
191
/*
* Checks to make sure the joystick is valid.
*/
int
SDL_PrivateJoystickValid(SDL_Joystick ** joystick)
192
{
193
194
195
196
197
198
199
200
201
202
203
204
int valid;
if (*joystick == NULL) {
*joystick = default_joystick;
}
if (*joystick == NULL) {
SDL_SetError("Joystick hasn't been opened yet");
valid = 0;
} else {
valid = 1;
}
return valid;
205
206
207
208
209
}
/*
* Get the device index of an opened joystick.
*/
210
211
int
SDL_JoystickIndex(SDL_Joystick * joystick)
212
{
213
if (!SDL_PrivateJoystickValid(&joystick)) {
214
215
216
return (-1);
}
return (joystick->index);
217
218
219
220
221
}
/*
* Get the number of multi-dimensional axis controls on a joystick
*/
222
223
int
SDL_JoystickNumAxes(SDL_Joystick * joystick)
224
{
225
if (!SDL_PrivateJoystickValid(&joystick)) {
226
227
228
return (-1);
}
return (joystick->naxes);
229
230
231
232
233
}
/*
* Get the number of hats on a joystick
*/
234
235
int
SDL_JoystickNumHats(SDL_Joystick * joystick)
236
{
237
if (!SDL_PrivateJoystickValid(&joystick)) {
238
239
240
return (-1);
}
return (joystick->nhats);
241
242
243
244
245
}
/*
* Get the number of trackballs on a joystick
*/
246
247
int
SDL_JoystickNumBalls(SDL_Joystick * joystick)
248
{
249
if (!SDL_PrivateJoystickValid(&joystick)) {
250
251
252
return (-1);
}
return (joystick->nballs);
253
254
255
256
257
}
/*
* Get the number of buttons on a joystick
*/
258
259
int
SDL_JoystickNumButtons(SDL_Joystick * joystick)
260
{
261
if (!SDL_PrivateJoystickValid(&joystick)) {
262
263
264
return (-1);
}
return (joystick->nbuttons);
265
266
267
268
269
}
/*
* Get the current state of an axis control on a joystick
*/
270
271
Sint16
SDL_JoystickGetAxis(SDL_Joystick * joystick, int axis)
272
{
273
274
Sint16 state;
275
if (!SDL_PrivateJoystickValid(&joystick)) {
276
277
278
279
280
281
282
283
284
return (0);
}
if (axis < joystick->naxes) {
state = joystick->axes[axis];
} else {
SDL_SetError("Joystick only has %d axes", joystick->naxes);
state = 0;
}
return (state);
285
286
287
288
289
}
/*
* Get the current state of a hat on a joystick
*/
290
291
Uint8
SDL_JoystickGetHat(SDL_Joystick * joystick, int hat)
292
{
293
294
Uint8 state;
295
if (!SDL_PrivateJoystickValid(&joystick)) {
296
297
298
299
300
301
302
303
304
return (0);
}
if (hat < joystick->nhats) {
state = joystick->hats[hat];
} else {
SDL_SetError("Joystick only has %d hats", joystick->nhats);
state = 0;
}
return (state);
305
306
307
308
309
}
/*
* Get the ball axis change since the last poll
*/
310
311
int
SDL_JoystickGetBall(SDL_Joystick * joystick, int ball, int *dx, int *dy)
312
{
313
314
int retval;
315
if (!SDL_PrivateJoystickValid(&joystick)) {
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
return (-1);
}
retval = 0;
if (ball < joystick->nballs) {
if (dx) {
*dx = joystick->balls[ball].dx;
}
if (dy) {
*dy = joystick->balls[ball].dy;
}
joystick->balls[ball].dx = 0;
joystick->balls[ball].dy = 0;
} else {
SDL_SetError("Joystick only has %d balls", joystick->nballs);
retval = -1;
}
return (retval);
334
335
336
337
338
}
/*
* Get the current state of a button on a joystick
*/
339
340
Uint8
SDL_JoystickGetButton(SDL_Joystick * joystick, int button)
341
{
342
343
Uint8 state;
344
if (!SDL_PrivateJoystickValid(&joystick)) {
345
346
347
348
349
350
351
352
353
return (0);
}
if (button < joystick->nbuttons) {
state = joystick->buttons[button];
} else {
SDL_SetError("Joystick only has %d buttons", joystick->nbuttons);
state = 0;
}
return (state);
354
355
356
357
358
}
/*
* Close a joystick previously opened with SDL_JoystickOpen()
*/
359
360
void
SDL_JoystickClose(SDL_Joystick * joystick)
361
{
362
363
int i;
364
if (!SDL_PrivateJoystickValid(&joystick)) {
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
return;
}
/* First decrement ref count */
if (--joystick->ref_count > 0) {
return;
}
if (joystick == default_joystick) {
default_joystick = NULL;
}
SDL_SYS_JoystickClose(joystick);
/* Remove joystick from list */
for (i = 0; SDL_joysticks[i]; ++i) {
if (joystick == SDL_joysticks[i]) {
381
382
SDL_memmove(&SDL_joysticks[i], &SDL_joysticks[i + 1],
(SDL_numjoysticks - i) * sizeof(joystick));
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
break;
}
}
/* Free the data associated with this joystick */
if (joystick->axes) {
SDL_free(joystick->axes);
}
if (joystick->hats) {
SDL_free(joystick->hats);
}
if (joystick->balls) {
SDL_free(joystick->balls);
}
if (joystick->buttons) {
SDL_free(joystick->buttons);
}
SDL_free(joystick);
401
402
}
403
404
void
SDL_JoystickQuit(void)
405
{
406
407
408
409
410
411
412
413
414
/* Stop the event polling */
SDL_numjoysticks = 0;
/* Quit the joystick setup */
SDL_SYS_JoystickQuit();
if (SDL_joysticks) {
SDL_free(SDL_joysticks);
SDL_joysticks = NULL;
}
415
416
417
418
419
}
/* These are global for SDL_sysjoystick.c and SDL_events.c */
420
421
int
SDL_PrivateJoystickAxis(SDL_Joystick * joystick, Uint8 axis, Sint16 value)
422
{
423
int posted;
424
425
426
/* Update internal joystick state */
joystick->axes[axis] = value;
427
428
429
/* Post the event, if desired */
posted = 0;
430
#if !SDL_EVENTS_DISABLED
431
if (SDL_GetEventState(SDL_JOYAXISMOTION) == SDL_ENABLE) {
432
433
434
435
436
437
438
439
440
441
442
SDL_Event event;
event.type = SDL_JOYAXISMOTION;
event.jaxis.which = joystick->index;
event.jaxis.axis = axis;
event.jaxis.value = value;
if ((SDL_EventOK == NULL)
|| (*SDL_EventOK) (SDL_EventOKParam, &event)) {
posted = 1;
SDL_PushEvent(&event);
}
}
443
#endif /* !SDL_EVENTS_DISABLED */
444
return (posted);
445
446
}
447
448
int
SDL_PrivateJoystickHat(SDL_Joystick * joystick, Uint8 hat, Uint8 value)
449
{
450
int posted;
451
452
453
/* Update internal joystick state */
joystick->hats[hat] = value;
454
455
456
/* Post the event, if desired */
posted = 0;
457
#if !SDL_EVENTS_DISABLED
458
if (SDL_GetEventState(SDL_JOYHATMOTION) == SDL_ENABLE) {
459
460
461
462
463
464
465
466
467
468
469
SDL_Event event;
event.jhat.type = SDL_JOYHATMOTION;
event.jhat.which = joystick->index;
event.jhat.hat = hat;
event.jhat.value = value;
if ((SDL_EventOK == NULL)
|| (*SDL_EventOK) (SDL_EventOKParam, &event)) {
posted = 1;
SDL_PushEvent(&event);
}
}
470
#endif /* !SDL_EVENTS_DISABLED */
471
return (posted);
472
473
}
474
475
476
int
SDL_PrivateJoystickBall(SDL_Joystick * joystick, Uint8 ball,
Sint16 xrel, Sint16 yrel)
477
{
478
int posted;
479
480
481
482
/* Update internal mouse state */
joystick->balls[ball].dx += xrel;
joystick->balls[ball].dy += yrel;
483
484
485
/* Post the event, if desired */
posted = 0;
486
#if !SDL_EVENTS_DISABLED
487
if (SDL_GetEventState(SDL_JOYBALLMOTION) == SDL_ENABLE) {
488
489
490
491
492
493
494
495
496
497
498
499
SDL_Event event;
event.jball.type = SDL_JOYBALLMOTION;
event.jball.which = joystick->index;
event.jball.ball = ball;
event.jball.xrel = xrel;
event.jball.yrel = yrel;
if ((SDL_EventOK == NULL)
|| (*SDL_EventOK) (SDL_EventOKParam, &event)) {
posted = 1;
SDL_PushEvent(&event);
}
}
500
#endif /* !SDL_EVENTS_DISABLED */
501
return (posted);
502
503
}
504
505
int
SDL_PrivateJoystickButton(SDL_Joystick * joystick, Uint8 button, Uint8 state)
506
{
507
int posted;
508
#if !SDL_EVENTS_DISABLED
509
510
511
512
513
514
515
516
517
518
519
520
521
SDL_Event event;
switch (state) {
case SDL_PRESSED:
event.type = SDL_JOYBUTTONDOWN;
break;
case SDL_RELEASED:
event.type = SDL_JOYBUTTONUP;
break;
default:
/* Invalid state -- bail */
return (0);
}
522
#endif /* !SDL_EVENTS_DISABLED */
523
524
525
/* Update internal joystick state */
joystick->buttons[button] = state;
526
527
528
/* Post the event, if desired */
posted = 0;
529
#if !SDL_EVENTS_DISABLED
530
if (SDL_GetEventState(event.type) == SDL_ENABLE) {
531
532
533
534
535
536
537
538
539
event.jbutton.which = joystick->index;
event.jbutton.button = button;
event.jbutton.state = state;
if ((SDL_EventOK == NULL)
|| (*SDL_EventOK) (SDL_EventOKParam, &event)) {
posted = 1;
SDL_PushEvent(&event);
}
}
540
#endif /* !SDL_EVENTS_DISABLED */
541
return (posted);
542
543
}
544
545
void
SDL_JoystickUpdate(void)
546
{
547
int i;
548
549
550
551
for (i = 0; SDL_joysticks[i]; ++i) {
SDL_SYS_JoystickUpdate(SDL_joysticks[i]);
}
552
553
}
554
555
int
SDL_JoystickEventState(int state)
556
{
557
#if SDL_EVENTS_DISABLED
558
return SDL_IGNORE;
559
#else
560
const Uint32 event_list[] = {
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
SDL_JOYAXISMOTION, SDL_JOYBALLMOTION, SDL_JOYHATMOTION,
SDL_JOYBUTTONDOWN, SDL_JOYBUTTONUP,
};
unsigned int i;
switch (state) {
case SDL_QUERY:
state = SDL_IGNORE;
for (i = 0; i < SDL_arraysize(event_list); ++i) {
state = SDL_EventState(event_list[i], SDL_QUERY);
if (state == SDL_ENABLE) {
break;
}
}
break;
default:
for (i = 0; i < SDL_arraysize(event_list); ++i) {
SDL_EventState(event_list[i], state);
}
break;
}
return (state);
583
#endif /* SDL_EVENTS_DISABLED */
584
}
585
586
/* vi: set ts=4 sw=4 expandtab: */