From 1e9380584a5393e6b664bcf3ca566dcca70ac0d4 Mon Sep 17 00:00:00 2001 From: Locria Cyber <74560659+iacore@users.noreply.github.com> Date: Mon, 12 Jun 2023 22:44:29 +0000 Subject: [PATCH 1/3] Match gamepad by USB vendor+product on Linux --- src/input.c | 41 ++++++++++++++++++++++++++++++++++++++--- src/internal.h | 12 ++++++++++++ src/linux_joystick.c | 7 +++++++ 3 files changed, 57 insertions(+), 3 deletions(-) diff --git a/src/input.c b/src/input.c index 36128e10..f0e27369 100644 --- a/src/input.c +++ b/src/input.c @@ -67,15 +67,50 @@ static GLFWbool initJoysticks(void) return _glfw.joysticksInitialized = GLFW_TRUE; } +uint16_t parseHexDigit(char c) +{ + if (c >= '0' && c <= '9') return c - '0'; + if (c >= 'a' && c <= 'f') return c - 'a' + 10; + if (c >= 'A' && c <= 'F') return c - 'A' + 10; + assert(GLFW_FALSE); +} + +struct vendor_product +{ + uint16_t vendor; + uint16_t product; +}; + +static struct vendor_product parseGUID(const char* guid) +{ + struct vendor_product result; + result.vendor = parseHexDigit(guid[8]) | (parseHexDigit(guid[9]) << 8); + result.product = parseHexDigit(guid[16]) | (parseHexDigit(guid[17]) << 8); + return result; +} + // Finds a mapping based on joystick GUID // static _GLFWmapping* findMapping(const char* guid) { - int i; + struct vendor_product this; - for (i = 0; i < _glfw.mappingCount; i++) + // exact match + for (int i = 0; i < _glfw.mappingCount; i++) { - if (strcmp(_glfw.mappings[i].guid, guid) == 0) + if (strncmp(_glfw.mappings[i].guid, guid, 32) == 0) + return _glfw.mappings + i; + } + + // only match vendor product + + this = parseGUID(guid); + + for (int i = 0; i < _glfw.mappingCount; i++) + { + struct vendor_product that = parseGUID(_glfw.mappings[i].guid); + + if (memcmp(&this, &that, sizeof(struct vendor_product)) == 0) return _glfw.mappings + i; } diff --git a/src/internal.h b/src/internal.h index fe0369aa..72f85dd5 100644 --- a/src/internal.h +++ b/src/internal.h @@ -74,6 +74,7 @@ typedef struct _GLFWmonitor _GLFWmonitor; typedef struct _GLFWcursor _GLFWcursor; typedef struct _GLFWmapelement _GLFWmapelement; typedef struct _GLFWmapping _GLFWmapping; +typedef struct _GLFWusbinfo _GLFWusbinfo; typedef struct _GLFWjoystick _GLFWjoystick; typedef struct _GLFWtls _GLFWtls; typedef struct _GLFWmutex _GLFWmutex; @@ -631,6 +632,16 @@ struct _GLFWmapping _GLFWmapelement axes[6]; }; +// USB vendor, product, version +// +struct _GLFWusbinfo +{ + uint16_t bustype; + uint16_t vendor; + uint16_t product; + uint16_t version; +}; + // Joystick structure // struct _GLFWjoystick @@ -646,6 +657,7 @@ struct _GLFWjoystick char name[128]; void* userPointer; char guid[33]; + _GLFWusbinfo usbInfo; _GLFWmapping* mapping; // This is defined in platform.h diff --git a/src/linux_joystick.c b/src/linux_joystick.c index 26db853e..8f7e481a 100644 --- a/src/linux_joystick.c +++ b/src/linux_joystick.c @@ -172,6 +172,12 @@ static GLFWbool openJoystickDevice(const char* path) char guid[33] = ""; + _GLFWusbinfo usbinfo; + usbinfo.bustype = id.bustype; + usbinfo.vendor = id.vendor; + usbinfo.product = id.product; + usbinfo.version = id.version; + // Generate a joystick GUID that matches the SDL 2.0.5+ one if (id.vendor && id.product && id.version) { @@ -231,6 +237,7 @@ static GLFWbool openJoystickDevice(const char* path) close(linjs.fd); return GLFW_FALSE; } + js->usbInfo = usbinfo; strncpy(linjs.path, path, sizeof(linjs.path) - 1); memcpy(&js->linjs, &linjs, sizeof(linjs)); From 337a4589a94139bde9e77644d28624038faa221b Mon Sep 17 00:00:00 2001 From: Locria Cyber <74560659+iacore@users.noreply.github.com> Date: Tue, 13 Jun 2023 15:22:22 +0000 Subject: [PATCH 2/3] Add guard for linux joystick --- src/input.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/input.c b/src/input.c index f0e27369..5f107d41 100644 --- a/src/input.c +++ b/src/input.c @@ -102,8 +102,8 @@ static _GLFWmapping* findMapping(const char* guid) return _glfw.mappings + i; } +#if defined(GLFW_BUILD_LINUX_JOYSTICK) // only match vendor product - this = parseGUID(guid); for (int i = 0; i < _glfw.mappingCount; i++) @@ -113,6 +113,7 @@ static _GLFWmapping* findMapping(const char* guid) if (memcmp(&this, &that, sizeof(struct vendor_product)) == 0) return _glfw.mappings + i; } +#endif return NULL; } From 0c181dcf74d486fb3b0b9fd56640a7d3a354ad74 Mon Sep 17 00:00:00 2001 From: Locria Cyber <74560659+iacore@users.noreply.github.com> Date: Wed, 14 Jun 2023 18:53:01 +0000 Subject: [PATCH 3/3] Fix to Pass tests --- src/input.c | 39 +++++++++++++++++++++++++-------------- 1 file changed, 25 insertions(+), 14 deletions(-) diff --git a/src/input.c b/src/input.c index 5f107d41..3e2a4b39 100644 --- a/src/input.c +++ b/src/input.c @@ -67,6 +67,8 @@ static GLFWbool initJoysticks(void) return _glfw.joysticksInitialized = GLFW_TRUE; } +#if defined(GLFW_BUILD_LINUX_JOYSTICK) + uint16_t parseHexDigit(char c) { if (c >= '0' && c <= '9') return c - '0'; @@ -89,21 +91,10 @@ static struct vendor_product parseGUID(const char* guid) return result; } -// Finds a mapping based on joystick GUID -// -static _GLFWmapping* findMapping(const char* guid) +static _GLFWmapping* findMappingUSBVendorProduct(const char* guid) { + struct vendor_product this; - - // exact match - for (int i = 0; i < _glfw.mappingCount; i++) - { - if (strncmp(_glfw.mappings[i].guid, guid, 32) == 0) - return _glfw.mappings + i; - } - -#if defined(GLFW_BUILD_LINUX_JOYSTICK) - // only match vendor product this = parseGUID(guid); for (int i = 0; i < _glfw.mappingCount; i++) @@ -113,11 +104,31 @@ static _GLFWmapping* findMapping(const char* guid) if (memcmp(&this, &that, sizeof(struct vendor_product)) == 0) return _glfw.mappings + i; } -#endif return NULL; } +#endif + +// Finds a mapping based on joystick GUID +// +static _GLFWmapping* findMapping(const char* guid) +{ + // exact match + for (int i = 0; i < _glfw.mappingCount; i++) + { + if (strncmp(_glfw.mappings[i].guid, guid, 32) == 0) + return _glfw.mappings + i; + } + +#if defined(GLFW_BUILD_LINUX_JOYSTICK) + // only match vendor id, product id + return findMappingUSBVendorProduct(guid); +#else + return NULL; +#endif +} + // Checks whether a gamepad mapping element is present in the hardware // static GLFWbool isValidElementForJoystick(const _GLFWmapelement* e,