Use minimal amount of memory needed to enumerate in Windows
This commit is contained in:
parent
ea3c593776
commit
05aae9078a
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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, ¶mKey) == ERROR_SUCCESS)
|
if (RegOpenKeyExW(key, subkey, 0, KEY_QUERY_VALUE, ¶mKey) == 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, ¶mKey) == ERROR_SUCCESS)
|
if (RegOpenKeyExW(key, subkey, 0, KEY_SET_VALUE, ¶mKey) == 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, ¶mKey) == 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, ¶mKey) == 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;
|
||||||
|
|
|
@ -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__
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading…
Reference in New Issue