SDL_gamecontroller.c 45 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432
  1. /*
  2. Simple DirectMedia Layer
  3. Copyright (C) 1997-2016 Sam Lantinga <[email protected]>
  4. This software is provided 'as-is', without any express or implied
  5. warranty. In no event will the authors be held liable for any damages
  6. arising from the use of this software.
  7. Permission is granted to anyone to use this software for any purpose,
  8. including commercial applications, and to alter it and redistribute it
  9. freely, subject to the following restrictions:
  10. 1. The origin of this software must not be misrepresented; you must not
  11. claim that you wrote the original software. If you use this software
  12. in a product, an acknowledgment in the product documentation would be
  13. appreciated but is not required.
  14. 2. Altered source versions must be plainly marked as such, and must not be
  15. misrepresented as being the original software.
  16. 3. This notice may not be removed or altered from any source distribution.
  17. */
  18. #include "../SDL_internal.h"
  19. /* This is the game controller API for Simple DirectMedia Layer */
  20. #include "SDL_events.h"
  21. #include "SDL_assert.h"
  22. #include "SDL_sysjoystick.h"
  23. #include "SDL_hints.h"
  24. #include "SDL_gamecontrollerdb.h"
  25. #if !SDL_EVENTS_DISABLED
  26. #include "../events/SDL_events_c.h"
  27. #endif
  28. #define ABS(_x) ((_x) < 0 ? -(_x) : (_x))
  29. #define SDL_CONTROLLER_PLATFORM_FIELD "platform:"
  30. /* a list of currently opened game controllers */
  31. static SDL_GameController *SDL_gamecontrollers = NULL;
  32. /* keep track of the hat and mask value that transforms this hat movement into a button/axis press */
  33. struct _SDL_HatMapping
  34. {
  35. int hat;
  36. Uint8 mask;
  37. };
  38. /* We need 36 entries for Android (as of SDL v2.0.4) */
  39. #define k_nMaxReverseEntries 48
  40. /**
  41. * We are encoding the "HAT" as 0xhm. where h == hat ID and m == mask
  42. * MAX 4 hats supported
  43. */
  44. #define k_nMaxHatEntries 0x3f + 1
  45. /* our in memory mapping db between joystick objects and controller mappings */
  46. struct _SDL_ControllerMapping
  47. {
  48. SDL_JoystickGUID guid;
  49. const char *name;
  50. /* mapping of axis/button id to controller version */
  51. int axes[SDL_CONTROLLER_AXIS_MAX];
  52. int buttonasaxis[SDL_CONTROLLER_AXIS_MAX];
  53. int buttons[SDL_CONTROLLER_BUTTON_MAX];
  54. int axesasbutton[SDL_CONTROLLER_BUTTON_MAX];
  55. struct _SDL_HatMapping hatasbutton[SDL_CONTROLLER_BUTTON_MAX];
  56. /* reverse mapping, joystick indices to buttons */
  57. SDL_GameControllerAxis raxes[k_nMaxReverseEntries];
  58. SDL_GameControllerAxis rbuttonasaxis[k_nMaxReverseEntries];
  59. SDL_GameControllerButton rbuttons[k_nMaxReverseEntries];
  60. SDL_GameControllerButton raxesasbutton[k_nMaxReverseEntries];
  61. SDL_GameControllerButton rhatasbutton[k_nMaxHatEntries];
  62. };
  63. /* our hard coded list of mapping support */
  64. typedef enum
  65. {
  66. SDL_CONTROLLER_MAPPING_PRIORITY_DEFAULT,
  67. SDL_CONTROLLER_MAPPING_PRIORITY_API,
  68. SDL_CONTROLLER_MAPPING_PRIORITY_USER,
  69. } SDL_ControllerMappingPriority;
  70. typedef struct _ControllerMapping_t
  71. {
  72. SDL_JoystickGUID guid;
  73. char *name;
  74. char *mapping;
  75. SDL_ControllerMappingPriority priority;
  76. struct _ControllerMapping_t *next;
  77. } ControllerMapping_t;
  78. static ControllerMapping_t *s_pSupportedControllers = NULL;
  79. static ControllerMapping_t *s_pXInputMapping = NULL;
  80. static ControllerMapping_t *s_pEmscriptenMapping = NULL;
  81. /* The SDL game controller structure */
  82. struct _SDL_GameController
  83. {
  84. SDL_Joystick *joystick; /* underlying joystick device */
  85. int ref_count;
  86. Uint8 hatState[4]; /* the current hat state for this controller */
  87. struct _SDL_ControllerMapping mapping; /* the mapping object for this controller */
  88. struct _SDL_GameController *next; /* pointer to next game controller we have allocated */
  89. };
  90. int SDL_PrivateGameControllerAxis(SDL_GameController * gamecontroller, SDL_GameControllerAxis axis, Sint16 value);
  91. int SDL_PrivateGameControllerButton(SDL_GameController * gamecontroller, SDL_GameControllerButton button, Uint8 state);
  92. /*
  93. * If there is an existing add event in the queue, it needs to be modified
  94. * to have the right value for which, because the number of controllers in
  95. * the system is now one less.
  96. */
  97. static void UpdateEventsForDeviceRemoval()
  98. {
  99. int i, num_events;
  100. SDL_Event *events;
  101. num_events = SDL_PeepEvents(NULL, 0, SDL_PEEKEVENT, SDL_CONTROLLERDEVICEADDED, SDL_CONTROLLERDEVICEADDED);
  102. if (num_events <= 0) {
  103. return;
  104. }
  105. events = SDL_stack_alloc(SDL_Event, num_events);
  106. if (!events) {
  107. return;
  108. }
  109. num_events = SDL_PeepEvents(events, num_events, SDL_GETEVENT, SDL_CONTROLLERDEVICEADDED, SDL_CONTROLLERDEVICEADDED);
  110. for (i = 0; i < num_events; ++i) {
  111. --events[i].cdevice.which;
  112. }
  113. SDL_PeepEvents(events, num_events, SDL_ADDEVENT, 0, 0);
  114. SDL_stack_free(events);
  115. }
  116. /*
  117. * Event filter to fire controller events from joystick ones
  118. */
  119. static int SDL_GameControllerEventWatcher(void *userdata, SDL_Event * event)
  120. {
  121. switch(event->type) {
  122. case SDL_JOYAXISMOTION:
  123. {
  124. SDL_GameController *controllerlist;
  125. if (event->jaxis.axis >= k_nMaxReverseEntries)
  126. {
  127. SDL_SetError("SDL_GameControllerEventWatcher: Axis index %d too large, ignoring motion", (int)event->jaxis.axis);
  128. break;
  129. }
  130. controllerlist = SDL_gamecontrollers;
  131. while (controllerlist) {
  132. if (controllerlist->joystick->instance_id == event->jaxis.which) {
  133. if (controllerlist->mapping.raxes[event->jaxis.axis] >= 0) /* simple axis to axis, send it through */ {
  134. SDL_GameControllerAxis axis = controllerlist->mapping.raxes[event->jaxis.axis];
  135. Sint16 value = event->jaxis.value;
  136. switch (axis) {
  137. case SDL_CONTROLLER_AXIS_TRIGGERLEFT:
  138. case SDL_CONTROLLER_AXIS_TRIGGERRIGHT:
  139. value = value / 2 + 16384;
  140. break;
  141. default:
  142. break;
  143. }
  144. SDL_PrivateGameControllerAxis(controllerlist, axis, value);
  145. } else if (controllerlist->mapping.raxesasbutton[event->jaxis.axis] >= 0) { /* simulate an axis as a button */
  146. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.raxesasbutton[event->jaxis.axis], ABS(event->jaxis.value) > 32768/2 ? SDL_PRESSED : SDL_RELEASED);
  147. }
  148. break;
  149. }
  150. controllerlist = controllerlist->next;
  151. }
  152. }
  153. break;
  154. case SDL_JOYBUTTONDOWN:
  155. case SDL_JOYBUTTONUP:
  156. {
  157. SDL_GameController *controllerlist;
  158. if (event->jbutton.button >= k_nMaxReverseEntries)
  159. {
  160. SDL_SetError("SDL_GameControllerEventWatcher: Button index %d too large, ignoring update", (int)event->jbutton.button);
  161. break;
  162. }
  163. controllerlist = SDL_gamecontrollers;
  164. while (controllerlist) {
  165. if (controllerlist->joystick->instance_id == event->jbutton.which) {
  166. if (controllerlist->mapping.rbuttons[event->jbutton.button] >= 0) { /* simple button as button */
  167. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rbuttons[event->jbutton.button], event->jbutton.state);
  168. } else if (controllerlist->mapping.rbuttonasaxis[event->jbutton.button] >= 0) { /* an button pretending to be an axis */
  169. SDL_PrivateGameControllerAxis(controllerlist, controllerlist->mapping.rbuttonasaxis[event->jbutton.button], event->jbutton.state > 0 ? 32767 : 0);
  170. }
  171. break;
  172. }
  173. controllerlist = controllerlist->next;
  174. }
  175. }
  176. break;
  177. case SDL_JOYHATMOTION:
  178. {
  179. SDL_GameController *controllerlist;
  180. if (event->jhat.hat >= 4) break;
  181. controllerlist = SDL_gamecontrollers;
  182. while (controllerlist) {
  183. if (controllerlist->joystick->instance_id == event->jhat.which) {
  184. Uint8 bSame = controllerlist->hatState[event->jhat.hat] & event->jhat.value;
  185. /* Get list of removed bits (button release) */
  186. Uint8 bChanged = controllerlist->hatState[event->jhat.hat] ^ bSame;
  187. /* the hat idx in the high nibble */
  188. int bHighHat = event->jhat.hat << 4;
  189. if (bChanged & SDL_HAT_DOWN)
  190. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_DOWN], SDL_RELEASED);
  191. if (bChanged & SDL_HAT_UP)
  192. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_UP], SDL_RELEASED);
  193. if (bChanged & SDL_HAT_LEFT)
  194. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_LEFT], SDL_RELEASED);
  195. if (bChanged & SDL_HAT_RIGHT)
  196. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_RIGHT], SDL_RELEASED);
  197. /* Get list of added bits (button press) */
  198. bChanged = event->jhat.value ^ bSame;
  199. if (bChanged & SDL_HAT_DOWN)
  200. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_DOWN], SDL_PRESSED);
  201. if (bChanged & SDL_HAT_UP)
  202. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_UP], SDL_PRESSED);
  203. if (bChanged & SDL_HAT_LEFT)
  204. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_LEFT], SDL_PRESSED);
  205. if (bChanged & SDL_HAT_RIGHT)
  206. SDL_PrivateGameControllerButton(controllerlist, controllerlist->mapping.rhatasbutton[bHighHat | SDL_HAT_RIGHT], SDL_PRESSED);
  207. /* update our state cache */
  208. controllerlist->hatState[event->jhat.hat] = event->jhat.value;
  209. break;
  210. }
  211. controllerlist = controllerlist->next;
  212. }
  213. }
  214. break;
  215. case SDL_JOYDEVICEADDED:
  216. {
  217. if (SDL_IsGameController(event->jdevice.which)) {
  218. SDL_Event deviceevent;
  219. deviceevent.type = SDL_CONTROLLERDEVICEADDED;
  220. deviceevent.cdevice.which = event->jdevice.which;
  221. SDL_PushEvent(&deviceevent);
  222. }
  223. }
  224. break;
  225. case SDL_JOYDEVICEREMOVED:
  226. {
  227. SDL_GameController *controllerlist = SDL_gamecontrollers;
  228. while (controllerlist) {
  229. if (controllerlist->joystick->instance_id == event->jdevice.which) {
  230. SDL_Event deviceevent;
  231. deviceevent.type = SDL_CONTROLLERDEVICEREMOVED;
  232. deviceevent.cdevice.which = event->jdevice.which;
  233. SDL_PushEvent(&deviceevent);
  234. UpdateEventsForDeviceRemoval();
  235. break;
  236. }
  237. controllerlist = controllerlist->next;
  238. }
  239. }
  240. break;
  241. default:
  242. break;
  243. }
  244. return 1;
  245. }
  246. /*
  247. * Helper function to scan the mappings database for a controller with the specified GUID
  248. */
  249. static ControllerMapping_t *SDL_PrivateGetControllerMappingForGUID(SDL_JoystickGUID *guid)
  250. {
  251. ControllerMapping_t *pSupportedController = s_pSupportedControllers;
  252. while (pSupportedController) {
  253. if (SDL_memcmp(guid, &pSupportedController->guid, sizeof(*guid)) == 0) {
  254. return pSupportedController;
  255. }
  256. pSupportedController = pSupportedController->next;
  257. }
  258. return NULL;
  259. }
  260. static const char* map_StringForControllerAxis[] = {
  261. "leftx",
  262. "lefty",
  263. "rightx",
  264. "righty",
  265. "lefttrigger",
  266. "righttrigger",
  267. NULL
  268. };
  269. /*
  270. * convert a string to its enum equivalent
  271. */
  272. SDL_GameControllerAxis SDL_GameControllerGetAxisFromString(const char *pchString)
  273. {
  274. int entry;
  275. if (!pchString || !pchString[0])
  276. return SDL_CONTROLLER_AXIS_INVALID;
  277. for (entry = 0; map_StringForControllerAxis[entry]; ++entry) {
  278. if (!SDL_strcasecmp(pchString, map_StringForControllerAxis[entry]))
  279. return entry;
  280. }
  281. return SDL_CONTROLLER_AXIS_INVALID;
  282. }
  283. /*
  284. * convert an enum to its string equivalent
  285. */
  286. const char* SDL_GameControllerGetStringForAxis(SDL_GameControllerAxis axis)
  287. {
  288. if (axis > SDL_CONTROLLER_AXIS_INVALID && axis < SDL_CONTROLLER_AXIS_MAX) {
  289. return map_StringForControllerAxis[axis];
  290. }
  291. return NULL;
  292. }
  293. static const char* map_StringForControllerButton[] = {
  294. "a",
  295. "b",
  296. "x",
  297. "y",
  298. "back",
  299. "guide",
  300. "start",
  301. "leftstick",
  302. "rightstick",
  303. "leftshoulder",
  304. "rightshoulder",
  305. "dpup",
  306. "dpdown",
  307. "dpleft",
  308. "dpright",
  309. NULL
  310. };
  311. /*
  312. * convert a string to its enum equivalent
  313. */
  314. SDL_GameControllerButton SDL_GameControllerGetButtonFromString(const char *pchString)
  315. {
  316. int entry;
  317. if (!pchString || !pchString[0])
  318. return SDL_CONTROLLER_BUTTON_INVALID;
  319. for (entry = 0; map_StringForControllerButton[entry]; ++entry) {
  320. if (SDL_strcasecmp(pchString, map_StringForControllerButton[entry]) == 0)
  321. return entry;
  322. }
  323. return SDL_CONTROLLER_BUTTON_INVALID;
  324. }
  325. /*
  326. * convert an enum to its string equivalent
  327. */
  328. const char* SDL_GameControllerGetStringForButton(SDL_GameControllerButton axis)
  329. {
  330. if (axis > SDL_CONTROLLER_BUTTON_INVALID && axis < SDL_CONTROLLER_BUTTON_MAX) {
  331. return map_StringForControllerButton[axis];
  332. }
  333. return NULL;
  334. }
  335. /*
  336. * given a controller button name and a joystick name update our mapping structure with it
  337. */
  338. static void SDL_PrivateGameControllerParseButton(const char *szGameButton, const char *szJoystickButton, struct _SDL_ControllerMapping *pMapping)
  339. {
  340. int iSDLButton = 0;
  341. SDL_GameControllerButton button;
  342. SDL_GameControllerAxis axis;
  343. button = SDL_GameControllerGetButtonFromString(szGameButton);
  344. axis = SDL_GameControllerGetAxisFromString(szGameButton);
  345. iSDLButton = SDL_atoi(&szJoystickButton[1]);
  346. if (szJoystickButton[0] == 'a') {
  347. if (iSDLButton >= k_nMaxReverseEntries) {
  348. SDL_SetError("Axis index too large: %d", iSDLButton);
  349. return;
  350. }
  351. if (axis != SDL_CONTROLLER_AXIS_INVALID) {
  352. pMapping->axes[axis] = iSDLButton;
  353. pMapping->raxes[iSDLButton] = axis;
  354. } else if (button != SDL_CONTROLLER_BUTTON_INVALID) {
  355. pMapping->axesasbutton[button] = iSDLButton;
  356. pMapping->raxesasbutton[iSDLButton] = button;
  357. } else {
  358. SDL_assert(!"How did we get here?");
  359. }
  360. } else if (szJoystickButton[0] == 'b') {
  361. if (iSDLButton >= k_nMaxReverseEntries) {
  362. SDL_SetError("Button index too large: %d", iSDLButton);
  363. return;
  364. }
  365. if (button != SDL_CONTROLLER_BUTTON_INVALID) {
  366. pMapping->buttons[button] = iSDLButton;
  367. pMapping->rbuttons[iSDLButton] = button;
  368. } else if (axis != SDL_CONTROLLER_AXIS_INVALID) {
  369. pMapping->buttonasaxis[axis] = iSDLButton;
  370. pMapping->rbuttonasaxis[iSDLButton] = axis;
  371. } else {
  372. SDL_assert(!"How did we get here?");
  373. }
  374. } else if (szJoystickButton[0] == 'h') {
  375. int hat = SDL_atoi(&szJoystickButton[1]);
  376. int mask = SDL_atoi(&szJoystickButton[3]);
  377. if (hat >= 4) {
  378. SDL_SetError("Hat index too large: %d", iSDLButton);
  379. }
  380. if (button != SDL_CONTROLLER_BUTTON_INVALID) {
  381. int ridx;
  382. pMapping->hatasbutton[button].hat = hat;
  383. pMapping->hatasbutton[button].mask = mask;
  384. ridx = (hat << 4) | mask;
  385. pMapping->rhatasbutton[ridx] = button;
  386. } else if (axis != SDL_CONTROLLER_AXIS_INVALID) {
  387. SDL_assert(!"Support hat as axis");
  388. } else {
  389. SDL_assert(!"How did we get here?");
  390. }
  391. }
  392. }
  393. /*
  394. * given a controller mapping string update our mapping object
  395. */
  396. static void
  397. SDL_PrivateGameControllerParseControllerConfigString(struct _SDL_ControllerMapping *pMapping, const char *pchString)
  398. {
  399. char szGameButton[20];
  400. char szJoystickButton[20];
  401. SDL_bool bGameButton = SDL_TRUE;
  402. int i = 0;
  403. const char *pchPos = pchString;
  404. SDL_memset(szGameButton, 0x0, sizeof(szGameButton));
  405. SDL_memset(szJoystickButton, 0x0, sizeof(szJoystickButton));
  406. while (pchPos && *pchPos) {
  407. if (*pchPos == ':') {
  408. i = 0;
  409. bGameButton = SDL_FALSE;
  410. } else if (*pchPos == ' ') {
  411. } else if (*pchPos == ',') {
  412. i = 0;
  413. bGameButton = SDL_TRUE;
  414. SDL_PrivateGameControllerParseButton(szGameButton, szJoystickButton, pMapping);
  415. SDL_memset(szGameButton, 0x0, sizeof(szGameButton));
  416. SDL_memset(szJoystickButton, 0x0, sizeof(szJoystickButton));
  417. } else if (bGameButton) {
  418. if (i >= sizeof(szGameButton)) {
  419. SDL_SetError("Button name too large: %s", szGameButton);
  420. return;
  421. }
  422. szGameButton[i] = *pchPos;
  423. i++;
  424. } else {
  425. if (i >= sizeof(szJoystickButton)) {
  426. SDL_SetError("Joystick button name too large: %s", szJoystickButton);
  427. return;
  428. }
  429. szJoystickButton[i] = *pchPos;
  430. i++;
  431. }
  432. pchPos++;
  433. }
  434. SDL_PrivateGameControllerParseButton(szGameButton, szJoystickButton, pMapping);
  435. }
  436. /*
  437. * Make a new button mapping struct
  438. */
  439. static void SDL_PrivateLoadButtonMapping(struct _SDL_ControllerMapping *pMapping, SDL_JoystickGUID guid, const char *pchName, const char *pchMapping)
  440. {
  441. int j;
  442. pMapping->guid = guid;
  443. pMapping->name = pchName;
  444. /* set all the button mappings to non defaults */
  445. for (j = 0; j < SDL_CONTROLLER_AXIS_MAX; j++) {
  446. pMapping->axes[j] = -1;
  447. pMapping->buttonasaxis[j] = -1;
  448. }
  449. for (j = 0; j < SDL_CONTROLLER_BUTTON_MAX; j++) {
  450. pMapping->buttons[j] = -1;
  451. pMapping->axesasbutton[j] = -1;
  452. pMapping->hatasbutton[j].hat = -1;
  453. }
  454. for (j = 0; j < k_nMaxReverseEntries; j++) {
  455. pMapping->raxes[j] = SDL_CONTROLLER_AXIS_INVALID;
  456. pMapping->rbuttonasaxis[j] = SDL_CONTROLLER_AXIS_INVALID;
  457. pMapping->rbuttons[j] = SDL_CONTROLLER_BUTTON_INVALID;
  458. pMapping->raxesasbutton[j] = SDL_CONTROLLER_BUTTON_INVALID;
  459. }
  460. for (j = 0; j < k_nMaxHatEntries; j++) {
  461. pMapping->rhatasbutton[j] = SDL_CONTROLLER_BUTTON_INVALID;
  462. }
  463. SDL_PrivateGameControllerParseControllerConfigString(pMapping, pchMapping);
  464. }
  465. /*
  466. * grab the guid string from a mapping string
  467. */
  468. static char *SDL_PrivateGetControllerGUIDFromMappingString(const char *pMapping)
  469. {
  470. const char *pFirstComma = SDL_strchr(pMapping, ',');
  471. if (pFirstComma) {
  472. char *pchGUID = SDL_malloc(pFirstComma - pMapping + 1);
  473. if (!pchGUID) {
  474. SDL_OutOfMemory();
  475. return NULL;
  476. }
  477. SDL_memcpy(pchGUID, pMapping, pFirstComma - pMapping);
  478. pchGUID[pFirstComma - pMapping] = '\0';
  479. /* Convert old style GUIDs to the new style in 2.0.5 */
  480. #if __WIN32__
  481. if (SDL_strlen(pchGUID) == 32 &&
  482. SDL_memcmp(&pchGUID[20], "504944564944", 12) == 0) {
  483. SDL_memcpy(&pchGUID[20], "000000000000", 12);
  484. SDL_memcpy(&pchGUID[16], &pchGUID[4], 4);
  485. SDL_memcpy(&pchGUID[8], &pchGUID[0], 4);
  486. SDL_memcpy(&pchGUID[0], "03000000", 8);
  487. }
  488. #elif __MACOSX__
  489. if (SDL_strlen(pchGUID) == 32 &&
  490. SDL_memcmp(&pchGUID[4], "000000000000", 12) == 0 &&
  491. SDL_memcmp(&pchGUID[20], "000000000000", 12) == 0) {
  492. SDL_memcpy(&pchGUID[20], "000000000000", 12);
  493. SDL_memcpy(&pchGUID[8], &pchGUID[0], 4);
  494. SDL_memcpy(&pchGUID[0], "03000000", 8);
  495. }
  496. #endif
  497. return pchGUID;
  498. }
  499. return NULL;
  500. }
  501. /*
  502. * grab the name string from a mapping string
  503. */
  504. static char *SDL_PrivateGetControllerNameFromMappingString(const char *pMapping)
  505. {
  506. const char *pFirstComma, *pSecondComma;
  507. char *pchName;
  508. pFirstComma = SDL_strchr(pMapping, ',');
  509. if (!pFirstComma)
  510. return NULL;
  511. pSecondComma = SDL_strchr(pFirstComma + 1, ',');
  512. if (!pSecondComma)
  513. return NULL;
  514. pchName = SDL_malloc(pSecondComma - pFirstComma);
  515. if (!pchName) {
  516. SDL_OutOfMemory();
  517. return NULL;
  518. }
  519. SDL_memcpy(pchName, pFirstComma + 1, pSecondComma - pFirstComma);
  520. pchName[pSecondComma - pFirstComma - 1] = 0;
  521. return pchName;
  522. }
  523. /*
  524. * grab the button mapping string from a mapping string
  525. */
  526. static char *SDL_PrivateGetControllerMappingFromMappingString(const char *pMapping)
  527. {
  528. const char *pFirstComma, *pSecondComma;
  529. pFirstComma = SDL_strchr(pMapping, ',');
  530. if (!pFirstComma)
  531. return NULL;
  532. pSecondComma = SDL_strchr(pFirstComma + 1, ',');
  533. if (!pSecondComma)
  534. return NULL;
  535. return SDL_strdup(pSecondComma + 1); /* mapping is everything after the 3rd comma */
  536. }
  537. /*
  538. * Helper function to refresh a mapping
  539. */
  540. static void SDL_PrivateGameControllerRefreshMapping(ControllerMapping_t *pControllerMapping)
  541. {
  542. SDL_GameController *gamecontrollerlist = SDL_gamecontrollers;
  543. while (gamecontrollerlist) {
  544. if (!SDL_memcmp(&gamecontrollerlist->mapping.guid, &pControllerMapping->guid, sizeof(pControllerMapping->guid))) {
  545. SDL_Event event;
  546. event.type = SDL_CONTROLLERDEVICEREMAPPED;
  547. event.cdevice.which = gamecontrollerlist->joystick->instance_id;
  548. SDL_PushEvent(&event);
  549. /* Not really threadsafe. Should this lock access within SDL_GameControllerEventWatcher? */
  550. SDL_PrivateLoadButtonMapping(&gamecontrollerlist->mapping, pControllerMapping->guid, pControllerMapping->name, pControllerMapping->mapping);
  551. }
  552. gamecontrollerlist = gamecontrollerlist->next;
  553. }
  554. }
  555. /*
  556. * Helper function to add a mapping for a guid
  557. */
  558. static ControllerMapping_t *
  559. SDL_PrivateAddMappingForGUID(SDL_JoystickGUID jGUID, const char *mappingString, SDL_bool *existing, SDL_ControllerMappingPriority priority)
  560. {
  561. char *pchName;
  562. char *pchMapping;
  563. ControllerMapping_t *pControllerMapping;
  564. pchName = SDL_PrivateGetControllerNameFromMappingString(mappingString);
  565. if (!pchName) {
  566. SDL_SetError("Couldn't parse name from %s", mappingString);
  567. return NULL;
  568. }
  569. pchMapping = SDL_PrivateGetControllerMappingFromMappingString(mappingString);
  570. if (!pchMapping) {
  571. SDL_free(pchName);
  572. SDL_SetError("Couldn't parse %s", mappingString);
  573. return NULL;
  574. }
  575. pControllerMapping = SDL_PrivateGetControllerMappingForGUID(&jGUID);
  576. if (pControllerMapping) {
  577. /* Only overwrite the mapping if the priority is the same or higher. */
  578. if (pControllerMapping->priority <= priority) {
  579. /* Update existing mapping */
  580. SDL_free(pControllerMapping->name);
  581. pControllerMapping->name = pchName;
  582. SDL_free(pControllerMapping->mapping);
  583. pControllerMapping->mapping = pchMapping;
  584. pControllerMapping->priority = priority;
  585. /* refresh open controllers */
  586. SDL_PrivateGameControllerRefreshMapping(pControllerMapping);
  587. } else {
  588. SDL_free(pchName);
  589. SDL_free(pchMapping);
  590. }
  591. *existing = SDL_TRUE;
  592. } else {
  593. pControllerMapping = SDL_malloc(sizeof(*pControllerMapping));
  594. if (!pControllerMapping) {
  595. SDL_free(pchName);
  596. SDL_free(pchMapping);
  597. SDL_OutOfMemory();
  598. return NULL;
  599. }
  600. pControllerMapping->guid = jGUID;
  601. pControllerMapping->name = pchName;
  602. pControllerMapping->mapping = pchMapping;
  603. pControllerMapping->next = s_pSupportedControllers;
  604. pControllerMapping->priority = priority;
  605. s_pSupportedControllers = pControllerMapping;
  606. *existing = SDL_FALSE;
  607. }
  608. return pControllerMapping;
  609. }
  610. /*
  611. * Helper function to determine pre-calculated offset to certain joystick mappings
  612. */
  613. static ControllerMapping_t *SDL_PrivateGetControllerMapping(int device_index)
  614. {
  615. SDL_JoystickGUID jGUID = SDL_JoystickGetDeviceGUID(device_index);
  616. ControllerMapping_t *mapping;
  617. mapping = SDL_PrivateGetControllerMappingForGUID(&jGUID);
  618. #if SDL_JOYSTICK_XINPUT
  619. if (!mapping && SDL_SYS_IsXInputGamepad_DeviceIndex(device_index)) {
  620. mapping = s_pXInputMapping;
  621. }
  622. #endif
  623. #if defined(SDL_JOYSTICK_EMSCRIPTEN)
  624. if (!mapping && s_pEmscriptenMapping) {
  625. mapping = s_pEmscriptenMapping;
  626. }
  627. #endif
  628. #ifdef __LINUX__
  629. if (!mapping) {
  630. const char *name = SDL_JoystickNameForIndex(device_index);
  631. if (name) {
  632. if (SDL_strstr(name, "Xbox 360 Wireless Receiver")) {
  633. /* The Linux driver xpad.c maps the wireless dpad to buttons */
  634. SDL_bool existing;
  635. mapping = SDL_PrivateAddMappingForGUID(jGUID,
  636. "none,X360 Wireless Controller,a:b0,b:b1,back:b6,dpdown:b14,dpleft:b11,dpright:b12,dpup:b13,guide:b8,leftshoulder:b4,leftstick:b9,lefttrigger:a2,leftx:a0,lefty:a1,rightshoulder:b5,rightstick:b10,righttrigger:a5,rightx:a3,righty:a4,start:b7,x:b2,y:b3,",
  637. &existing, SDL_CONTROLLER_MAPPING_PRIORITY_DEFAULT);
  638. }
  639. }
  640. }
  641. #endif /* __LINUX__ */
  642. if (!mapping) {
  643. const char *name = SDL_JoystickNameForIndex(device_index);
  644. if (name) {
  645. if (SDL_strstr(name, "Xbox") || SDL_strstr(name, "X-Box")) {
  646. mapping = s_pXInputMapping;
  647. }
  648. }
  649. }
  650. return mapping;
  651. }
  652. /*
  653. * Add or update an entry into the Mappings Database
  654. */
  655. int
  656. SDL_GameControllerAddMappingsFromRW(SDL_RWops * rw, int freerw)
  657. {
  658. const char *platform = SDL_GetPlatform();
  659. int controllers = 0;
  660. char *buf, *line, *line_end, *tmp, *comma, line_platform[64];
  661. size_t db_size, platform_len;
  662. if (rw == NULL) {
  663. return SDL_SetError("Invalid RWops");
  664. }
  665. db_size = (size_t)SDL_RWsize(rw);
  666. buf = (char *)SDL_malloc(db_size + 1);
  667. if (buf == NULL) {
  668. if (freerw) {
  669. SDL_RWclose(rw);
  670. }
  671. return SDL_SetError("Could not allocate space to read DB into memory");
  672. }
  673. if (SDL_RWread(rw, buf, db_size, 1) != 1) {
  674. if (freerw) {
  675. SDL_RWclose(rw);
  676. }
  677. SDL_free(buf);
  678. return SDL_SetError("Could not read DB");
  679. }
  680. if (freerw) {
  681. SDL_RWclose(rw);
  682. }
  683. buf[db_size] = '\0';
  684. line = buf;
  685. while (line < buf + db_size) {
  686. line_end = SDL_strchr(line, '\n');
  687. if (line_end != NULL) {
  688. *line_end = '\0';
  689. } else {
  690. line_end = buf + db_size;
  691. }
  692. /* Extract and verify the platform */
  693. tmp = SDL_strstr(line, SDL_CONTROLLER_PLATFORM_FIELD);
  694. if (tmp != NULL) {
  695. tmp += SDL_strlen(SDL_CONTROLLER_PLATFORM_FIELD);
  696. comma = SDL_strchr(tmp, ',');
  697. if (comma != NULL) {
  698. platform_len = comma - tmp + 1;
  699. if (platform_len + 1 < SDL_arraysize(line_platform)) {
  700. SDL_strlcpy(line_platform, tmp, platform_len);
  701. if (SDL_strncasecmp(line_platform, platform, platform_len) == 0 &&
  702. SDL_GameControllerAddMapping(line) > 0) {
  703. controllers++;
  704. }
  705. }
  706. }
  707. }
  708. line = line_end + 1;
  709. }
  710. SDL_free(buf);
  711. return controllers;
  712. }
  713. /*
  714. * Add or update an entry into the Mappings Database with a priority
  715. */
  716. static int
  717. SDL_PrivateGameControllerAddMapping(const char *mappingString, SDL_ControllerMappingPriority priority)
  718. {
  719. char *pchGUID;
  720. SDL_JoystickGUID jGUID;
  721. SDL_bool is_xinput_mapping = SDL_FALSE;
  722. SDL_bool is_emscripten_mapping = SDL_FALSE;
  723. SDL_bool existing = SDL_FALSE;
  724. ControllerMapping_t *pControllerMapping;
  725. if (!mappingString) {
  726. return SDL_InvalidParamError("mappingString");
  727. }
  728. pchGUID = SDL_PrivateGetControllerGUIDFromMappingString(mappingString);
  729. if (!pchGUID) {
  730. return SDL_SetError("Couldn't parse GUID from %s", mappingString);
  731. }
  732. if (!SDL_strcasecmp(pchGUID, "xinput")) {
  733. is_xinput_mapping = SDL_TRUE;
  734. }
  735. if (!SDL_strcasecmp(pchGUID, "emscripten")) {
  736. is_emscripten_mapping = SDL_TRUE;
  737. }
  738. jGUID = SDL_JoystickGetGUIDFromString(pchGUID);
  739. SDL_free(pchGUID);
  740. pControllerMapping = SDL_PrivateAddMappingForGUID(jGUID, mappingString, &existing, priority);
  741. if (!pControllerMapping) {
  742. return -1;
  743. }
  744. if (existing) {
  745. return 0;
  746. } else {
  747. if (is_xinput_mapping) {
  748. s_pXInputMapping = pControllerMapping;
  749. }
  750. if (is_emscripten_mapping) {
  751. s_pEmscriptenMapping = pControllerMapping;
  752. }
  753. return 1;
  754. }
  755. }
  756. /*
  757. * Add or update an entry into the Mappings Database
  758. */
  759. int
  760. SDL_GameControllerAddMapping(const char *mappingString)
  761. {
  762. return SDL_PrivateGameControllerAddMapping(mappingString, SDL_CONTROLLER_MAPPING_PRIORITY_API);
  763. }
  764. /*
  765. * Get the mapping string for this GUID
  766. */
  767. char *
  768. SDL_GameControllerMappingForGUID(SDL_JoystickGUID guid)
  769. {
  770. char *pMappingString = NULL;
  771. ControllerMapping_t *mapping = SDL_PrivateGetControllerMappingForGUID(&guid);
  772. if (mapping) {
  773. char pchGUID[33];
  774. size_t needed;
  775. SDL_JoystickGetGUIDString(guid, pchGUID, sizeof(pchGUID));
  776. /* allocate enough memory for GUID + ',' + name + ',' + mapping + \0 */
  777. needed = SDL_strlen(pchGUID) + 1 + SDL_strlen(mapping->name) + 1 + SDL_strlen(mapping->mapping) + 1;
  778. pMappingString = SDL_malloc(needed);
  779. if (!pMappingString) {
  780. SDL_OutOfMemory();
  781. return NULL;
  782. }
  783. SDL_snprintf(pMappingString, needed, "%s,%s,%s", pchGUID, mapping->name, mapping->mapping);
  784. }
  785. return pMappingString;
  786. }
  787. /*
  788. * Get the mapping string for this device
  789. */
  790. char *
  791. SDL_GameControllerMapping(SDL_GameController * gamecontroller)
  792. {
  793. if (!gamecontroller) {
  794. return NULL;
  795. }
  796. return SDL_GameControllerMappingForGUID(gamecontroller->mapping.guid);
  797. }
  798. static void
  799. SDL_GameControllerLoadHints()
  800. {
  801. const char *hint = SDL_GetHint(SDL_HINT_GAMECONTROLLERCONFIG);
  802. if (hint && hint[0]) {
  803. size_t nchHints = SDL_strlen(hint);
  804. char *pUserMappings = SDL_malloc(nchHints + 1);
  805. char *pTempMappings = pUserMappings;
  806. SDL_memcpy(pUserMappings, hint, nchHints);
  807. pUserMappings[nchHints] = '\0';
  808. while (pUserMappings) {
  809. char *pchNewLine = NULL;
  810. pchNewLine = SDL_strchr(pUserMappings, '\n');
  811. if (pchNewLine)
  812. *pchNewLine = '\0';
  813. SDL_PrivateGameControllerAddMapping(pUserMappings, SDL_CONTROLLER_MAPPING_PRIORITY_USER);
  814. if (pchNewLine) {
  815. pUserMappings = pchNewLine + 1;
  816. } else {
  817. pUserMappings = NULL;
  818. }
  819. }
  820. SDL_free(pTempMappings);
  821. }
  822. }
  823. /*
  824. * Initialize the game controller system, mostly load our DB of controller config mappings
  825. */
  826. int
  827. SDL_GameControllerInit(void)
  828. {
  829. int i = 0;
  830. const char *pMappingString = NULL;
  831. pMappingString = s_ControllerMappings[i];
  832. while (pMappingString) {
  833. SDL_PrivateGameControllerAddMapping(pMappingString, SDL_CONTROLLER_MAPPING_PRIORITY_DEFAULT);
  834. i++;
  835. pMappingString = s_ControllerMappings[i];
  836. }
  837. /* load in any user supplied config */
  838. SDL_GameControllerLoadHints();
  839. /* watch for joy events and fire controller ones if needed */
  840. SDL_AddEventWatch(SDL_GameControllerEventWatcher, NULL);
  841. /* Send added events for controllers currently attached */
  842. for (i = 0; i < SDL_NumJoysticks(); ++i) {
  843. if (SDL_IsGameController(i)) {
  844. SDL_Event deviceevent;
  845. deviceevent.type = SDL_CONTROLLERDEVICEADDED;
  846. deviceevent.cdevice.which = i;
  847. SDL_PushEvent(&deviceevent);
  848. }
  849. }
  850. return (0);
  851. }
  852. /*
  853. * Get the implementation dependent name of a controller
  854. */
  855. const char *
  856. SDL_GameControllerNameForIndex(int device_index)
  857. {
  858. ControllerMapping_t *pSupportedController = SDL_PrivateGetControllerMapping(device_index);
  859. if (pSupportedController) {
  860. return pSupportedController->name;
  861. }
  862. return NULL;
  863. }
  864. /*
  865. * Return 1 if the joystick at this device index is a supported controller
  866. */
  867. SDL_bool
  868. SDL_IsGameController(int device_index)
  869. {
  870. ControllerMapping_t *pSupportedController = SDL_PrivateGetControllerMapping(device_index);
  871. if (pSupportedController) {
  872. return SDL_TRUE;
  873. }
  874. return SDL_FALSE;
  875. }
  876. /*
  877. * Open a controller for use - the index passed as an argument refers to
  878. * the N'th controller on the system. This index is the value which will
  879. * identify this controller in future controller events.
  880. *
  881. * This function returns a controller identifier, or NULL if an error occurred.
  882. */
  883. SDL_GameController *
  884. SDL_GameControllerOpen(int device_index)
  885. {
  886. SDL_GameController *gamecontroller;
  887. SDL_GameController *gamecontrollerlist;
  888. ControllerMapping_t *pSupportedController = NULL;
  889. if ((device_index < 0) || (device_index >= SDL_NumJoysticks())) {
  890. SDL_SetError("There are %d joysticks available", SDL_NumJoysticks());
  891. return (NULL);
  892. }
  893. gamecontrollerlist = SDL_gamecontrollers;
  894. /* If the controller is already open, return it */
  895. while (gamecontrollerlist) {
  896. if (SDL_SYS_GetInstanceIdOfDeviceIndex(device_index) == gamecontrollerlist->joystick->instance_id) {
  897. gamecontroller = gamecontrollerlist;
  898. ++gamecontroller->ref_count;
  899. return (gamecontroller);
  900. }
  901. gamecontrollerlist = gamecontrollerlist->next;
  902. }
  903. /* Find a controller mapping */
  904. pSupportedController = SDL_PrivateGetControllerMapping(device_index);
  905. if (!pSupportedController) {
  906. SDL_SetError("Couldn't find mapping for device (%d)", device_index);
  907. return (NULL);
  908. }
  909. /* Create and initialize the joystick */
  910. gamecontroller = (SDL_GameController *) SDL_malloc((sizeof *gamecontroller));
  911. if (gamecontroller == NULL) {
  912. SDL_OutOfMemory();
  913. return NULL;
  914. }
  915. SDL_memset(gamecontroller, 0, (sizeof *gamecontroller));
  916. gamecontroller->joystick = SDL_JoystickOpen(device_index);
  917. if (!gamecontroller->joystick) {
  918. SDL_free(gamecontroller);
  919. return NULL;
  920. }
  921. SDL_PrivateLoadButtonMapping(&gamecontroller->mapping, pSupportedController->guid, pSupportedController->name, pSupportedController->mapping);
  922. /* The triggers are mapped from -32768 to 32767, where -32768 is the 'unpressed' value */
  923. {
  924. int leftTriggerMapping = gamecontroller->mapping.axes[SDL_CONTROLLER_AXIS_TRIGGERLEFT];
  925. int rightTriggerMapping = gamecontroller->mapping.axes[SDL_CONTROLLER_AXIS_TRIGGERRIGHT];
  926. if (leftTriggerMapping >= 0) {
  927. gamecontroller->joystick->axes[leftTriggerMapping] =
  928. gamecontroller->joystick->axes_zero[leftTriggerMapping] = (Sint16)-32768;
  929. }
  930. if (rightTriggerMapping >= 0) {
  931. gamecontroller->joystick->axes[rightTriggerMapping] =
  932. gamecontroller->joystick->axes_zero[rightTriggerMapping] = (Sint16)-32768;
  933. }
  934. }
  935. /* Add joystick to list */
  936. ++gamecontroller->ref_count;
  937. /* Link the joystick in the list */
  938. gamecontroller->next = SDL_gamecontrollers;
  939. SDL_gamecontrollers = gamecontroller;
  940. SDL_SYS_JoystickUpdate(gamecontroller->joystick);
  941. return (gamecontroller);
  942. }
  943. /*
  944. * Manually pump for controller updates.
  945. */
  946. void
  947. SDL_GameControllerUpdate(void)
  948. {
  949. /* Just for API completeness; the joystick API does all the work. */
  950. SDL_JoystickUpdate();
  951. }
  952. /*
  953. * Get the current state of an axis control on a controller
  954. */
  955. Sint16
  956. SDL_GameControllerGetAxis(SDL_GameController * gamecontroller, SDL_GameControllerAxis axis)
  957. {
  958. if (!gamecontroller)
  959. return 0;
  960. if (gamecontroller->mapping.axes[axis] >= 0) {
  961. Sint16 value = (SDL_JoystickGetAxis(gamecontroller->joystick, gamecontroller->mapping.axes[axis]));
  962. switch (axis) {
  963. case SDL_CONTROLLER_AXIS_TRIGGERLEFT:
  964. case SDL_CONTROLLER_AXIS_TRIGGERRIGHT:
  965. /* Shift it to be 0 - 32767 */
  966. value = value / 2 + 16384;
  967. default:
  968. break;
  969. }
  970. return value;
  971. } else if (gamecontroller->mapping.buttonasaxis[axis] >= 0) {
  972. Uint8 value;
  973. value = SDL_JoystickGetButton(gamecontroller->joystick, gamecontroller->mapping.buttonasaxis[axis]);
  974. if (value > 0)
  975. return 32767;
  976. return 0;
  977. }
  978. return 0;
  979. }
  980. /*
  981. * Get the current state of a button on a controller
  982. */
  983. Uint8
  984. SDL_GameControllerGetButton(SDL_GameController * gamecontroller, SDL_GameControllerButton button)
  985. {
  986. if (!gamecontroller)
  987. return 0;
  988. if (gamecontroller->mapping.buttons[button] >= 0) {
  989. return (SDL_JoystickGetButton(gamecontroller->joystick, gamecontroller->mapping.buttons[button]));
  990. } else if (gamecontroller->mapping.axesasbutton[button] >= 0) {
  991. Sint16 value;
  992. value = SDL_JoystickGetAxis(gamecontroller->joystick, gamecontroller->mapping.axesasbutton[button]);
  993. if (ABS(value) > 32768/2)
  994. return 1;
  995. return 0;
  996. } else if (gamecontroller->mapping.hatasbutton[button].hat >= 0) {
  997. Uint8 value;
  998. value = SDL_JoystickGetHat(gamecontroller->joystick, gamecontroller->mapping.hatasbutton[button].hat);
  999. if (value & gamecontroller->mapping.hatasbutton[button].mask)
  1000. return 1;
  1001. return 0;
  1002. }
  1003. return 0;
  1004. }
  1005. const char *
  1006. SDL_GameControllerName(SDL_GameController * gamecontroller)
  1007. {
  1008. if (!gamecontroller)
  1009. return NULL;
  1010. return gamecontroller->mapping.name;
  1011. }
  1012. Uint16
  1013. SDL_GameControllerGetVendor(SDL_GameController * gamecontroller)
  1014. {
  1015. return SDL_JoystickGetVendor(SDL_GameControllerGetJoystick(gamecontroller));
  1016. }
  1017. Uint16
  1018. SDL_GameControllerGetProduct(SDL_GameController * gamecontroller)
  1019. {
  1020. return SDL_JoystickGetProduct(SDL_GameControllerGetJoystick(gamecontroller));
  1021. }
  1022. Uint16
  1023. SDL_GameControllerGetProductVersion(SDL_GameController * gamecontroller)
  1024. {
  1025. return SDL_JoystickGetProductVersion(SDL_GameControllerGetJoystick(gamecontroller));
  1026. }
  1027. /*
  1028. * Return if the joystick in question is currently attached to the system,
  1029. * \return 0 if not plugged in, 1 if still present.
  1030. */
  1031. SDL_bool
  1032. SDL_GameControllerGetAttached(SDL_GameController * gamecontroller)
  1033. {
  1034. if (!gamecontroller)
  1035. return SDL_FALSE;
  1036. return SDL_JoystickGetAttached(gamecontroller->joystick);
  1037. }
  1038. /*
  1039. * Get the joystick for this controller
  1040. */
  1041. SDL_Joystick *SDL_GameControllerGetJoystick(SDL_GameController * gamecontroller)
  1042. {
  1043. if (!gamecontroller)
  1044. return NULL;
  1045. return gamecontroller->joystick;
  1046. }
  1047. /*
  1048. * Find the SDL_GameController that owns this instance id
  1049. */
  1050. SDL_GameController *
  1051. SDL_GameControllerFromInstanceID(SDL_JoystickID joyid)
  1052. {
  1053. SDL_GameController *gamecontroller = SDL_gamecontrollers;
  1054. while (gamecontroller) {
  1055. if (gamecontroller->joystick->instance_id == joyid) {
  1056. return gamecontroller;
  1057. }
  1058. gamecontroller = gamecontroller->next;
  1059. }
  1060. return NULL;
  1061. }
  1062. /**
  1063. * Get the SDL joystick layer binding for this controller axis mapping
  1064. */
  1065. SDL_GameControllerButtonBind SDL_GameControllerGetBindForAxis(SDL_GameController * gamecontroller, SDL_GameControllerAxis axis)
  1066. {
  1067. SDL_GameControllerButtonBind bind;
  1068. SDL_memset(&bind, 0x0, sizeof(bind));
  1069. if (!gamecontroller || axis == SDL_CONTROLLER_AXIS_INVALID)
  1070. return bind;
  1071. if (gamecontroller->mapping.axes[axis] >= 0) {
  1072. bind.bindType = SDL_CONTROLLER_BINDTYPE_AXIS;
  1073. bind.value.button = gamecontroller->mapping.axes[axis];
  1074. } else if (gamecontroller->mapping.buttonasaxis[axis] >= 0) {
  1075. bind.bindType = SDL_CONTROLLER_BINDTYPE_BUTTON;
  1076. bind.value.button = gamecontroller->mapping.buttonasaxis[axis];
  1077. }
  1078. return bind;
  1079. }
  1080. /**
  1081. * Get the SDL joystick layer binding for this controller button mapping
  1082. */
  1083. SDL_GameControllerButtonBind SDL_GameControllerGetBindForButton(SDL_GameController * gamecontroller, SDL_GameControllerButton button)
  1084. {
  1085. SDL_GameControllerButtonBind bind;
  1086. SDL_memset(&bind, 0x0, sizeof(bind));
  1087. if (!gamecontroller || button == SDL_CONTROLLER_BUTTON_INVALID)
  1088. return bind;
  1089. if (gamecontroller->mapping.buttons[button] >= 0) {
  1090. bind.bindType = SDL_CONTROLLER_BINDTYPE_BUTTON;
  1091. bind.value.button = gamecontroller->mapping.buttons[button];
  1092. } else if (gamecontroller->mapping.axesasbutton[button] >= 0) {
  1093. bind.bindType = SDL_CONTROLLER_BINDTYPE_AXIS;
  1094. bind.value.axis = gamecontroller->mapping.axesasbutton[button];
  1095. } else if (gamecontroller->mapping.hatasbutton[button].hat >= 0) {
  1096. bind.bindType = SDL_CONTROLLER_BINDTYPE_HAT;
  1097. bind.value.hat.hat = gamecontroller->mapping.hatasbutton[button].hat;
  1098. bind.value.hat.hat_mask = gamecontroller->mapping.hatasbutton[button].mask;
  1099. }
  1100. return bind;
  1101. }
  1102. void
  1103. SDL_GameControllerClose(SDL_GameController * gamecontroller)
  1104. {
  1105. SDL_GameController *gamecontrollerlist, *gamecontrollerlistprev;
  1106. if (!gamecontroller)
  1107. return;
  1108. /* First decrement ref count */
  1109. if (--gamecontroller->ref_count > 0) {
  1110. return;
  1111. }
  1112. SDL_JoystickClose(gamecontroller->joystick);
  1113. gamecontrollerlist = SDL_gamecontrollers;
  1114. gamecontrollerlistprev = NULL;
  1115. while (gamecontrollerlist) {
  1116. if (gamecontroller == gamecontrollerlist) {
  1117. if (gamecontrollerlistprev) {
  1118. /* unlink this entry */
  1119. gamecontrollerlistprev->next = gamecontrollerlist->next;
  1120. } else {
  1121. SDL_gamecontrollers = gamecontroller->next;
  1122. }
  1123. break;
  1124. }
  1125. gamecontrollerlistprev = gamecontrollerlist;
  1126. gamecontrollerlist = gamecontrollerlist->next;
  1127. }
  1128. SDL_free(gamecontroller);
  1129. }
  1130. /*
  1131. * Quit the controller subsystem
  1132. */
  1133. void
  1134. SDL_GameControllerQuit(void)
  1135. {
  1136. ControllerMapping_t *pControllerMap;
  1137. while (SDL_gamecontrollers) {
  1138. SDL_gamecontrollers->ref_count = 1;
  1139. SDL_GameControllerClose(SDL_gamecontrollers);
  1140. }
  1141. while (s_pSupportedControllers) {
  1142. pControllerMap = s_pSupportedControllers;
  1143. s_pSupportedControllers = s_pSupportedControllers->next;
  1144. SDL_free(pControllerMap->name);
  1145. SDL_free(pControllerMap->mapping);
  1146. SDL_free(pControllerMap);
  1147. }
  1148. SDL_DelEventWatch(SDL_GameControllerEventWatcher, NULL);
  1149. }
  1150. /*
  1151. * Event filter to transform joystick events into appropriate game controller ones
  1152. */
  1153. int
  1154. SDL_PrivateGameControllerAxis(SDL_GameController * gamecontroller, SDL_GameControllerAxis axis, Sint16 value)
  1155. {
  1156. int posted;
  1157. /* translate the event, if desired */
  1158. posted = 0;
  1159. #if !SDL_EVENTS_DISABLED
  1160. if (SDL_GetEventState(SDL_CONTROLLERAXISMOTION) == SDL_ENABLE) {
  1161. SDL_Event event;
  1162. event.type = SDL_CONTROLLERAXISMOTION;
  1163. event.caxis.which = gamecontroller->joystick->instance_id;
  1164. event.caxis.axis = axis;
  1165. event.caxis.value = value;
  1166. posted = SDL_PushEvent(&event) == 1;
  1167. }
  1168. #endif /* !SDL_EVENTS_DISABLED */
  1169. return (posted);
  1170. }
  1171. /*
  1172. * Event filter to transform joystick events into appropriate game controller ones
  1173. */
  1174. int
  1175. SDL_PrivateGameControllerButton(SDL_GameController * gamecontroller, SDL_GameControllerButton button, Uint8 state)
  1176. {
  1177. int posted;
  1178. #if !SDL_EVENTS_DISABLED
  1179. SDL_Event event;
  1180. if (button == SDL_CONTROLLER_BUTTON_INVALID)
  1181. return (0);
  1182. switch (state) {
  1183. case SDL_PRESSED:
  1184. event.type = SDL_CONTROLLERBUTTONDOWN;
  1185. break;
  1186. case SDL_RELEASED:
  1187. event.type = SDL_CONTROLLERBUTTONUP;
  1188. break;
  1189. default:
  1190. /* Invalid state -- bail */
  1191. return (0);
  1192. }
  1193. #endif /* !SDL_EVENTS_DISABLED */
  1194. /* translate the event, if desired */
  1195. posted = 0;
  1196. #if !SDL_EVENTS_DISABLED
  1197. if (SDL_GetEventState(event.type) == SDL_ENABLE) {
  1198. event.cbutton.which = gamecontroller->joystick->instance_id;
  1199. event.cbutton.button = button;
  1200. event.cbutton.state = state;
  1201. posted = SDL_PushEvent(&event) == 1;
  1202. }
  1203. #endif /* !SDL_EVENTS_DISABLED */
  1204. return (posted);
  1205. }
  1206. /*
  1207. * Turn off controller events
  1208. */
  1209. int
  1210. SDL_GameControllerEventState(int state)
  1211. {
  1212. #if SDL_EVENTS_DISABLED
  1213. return SDL_IGNORE;
  1214. #else
  1215. const Uint32 event_list[] = {
  1216. SDL_CONTROLLERAXISMOTION, SDL_CONTROLLERBUTTONDOWN, SDL_CONTROLLERBUTTONUP,
  1217. SDL_CONTROLLERDEVICEADDED, SDL_CONTROLLERDEVICEREMOVED, SDL_CONTROLLERDEVICEREMAPPED,
  1218. };
  1219. unsigned int i;
  1220. switch (state) {
  1221. case SDL_QUERY:
  1222. state = SDL_IGNORE;
  1223. for (i = 0; i < SDL_arraysize(event_list); ++i) {
  1224. state = SDL_EventState(event_list[i], SDL_QUERY);
  1225. if (state == SDL_ENABLE) {
  1226. break;
  1227. }
  1228. }
  1229. break;
  1230. default:
  1231. for (i = 0; i < SDL_arraysize(event_list); ++i) {
  1232. SDL_EventState(event_list[i], state);
  1233. }
  1234. break;
  1235. }
  1236. return (state);
  1237. #endif /* SDL_EVENTS_DISABLED */
  1238. }
  1239. /* vi: set ts=4 sw=4 expandtab: */