Use minimal amount of memory needed to enumerate in Windows

This commit is contained in:
Will Hedgecock 2022-02-17 15:17:34 -06:00
parent ea3c593776
commit 05aae9078a
7 changed files with 176 additions and 86 deletions

View File

@ -2,7 +2,7 @@
* SerialPort_Windows.c * SerialPort_Windows.c
* *
* Created on: Feb 25, 2012 * Created on: Feb 25, 2012
* Last Updated on: Feb 16, 2022 * Last Updated on: Feb 17, 2022
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2022 Fazecast, Inc. * Copyright (C) 2012-2022 Fazecast, Inc.
@ -108,7 +108,6 @@ static void enumeratePorts(void)
serialPorts.ports[i]->enumerated = (serialPorts.ports[i]->handle != INVALID_HANDLE_VALUE); serialPorts.ports[i]->enumerated = (serialPorts.ports[i]->handle != INVALID_HANDLE_VALUE);
// Enumerate all serial ports present on the current system // Enumerate all serial ports present on the current system
wchar_t comPort[128];
const struct { GUID guid; DWORD flags; } setupClasses[] = { const struct { GUID guid; DWORD flags; } setupClasses[] = {
{ .guid = GUID_DEVCLASS_PORTS, .flags = DIGCF_PRESENT }, { .guid = GUID_DEVCLASS_PORTS, .flags = DIGCF_PRESENT },
{ .guid = GUID_DEVCLASS_MODEM, .flags = DIGCF_PRESENT }, { .guid = GUID_DEVCLASS_MODEM, .flags = DIGCF_PRESENT },
@ -129,57 +128,81 @@ static void enumeratePorts(void)
while (SetupDiEnumDeviceInfo(devList, devInterfaceIndex++, &devInfoData)) while (SetupDiEnumDeviceInfo(devList, devInterfaceIndex++, &devInfoData))
{ {
// Fetch the corresponding COM port for this device // Fetch the corresponding COM port for this device
wchar_t *comPortString = NULL; DWORD comPortLength = 0;
DWORD comPortLength = sizeof(comPort) / sizeof(wchar_t); wchar_t *comPort = NULL, *comPortString = NULL;
char friendlyNameMemory = 0, portDescriptionMemory = 0;
HKEY key = SetupDiOpenDevRegKey(devList, &devInfoData, DICS_FLAG_GLOBAL, 0, DIREG_DEV, KEY_QUERY_VALUE); HKEY key = SetupDiOpenDevRegKey(devList, &devInfoData, DICS_FLAG_GLOBAL, 0, DIREG_DEV, KEY_QUERY_VALUE);
if ((key != INVALID_HANDLE_VALUE) && (RegQueryValueExW(key, L"PortName", NULL, NULL, (BYTE*)comPort, &comPortLength) == ERROR_SUCCESS))
comPortString = (comPort[0] == L'\\') ? (wcsrchr(comPort, L'\\') + 1) : comPort;
if (key != INVALID_HANDLE_VALUE) if (key != INVALID_HANDLE_VALUE)
{
if ((RegQueryValueExW(key, L"PortName", NULL, NULL, NULL, &comPortLength) == ERROR_SUCCESS) && (comPortLength < 32))
{
comPortLength += sizeof(wchar_t);
comPort = (wchar_t*)malloc(comPortLength);
if (comPort && (RegGetValueW(key, NULL, L"PortName", RRF_RT_REG_SZ, NULL, (LPBYTE)comPort, &comPortLength) == ERROR_SUCCESS))
comPortString = (comPort[0] == L'\\') ? (wcsrchr(comPort, L'\\') + 1) : comPort;
}
RegCloseKey(key); RegCloseKey(key);
}
if (!comPortString || wcsstr(comPortString, L"LPT")) if (!comPortString || wcsstr(comPortString, L"LPT"))
{
if (comPort)
free(comPort);
continue; continue;
}
// Fetch the friendly name for this device // Fetch the friendly name for this device
DWORD friendlyNameLength = 0; DWORD friendlyNameLength = 0;
wchar_t *friendlyNameString = NULL; wchar_t *friendlyNameString = NULL;
SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_FriendlyName, &devInfoPropType, NULL, 0, &friendlyNameLength, 0); if ((!SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_FriendlyName, &devInfoPropType, NULL, 0, &friendlyNameLength, 0) && (GetLastError() != ERROR_INSUFFICIENT_BUFFER)) || !friendlyNameLength)
if (!friendlyNameLength)
SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_FRIENDLYNAME, NULL, NULL, 0, &friendlyNameLength); SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_FRIENDLYNAME, NULL, NULL, 0, &friendlyNameLength);
if (friendlyNameLength) if (friendlyNameLength && (friendlyNameLength < 256))
{ {
friendlyNameLength += sizeof(wchar_t);
friendlyNameString = (wchar_t*)malloc(friendlyNameLength); friendlyNameString = (wchar_t*)malloc(friendlyNameLength);
if (!SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_FriendlyName, &devInfoPropType, (BYTE*)friendlyNameString, friendlyNameLength, NULL, 0) && if (!friendlyNameString || (SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_FriendlyName, &devInfoPropType, (BYTE*)friendlyNameString, friendlyNameLength, NULL, 0) &&
!SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_FRIENDLYNAME, NULL, (BYTE*)friendlyNameString, friendlyNameLength, NULL)) !SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_FRIENDLYNAME, NULL, (BYTE*)friendlyNameString, friendlyNameLength, NULL)))
{ {
if (friendlyNameString)
free(friendlyNameString);
friendlyNameString = comPortString;
friendlyNameLength = comPortLength; friendlyNameLength = comPortLength;
friendlyNameString = (wchar_t*)realloc(friendlyNameString, comPortLength); }
wcscpy_s(friendlyNameString, comPortLength / sizeof(wchar_t), comPortString); else
{
friendlyNameMemory = 1;
friendlyNameString[(friendlyNameLength / sizeof(wchar_t)) - 1] = 0;
} }
} }
else else
{ {
friendlyNameString = comPortString;
friendlyNameLength = comPortLength; friendlyNameLength = comPortLength;
friendlyNameString = (wchar_t*)malloc(comPortLength);
wcscpy_s(friendlyNameString, comPortLength / sizeof(wchar_t), comPortString);
} }
// Fetch the bus-reported device description // Fetch the bus-reported device description
DWORD portDescriptionLength = 0; DWORD portDescriptionLength = 0;
wchar_t *portDescriptionString = NULL; wchar_t *portDescriptionString = NULL;
SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_BusReportedDeviceDesc, &devInfoPropType, NULL, 0, &portDescriptionLength, 0); if ((SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_BusReportedDeviceDesc, &devInfoPropType, NULL, 0, &portDescriptionLength, 0) || (GetLastError() == ERROR_INSUFFICIENT_BUFFER)) && portDescriptionLength && (portDescriptionLength < 256))
if (portDescriptionLength)
{ {
portDescriptionLength += sizeof(wchar_t);
portDescriptionString = (wchar_t*)malloc(portDescriptionLength); portDescriptionString = (wchar_t*)malloc(portDescriptionLength);
if (!SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_BusReportedDeviceDesc, &devInfoPropType, (BYTE*)portDescriptionString, portDescriptionLength, NULL, 0)) if (!portDescriptionString || !SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_BusReportedDeviceDesc, &devInfoPropType, (BYTE*)portDescriptionString, portDescriptionLength, NULL, 0))
{ {
portDescriptionString = (wchar_t*)realloc(portDescriptionString, friendlyNameLength); if (portDescriptionString)
wcscpy_s(portDescriptionString, friendlyNameLength / sizeof(wchar_t), friendlyNameString); free(portDescriptionString);
portDescriptionString = friendlyNameString;
portDescriptionLength = friendlyNameLength;
}
else
{
portDescriptionMemory = 1;
portDescriptionString[(portDescriptionLength / sizeof(wchar_t)) - 1] = 0;
} }
} }
else else
{ {
portDescriptionString = (wchar_t*)malloc(friendlyNameLength); portDescriptionString = friendlyNameString;
wcscpy_s(portDescriptionString, friendlyNameLength / sizeof(wchar_t), friendlyNameString); portDescriptionLength = friendlyNameLength;
} }
// Fetch the physical location for this device // Fetch the physical location for this device
@ -191,15 +214,16 @@ static void enumeratePorts(void)
if (!SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_Address, &devInfoPropType, (BYTE*)&portNumber, sizeof(portNumber), NULL, 0) && if (!SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_Address, &devInfoPropType, (BYTE*)&portNumber, sizeof(portNumber), NULL, 0) &&
!SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_ADDRESS, NULL, (BYTE*)&portNumber, sizeof(portNumber), NULL)) !SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_ADDRESS, NULL, (BYTE*)&portNumber, sizeof(portNumber), NULL))
portNumber = -1; portNumber = -1;
SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_LocationInfo, &devInfoPropType, NULL, 0, &locationLength, 0); if ((!SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_LocationInfo, &devInfoPropType, NULL, 0, &locationLength, 0) && (GetLastError() != ERROR_INSUFFICIENT_BUFFER)) || !locationLength)
if (!locationLength)
SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_LOCATION_INFORMATION, NULL, NULL, 0, &locationLength); SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_LOCATION_INFORMATION, NULL, NULL, 0, &locationLength);
if (locationLength) if (locationLength && (locationLength < 256))
{ {
locationLength += sizeof(wchar_t);
locationString = (wchar_t*)malloc(locationLength); locationString = (wchar_t*)malloc(locationLength);
if (SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_LocationInfo, &devInfoPropType, (BYTE*)locationString, locationLength, NULL, 0) || if (locationString && (SetupDiGetDevicePropertyW(devList, &devInfoData, &DEVPKEY_Device_LocationInfo, &devInfoPropType, (BYTE*)locationString, locationLength, NULL, 0) ||
SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_LOCATION_INFORMATION, NULL, (BYTE*)locationString, locationLength, NULL)) SetupDiGetDeviceRegistryPropertyW(devList, &devInfoData, SPDRP_LOCATION_INFORMATION, NULL, (BYTE*)locationString, locationLength, NULL)))
{ {
locationString[(locationLength / sizeof(wchar_t)) - 1] = 0;
if (wcsstr(locationString, L"Hub")) if (wcsstr(locationString, L"Hub"))
hubNumber = _wtoi(wcschr(wcsstr(locationString, L"Hub"), L'#') + 1); hubNumber = _wtoi(wcschr(wcsstr(locationString, L"Hub"), L'#') + 1);
if ((portNumber == -1) && wcsstr(locationString, L"Port")) if ((portNumber == -1) && wcsstr(locationString, L"Port"))
@ -214,6 +238,7 @@ static void enumeratePorts(void)
portNumber = _wtoi(portString); portNumber = _wtoi(portString);
} }
} }
if (locationString)
free(locationString); free(locationString);
} }
if (busNumber == -1) if (busNumber == -1)
@ -222,8 +247,18 @@ static void enumeratePorts(void)
hubNumber = 0; hubNumber = 0;
if (portNumber == -1) if (portNumber == -1)
portNumber = 0; portNumber = 0;
locationString = (wchar_t*)malloc(32*sizeof(wchar_t)); locationString = (wchar_t*)malloc(16*sizeof(wchar_t));
_snwprintf_s(locationString, 32, 32, L"%d-%d.%d", busNumber, hubNumber, portNumber); if (locationString)
_snwprintf_s(locationString, 16, 16, L"%d-%d.%d", busNumber, hubNumber, portNumber);
else
{
free(comPort);
if (friendlyNameMemory)
free(friendlyNameString);
if (portDescriptionMemory)
free(portDescriptionString);
continue;
}
// Check if port is already enumerated // Check if port is already enumerated
serialPort *port = fetchPort(&serialPorts, comPortString); serialPort *port = fetchPort(&serialPorts, comPortString);
@ -231,23 +266,32 @@ static void enumeratePorts(void)
{ {
// See if device has changed locations // See if device has changed locations
port->enumerated = 1; port->enumerated = 1;
int oldLength = wcslen(port->portLocation); int oldLength = 1 + wcslen(port->portLocation);
int newLength = wcslen(locationString); int newLength = 1 + wcslen(locationString);
if (oldLength != newLength) if (oldLength != newLength)
{ {
port->portLocation = (wchar_t*)realloc(port->portLocation, (1 + newLength) * sizeof(wchar_t)); wchar_t *newMemory = (wchar_t*)realloc(port->portLocation, newLength * sizeof(wchar_t));
wcscpy_s(port->portLocation, 32, locationString); if (newMemory)
{
port->portLocation = newMemory;
wcscpy_s(port->portLocation, newLength, locationString);
}
else
wcscpy_s(port->portLocation, oldLength, locationString);
} }
else if (wcscmp(port->portLocation, locationString)) else if (wcscmp(port->portLocation, locationString))
wcscpy_s(port->portLocation, 32, locationString); wcscpy_s(port->portLocation, newLength, locationString);
} }
else else
pushBack(&serialPorts, comPortString, friendlyNameString, portDescriptionString, locationString); pushBack(&serialPorts, comPortString, friendlyNameString, portDescriptionString, locationString);
// Clean up memory and reset device info structure // Clean up memory and reset device info structure
free(comPort);
free(locationString); free(locationString);
free(portDescriptionString); if (friendlyNameMemory)
free(friendlyNameString); free(friendlyNameString);
if (portDescriptionMemory)
free(portDescriptionString);
devInfoData.cbSize = sizeof(devInfoData); devInfoData.cbSize = sizeof(devInfoData);
} }
SetupDiDestroyDeviceInfoList(devList); SetupDiDestroyDeviceInfoList(devList);
@ -266,12 +310,12 @@ static void enumeratePorts(void)
if ((FT_CreateDeviceInfoList(&numDevs) == FT_OK) && (numDevs > 0)) if ((FT_CreateDeviceInfoList(&numDevs) == FT_OK) && (numDevs > 0))
{ {
FT_DEVICE_LIST_INFO_NODE *devInfo = (FT_DEVICE_LIST_INFO_NODE*)malloc(sizeof(FT_DEVICE_LIST_INFO_NODE)*numDevs); FT_DEVICE_LIST_INFO_NODE *devInfo = (FT_DEVICE_LIST_INFO_NODE*)malloc(sizeof(FT_DEVICE_LIST_INFO_NODE)*numDevs);
if (FT_GetDeviceInfoList(devInfo, &numDevs) == FT_OK) if (devInfo && (FT_GetDeviceInfoList(devInfo, &numDevs) == FT_OK))
{ {
for (int i = 0; i < numDevs; ++i) for (int i = 0; i < numDevs; ++i)
{ {
// Determine if the port is currently enumerated and already open // Determine if the port is currently enumerated and already open
char isOpen = ((devInfo[i].Flags & FT_FLAGS_OPENED) || !strlen(devInfo[i].SerialNumber)) ? 1 : 0; char isOpen = ((devInfo[i].Flags & FT_FLAGS_OPENED) || (devInfo[i].SerialNumber[0] == 0)) ? 1 : 0;
if (!isOpen) if (!isOpen)
for (int j = 0; j < serialPorts.length; ++j) for (int j = 0; j < serialPorts.length; ++j)
if ((memcmp(serialPorts.ports[j]->serialNumber, devInfo[i].SerialNumber, sizeof(serialPorts.ports[j]->serialNumber)) == 0) && (serialPorts.ports[j]->handle != INVALID_HANDLE_VALUE)) if ((memcmp(serialPorts.ports[j]->serialNumber, devInfo[i].SerialNumber, sizeof(serialPorts.ports[j]->serialNumber)) == 0) && (serialPorts.ports[j]->handle != INVALID_HANDLE_VALUE))
@ -282,7 +326,11 @@ static void enumeratePorts(void)
} }
// Update the port description if not already open // Update the port description if not already open
if (!isOpen && getPortPathFromSerial(comPort, devInfo[i].SerialNumber)) const int comPortLength = 16;
wchar_t *comPort = (wchar_t*)malloc(comPortLength);
devInfo[i].Description[sizeof(devInfo[i].Description)-1] = 0;
devInfo[i].SerialNumber[sizeof(devInfo[i].SerialNumber)-1] = 0;
if (!isOpen && comPort && getPortPathFromSerial(comPort, comPortLength, devInfo[i].SerialNumber))
{ {
// Check if actually connected and present in the port list // Check if actually connected and present in the port list
for (int j = 0; j < serialPorts.length; ++j) for (int j = 0; j < serialPorts.length; ++j)
@ -301,8 +349,11 @@ static void enumeratePorts(void)
break; break;
} }
} }
if (comPort)
free(comPort);
} }
} }
if (devInfo)
free(devInfo); free(devInfo);
} }
} }

