1 /*
2 SDL - Simple DirectMedia Layer
3 Copyright (C) 1997-2012 Sam Lantinga
4
5 This library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 This library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with this library; if not, write to the Free Software
17 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19 Sam Lantinga
20 slouken@libsdl.org
21 */
22 #include "SDL_config.h"
23
24 #ifdef SDL_JOYSTICK_BEOS
25
26 /* This is the system specific header for the SDL joystick API */
27
28 #include <be/support/String.h>
29 #include <be/device/Joystick.h>
30
31 extern "C" {
32
33 #include "SDL_joystick.h"
34 #include "../SDL_sysjoystick.h"
35 #include "../SDL_joystick_c.h"
36
37
38 /* The maximum number of joysticks we'll detect */
39 #define MAX_JOYSTICKS 16
40
41 /* A list of available joysticks */
42 static char *SDL_joyport[MAX_JOYSTICKS];
43 static char *SDL_joyname[MAX_JOYSTICKS];
44
45 /* The private structure used to keep track of a joystick */
46 struct joystick_hwdata {
47 BJoystick *stick;
48 uint8 *new_hats;
49 int16 *new_axes;
50 };
51
52 /* Function to scan the system for joysticks.
53 * This function should set SDL_numjoysticks to the number of available
54 * joysticks. Joystick 0 should be the system default joystick.
55 * It should return 0, or -1 on an unrecoverable fatal error.
56 */
SDL_SYS_JoystickInit(void)57 int SDL_SYS_JoystickInit(void)
58 {
59 BJoystick joystick;
60 int numjoysticks;
61 int i;
62 int32 nports;
63 char name[B_OS_NAME_LENGTH];
64
65 /* Search for attached joysticks */
66 nports = joystick.CountDevices();
67 numjoysticks = 0;
68 SDL_memset(SDL_joyport, 0, (sizeof SDL_joyport));
69 SDL_memset(SDL_joyname, 0, (sizeof SDL_joyname));
70 for ( i=0; (SDL_numjoysticks < MAX_JOYSTICKS) && (i < nports); ++i ) {
71 if ( joystick.GetDeviceName(i, name) == B_OK ) {
72 if ( joystick.Open(name) != B_ERROR ) {
73 BString stick_name;
74 joystick.GetControllerName(&stick_name);
75 SDL_joyport[numjoysticks] = strdup(name);
76 SDL_joyname[numjoysticks] =
77 strdup(stick_name.String());
78 numjoysticks++;
79 joystick.Close();
80 }
81 }
82 }
83 return(numjoysticks);
84 }
85
86 /* Function to get the device-dependent name of a joystick */
SDL_SYS_JoystickName(int index)87 const char *SDL_SYS_JoystickName(int index)
88 {
89 return SDL_joyname[index];
90 }
91
92 /* Function to open a joystick for use.
93 The joystick to open is specified by the index field of the joystick.
94 This should fill the nbuttons and naxes fields of the joystick structure.
95 It returns 0, or -1 if there is an error.
96 */
SDL_SYS_JoystickOpen(SDL_Joystick * joystick)97 int SDL_SYS_JoystickOpen(SDL_Joystick *joystick)
98 {
99 BJoystick *stick;
100
101 /* Create the joystick data structure */
102 joystick->hwdata = (struct joystick_hwdata *)
103 SDL_malloc(sizeof(*joystick->hwdata));
104 if ( joystick->hwdata == NULL ) {
105 SDL_OutOfMemory();
106 return(-1);
107 }
108 SDL_memset(joystick->hwdata, 0, sizeof(*joystick->hwdata));
109 stick = new BJoystick;
110 joystick->hwdata->stick = stick;
111
112 /* Open the requested joystick for use */
113 if ( stick->Open(SDL_joyport[joystick->index]) == B_ERROR ) {
114 SDL_SetError("Unable to open joystick");
115 SDL_SYS_JoystickClose(joystick);
116 return(-1);
117 }
118
119 /* Set the joystick to calibrated mode */
120 stick->EnableCalibration();
121
122 /* Get the number of buttons, hats, and axes on the joystick */
123 joystick->nbuttons = stick->CountButtons();
124 joystick->naxes = stick->CountAxes();
125 joystick->nhats = stick->CountHats();
126
127 joystick->hwdata->new_axes = (int16 *)
128 SDL_malloc(joystick->naxes*sizeof(int16));
129 joystick->hwdata->new_hats = (uint8 *)
130 SDL_malloc(joystick->nhats*sizeof(uint8));
131 if ( ! joystick->hwdata->new_hats || ! joystick->hwdata->new_axes ) {
132 SDL_OutOfMemory();
133 SDL_SYS_JoystickClose(joystick);
134 return(-1);
135 }
136
137 /* We're done! */
138 return(0);
139 }
140
141 /* Function to update the state of a joystick - called as a device poll.
142 * This function shouldn't update the joystick structure directly,
143 * but instead should call SDL_PrivateJoystick*() to deliver events
144 * and update joystick device state.
145 */
SDL_SYS_JoystickUpdate(SDL_Joystick * joystick)146 void SDL_SYS_JoystickUpdate(SDL_Joystick *joystick)
147 {
148 static const Uint8 hat_map[9] = {
149 SDL_HAT_CENTERED,
150 SDL_HAT_UP,
151 SDL_HAT_RIGHTUP,
152 SDL_HAT_RIGHT,
153 SDL_HAT_RIGHTDOWN,
154 SDL_HAT_DOWN,
155 SDL_HAT_LEFTDOWN,
156 SDL_HAT_LEFT,
157 SDL_HAT_LEFTUP
158 };
159 const int JITTER = (32768/10); /* 10% jitter threshold (ok?) */
160
161 BJoystick *stick;
162 int i, change;
163 int16 *axes;
164 uint8 *hats;
165 uint32 buttons;
166
167 /* Set up data pointers */
168 stick = joystick->hwdata->stick;
169 axes = joystick->hwdata->new_axes;
170 hats = joystick->hwdata->new_hats;
171
172 /* Get the new joystick state */
173 stick->Update();
174 stick->GetAxisValues(axes);
175 stick->GetHatValues(hats);
176 buttons = stick->ButtonValues();
177
178 /* Generate axis motion events */
179 for ( i=0; i<joystick->naxes; ++i ) {
180 change = ((int32)axes[i] - joystick->axes[i]);
181 if ( (change > JITTER) || (change < -JITTER) ) {
182 SDL_PrivateJoystickAxis(joystick, i, axes[i]);
183 }
184 }
185
186 /* Generate hat change events */
187 for ( i=0; i<joystick->nhats; ++i ) {
188 if ( hats[i] != joystick->hats[i] ) {
189 SDL_PrivateJoystickHat(joystick, i, hat_map[hats[i]]);
190 }
191 }
192
193 /* Generate button events */
194 for ( i=0; i<joystick->nbuttons; ++i ) {
195 if ( (buttons&0x01) != joystick->buttons[i] ) {
196 SDL_PrivateJoystickButton(joystick, i, (buttons&0x01));
197 }
198 buttons >>= 1;
199 }
200 }
201
202 /* Function to close a joystick after use */
SDL_SYS_JoystickClose(SDL_Joystick * joystick)203 void SDL_SYS_JoystickClose(SDL_Joystick *joystick)
204 {
205 if ( joystick->hwdata ) {
206 joystick->hwdata->stick->Close();
207 delete joystick->hwdata->stick;
208 if ( joystick->hwdata->new_hats ) {
209 SDL_free(joystick->hwdata->new_hats);
210 }
211 if ( joystick->hwdata->new_axes ) {
212 SDL_free(joystick->hwdata->new_axes);
213 }
214 SDL_free(joystick->hwdata);
215 joystick->hwdata = NULL;
216 }
217 }
218
219 /* Function to perform any system-specific joystick related cleanup */
SDL_SYS_JoystickQuit(void)220 void SDL_SYS_JoystickQuit(void)
221 {
222 int i;
223
224 for ( i=0; SDL_joyport[i]; ++i ) {
225 SDL_free(SDL_joyport[i]);
226 }
227 SDL_joyport[0] = NULL;
228
229 for ( i=0; SDL_joyname[i]; ++i ) {
230 SDL_free(SDL_joyname[i]);
231 }
232 SDL_joyname[0] = NULL;
233 }
234
235 }; // extern "C"
236
237 #endif /* SDL_JOYSTICK_BEOS */
238