| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2014 Sam Lantinga <slouken@libsdl.org>
- 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.
- 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:
- 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.
- */
- #include "../../SDL_internal.h"
- #ifdef SDL_JOYSTICK_WINMM
- /* Win32 MultiMedia Joystick driver, contributed by Andrei de A. Formiga */
- #include "../../core/windows/SDL_windows.h"
- #include <mmsystem.h>
- #include <regstr.h>
- #include "SDL_events.h"
- #include "SDL_joystick.h"
- #include "../SDL_sysjoystick.h"
- #include "../SDL_joystick_c.h"
- #ifdef REGSTR_VAL_JOYOEMNAME
- #undef REGSTR_VAL_JOYOEMNAME
- #endif
- #define REGSTR_VAL_JOYOEMNAME "OEMName"
- #define MAX_JOYSTICKS 16
- #define MAX_AXES 6 /* each joystick can have up to 6 axes */
- #define MAX_BUTTONS 32 /* and 32 buttons */
- #define AXIS_MIN -32768 /* minimum value for axis coordinate */
- #define AXIS_MAX 32767 /* maximum value for axis coordinate */
- /* limit axis to 256 possible positions to filter out noise */
- #define JOY_AXIS_THRESHOLD (((AXIS_MAX)-(AXIS_MIN))/256)
- #define JOY_BUTTON_FLAG(n) (1<<n)
- /* array to hold joystick ID values */
- static UINT SYS_JoystickID[MAX_JOYSTICKS];
- static JOYCAPSA SYS_Joystick[MAX_JOYSTICKS];
- static char *SYS_JoystickName[MAX_JOYSTICKS];
- /* The private structure used to keep track of a joystick */
- struct joystick_hwdata
- {
- /* joystick ID */
- UINT id;
- /* values used to translate device-specific coordinates into
- SDL-standard ranges */
- struct _transaxis
- {
- int offset;
- float scale;
- } transaxis[6];
- };
- /* Convert a Windows Multimedia API return code to a text message */
- static void SetMMerror(char *function, int code);
- static char *
- GetJoystickName(int index, const char *szRegKey)
- {
- /* added 7/24/2004 by Eckhard Stolberg */
- /*
- see if there is a joystick for the current
- index (1-16) listed in the registry
- */
- char *name = NULL;
- HKEY hTopKey;
- HKEY hKey;
- DWORD regsize;
- LONG regresult;
- char regkey[256];
- char regvalue[256];
- char regname[256];
- SDL_snprintf(regkey, SDL_arraysize(regkey), "%s\\%s\\%s",
- REGSTR_PATH_JOYCONFIG, szRegKey, REGSTR_KEY_JOYCURR);
- hTopKey = HKEY_LOCAL_MACHINE;
- regresult = RegOpenKeyExA(hTopKey, regkey, 0, KEY_READ, &hKey);
- if (regresult != ERROR_SUCCESS) {
- hTopKey = HKEY_CURRENT_USER;
- regresult = RegOpenKeyExA(hTopKey, regkey, 0, KEY_READ, &hKey);
- }
- if (regresult != ERROR_SUCCESS) {
- return NULL;
- }
- /* find the registry key name for the joystick's properties */
- regsize = sizeof(regname);
- SDL_snprintf(regvalue, SDL_arraysize(regvalue), "Joystick%d%s", index + 1,
- REGSTR_VAL_JOYOEMNAME);
- regresult =
- RegQueryValueExA(hKey, regvalue, 0, 0, (LPBYTE) regname, ®size);
- RegCloseKey(hKey);
- if (regresult != ERROR_SUCCESS) {
- return NULL;
- }
- /* open that registry key */
- SDL_snprintf(regkey, SDL_arraysize(regkey), "%s\\%s", REGSTR_PATH_JOYOEM,
- regname);
- regresult = RegOpenKeyExA(hTopKey, regkey, 0, KEY_READ, &hKey);
- if (regresult != ERROR_SUCCESS) {
- return NULL;
- }
- /* find the size for the OEM name text */
- regsize = sizeof(regvalue);
- regresult =
- RegQueryValueExA(hKey, REGSTR_VAL_JOYOEMNAME, 0, 0, NULL, ®size);
- if (regresult == ERROR_SUCCESS) {
- /* allocate enough memory for the OEM name text ... */
- name = (char *) SDL_malloc(regsize);
- if (name) {
- /* ... and read it from the registry */
- regresult = RegQueryValueExA(hKey,
- REGSTR_VAL_JOYOEMNAME, 0, 0,
- (LPBYTE) name, ®size);
- }
- }
- RegCloseKey(hKey);
- return (name);
- }
- static int SDL_SYS_numjoysticks = 0;
- /* Function to scan the system for joysticks.
- * This function should set SDL_numjoysticks to the number of available
- * joysticks. Joystick 0 should be the system default joystick.
- * It should return 0, or -1 on an unrecoverable fatal error.
- */
- int
- SDL_SYS_JoystickInit(void)
- {
- int i;
- int maxdevs;
- JOYINFOEX joyinfo;
- JOYCAPSA joycaps;
- MMRESULT result;
- /* Reset the joystick ID & name mapping tables */
- for (i = 0; i < MAX_JOYSTICKS; ++i) {
- SYS_JoystickID[i] = 0;
- SYS_JoystickName[i] = NULL;
- }
- /* Loop over all potential joystick devices */
- SDL_SYS_numjoysticks = 0;
- maxdevs = joyGetNumDevs();
- for (i = JOYSTICKID1; i < maxdevs && SDL_SYS_numjoysticks < MAX_JOYSTICKS; ++i) {
- joyinfo.dwSize = sizeof(joyinfo);
- joyinfo.dwFlags = JOY_RETURNALL;
- result = joyGetPosEx(i, &joyinfo);
- if (result == JOYERR_NOERROR) {
- result = joyGetDevCapsA(i, &joycaps, sizeof(joycaps));
- if (result == JOYERR_NOERROR) {
- SYS_JoystickID[SDL_SYS_numjoysticks] = i;
- SYS_Joystick[SDL_SYS_numjoysticks] = joycaps;
- SYS_JoystickName[SDL_SYS_numjoysticks] =
- GetJoystickName(i, joycaps.szRegKey);
- SDL_SYS_numjoysticks++;
- }
- }
- }
- return (SDL_SYS_numjoysticks);
- }
- int SDL_SYS_NumJoysticks()
- {
- return SDL_SYS_numjoysticks;
- }
- void SDL_SYS_JoystickDetect()
- {
- }
- SDL_bool SDL_SYS_JoystickNeedsPolling()
- {
- return SDL_FALSE;
- }
- /* Function to get the device-dependent name of a joystick */
- const char *
- SDL_SYS_JoystickNameForDeviceIndex(int device_index)
- {
- if (SYS_JoystickName[device_index] != NULL) {
- return (SYS_JoystickName[device_index]);
- } else {
- return (SYS_Joystick[device_index].szPname);
- }
- }
- /* Function to perform the mapping from device index to the instance id for this index */
- SDL_JoystickID SDL_SYS_GetInstanceIdOfDeviceIndex(int device_index)
- {
- return device_index;
- }
- /* Function to open a joystick for use.
- The joystick to open is specified by the index field of the joystick.
- This should fill the nbuttons and naxes fields of the joystick structure.
- It returns 0, or -1 if there is an error.
- */
- int
- SDL_SYS_JoystickOpen(SDL_Joystick * joystick, int device_index)
- {
- int index, i;
- int caps_flags[MAX_AXES - 2] =
- { JOYCAPS_HASZ, JOYCAPS_HASR, JOYCAPS_HASU, JOYCAPS_HASV };
- int axis_min[MAX_AXES], axis_max[MAX_AXES];
- /* shortcut */
- index = device_index;
- axis_min[0] = SYS_Joystick[index].wXmin;
- axis_max[0] = SYS_Joystick[index].wXmax;
- axis_min[1] = SYS_Joystick[index].wYmin;
- axis_max[1] = SYS_Joystick[index].wYmax;
- axis_min[2] = SYS_Joystick[index].wZmin;
- axis_max[2] = SYS_Joystick[index].wZmax;
- axis_min[3] = SYS_Joystick[index].wRmin;
- axis_max[3] = SYS_Joystick[index].wRmax;
- axis_min[4] = SYS_Joystick[index].wUmin;
- axis_max[4] = SYS_Joystick[index].wUmax;
- axis_min[5] = SYS_Joystick[index].wVmin;
- axis_max[5] = SYS_Joystick[index].wVmax;
- /* allocate memory for system specific hardware data */
- joystick->instance_id = device_index;
- joystick->hwdata =
- (struct joystick_hwdata *) SDL_malloc(sizeof(*joystick->hwdata));
- if (joystick->hwdata == NULL) {
- return SDL_OutOfMemory();
- }
- SDL_memset(joystick->hwdata, 0, sizeof(*joystick->hwdata));
- /* set hardware data */
- joystick->hwdata->id = SYS_JoystickID[index];
- for (i = 0; i < MAX_AXES; ++i) {
- if ((i < 2) || (SYS_Joystick[index].wCaps & caps_flags[i - 2])) {
- joystick->hwdata->transaxis[i].offset = AXIS_MIN - axis_min[i];
- joystick->hwdata->transaxis[i].scale =
- (float) (AXIS_MAX - AXIS_MIN) / (axis_max[i] - axis_min[i]);
- } else {
- joystick->hwdata->transaxis[i].offset = 0;
- joystick->hwdata->transaxis[i].scale = 1.0; /* Just in case */
- }
- }
- /* fill nbuttons, naxes, and nhats fields */
- joystick->nbuttons = SYS_Joystick[index].wNumButtons;
- joystick->naxes = SYS_Joystick[index].wNumAxes;
- if (SYS_Joystick[index].wCaps & JOYCAPS_HASPOV) {
- joystick->nhats = 1;
- } else {
- joystick->nhats = 0;
- }
- return (0);
- }
- /* Function to determine is this joystick is attached to the system right now */
- SDL_bool SDL_SYS_JoystickAttached(SDL_Joystick *joystick)
- {
- return SDL_TRUE;
- }
- static Uint8
- TranslatePOV(DWORD value)
- {
- Uint8 pos;
- pos = SDL_HAT_CENTERED;
- if (value != JOY_POVCENTERED) {
- if ((value > JOY_POVLEFT) || (value < JOY_POVRIGHT)) {
- pos |= SDL_HAT_UP;
- }
- if ((value > JOY_POVFORWARD) && (value < JOY_POVBACKWARD)) {
- pos |= SDL_HAT_RIGHT;
- }
- if ((value > JOY_POVRIGHT) && (value < JOY_POVLEFT)) {
- pos |= SDL_HAT_DOWN;
- }
- if (value > JOY_POVBACKWARD) {
- pos |= SDL_HAT_LEFT;
- }
- }
- return (pos);
- }
- /* Function to update the state of a joystick - called as a device poll.
- * This function shouldn't update the joystick structure directly,
- * but instead should call SDL_PrivateJoystick*() to deliver events
- * and update joystick device state.
- */
- void
- SDL_SYS_JoystickUpdate(SDL_Joystick * joystick)
- {
- MMRESULT result;
- int i;
- DWORD flags[MAX_AXES] = { JOY_RETURNX, JOY_RETURNY, JOY_RETURNZ,
- JOY_RETURNR, JOY_RETURNU, JOY_RETURNV
- };
- DWORD pos[MAX_AXES];
- struct _transaxis *transaxis;
- int value, change;
- JOYINFOEX joyinfo;
- joyinfo.dwSize = sizeof(joyinfo);
- joyinfo.dwFlags = JOY_RETURNALL | JOY_RETURNPOVCTS;
- if (!joystick->hats) {
- joyinfo.dwFlags &= ~(JOY_RETURNPOV | JOY_RETURNPOVCTS);
- }
- result = joyGetPosEx(joystick->hwdata->id, &joyinfo);
- if (result != JOYERR_NOERROR) {
- SetMMerror("joyGetPosEx", result);
- return;
- }
- /* joystick motion events */
- pos[0] = joyinfo.dwXpos;
- pos[1] = joyinfo.dwYpos;
- pos[2] = joyinfo.dwZpos;
- pos[3] = joyinfo.dwRpos;
- pos[4] = joyinfo.dwUpos;
- pos[5] = joyinfo.dwVpos;
- transaxis = joystick->hwdata->transaxis;
- for (i = 0; i < joystick->naxes; i++) {
- if (joyinfo.dwFlags & flags[i]) {
- value =
- (int) (((float) pos[i] +
- transaxis[i].offset) * transaxis[i].scale);
- change = (value - joystick->axes[i]);
- if ((change < -JOY_AXIS_THRESHOLD)
- || (change > JOY_AXIS_THRESHOLD)) {
- SDL_PrivateJoystickAxis(joystick, (Uint8) i, (Sint16) value);
- }
- }
- }
- /* joystick button events */
- if (joyinfo.dwFlags & JOY_RETURNBUTTONS) {
- for (i = 0; i < joystick->nbuttons; ++i) {
- if (joyinfo.dwButtons & JOY_BUTTON_FLAG(i)) {
- if (!joystick->buttons[i]) {
- SDL_PrivateJoystickButton(joystick, (Uint8) i,
- SDL_PRESSED);
- }
- } else {
- if (joystick->buttons[i]) {
- SDL_PrivateJoystickButton(joystick, (Uint8) i,
- SDL_RELEASED);
- }
- }
- }
- }
- /* joystick hat events */
- if (joyinfo.dwFlags & JOY_RETURNPOV) {
- Uint8 pos;
- pos = TranslatePOV(joyinfo.dwPOV);
- if (pos != joystick->hats[0]) {
- SDL_PrivateJoystickHat(joystick, 0, pos);
- }
- }
- }
- /* Function to close a joystick after use */
- void
- SDL_SYS_JoystickClose(SDL_Joystick * joystick)
- {
- /* free system specific hardware data */
- SDL_free(joystick->hwdata);
- joystick->hwdata = NULL;
- }
- /* Function to perform any system-specific joystick related cleanup */
- void
- SDL_SYS_JoystickQuit(void)
- {
- int i;
- for (i = 0; i < MAX_JOYSTICKS; i++) {
- SDL_free(SYS_JoystickName[i]);
- SYS_JoystickName[i] = NULL;
- }
- }
- SDL_JoystickGUID SDL_SYS_JoystickGetDeviceGUID( int device_index )
- {
- SDL_JoystickGUID guid;
- /* the GUID is just the first 16 chars of the name for now */
- const char *name = SDL_SYS_JoystickNameForDeviceIndex( device_index );
- SDL_zero( guid );
- SDL_memcpy( &guid, name, SDL_min( sizeof(guid), SDL_strlen( name ) ) );
- return guid;
- }
- SDL_JoystickGUID SDL_SYS_JoystickGetGUID(SDL_Joystick * joystick)
- {
- SDL_JoystickGUID guid;
- /* the GUID is just the first 16 chars of the name for now */
- const char *name = joystick->name;
- SDL_zero( guid );
- SDL_memcpy( &guid, name, SDL_min( sizeof(guid), SDL_strlen( name ) ) );
- return guid;
- }
- /* implementation functions */
- void
- SetMMerror(char *function, int code)
- {
- static char *error;
- static char errbuf[1024];
- errbuf[0] = 0;
- switch (code) {
- case MMSYSERR_NODRIVER:
- error = "Joystick driver not present";
- break;
- case MMSYSERR_INVALPARAM:
- case JOYERR_PARMS:
- error = "Invalid parameter(s)";
- break;
- case MMSYSERR_BADDEVICEID:
- error = "Bad device ID";
- break;
- case JOYERR_UNPLUGGED:
- error = "Joystick not attached";
- break;
- case JOYERR_NOCANDO:
- error = "Can't capture joystick input";
- break;
- default:
- SDL_snprintf(errbuf, SDL_arraysize(errbuf),
- "%s: Unknown Multimedia system error: 0x%x",
- function, code);
- break;
- }
- if (!errbuf[0]) {
- SDL_snprintf(errbuf, SDL_arraysize(errbuf), "%s: %s", function,
- error);
- }
- SDL_SetError("%s", errbuf);
- }
- #endif /* SDL_JOYSTICK_WINMM */
- /* vi: set ts=4 sw=4 expandtab: */
|