View File

@ -2,7 +2,7 @@
* WindowsHelperFunctions.c * WindowsHelperFunctions.c
* *
* Created on: May 05, 2015 * Created on: May 05, 2015
* Last Updated on: Feb 16, 2022 * Last Updated on: Feb 17, 2022
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2022 Fazecast, Inc. * Copyright (C) 2012-2022 Fazecast, Inc.
@ -33,6 +33,7 @@
#include <shellapi.h> #include <shellapi.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <errno.h>
#include "WindowsHelperFunctions.h" #include "WindowsHelperFunctions.h"
// Common storage functionality // Common storage functionality
@ -65,6 +66,21 @@ serialPort* pushBack(serialPortVector* vector, const wchar_t* key, const wchar_t
port->portLocation = (wchar_t*)malloc((wcslen(location)+1)*sizeof(wchar_t)); port->portLocation = (wchar_t*)malloc((wcslen(location)+1)*sizeof(wchar_t));
port->friendlyName = (wchar_t*)malloc((wcslen(friendlyName)+1)*sizeof(wchar_t)); port->friendlyName = (wchar_t*)malloc((wcslen(friendlyName)+1)*sizeof(wchar_t));
port->portDescription = (wchar_t*)malloc((wcslen(description)+1)*sizeof(wchar_t)); port->portDescription = (wchar_t*)malloc((wcslen(description)+1)*sizeof(wchar_t));
if (!port->portPath || !port->portLocation || !port->friendlyName || !port->portDescription)
{
// Clean up memory associated with the port
vector->length--;
if (port->portPath)
free(port->portPath);
if (port->portLocation)
free(port->portLocation);
if (port->friendlyName)
free(port->friendlyName);
if (port->portDescription)
free(port->portDescription);
free(port);
return NULL;
}
// Store port strings // Store port strings
if (containsSlashes) if (containsSlashes)
@ -120,32 +136,43 @@ void removePort(serialPortVector* vector, serialPort* port)
void reduceLatencyToMinimum(const wchar_t* portName, unsigned char requestElevatedPermissions) void reduceLatencyToMinimum(const wchar_t* portName, unsigned char requestElevatedPermissions)
{ {
// Search for this port in all FTDI enumerated ports // Search for this port in all FTDI enumerated ports
HKEY key, paramKey = 0; HKEY key, paramKey;
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SYSTEM\\CurrentControlSet\\Enum\\FTDIBUS", 0, KEY_READ, &key) == ERROR_SUCCESS) DWORD maxSubkeySize, maxPortNameSize = 8;
if ((RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SYSTEM\\CurrentControlSet\\Enum\\FTDIBUS", 0, KEY_READ, &key) == ERROR_SUCCESS) &&
(RegQueryInfoKeyW(key, NULL, NULL, NULL, NULL, &maxSubkeySize, NULL, NULL, NULL, NULL, NULL, NULL) == ERROR_SUCCESS))
{ {
DWORD index = 0, subkeySize = 255, portNameSize = 16; maxSubkeySize += 32;
wchar_t *subkey = (wchar_t*)malloc(subkeySize*sizeof(wchar_t)), *regPortName = (wchar_t*)malloc(portNameSize*sizeof(wchar_t)); DWORD index = 0, subkeySize = maxSubkeySize;
while (RegEnumKeyExW(key, index++, subkey, &subkeySize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS) wchar_t *subkey = (wchar_t*)malloc(maxSubkeySize * sizeof(wchar_t)), *portPath = (wchar_t*)malloc(maxPortNameSize * sizeof(wchar_t));
while (subkey && portPath && (RegEnumKeyExW(key, index++, subkey, &subkeySize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS))
{
// Retrieve the current port latency value
size_t convertedSize;
char *subkeyString = NULL;
subkeySize = maxSubkeySize;
DWORD desiredLatency = 2, oldLatency = 2;
if ((wcstombs_s(&convertedSize, NULL, 0, subkey, 0) == 0) && (convertedSize < 255))
{
subkeyString = (char*)malloc(convertedSize);
if (subkeyString && (wcstombs_s(NULL, subkeyString, convertedSize, subkey, convertedSize - 1) == 0) &&
(wcscat_s(subkey, maxSubkeySize, L"\\0000\\Device Parameters") == 0))
{ {
DWORD latency = 2, oldLatency = 2, oldLatencySize = sizeof(DWORD);
char *subkeyString = (char*)malloc(subkeySize + 2);
memset(subkeyString, 0, subkeySize + 2);
wcstombs(subkeyString, subkey, subkeySize + 1);
subkeySize = 255;
portNameSize = 16;
memset(regPortName, 0, portNameSize * sizeof(wchar_t));
wcscat_s(subkey, subkeySize, L"\\0000\\Device Parameters");
if (RegOpenKeyExW(key, subkey, 0, KEY_QUERY_VALUE, &paramKey) == ERROR_SUCCESS) if (RegOpenKeyExW(key, subkey, 0, KEY_QUERY_VALUE, &paramKey) == ERROR_SUCCESS)
{ {
if ((RegQueryValueExW(paramKey, L"PortName", NULL, NULL, (LPBYTE)regPortName, &portNameSize) == ERROR_SUCCESS) && (wcscmp(portName, regPortName) == 0)) DWORD oldLatencySize = sizeof(DWORD), portNameSize = maxPortNameSize * sizeof(wchar_t);
RegQueryValueExW(paramKey, L"LatencyTimer", NULL, NULL, (LPBYTE)&oldLatency, &oldLatencySize); if ((RegGetValueW(paramKey, NULL, L"PortName", RRF_RT_REG_SZ, NULL, (LPBYTE)portPath, &portNameSize) == ERROR_SUCCESS) && (wcscmp(portName, portPath) == 0))
RegGetValueW(paramKey, NULL, L"LatencyTimer", RRF_RT_REG_DWORD, NULL, (LPBYTE)&oldLatency, &oldLatencySize);
RegCloseKey(paramKey); RegCloseKey(paramKey);
} }
if (oldLatency > latency) }
}
// Update the port latency value if it is too high
if (oldLatency > desiredLatency)
{ {
if (RegOpenKeyExW(key, subkey, 0, KEY_SET_VALUE, &paramKey) == ERROR_SUCCESS) if (RegOpenKeyExW(key, subkey, 0, KEY_SET_VALUE, &paramKey) == ERROR_SUCCESS)
{ {
RegSetValueExW(paramKey, L"LatencyTimer", 0, REG_DWORD, (LPBYTE)&latency, sizeof(latency)); RegSetValueExW(paramKey, L"LatencyTimer", 0, REG_DWORD, (LPBYTE)&desiredLatency, sizeof(desiredLatency));
RegCloseKey(paramKey); RegCloseKey(paramKey);
} }
else if (requestElevatedPermissions) else if (requestElevatedPermissions)
@ -192,45 +219,57 @@ void reduceLatencyToMinimum(const wchar_t* portName, unsigned char requestElevat
free(paramsString); free(paramsString);
} }
} }
// Clean up memory
if (subkeyString)
free(subkeyString); free(subkeyString);
} }
RegCloseKey(key); RegCloseKey(key);
free(regPortName); free(portPath);
free(subkey); free(subkey);
} }
} }
int getPortPathFromSerial(wchar_t* portPath, const char* serialNumber) int getPortPathFromSerial(wchar_t* portPath, int portPathLength, const char* serialNumber)
{ {
// Search for this port in all FTDI enumerated ports // Search for this port in all FTDI enumerated ports
int found = 0; int found = 0;
HKEY key, paramKey = 0; HKEY key, paramKey;
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SYSTEM\\CurrentControlSet\\Enum\\FTDIBUS", 0, KEY_READ, &key) == ERROR_SUCCESS) DWORD maxSubkeySize;
if ((RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SYSTEM\\CurrentControlSet\\Enum\\FTDIBUS", 0, KEY_READ, &key) == ERROR_SUCCESS) &&
(RegQueryInfoKeyW(key, NULL, NULL, NULL, NULL, &maxSubkeySize, NULL, NULL, NULL, NULL, NULL, NULL) == ERROR_SUCCESS))
{ {
DWORD index = 0, subkeySize = 255, portNameSize = 16; maxSubkeySize += 32;
wchar_t *subkey = (wchar_t*)malloc(subkeySize * sizeof(wchar_t)); DWORD index = 0, subkeySize = maxSubkeySize;
while (RegEnumKeyExW(key, index++, subkey, &subkeySize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS) wchar_t *subkey = (wchar_t*)malloc(maxSubkeySize * sizeof(wchar_t));
while (subkey && (RegEnumKeyExW(key, index++, subkey, &subkeySize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS))
{
// Convert this string from wchar* to char*
size_t convertedSize;
subkeySize = maxSubkeySize;
if ((wcstombs_s(&convertedSize, NULL, 0, subkey, 0) == 0) && (convertedSize < 255))
{
char *subkeyString = (char*)malloc(convertedSize);
if (subkeyString && (wcstombs_s(NULL, subkeyString, convertedSize, subkey, convertedSize - 1) == 0))
{ {
// Determine if this device matches the specified serial number // Determine if this device matches the specified serial number
char *subkeyString = (char*)malloc(subkeySize + 2); if (serialNumber && strstr(subkeyString, serialNumber) && (wcscat_s(subkey, maxSubkeySize, L"\\0000\\Device Parameters") == 0))
memset(subkeyString, 0, subkeySize + 2);
wcstombs(subkeyString, subkey, subkeySize + 1);
subkeySize = 255;
if (strstr(subkeyString, serialNumber))
{ {
portNameSize = 16; DWORD portNameSize = portPathLength;
memset(portPath, 0, portNameSize * sizeof(wchar_t)); if ((RegOpenKeyExW(key, subkey, 0, KEY_QUERY_VALUE, &paramKey) == ERROR_SUCCESS) &&
wcscat_s(subkey, subkeySize, L"\\0000\\Device Parameters"); (RegGetValueW(paramKey, NULL, L"PortName", RRF_RT_REG_SZ, NULL, (LPBYTE)portPath, &portNameSize) == ERROR_SUCCESS))
if (RegOpenKeyExW(key, subkey, 0, KEY_QUERY_VALUE, &paramKey) == ERROR_SUCCESS)
{ {
if (RegQueryValueExW(paramKey, L"PortName", NULL, NULL, (LPBYTE)portPath, &portNameSize) == ERROR_SUCCESS)
found = 1; found = 1;
RegCloseKey(paramKey); RegCloseKey(paramKey);
} }
} }
}
if (subkeyString)
free(subkeyString); free(subkeyString);
} }
}
RegCloseKey(key); RegCloseKey(key);
if (subkey)
free(subkey); free(subkey);
} }
return found; return found;

View File

@ -2,7 +2,7 @@
* WindowsHelperFunctions.h * WindowsHelperFunctions.h
* *
* Created on: May 05, 2015 * Created on: May 05, 2015
* Last Updated on: Jan 28, 2022 * Last Updated on: Feb 17, 2022
* Author: Will Hedgecock * Author: Will Hedgecock
* *
* Copyright (C) 2012-2022 Fazecast, Inc. * Copyright (C) 2012-2022 Fazecast, Inc.
@ -52,6 +52,6 @@ void removePort(serialPortVector* vector, serialPort* port);
// Windows-specific functionality // Windows-specific functionality
void reduceLatencyToMinimum(const wchar_t* portName, unsigned char requestElevatedPermissions); void reduceLatencyToMinimum(const wchar_t* portName, unsigned char requestElevatedPermissions);
int getPortPathFromSerial(wchar_t* portPath, const char* serialNumber); int getPortPathFromSerial(wchar_t* portPath, int portPathLength, const char* serialNumber);
#endif // #ifndef __WINDOWS_HELPER_FUNCTIONS_HEADER_H__ #endif // #ifndef __WINDOWS_HELPER_FUNCTIONS_HEADER_H__