reactos/dll/win32/setupapi/cfgmgr.c
Serge Gautherie ffc99d08e8
[SETUPAPI] CMP_RegisterNotification(): Remove useless/broken check (#6210)
Revert new broken check from commit  b0a73746c,
and even remove initial a6eabc004 (r73394) useless check.
2023-12-23 21:49:47 +01:00

8510 lines
216 KiB
C

/*
* Configuration manager functions
*
* Copyright 2000 James Hatheway
* Copyright 2005, 2006 Eric Kohl
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include "setupapi_private.h"
#include <dbt.h>
#include <pnp_c.h>
#include <winsvc.h>
#include <pseh/pseh2.h>
#include "rpc_private.h"
DWORD
WINAPI
I_ScPnPGetServiceName(IN SERVICE_STATUS_HANDLE hServiceStatus,
OUT LPWSTR lpServiceName,
IN DWORD cchServiceName);
/* Registry key and value names */
static const WCHAR BackslashOpenBrace[] = {'\\', '{', 0};
static const WCHAR CloseBrace[] = {'}', 0};
static const WCHAR Class[] = {'C','l','a','s','s',0};
static const WCHAR ControlClass[] = {'S','y','s','t','e','m','\\',
'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\',
'C','o','n','t','r','o','l','\\',
'C','l','a','s','s',0};
static const WCHAR DeviceClasses[] = {'S','y','s','t','e','m','\\',
'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\',
'C','o','n','t','r','o','l','\\',
'D','e','v','i','c','e','C','l','a','s','s','e','s',0};
typedef struct _MACHINE_INFO
{
WCHAR szMachineName[SP_MAX_MACHINENAME_LENGTH];
RPC_BINDING_HANDLE BindingHandle;
HSTRING_TABLE StringTable;
BOOL bLocal;
} MACHINE_INFO, *PMACHINE_INFO;
typedef struct _LOG_CONF_INFO
{
ULONG ulMagic;
DEVINST dnDevInst;
ULONG ulType;
ULONG ulTag;
} LOG_CONF_INFO, *PLOG_CONF_INFO;
#define LOG_CONF_MAGIC 0x464E434C /* "LCNF" */
typedef struct _NOTIFY_DATA
{
ULONG ulMagic;
PVOID hNotifyHandle;
} NOTIFY_DATA, *PNOTIFY_DATA;
#define NOTIFY_MAGIC 0x44556677
typedef struct _INTERNAL_RANGE
{
LIST_ENTRY ListEntry;
struct _INTERNAL_RANGE_LIST *pRangeList;
DWORDLONG ullStart;
DWORDLONG ullEnd;
} INTERNAL_RANGE, *PINTERNAL_RANGE;
typedef struct _INTERNAL_RANGE_LIST
{
ULONG ulMagic;
HANDLE hMutex;
LIST_ENTRY ListHead;
} INTERNAL_RANGE_LIST, *PINTERNAL_RANGE_LIST;
#define RANGE_LIST_MAGIC 0x33445566
typedef struct _CONFLICT_DATA
{
ULONG ulMagic;
PPNP_CONFLICT_LIST pConflictList;
} CONFLICT_DATA, *PCONFLICT_DATA;
#define CONFLICT_MAGIC 0x11225588
/* FUNCTIONS ****************************************************************/
static
BOOL
GuidToString(
_In_ LPGUID Guid,
_Out_ LPWSTR String)
{
LPWSTR lpString;
if (UuidToStringW(Guid, &lpString) != RPC_S_OK)
return FALSE;
lstrcpyW(&String[1], lpString);
String[0] = '{';
String[MAX_GUID_STRING_LEN - 2] = '}';
String[MAX_GUID_STRING_LEN - 1] = UNICODE_NULL;
RpcStringFreeW(&lpString);
return TRUE;
}
static
CONFIGRET
RpcStatusToCmStatus(
_In_ RPC_STATUS Status)
{
return CR_FAILURE;
}
static
ULONG
GetRegistryPropertyType(
_In_ ULONG ulProperty)
{
switch (ulProperty)
{
case CM_DRP_DEVICEDESC:
case CM_DRP_SERVICE:
case CM_DRP_CLASS:
case CM_DRP_CLASSGUID:
case CM_DRP_DRIVER:
case CM_DRP_MFG:
case CM_DRP_FRIENDLYNAME:
case CM_DRP_LOCATION_INFORMATION:
case CM_DRP_PHYSICAL_DEVICE_OBJECT_NAME:
case CM_DRP_ENUMERATOR_NAME:
case CM_DRP_SECURITY_SDS:
case CM_DRP_UI_NUMBER_DESC_FORMAT:
return REG_SZ;
case CM_DRP_HARDWAREID:
case CM_DRP_COMPATIBLEIDS:
case CM_DRP_UPPERFILTERS:
case CM_DRP_LOWERFILTERS:
return REG_MULTI_SZ;
case CM_DRP_CONFIGFLAGS:
case CM_DRP_CAPABILITIES:
case CM_DRP_UI_NUMBER:
case CM_DRP_LEGACYBUSTYPE:
case CM_DRP_BUSNUMBER:
case CM_DRP_DEVTYPE:
case CM_DRP_EXCLUSIVE:
case CM_DRP_CHARACTERISTICS:
case CM_DRP_ADDRESS:
case CM_DRP_REMOVAL_POLICY:
case CM_DRP_REMOVAL_POLICY_HW_DEFAULT:
case CM_DRP_REMOVAL_POLICY_OVERRIDE:
case CM_DRP_INSTALL_STATE:
return REG_DWORD;
case CM_DRP_BUSTYPEGUID:
case CM_DRP_SECURITY:
case CM_DRP_DEVICE_POWER_DATA:
default:
return REG_BINARY;
}
return REG_NONE;
}
static
VOID
SplitDeviceInstanceId(
_In_ PWSTR pszDeviceInstanceId,
_Out_ PWSTR pszDeviceId,
_Out_ PWSTR pszInstanceId)
{
PWCHAR ptr;
wcscpy(pszDeviceId, pszDeviceInstanceId);
ptr = wcschr(pszDeviceId, L'\\');
if (ptr != NULL)
{
*ptr = UNICODE_NULL;
ptr++;
wcscpy(pszInstanceId, ptr);
}
else
{
*pszInstanceId = UNICODE_NULL;
}
}
static
CONFIGRET
GetDeviceInstanceKeyPath(
_In_ RPC_BINDING_HANDLE BindingHandle,
_In_ PWSTR pszDeviceInst,
_Out_ PWSTR pszKeyPath,
_Out_ PWSTR pszInstancePath,
_In_ ULONG ulHardwareProfile,
_In_ ULONG ulFlags)
{
PWSTR pszBuffer = NULL;
ULONG ulType = 0;
ULONG ulTransferLength, ulLength;
CONFIGRET ret = CR_SUCCESS;
TRACE("GetDeviceInstanceKeyPath()\n");
/* Allocate a buffer for the device id */
pszBuffer = MyMalloc(300 * sizeof(WCHAR));
if (pszBuffer == NULL)
{
ERR("MyMalloc() failed\n");
return CR_OUT_OF_MEMORY;
}
if (ulFlags & CM_REGISTRY_SOFTWARE)
{
/* Software Key Path */
ulTransferLength = 300 * sizeof(WCHAR);
ulLength = 300 * sizeof(WCHAR);
RpcTryExcept
{
ret = PNP_GetDeviceRegProp(BindingHandle,
pszDeviceInst,
CM_DRP_DRIVER,
&ulType,
(PVOID)pszBuffer,
&ulTransferLength,
&ulLength,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
{
RpcTryExcept
{
ret = PNP_GetClassInstance(BindingHandle,
pszDeviceInst,
(PVOID)pszBuffer,
300);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
{
goto done;
}
}
TRACE("szBuffer: %S\n", pszBuffer);
SplitDeviceInstanceId(pszBuffer,
pszBuffer,
pszInstancePath);
TRACE("szBuffer: %S\n", pszBuffer);
if (ulFlags & CM_REGISTRY_CONFIG)
{
if (ulHardwareProfile == 0)
{
wsprintfW(pszKeyPath,
L"%s\\%s\\%s\\%s",
L"System\\CurrentControlSet\\Hardware Profiles",
L"Current",
L"System\\CurrentControlSet\\Control\\Class",
pszBuffer);
}
else
{
wsprintfW(pszKeyPath,
L"%s\\%04lu\\%s\\%s",
L"System\\CurrentControlSet\\Hardware Profiles",
ulHardwareProfile,
L"System\\CurrentControlSet\\Control\\Class",
pszBuffer);
}
}
else
{
wsprintfW(pszKeyPath,
L"%s\\%s",
L"System\\CurrentControlSet\\Control\\Class",
pszBuffer);
}
}
else
{
/* Hardware Key Path */
if (ulFlags & CM_REGISTRY_CONFIG)
{
SplitDeviceInstanceId(pszDeviceInst,
pszBuffer,
pszInstancePath);
if (ulHardwareProfile == 0)
{
wsprintfW(pszKeyPath,
L"%s\\%s\\%s\\%s",
L"System\\CurrentControlSet\\Hardware Profiles",
L"Current",
L"System\\CurrentControlSet\\Enum",
pszBuffer);
}
else
{
wsprintfW(pszKeyPath,
L"%s\\%04lu\\%s\\%s",
L"System\\CurrentControlSet\\Hardware Profiles",
ulHardwareProfile,
L"System\\CurrentControlSet\\Enum",
pszBuffer);
}
}
else if (ulFlags & CM_REGISTRY_USER)
{
wsprintfW(pszKeyPath,
L"%s\\%s",
L"System\\CurrentControlSet\\Enum",
pszDeviceInst);
wcscpy(pszInstancePath,
L"Device Parameters");
}
else
{
SplitDeviceInstanceId(pszDeviceInst,
pszBuffer,
pszInstancePath);
wsprintfW(pszKeyPath,
L"%s\\%s",
L"System\\CurrentControlSet\\Enum",
pszBuffer);
}
}
done:
if (pszBuffer != NULL)
MyFree(pszBuffer);
return ret;
}
BOOL
IsValidRangeList(
_In_opt_ PINTERNAL_RANGE_LIST pRangeList)
{
BOOL bValid = TRUE;
if (pRangeList == NULL)
return FALSE;
_SEH2_TRY
{
if (pRangeList->ulMagic != RANGE_LIST_MAGIC)
bValid = FALSE;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
bValid = FALSE;
}
_SEH2_END;
return bValid;
}
BOOL
IsValidLogConf(
_In_opt_ PLOG_CONF_INFO pLogConfInfo)
{
BOOL bValid = TRUE;
if (pLogConfInfo == NULL)
return FALSE;
_SEH2_TRY
{
if (pLogConfInfo->ulMagic != LOG_CONF_MAGIC)
bValid = FALSE;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
bValid = FALSE;
}
_SEH2_END;
return bValid;
}
BOOL
IsValidConflictData(
_In_opt_ PCONFLICT_DATA pConflictData)
{
BOOL bValid = TRUE;
if (pConflictData == NULL)
return FALSE;
_SEH2_TRY
{
if (pConflictData->ulMagic != CONFLICT_MAGIC)
bValid = FALSE;
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
bValid = FALSE;
}
_SEH2_END;
return bValid;
}
/***********************************************************************
* CMP_GetBlockedDriverInfo [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_GetBlockedDriverInfo(
_Out_opt_ LPWSTR pszNames,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
ULONG ulTransferLength;
CONFIGRET ret;
TRACE("CMP_GetBlockedDriverInfo(%p %p %lx %p)\n",
pszNames, pulLength, ulFlags, hMachine);
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
ulTransferLength = *pulLength;
RpcTryExcept
{
ret = PNP_GetBlockedDriverInfo(BindingHandle,
(PBYTE)pszNames,
&ulTransferLength,
pulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CMP_GetServerSideDeviceInstallFlags [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_GetServerSideDeviceInstallFlags(
_Out_ PULONG pulSSDIFlags,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CMP_GetServerSideDeviceInstallFlags(%p %lx %p)\n",
pulSSDIFlags, ulFlags, hMachine);
if (pulSSDIFlags == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_GetServerSideDeviceInstallFlags(BindingHandle,
pulSSDIFlags,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CMP_Init_Detection [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_Init_Detection(
_In_ ULONG ulMagic)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CMP_Init_Detection(%lu)\n", ulMagic);
if (ulMagic != CMP_MAGIC)
return CR_INVALID_DATA;
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
RpcTryExcept
{
ret = PNP_InitDetection(BindingHandle);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CMP_RegisterNotification [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_RegisterNotification(
_In_ HANDLE hRecipient,
_In_ LPVOID lpvNotificationFilter,
_In_ ULONG ulFlags,
_Out_ PHDEVNOTIFY phDevNotify)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
PNOTIFY_DATA pNotifyData;
WCHAR szNameBuffer[256];
INT nLength;
DWORD ulUnknown9 = 0;
DWORD dwError;
CONFIGRET ret = CR_SUCCESS;
FIXME("CMP_RegisterNotification(%p %p %lu %p)\n",
hRecipient, lpvNotificationFilter, ulFlags, phDevNotify);
if ((hRecipient == NULL) ||
(lpvNotificationFilter == NULL) ||
(phDevNotify == NULL))
return CR_INVALID_POINTER;
if (ulFlags & ~0x7)
return CR_INVALID_FLAG;
if (((PDEV_BROADCAST_HDR)lpvNotificationFilter)->dbch_size < sizeof(DEV_BROADCAST_HDR))
return CR_INVALID_DATA;
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
pNotifyData = HeapAlloc(GetProcessHeap(),
HEAP_ZERO_MEMORY,
sizeof(NOTIFY_DATA));
if (pNotifyData == NULL)
return CR_OUT_OF_MEMORY;
pNotifyData->ulMagic = NOTIFY_MAGIC;
pNotifyData->hNotifyHandle = NULL;
ZeroMemory(szNameBuffer, sizeof(szNameBuffer));
if ((ulFlags & DEVICE_NOTIFY_SERVICE_HANDLE) == DEVICE_NOTIFY_WINDOW_HANDLE)
{
FIXME("Register a window\n");
nLength = GetWindowTextW((HWND)hRecipient,
szNameBuffer,
ARRAYSIZE(szNameBuffer));
if (nLength == 0)
{
szNameBuffer[0] = UNICODE_NULL;
}
FIXME("Register window: %S\n", szNameBuffer);
}
else if ((ulFlags & DEVICE_NOTIFY_SERVICE_HANDLE) == DEVICE_NOTIFY_SERVICE_HANDLE)
{
FIXME("Register a service\n");
dwError = I_ScPnPGetServiceName((SERVICE_STATUS_HANDLE)hRecipient,
szNameBuffer,
ARRAYSIZE(szNameBuffer));
if (dwError != ERROR_SUCCESS)
{
HeapFree(GetProcessHeap(), 0, pNotifyData);
return CR_INVALID_DATA;
}
FIXME("Register service: %S\n", szNameBuffer);
}
RpcTryExcept
{
ret = PNP_RegisterNotification(BindingHandle,
(DWORD_PTR)hRecipient,
szNameBuffer,
(BYTE*)lpvNotificationFilter,
((DEV_BROADCAST_HDR*)lpvNotificationFilter)->dbch_size,
ulFlags,
&pNotifyData->hNotifyHandle,
GetCurrentProcessId(),
&ulUnknown9); /* ??? */
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
TRACE("hNotifyHandle: %p\n", pNotifyData->hNotifyHandle);
*phDevNotify = (HDEVNOTIFY)pNotifyData;
}
else
{
HeapFree(GetProcessHeap(), 0, pNotifyData);
*phDevNotify = (HDEVNOTIFY)NULL;
}
return ret;
}
/***********************************************************************
* CMP_Report_LogOn [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_Report_LogOn(
_In_ DWORD dwMagic,
_In_ DWORD dwProcessId)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret = CR_SUCCESS;
BOOL bAdmin;
DWORD i;
TRACE("CMP_Report_LogOn(%lu %lu)\n", dwMagic, dwProcessId);
if (dwMagic != CMP_MAGIC)
return CR_INVALID_DATA;
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
bAdmin = pSetupIsUserAdmin();
for (i = 0; i < 30; i++)
{
RpcTryExcept
{
ret = PNP_ReportLogOn(BindingHandle,
bAdmin,
dwProcessId);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
break;
Sleep(5000);
}
return ret;
}
/***********************************************************************
* CMP_UnregisterNotification [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_UnregisterNotification(
_In_ HDEVNOTIFY hDevNotify)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
PNOTIFY_DATA pNotifyData;
CONFIGRET ret = CR_SUCCESS;
TRACE("CMP_UnregisterNotification(%p)\n", hDevNotify);
pNotifyData = (PNOTIFY_DATA)hDevNotify;
if ((pNotifyData == NULL) ||
(pNotifyData->ulMagic != NOTIFY_MAGIC))
return CR_INVALID_POINTER;
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
RpcTryExcept
{
ret = PNP_UnregisterNotification(BindingHandle,
&pNotifyData->hNotifyHandle);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
pNotifyData->hNotifyHandle = NULL;
HeapFree(GetProcessHeap(), 0, pNotifyData);
}
return ret;
}
/***********************************************************************
* CMP_WaitNoPendingInstallEvents [SETUPAPI.@]
*/
DWORD
WINAPI
CMP_WaitNoPendingInstallEvents(
_In_ DWORD dwTimeout)
{
HANDLE hEvent;
DWORD ret;
TRACE("CMP_WaitNoPendingInstallEvents(%lu)\n", dwTimeout);
hEvent = OpenEventW(SYNCHRONIZE, FALSE, L"Global\\PnP_No_Pending_Install_Events");
if (hEvent == NULL)
return WAIT_FAILED;
ret = WaitForSingleObject(hEvent, dwTimeout);
CloseHandle(hEvent);
return ret;
}
/***********************************************************************
* CMP_WaitServicesAvailable [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CMP_WaitServicesAvailable(
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret = CR_SUCCESS;
WORD Version;
TRACE("CMP_WaitServicesAvailable(%p)\n", hMachine);
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_GetVersion(BindingHandle, &Version);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Add_Empty_Log_Conf [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_Empty_Log_Conf(
_Out_ PLOG_CONF plcLogConf,
_In_ DEVINST dnDevInst,
_In_ PRIORITY Priority,
_In_ ULONG ulFlags)
{
TRACE("CM_Add_Empty_Log_Conf(%p %p %lu %lx)\n",
plcLogConf, dnDevInst, Priority, ulFlags);
return CM_Add_Empty_Log_Conf_Ex(plcLogConf, dnDevInst, Priority,
ulFlags, NULL);
}
/***********************************************************************
* CM_Add_Empty_Log_Conf_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_Empty_Log_Conf_Ex(
_Out_ PLOG_CONF plcLogConf,
_In_ DEVINST dnDevInst,
_In_ PRIORITY Priority,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
ULONG ulLogConfTag = 0;
LPWSTR lpDevInst;
PLOG_CONF_INFO pLogConfInfo;
CONFIGRET ret = CR_SUCCESS;
FIXME("CM_Add_Empty_Log_Conf_Ex(%p %p %lu %lx %p)\n",
plcLogConf, dnDevInst, Priority, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (plcLogConf == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (Priority > 0xFFFF)
return CR_INVALID_PRIORITY;
if (ulFlags & ~(LOG_CONF_BITS | PRIORITY_BIT))
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_AddEmptyLogConf(BindingHandle, lpDevInst, Priority,
&ulLogConfTag, ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
pLogConfInfo = HeapAlloc(GetProcessHeap(), 0, sizeof(LOG_CONF_INFO));
if (pLogConfInfo == NULL)
{
ret = CR_OUT_OF_MEMORY;
}
else
{
pLogConfInfo->ulMagic = LOG_CONF_MAGIC;
pLogConfInfo->dnDevInst = dnDevInst;
pLogConfInfo->ulType = ulFlags;
pLogConfInfo->ulTag = ulLogConfTag;
*plcLogConf = (LOG_CONF)pLogConfInfo;
ret = CR_SUCCESS;
}
}
return ret;
}
/***********************************************************************
* CM_Add_IDA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_IDA(
_In_ DEVINST dnDevInst,
_In_ PSTR pszID,
_In_ ULONG ulFlags)
{
TRACE("CM_Add_IDA(%p %s %lx)\n",
dnDevInst, debugstr_a(pszID), ulFlags);
return CM_Add_ID_ExA(dnDevInst, pszID, ulFlags, NULL);
}
/***********************************************************************
* CM_Add_IDW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_IDW(
_In_ DEVINST dnDevInst,
_In_ PWSTR pszID,
_In_ ULONG ulFlags)
{
TRACE("CM_Add_IDW(%p %s %lx)\n",
dnDevInst, debugstr_w(pszID), ulFlags);
return CM_Add_ID_ExW(dnDevInst, pszID, ulFlags, NULL);
}
/***********************************************************************
* CM_Add_ID_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_ID_ExA(
_In_ DEVINST dnDevInst,
_In_ PSTR pszID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
PWSTR pszIDW;
CONFIGRET ret;
TRACE("CM_Add_ID_ExA(%p %s %lx %p)\n",
dnDevInst, debugstr_a(pszID), ulFlags, hMachine);
if (pSetupCaptureAndConvertAnsiArg(pszID, &pszIDW))
return CR_INVALID_DATA;
ret = CM_Add_ID_ExW(dnDevInst, pszIDW, ulFlags, hMachine);
MyFree(pszIDW);
return ret;
}
/***********************************************************************
* CM_Add_ID_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_ID_ExW(
_In_ DEVINST dnDevInst,
_In_ PWSTR pszID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Add_ID_ExW(%p %s %lx %p)\n",
dnDevInst, debugstr_w(pszID), ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (pszID == NULL)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_ADD_ID_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_AddID(BindingHandle,
lpDevInst,
pszID,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Add_Range [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_Range(
_In_ DWORDLONG ullStartValue,
_In_ DWORDLONG ullEndValue,
_In_ RANGE_LIST rlh,
_In_ ULONG ulFlags)
{
PINTERNAL_RANGE_LIST pRangeList;
PINTERNAL_RANGE pRange;
CONFIGRET ret = CR_SUCCESS;
FIXME("CM_Add_Range(%I64u %I64u %p %lx)\n",
ullStartValue, ullEndValue, rlh, ulFlags);
pRangeList = (PINTERNAL_RANGE_LIST)rlh;
if (!IsValidRangeList(pRangeList))
return CR_INVALID_RANGE_LIST;
if (ulFlags & ~CM_ADD_RANGE_BITS)
return CR_INVALID_FLAG;
if (ullEndValue < ullStartValue)
return CR_INVALID_RANGE;
/* Lock the range list */
WaitForSingleObject(pRangeList->hMutex, INFINITE);
/* Allocate the new range */
pRange = HeapAlloc(GetProcessHeap(), 0, sizeof(INTERNAL_RANGE));
if (pRange == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
pRange->pRangeList = pRangeList;
pRange->ullStart = ullStartValue;
pRange->ullEnd = ullEndValue;
/* Insert the range */
if (IsListEmpty(&pRangeList->ListHead))
{
InsertTailList(&pRangeList->ListHead, &pRange->ListEntry);
}
else
{
HeapFree(GetProcessHeap(), 0, pRange);
UNIMPLEMENTED;
}
done:
/* Unlock the range list */
ReleaseMutex(pRangeList->hMutex);
return ret;
}
/***********************************************************************
* CM_Add_Res_Des [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_Res_Des(
_Out_opt_ PRES_DES prdResDes,
_In_ LOG_CONF lcLogConf,
_In_ RESOURCEID ResourceID,
_In_reads_bytes_(ResourceLen) PCVOID ResourceData,
_In_ ULONG ResourceLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Add_Res_Des(%p %p %lu %p %lu %lx)\n",
prdResDes, lcLogConf, ResourceID, ResourceData, ResourceLen, ulFlags);
return CM_Add_Res_Des_Ex(prdResDes, lcLogConf, ResourceID, ResourceData,
ResourceLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Add_Res_Des_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Add_Res_Des_Ex(
_Out_opt_ PRES_DES prdResDes,
_In_ LOG_CONF lcLogConf,
_In_ RESOURCEID ResourceID,
_In_reads_bytes_(ResourceLen) PCVOID ResourceData,
_In_ ULONG ResourceLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Add_Res_Des_Ex(%p %p %lu %p %lu %lx %p)\n",
prdResDes, lcLogConf, ResourceID,
ResourceData, ResourceLen, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Connect_MachineA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Connect_MachineA(
_In_opt_ PCSTR UNCServerName,
_Out_ PHMACHINE phMachine)
{
PWSTR pServerNameW;
CONFIGRET ret;
TRACE("CM_Connect_MachineA(%s %p)\n",
debugstr_a(UNCServerName), phMachine);
if (UNCServerName == NULL || *UNCServerName == 0)
return CM_Connect_MachineW(NULL, phMachine);
if (pSetupCaptureAndConvertAnsiArg(UNCServerName, &pServerNameW))
return CR_INVALID_DATA;
ret = CM_Connect_MachineW(pServerNameW, phMachine);
MyFree(pServerNameW);
return ret;
}
/***********************************************************************
* CM_Connect_MachineW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Connect_MachineW(
_In_opt_ PCWSTR UNCServerName,
_Out_ PHMACHINE phMachine)
{
PMACHINE_INFO pMachine;
TRACE("CM_Connect_MachineW(%s %p)\n",
debugstr_w(UNCServerName), phMachine);
if (phMachine == NULL)
return CR_INVALID_POINTER;
*phMachine = NULL;
pMachine = HeapAlloc(GetProcessHeap(), 0, sizeof(MACHINE_INFO));
if (pMachine == NULL)
return CR_OUT_OF_MEMORY;
if (UNCServerName == NULL || *UNCServerName == 0)
{
pMachine->bLocal = TRUE;
/* FIXME: store the computers name in pMachine->szMachineName */
if (!PnpGetLocalHandles(&pMachine->BindingHandle,
&pMachine->StringTable))
{
HeapFree(GetProcessHeap(), 0, pMachine);
return CR_FAILURE;
}
}
else
{
pMachine->bLocal = FALSE;
if (wcslen(UNCServerName) >= SP_MAX_MACHINENAME_LENGTH - 1)
{
HeapFree(GetProcessHeap(), 0, pMachine);
return CR_INVALID_MACHINENAME;
}
lstrcpyW(pMachine->szMachineName, UNCServerName);
pMachine->StringTable = pSetupStringTableInitialize();
if (pMachine->StringTable == NULL)
{
HeapFree(GetProcessHeap(), 0, pMachine);
return CR_FAILURE;
}
pSetupStringTableAddString(pMachine->StringTable, L"PLT", 1);
if (!PnpBindRpc(UNCServerName, &pMachine->BindingHandle))
{
pSetupStringTableDestroy(pMachine->StringTable);
HeapFree(GetProcessHeap(), 0, pMachine);
return CR_INVALID_MACHINENAME;
}
}
*phMachine = (PHMACHINE)pMachine;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Create_DevNodeA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Create_DevNodeA(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINSTID_A pDeviceID,
_In_ DEVINST dnParent,
_In_ ULONG ulFlags)
{
TRACE("CM_Create_DevNodeA(%p %s %p %lx)\n",
pdnDevInst, debugstr_a(pDeviceID), dnParent, ulFlags);
return CM_Create_DevNode_ExA(pdnDevInst, pDeviceID, dnParent,
ulFlags, NULL);
}
/***********************************************************************
* CM_Create_DevNodeW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Create_DevNodeW(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINSTID_W pDeviceID,
_In_ DEVINST dnParent,
_In_ ULONG ulFlags)
{
TRACE("CM_Create_DevNodeW(%p %s %p %lx)\n",
pdnDevInst, debugstr_w(pDeviceID), dnParent, ulFlags);
return CM_Create_DevNode_ExW(pdnDevInst, pDeviceID, dnParent,
ulFlags, NULL);
}
/***********************************************************************
* CM_Create_DevNode_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Create_DevNode_ExA(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINSTID_A pDeviceID,
_In_ DEVINST dnParent,
_In_ ULONG ulFlags,
_In_opt_ HANDLE hMachine)
{
DEVINSTID_W pDeviceIDW;
CONFIGRET ret;
TRACE("CM_Create_DevNode_ExA(%p %s %p %lx %p)\n",
pdnDevInst, debugstr_a(pDeviceID), dnParent, ulFlags, hMachine);
if (pSetupCaptureAndConvertAnsiArg(pDeviceID, &pDeviceIDW))
return CR_INVALID_DATA;
ret = CM_Create_DevNode_ExW(pdnDevInst, pDeviceIDW, dnParent, ulFlags,
hMachine);
MyFree(pDeviceIDW);
return ret;
}
/***********************************************************************
* CM_Create_DevNode_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Create_DevNode_ExW(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINSTID_W pDeviceID,
_In_ DEVINST dnParent,
_In_ ULONG ulFlags,
_In_opt_ HANDLE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpParentDevInst;
CONFIGRET ret = CR_SUCCESS;
WCHAR szLocalDeviceID[MAX_DEVICE_ID_LEN];
TRACE("CM_Create_DevNode_ExW(%p %s %p %lx %p)\n",
pdnDevInst, debugstr_w(pDeviceID), dnParent, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (pdnDevInst == NULL)
return CR_INVALID_POINTER;
if (pDeviceID == NULL || wcslen(pDeviceID) == 0 || wcslen(pDeviceID) >= MAX_DEVICE_ID_LEN)
return CR_INVALID_DEVICE_ID;
if (dnParent == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_CREATE_DEVNODE_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpParentDevInst = pSetupStringTableStringFromId(StringTable, dnParent);
if (lpParentDevInst == NULL)
return CR_INVALID_DEVNODE;
wcscpy(szLocalDeviceID, pDeviceID);
RpcTryExcept
{
ret = PNP_CreateDevInst(BindingHandle,
szLocalDeviceID,
lpParentDevInst,
MAX_DEVICE_ID_LEN,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
/* If CM_CREATE_DEVINST_GENERATE_ID was passed in, PNP_CreateDevInst
* will return the generated device ID in szLocalDeviceID */
*pdnDevInst = pSetupStringTableAddString(StringTable, szLocalDeviceID, 1);
if (*pdnDevInst == 0)
ret = CR_NO_SUCH_DEVNODE;
}
return ret;
}
/***********************************************************************
* CM_Create_Range_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Create_Range_List(
_Out_ PRANGE_LIST prlh,
_In_ ULONG ulFlags)
{
PINTERNAL_RANGE_LIST pRangeList;
FIXME("CM_Create_Range_List(%p %lx)\n",
prlh, ulFlags);
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (prlh == NULL)
return CR_INVALID_POINTER;
/* Allocate the range list */
pRangeList = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(INTERNAL_RANGE_LIST));
if (pRangeList == NULL)
return CR_OUT_OF_MEMORY;
/* Set the magic value */
pRangeList->ulMagic = RANGE_LIST_MAGIC;
/* Initialize the mutex for synchonized access */
pRangeList->hMutex = CreateMutex(NULL, FALSE, NULL);
if (pRangeList->hMutex == NULL)
{
HeapFree(GetProcessHeap(), 0, pRangeList);
return CR_FAILURE;
}
InitializeListHead(&pRangeList->ListHead);
*prlh = (RANGE_LIST)pRangeList;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Delete_Class_Key [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Delete_Class_Key(
_In_ LPGUID ClassGuid,
_In_ ULONG ulFlags)
{
TRACE("CM_Delete_Class_Key(%p %lx)\n",
ClassGuid, ulFlags);
return CM_Delete_Class_Key_Ex(ClassGuid, ulFlags, NULL);
}
/***********************************************************************
* CM_Delete_Class_Key_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Delete_Class_Key_Ex(
_In_ LPGUID ClassGuid,
_In_ ULONG ulFlags,
_In_opt_ HANDLE hMachine)
{
WCHAR szGuidString[MAX_GUID_STRING_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Delete_Class_Key_Ex(%p %lx %p)\n",
ClassGuid, ulFlags, hMachine);
if (ClassGuid == NULL)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_DELETE_CLASS_BITS)
return CR_INVALID_FLAG;
if (!GuidToString(ClassGuid, szGuidString))
return CR_INVALID_DATA;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_DeleteClassKey(BindingHandle,
szGuidString,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Delete_DevNode_Key [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Delete_DevNode_Key(
_In_ DEVINST dnDevInst,
_In_ ULONG ulHardwareProfile,
_In_ ULONG ulFlags)
{
TRACE("CM_Delete_DevNode_Key(%p %lu %lx)\n",
dnDevInst, ulHardwareProfile, ulFlags);
return CM_Delete_DevNode_Key_Ex(dnDevInst, ulHardwareProfile, ulFlags,
NULL);
}
/***********************************************************************
* CM_Delete_DevNode_Key_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Delete_DevNode_Key_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulHardwareProfile,
_In_ ULONG ulFlags,
_In_opt_ HANDLE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
PWSTR pszDevInst, pszKeyPath = NULL, pszInstancePath = NULL;
CONFIGRET ret;
FIXME("CM_Delete_DevNode_Key_Ex(%p %lu %lx %p)\n",
dnDevInst, ulHardwareProfile, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags & ~CM_REGISTRY_BITS)
return CR_INVALID_FLAG;
if ((ulFlags & CM_REGISTRY_USER) && (ulFlags & CM_REGISTRY_CONFIG))
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
pszDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (pszDevInst == NULL)
return CR_INVALID_DEVNODE;
TRACE("pszDevInst: %S\n", pszDevInst);
pszKeyPath = MyMalloc(512 * sizeof(WCHAR));
if (pszKeyPath == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
pszInstancePath = MyMalloc(512 * sizeof(WCHAR));
if (pszInstancePath == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
ret = GetDeviceInstanceKeyPath(BindingHandle,
pszDevInst,
pszKeyPath,
pszInstancePath,
ulHardwareProfile,
ulFlags);
if (ret != CR_SUCCESS)
goto done;
TRACE("pszKeyPath: %S\n", pszKeyPath);
TRACE("pszInstancePath: %S\n", pszInstancePath);
if (ulFlags & CM_REGISTRY_USER)
{
FIXME("The CM_REGISTRY_USER flag is not supported yet!\n");
}
else
{
#if 0
if (!pSetupIsUserAdmin())
{
ret = CR_ACCESS_DENIED;
goto done;
}
#endif
if (!(ulFlags & CM_REGISTRY_CONFIG))
ulHardwareProfile = 0;
RpcTryExcept
{
ret = PNP_DeleteRegistryKey(BindingHandle,
pszDevInst,
pszKeyPath,
pszInstancePath,
ulHardwareProfile);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
}
done:
if (pszInstancePath != NULL)
MyFree(pszInstancePath);
if (pszKeyPath != NULL)
MyFree(pszKeyPath);
return ret;
}
/***********************************************************************
* CM_Delete_Range [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Delete_Range(
_In_ DWORDLONG ullStartValue,
_In_ DWORDLONG ullEndValue,
_In_ RANGE_LIST rlh,
_In_ ULONG ulFlags)
{
FIXME("CM_Delete_Range(%I64u %I64u %p %lx)\n",
ullStartValue, ullEndValue, rlh, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Detect_Resource_Conflict [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Detect_Resource_Conflict(
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_reads_bytes_(ResourceLen) PCVOID ResourceData,
_In_ ULONG ResourceLen,
_Out_ PBOOL pbConflictDetected,
_In_ ULONG ulFlags)
{
TRACE("CM_Detect_Resource_Conflict(%p %lu %p %lu %p 0x%lx)\n",
dnDevInst, ResourceID, ResourceData, ResourceLen,
pbConflictDetected, ulFlags);
return CM_Detect_Resource_Conflict_Ex(dnDevInst,
ResourceID,
ResourceData,
ResourceLen,
pbConflictDetected,
ulFlags,
NULL);
}
/***********************************************************************
* CM_Detect_Resource_Conflict_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Detect_Resource_Conflict_Ex(
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_reads_bytes_(ResourceLen) PCVOID ResourceData,
_In_ ULONG ResourceLen,
_Out_ PBOOL pbConflictDetected,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Detect_Resource_Conflict_Ex(%p %lu %p %lu %p 0x%lx %p)\n",
dnDevInst, ResourceID, ResourceData, ResourceLen,
pbConflictDetected, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Disable_DevNode [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Disable_DevNode(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Disable_DevNode(%p %lx)\n",
dnDevInst, ulFlags);
return CM_Disable_DevNode_Ex(dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Disable_DevNode_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Disable_DevNode_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Disable_DevNode_Ex(%p %lx %p)\n",
dnDevInst, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_DisableDevInst(BindingHandle,
lpDevInst,
NULL,
NULL,
0,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Disconnect_Machine [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Disconnect_Machine(
_In_opt_ HMACHINE hMachine)
{
PMACHINE_INFO pMachine;
TRACE("CM_Disconnect_Machine(%p)\n", hMachine);
pMachine = (PMACHINE_INFO)hMachine;
if (pMachine == NULL)
return CR_SUCCESS;
if (pMachine->bLocal == FALSE)
{
if (pMachine->StringTable != NULL)
pSetupStringTableDestroy(pMachine->StringTable);
if (!PnpUnbindRpc(pMachine->BindingHandle))
return CR_ACCESS_DENIED;
}
HeapFree(GetProcessHeap(), 0, pMachine);
return CR_SUCCESS;
}
/***********************************************************************
* CM_Dup_Range_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Dup_Range_List(
_In_ RANGE_LIST rlhOld,
_In_ RANGE_LIST rlhNew,
_In_ ULONG ulFlags)
{
FIXME("CM_Dup_Range_List(%p %p %lx)\n",
rlhOld, rlhNew, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Enable_DevNode [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enable_DevNode(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Enable_DevNode(%p %lx)\n",
dnDevInst, ulFlags);
return CM_Enable_DevNode_Ex(dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Enable_DevNode_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enable_DevNode_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Enable_DevNode_Ex(%p %lx %p)\n",
dnDevInst, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_DeviceInstanceAction(BindingHandle,
PNP_DEVINST_ENABLE,
ulFlags,
lpDevInst,
NULL);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Enumerate_Classes [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enumerate_Classes(
_In_ ULONG ulClassIndex,
_Out_ LPGUID ClassGuid,
_In_ ULONG ulFlags)
{
TRACE("CM_Enumerate_Classes(%lx %p %lx)\n",
ulClassIndex, ClassGuid, ulFlags);
return CM_Enumerate_Classes_Ex(ulClassIndex, ClassGuid, ulFlags, NULL);
}
/***********************************************************************
* CM_Enumerate_Classes_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enumerate_Classes_Ex(
_In_ ULONG ulClassIndex,
_Out_ LPGUID ClassGuid,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szBuffer[MAX_GUID_STRING_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret = CR_SUCCESS;
ULONG ulLength = MAX_GUID_STRING_LEN;
TRACE("CM_Enumerate_Classes_Ex(%lx %p %lx %p)\n",
ulClassIndex, ClassGuid, ulFlags, hMachine);
if (ClassGuid == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_EnumerateSubKeys(BindingHandle,
PNP_CLASS_SUBKEYS,
ulClassIndex,
szBuffer,
MAX_GUID_STRING_LEN,
&ulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
/* Remove the {} */
szBuffer[MAX_GUID_STRING_LEN - 2] = UNICODE_NULL;
/* Convert the buffer to a GUID */
if (UuidFromStringW(&szBuffer[1], ClassGuid) != RPC_S_OK)
return CR_FAILURE;
}
return ret;
}
/***********************************************************************
* CM_Enumerate_EnumeratorsA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enumerate_EnumeratorsA(
_In_ ULONG ulEnumIndex,
_Out_writes_(*pulLength) PCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Enumerate_EnumeratorsA(%lu %p %p %lx)\n",
ulEnumIndex, Buffer, pulLength, ulFlags);
return CM_Enumerate_Enumerators_ExA(ulEnumIndex, Buffer, pulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Enumerate_EnumeratorsW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enumerate_EnumeratorsW(
_In_ ULONG ulEnumIndex,
_Out_writes_(*pulLength) PWCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Enumerate_EnumeratorsW(%lu %p %p %lx)\n",
ulEnumIndex, Buffer, pulLength, ulFlags);
return CM_Enumerate_Enumerators_ExW(ulEnumIndex, Buffer, pulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Enumerate_Enumerators_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enumerate_Enumerators_ExA(
_In_ ULONG ulEnumIndex,
_Out_writes_(*pulLength) PCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szBuffer[MAX_DEVICE_ID_LEN];
ULONG ulOrigLength;
ULONG ulLength;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Enumerate_Enumerators_ExA(%lu %p %p %lx %p)\n",
ulEnumIndex, Buffer, pulLength, ulFlags, hMachine);
if (Buffer == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
ulOrigLength = *pulLength;
*pulLength = 0;
ulLength = MAX_DEVICE_ID_LEN;
ret = CM_Enumerate_Enumerators_ExW(ulEnumIndex, szBuffer, &ulLength,
ulFlags, hMachine);
if (ret == CR_SUCCESS)
{
if (WideCharToMultiByte(CP_ACP,
0,
szBuffer,
ulLength,
Buffer,
ulOrigLength,
NULL,
NULL) == 0)
ret = CR_FAILURE;
else
*pulLength = lstrlenA(Buffer) + 1;
}
return ret;
}
/***********************************************************************
* CM_Enumerate_Enumerators_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Enumerate_Enumerators_ExW(
_In_ ULONG ulEnumIndex,
_Out_writes_(*pulLength) PWCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Enumerate_Enumerators_ExW(%lu %p %p %lx %p)\n",
ulEnumIndex, Buffer, pulLength, ulFlags, hMachine);
if (Buffer == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
*Buffer = UNICODE_NULL;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_EnumerateSubKeys(BindingHandle,
PNP_ENUMERATOR_SUBKEYS,
ulEnumIndex,
Buffer,
*pulLength,
pulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Find_Range [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Find_Range(
_Out_ PDWORDLONG pullStart,
_In_ DWORDLONG ullStart,
_In_ ULONG ulLength,
_In_ DWORDLONG ullAlignment,
_In_ DWORDLONG ullEnd,
_In_ RANGE_LIST rlh,
_In_ ULONG ulFlags)
{
FIXME("CM_Find_Range(%p %I64u %lu %I64u %I64u %p %lx)\n",
pullStart, ullStart, ulLength, ullAlignment, ullEnd, rlh, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_First_Range [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_First_Range(
_In_ RANGE_LIST rlh,
_Out_ PDWORDLONG pullStart,
_Out_ PDWORDLONG pullEnd,
_Out_ PRANGE_ELEMENT preElement,
_In_ ULONG ulFlags)
{
PINTERNAL_RANGE_LIST pRangeList;
PINTERNAL_RANGE pRange;
PLIST_ENTRY ListEntry;
CONFIGRET ret = CR_SUCCESS;
FIXME("CM_First_Range(%p %p %p %p %lx)\n",
rlh, pullStart, pullEnd, preElement, ulFlags);
pRangeList = (PINTERNAL_RANGE_LIST)rlh;
if (!IsValidRangeList(pRangeList))
return CR_INVALID_RANGE_LIST;
if (pullStart == NULL || pullEnd == NULL || preElement == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
/* Lock the range list */
WaitForSingleObject(pRangeList->hMutex, INFINITE);
/* Fail, if the list is empty */
if (IsListEmpty(&pRangeList->ListHead))
{
ret = CR_FAILURE;
goto done;
}
/* Get the first range */
ListEntry = pRangeList->ListHead.Flink;
pRange = CONTAINING_RECORD(ListEntry, INTERNAL_RANGE, ListEntry);
/* Return the range data */
*pullStart = pRange->ullStart;
*pullEnd = pRange->ullEnd;
*preElement = (RANGE_ELEMENT)pRange;
done:
/* Unlock the range list */
ReleaseMutex(pRangeList->hMutex);
return ret;
}
/***********************************************************************
* CM_Free_Log_Conf [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Log_Conf(
_In_ LOG_CONF lcLogConfToBeFreed,
_In_ ULONG ulFlags)
{
TRACE("CM_Free_Log_Conf(%lx %lx)\n",
lcLogConfToBeFreed, ulFlags);
return CM_Free_Log_Conf_Ex(lcLogConfToBeFreed, ulFlags, NULL);
}
/***********************************************************************
* CM_Free_Log_Conf_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Log_Conf_Ex(
_In_ LOG_CONF lcLogConfToBeFreed,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
PLOG_CONF_INFO pLogConfInfo;
CONFIGRET ret;
TRACE("CM_Free_Log_Conf_Ex(%lx %lx %p)\n",
lcLogConfToBeFreed, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
pLogConfInfo = (PLOG_CONF_INFO)lcLogConfToBeFreed;
if (!IsValidLogConf(pLogConfInfo))
return CR_INVALID_LOG_CONF;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, pLogConfInfo->dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_FreeLogConf(BindingHandle,
lpDevInst,
pLogConfInfo->ulType,
pLogConfInfo->ulTag,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Free_Log_Conf_Handle [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Log_Conf_Handle(
_In_ LOG_CONF lcLogConf)
{
PLOG_CONF_INFO pLogConfInfo;
TRACE("CM_Free_Log_Conf_Handle(%lx)\n", lcLogConf);
pLogConfInfo = (PLOG_CONF_INFO)lcLogConf;
if (!IsValidLogConf(pLogConfInfo))
return CR_INVALID_LOG_CONF;
HeapFree(GetProcessHeap(), 0, pLogConfInfo);
return CR_SUCCESS;
}
/***********************************************************************
* CM_Free_Range_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Range_List(
_In_ RANGE_LIST RangeList,
_In_ ULONG ulFlags)
{
PINTERNAL_RANGE_LIST pRangeList;
PINTERNAL_RANGE pRange;
PLIST_ENTRY ListEntry;
FIXME("CM_Free_Range_List(%p %lx)\n",
RangeList, ulFlags);
pRangeList = (PINTERNAL_RANGE_LIST)RangeList;
if (!IsValidRangeList(pRangeList))
return CR_INVALID_RANGE_LIST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
/* Lock the range list */
WaitForSingleObject(pRangeList->hMutex, INFINITE);
/* Free the list of ranges */
while (!IsListEmpty(&pRangeList->ListHead))
{
ListEntry = RemoveHeadList(&pRangeList->ListHead);
pRange = CONTAINING_RECORD(ListEntry, INTERNAL_RANGE, ListEntry);
HeapFree(GetProcessHeap(), 0, pRange);
}
/* Unlock the range list */
ReleaseMutex(pRangeList->hMutex);
/* Close the mutex */
CloseHandle(pRangeList->hMutex);
/* Free the range list */
HeapFree(GetProcessHeap(), 0, pRangeList);
return CR_SUCCESS;
}
/***********************************************************************
* CM_Free_Res_Des [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Res_Des(
_Out_ PRES_DES prdResDes,
_In_ RES_DES rdResDes,
_In_ ULONG ulFlags)
{
TRACE("CM_Free_Res_Des(%p %p %lx)\n",
prdResDes, rdResDes, ulFlags);
return CM_Free_Res_Des_Ex(prdResDes, rdResDes, ulFlags, NULL);
}
/***********************************************************************
* CM_Free_Res_Des_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Res_Des_Ex(
_Out_ PRES_DES prdResDes,
_In_ RES_DES rdResDes,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Free_Res_Des_Ex(%p %p %lx %p)\n",
prdResDes, rdResDes, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Free_Res_Des_Handle [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Res_Des_Handle(
_In_ RES_DES rdResDes)
{
FIXME("CM_Free_Res_Des_Handle(%p)\n", rdResDes);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Free_Resource_Conflict_Handle [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Free_Resource_Conflict_Handle(
_In_ CONFLICT_LIST clConflictList)
{
PCONFLICT_DATA pConflictData;
FIXME("CM_Free_Resource_Conflict_Handle(%p)\n",
clConflictList);
pConflictData = (PCONFLICT_DATA)clConflictList;
if (!IsValidConflictData(pConflictData))
return CR_INVALID_CONFLICT_LIST;
if (pConflictData->pConflictList != NULL)
MyFree(pConflictData->pConflictList);
MyFree(pConflictData);
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Child [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Child(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Child(%p %p %lx)\n",
pdnDevInst, dnDevInst, ulFlags);
return CM_Get_Child_Ex(pdnDevInst, dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Child_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Child_Ex(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szRelatedDevInst[MAX_DEVICE_ID_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
DWORD dwIndex, dwLength = MAX_DEVICE_ID_LEN;
CONFIGRET ret;
TRACE("CM_Get_Child_Ex(%p %lx %lx %p)\n",
pdnDevInst, dnDevInst, ulFlags, hMachine);
if (pdnDevInst == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
*pdnDevInst = -1;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetRelatedDeviceInstance(BindingHandle,
PNP_GET_CHILD_DEVICE_INSTANCE,
lpDevInst,
szRelatedDevInst,
&dwLength,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return ret;
TRACE("szRelatedDevInst: %s\n", debugstr_w(szRelatedDevInst));
dwIndex = pSetupStringTableAddString(StringTable, szRelatedDevInst, 1);
if (dwIndex == -1)
return CR_FAILURE;
*pdnDevInst = dwIndex;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Class_Key_NameA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Key_NameA(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) LPSTR pszKeyName,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Class_Key_NameA(%p %p %p %lx)\n",
ClassGuid, pszKeyName, pulLength, ulFlags);
return CM_Get_Class_Key_Name_ExA(ClassGuid, pszKeyName, pulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Class_Key_NameW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Key_NameW(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) LPWSTR pszKeyName,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Class_Key_NameW(%p %p %p %lx)\n",
ClassGuid, pszKeyName, pulLength, ulFlags);
return CM_Get_Class_Key_Name_ExW(ClassGuid, pszKeyName, pulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Class_Key_Name_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Key_Name_ExA(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) LPSTR pszKeyName,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szBuffer[MAX_GUID_STRING_LEN];
CONFIGRET ret = CR_SUCCESS;
ULONG ulLength;
ULONG ulOrigLength;
TRACE("CM_Get_Class_Key_Name_ExA(%p %p %p %lx %p)\n",
ClassGuid, pszKeyName, pulLength, ulFlags, hMachine);
if (ClassGuid == NULL || pszKeyName == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
ulOrigLength = *pulLength;
*pulLength = 0;
ulLength = MAX_GUID_STRING_LEN;
ret = CM_Get_Class_Key_Name_ExW(ClassGuid, szBuffer, &ulLength,
ulFlags, hMachine);
if (ret == CR_SUCCESS)
{
if (WideCharToMultiByte(CP_ACP,
0,
szBuffer,
ulLength,
pszKeyName,
ulOrigLength,
NULL,
NULL) == 0)
ret = CR_FAILURE;
else
*pulLength = lstrlenA(pszKeyName) + 1;
}
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Class_Key_Name_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Key_Name_ExW(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) LPWSTR pszKeyName,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
TRACE("CM_Get_Class_Key_Name_ExW(%p %p %p %lx %p)\n",
ClassGuid, pszKeyName, pulLength, ulFlags, hMachine);
if (ClassGuid == NULL || pszKeyName == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (*pulLength < MAX_GUID_STRING_LEN)
{
*pulLength = 0;
return CR_BUFFER_SMALL;
}
if (!GuidToString(ClassGuid, pszKeyName))
return CR_INVALID_DATA;
*pulLength = MAX_GUID_STRING_LEN;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Class_NameA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_NameA(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) PCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Class_NameA(%p %p %p %lx)\n",
ClassGuid, Buffer, pulLength, ulFlags);
return CM_Get_Class_Name_ExA(ClassGuid, Buffer, pulLength, ulFlags,
NULL);
}
/***********************************************************************
* CM_Get_Class_NameW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_NameW(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) PWCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Class_NameW(%p %p %p %lx)\n",
ClassGuid, Buffer, pulLength, ulFlags);
return CM_Get_Class_Name_ExW(ClassGuid, Buffer, pulLength, ulFlags,
NULL);
}
/***********************************************************************
* CM_Get_Class_Name_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Name_ExA(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) PCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szBuffer[MAX_CLASS_NAME_LEN];
CONFIGRET ret = CR_SUCCESS;
ULONG ulLength;
ULONG ulOrigLength;
TRACE("CM_Get_Class_Name_ExA(%p %p %p %lx %p)\n",
ClassGuid, Buffer, pulLength, ulFlags, hMachine);
if (ClassGuid == NULL || Buffer == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
ulOrigLength = *pulLength;
*pulLength = 0;
ulLength = MAX_CLASS_NAME_LEN;
ret = CM_Get_Class_Name_ExW(ClassGuid, szBuffer, &ulLength,
ulFlags, hMachine);
if (ret == CR_SUCCESS)
{
if (WideCharToMultiByte(CP_ACP,
0,
szBuffer,
ulLength,
Buffer,
ulOrigLength,
NULL,
NULL) == 0)
ret = CR_FAILURE;
else
*pulLength = lstrlenA(Buffer) + 1;
}
return ret;
}
/***********************************************************************
* CM_Get_Class_Name_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Name_ExW(
_In_ LPGUID ClassGuid,
_Out_writes_opt_(*pulLength) PWCHAR Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szGuidString[MAX_GUID_STRING_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Get_Class_Name_ExW(%p %p %p %lx %p\n",
ClassGuid, Buffer, pulLength, ulFlags, hMachine);
if (ClassGuid == NULL || Buffer == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (!GuidToString(ClassGuid, szGuidString))
return CR_INVALID_DATA;
TRACE("Guid %s\n", debugstr_w(szGuidString));
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_GetClassName(BindingHandle,
szGuidString,
Buffer,
pulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Class_Registry_PropertyA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Registry_PropertyA(
LPGUID ClassGuid,
ULONG ulProperty,
PULONG pulRegDataType,
PVOID Buffer,
PULONG pulLength,
ULONG ulFlags,
HMACHINE hMachine)
{
PWSTR BufferW;
ULONG ulLength = 0;
ULONG ulType;
CONFIGRET ret;
TRACE("CM_Get_Class_Registry_PropertyA(%p %lu %p %p %p %lx %p)\n",
ClassGuid, ulProperty, pulRegDataType, Buffer, pulLength,
ulFlags, hMachine);
if (pulLength == NULL)
return CR_INVALID_POINTER;
if (ulProperty < CM_CRP_MIN || ulProperty > CM_CRP_MAX)
return CR_INVALID_PROPERTY;
ulType = GetRegistryPropertyType(ulProperty);
if (ulType == REG_SZ || ulType == REG_MULTI_SZ)
{
/* Get the required buffer size */
ret = CM_Get_Class_Registry_PropertyW(ClassGuid, ulProperty, pulRegDataType,
NULL, &ulLength, ulFlags, hMachine);
if (ret != CR_BUFFER_SMALL)
return ret;
/* Allocate the unicode buffer */
BufferW = HeapAlloc(GetProcessHeap(), 0, ulLength);
if (BufferW == NULL)
return CR_OUT_OF_MEMORY;
/* Get the property */
ret = CM_Get_Class_Registry_PropertyW(ClassGuid, ulProperty, pulRegDataType,
BufferW, &ulLength, ulFlags, hMachine);
if (ret != CR_SUCCESS)
{
HeapFree(GetProcessHeap(), 0, BufferW);
return ret;
}
/* Do W->A conversion */
*pulLength = WideCharToMultiByte(CP_ACP,
0,
BufferW,
ulLength,
Buffer,
*pulLength,
NULL,
NULL);
/* Release the unicode buffer */
HeapFree(GetProcessHeap(), 0, BufferW);
if (*pulLength == 0)
ret = CR_FAILURE;
}
else
{
/* Get the property */
ret = CM_Get_Class_Registry_PropertyW(ClassGuid, ulProperty, pulRegDataType,
Buffer, pulLength, ulFlags, hMachine);
}
return ret;
}
/***********************************************************************
* CM_Get_Class_Registry_PropertyW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Class_Registry_PropertyW(
LPGUID ClassGuid,
ULONG ulProperty,
PULONG pulRegDataType,
PVOID Buffer,
PULONG pulLength,
ULONG ulFlags,
HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
WCHAR szGuidString[PNP_MAX_GUID_STRING_LEN + 1];
ULONG ulType = 0;
ULONG ulTransferLength = 0;
CONFIGRET ret;
TRACE("CM_Get_Class_Registry_PropertyW(%p %lu %p %p %p %lx %p)\n",
ClassGuid, ulProperty, pulRegDataType, Buffer, pulLength,
ulFlags, hMachine);
if (ClassGuid == NULL || pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (pSetupStringFromGuid(ClassGuid,
szGuidString,
PNP_MAX_GUID_STRING_LEN) != 0)
return CR_INVALID_DATA;
if (ulProperty < CM_CRP_MIN || ulProperty > CM_CRP_MAX)
return CR_INVALID_PROPERTY;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
ulTransferLength = *pulLength;
RpcTryExcept
{
ret = PNP_GetClassRegProp(BindingHandle,
szGuidString,
ulProperty,
&ulType,
Buffer,
&ulTransferLength,
pulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
if (pulRegDataType != NULL)
*pulRegDataType = ulType;
}
return ret;
}
/***********************************************************************
* CM_Get_Depth [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Depth(
_Out_ PULONG pulDepth,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Depth(%p %lx %lx)\n",
pulDepth, dnDevInst, ulFlags);
return CM_Get_Depth_Ex(pulDepth, dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Depth_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Depth_Ex(
_Out_ PULONG pulDepth,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Get_Depth_Ex(%p %lx %lx %p)\n",
pulDepth, dnDevInst, ulFlags, hMachine);
if (pulDepth == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetDepth(BindingHandle,
lpDevInst,
pulDepth,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_DevNode_Custom_PropertyA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Custom_PropertyA(
_In_ DEVINST dnDevInst,
_In_ PCSTR pszCustomPropertyName,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_DevNode_Custom_PropertyA(%lx %s %p %p %p %lx)\n",
dnDevInst, pszCustomPropertyName, pulRegDataType,
Buffer, pulLength, ulFlags);
return CM_Get_DevNode_Custom_Property_ExA(dnDevInst, pszCustomPropertyName,
pulRegDataType, Buffer,
pulLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_DevNode_Custom_PropertyW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Custom_PropertyW(
_In_ DEVINST dnDevInst,
_In_ PCWSTR pszCustomPropertyName,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_DevNode_Custom_PropertyW(%lx %s %p %p %p %lx)\n",
dnDevInst, debugstr_w(pszCustomPropertyName), pulRegDataType,
Buffer, pulLength, ulFlags);
return CM_Get_DevNode_Custom_Property_ExW(dnDevInst, pszCustomPropertyName,
pulRegDataType, Buffer,
pulLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_DevNode_Custom_Property_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Custom_Property_ExA(
_In_ DEVINST dnDevInst,
_In_ PCSTR pszCustomPropertyName,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR pszPropertyNameW;
PVOID BufferW;
ULONG ulLengthW;
ULONG ulDataType = REG_NONE;
CONFIGRET ret;
TRACE("CM_Get_DevNode_Custom_Property_ExA(%lx %s %p %p %p %lx %p)\n",
dnDevInst, pszCustomPropertyName, pulRegDataType,
Buffer, pulLength, ulFlags, hMachine);
if (!pulLength)
return CR_INVALID_POINTER;
ulLengthW = *pulLength * sizeof(WCHAR);
BufferW = HeapAlloc(GetProcessHeap(), 0, ulLengthW);
if (!BufferW)
return CR_OUT_OF_MEMORY;
pszPropertyNameW = pSetupMultiByteToUnicode(pszCustomPropertyName,
CP_ACP);
if (pszPropertyNameW == NULL)
{
HeapFree(GetProcessHeap(), 0, BufferW);
return CR_OUT_OF_MEMORY;
}
ret = CM_Get_DevNode_Custom_Property_ExW(dnDevInst,
pszPropertyNameW,
&ulDataType,
BufferW,
&ulLengthW,
ulFlags,
hMachine);
if (ret == CR_SUCCESS)
{
if (ulDataType == REG_SZ ||
ulDataType == REG_EXPAND_SZ ||
ulDataType == REG_MULTI_SZ)
{
/* Do W->A conversion */
*pulLength = WideCharToMultiByte(CP_ACP,
0,
BufferW,
lstrlenW(BufferW) + 1,
Buffer,
*pulLength,
NULL,
NULL);
if (*pulLength == 0)
ret = CR_FAILURE;
}
else
{
/* Directly copy the value */
if (ulLengthW <= *pulLength)
memcpy(Buffer, BufferW, ulLengthW);
else
{
*pulLength = ulLengthW;
ret = CR_BUFFER_SMALL;
}
}
}
if (pulRegDataType)
*pulRegDataType = ulDataType;
HeapFree(GetProcessHeap(), 0, BufferW);
MyFree(pszPropertyNameW);
return ret;
}
/***********************************************************************
* CM_Get_DevNode_Custom_Property_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Custom_Property_ExW(
_In_ DEVINST dnDevInst,
_In_ PCWSTR pszCustomPropertyName,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
ULONG ulDataType = REG_NONE;
ULONG ulTransferLength;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_DevNode_Custom_Property_ExW(%lx %s %p %p %p %lx %p)\n",
dnDevInst, debugstr_w(pszCustomPropertyName), pulRegDataType,
Buffer, pulLength, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (pszCustomPropertyName == NULL ||
pulLength == NULL ||
*pulLength == 0)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_CUSTOMDEVPROP_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
ulTransferLength = *pulLength;
RpcTryExcept
{
ret = PNP_GetCustomDevProp(BindingHandle,
lpDevInst,
(LPWSTR)pszCustomPropertyName,
&ulDataType,
Buffer,
&ulTransferLength,
pulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
if (pulRegDataType != NULL)
*pulRegDataType = ulDataType;
}
return ret;
}
/***********************************************************************
* CM_Get_DevNode_Registry_PropertyA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Registry_PropertyA(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_DevNode_Registry_PropertyA(%lx %lu %p %p %p %lx)\n",
dnDevInst, ulProperty, pulRegDataType, Buffer, pulLength, ulFlags);
return CM_Get_DevNode_Registry_Property_ExA(dnDevInst, ulProperty,
pulRegDataType, Buffer,
pulLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_DevNode_Registry_PropertyW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Registry_PropertyW(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_DevNode_Registry_PropertyW(%lx %lu %p %p %p %lx)\n",
dnDevInst, ulProperty, pulRegDataType, Buffer, pulLength, ulFlags);
return CM_Get_DevNode_Registry_Property_ExW(dnDevInst, ulProperty,
pulRegDataType, Buffer,
pulLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_DevNode_Registry_Property_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Registry_Property_ExA(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
PVOID BufferW;
ULONG LengthW;
ULONG ulDataType = REG_NONE;
CONFIGRET ret;
TRACE("CM_Get_DevNode_Registry_Property_ExA(%lx %lu %p %p %p %lx %p)\n",
dnDevInst, ulProperty, pulRegDataType, Buffer, pulLength,
ulFlags, hMachine);
if (!pulLength)
return CR_INVALID_POINTER;
LengthW = *pulLength * sizeof(WCHAR);
BufferW = HeapAlloc(GetProcessHeap(), 0, LengthW);
if (!BufferW)
return CR_OUT_OF_MEMORY;
ret = CM_Get_DevNode_Registry_Property_ExW(dnDevInst,
ulProperty,
&ulDataType,
BufferW,
&LengthW,
ulFlags,
hMachine);
if (ret == CR_SUCCESS)
{
if (ulDataType == REG_SZ ||
ulDataType == REG_EXPAND_SZ ||
ulDataType == REG_MULTI_SZ)
{
/* Do W->A conversion */
*pulLength = WideCharToMultiByte(CP_ACP,
0,
BufferW,
lstrlenW(BufferW) + 1,
Buffer,
*pulLength,
NULL,
NULL);
if (*pulLength == 0)
ret = CR_FAILURE;
}
else
{
/* Directly copy the value */
if (LengthW <= *pulLength)
memcpy(Buffer, BufferW, LengthW);
else
{
*pulLength = LengthW;
ret = CR_BUFFER_SMALL;
}
}
}
if (pulRegDataType)
*pulRegDataType = ulDataType;
HeapFree(GetProcessHeap(), 0, BufferW);
return ret;
}
/***********************************************************************
* CM_Get_DevNode_Registry_Property_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Registry_Property_ExW(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_Out_opt_ PULONG pulRegDataType,
_Out_writes_bytes_opt_(*pulLength) PVOID Buffer,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
CONFIGRET ret = CR_SUCCESS;
LPWSTR lpDevInst;
ULONG ulDataType = REG_NONE;
ULONG ulTransferLength = 0;
TRACE("CM_Get_DevNode_Registry_Property_ExW(%lx %lu %p %p %p %lx %p)\n",
dnDevInst, ulProperty, pulRegDataType, Buffer, pulLength,
ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulProperty < CM_DRP_MIN || ulProperty > CM_DRP_MAX)
return CR_INVALID_PROPERTY;
/* pulRegDataType is optional */
/* Buffer is optional */
if (pulLength == NULL)
return CR_INVALID_POINTER;
if (*pulLength == 0)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
ulTransferLength = *pulLength;
RpcTryExcept
{
ret = PNP_GetDeviceRegProp(BindingHandle,
lpDevInst,
ulProperty,
&ulDataType,
Buffer,
&ulTransferLength,
pulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret == CR_SUCCESS)
{
if (pulRegDataType != NULL)
*pulRegDataType = ulDataType;
}
return ret;
}
/***********************************************************************
* CM_Get_DevNode_Status [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Status(
_Out_ PULONG pulStatus,
_Out_ PULONG pulProblemNumber,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_DevNode_Status(%p %p %lx %lx)\n",
pulStatus, pulProblemNumber, dnDevInst, ulFlags);
return CM_Get_DevNode_Status_Ex(pulStatus, pulProblemNumber, dnDevInst,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_DevNode_Status_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_DevNode_Status_Ex(
_Out_ PULONG pulStatus,
_Out_ PULONG pulProblemNumber,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Get_DevNode_Status_Ex(%p %p %lx %lx %p)\n",
pulStatus, pulProblemNumber, dnDevInst, ulFlags, hMachine);
if (pulStatus == NULL || pulProblemNumber == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetDeviceStatus(BindingHandle,
lpDevInst,
pulStatus,
pulProblemNumber,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Device_IDA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_IDA(
_In_ DEVINST dnDevInst,
_Out_writes_(BufferLen) PCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_IDA(%lx %p %lu %lx)\n",
dnDevInst, Buffer, BufferLen, ulFlags);
return CM_Get_Device_ID_ExA(dnDevInst, Buffer, BufferLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_IDW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_IDW(
_In_ DEVINST dnDevInst,
_Out_writes_(BufferLen) PWCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_IDW(%lx %p %lu %lx)\n",
dnDevInst, Buffer, BufferLen, ulFlags);
return CM_Get_Device_ID_ExW(dnDevInst, Buffer, BufferLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_ID_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_ExA(
_In_ DEVINST dnDevInst,
_Out_writes_(BufferLen) PCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szBufferW[MAX_DEVICE_ID_LEN];
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_ID_ExA(%lx %p %lu %lx %p)\n",
dnDevInst, Buffer, BufferLen, ulFlags, hMachine);
if (Buffer == NULL)
return CR_INVALID_POINTER;
ret = CM_Get_Device_ID_ExW(dnDevInst,
szBufferW,
MAX_DEVICE_ID_LEN,
ulFlags,
hMachine);
if (ret == CR_SUCCESS)
{
if (WideCharToMultiByte(CP_ACP,
0,
szBufferW,
lstrlenW(szBufferW) + 1,
Buffer,
BufferLen,
NULL,
NULL) == 0)
ret = CR_FAILURE;
}
return ret;
}
/***********************************************************************
* CM_Get_Device_ID_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_ExW(
_In_ DEVINST dnDevInst,
_Out_writes_(BufferLen) PWCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
HSTRING_TABLE StringTable = NULL;
TRACE("CM_Get_Device_ID_ExW(%lx %p %lu %lx %p)\n",
dnDevInst, Buffer, BufferLen, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (Buffer == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(NULL, &StringTable))
return CR_FAILURE;
}
if (!pSetupStringTableStringFromIdEx(StringTable,
dnDevInst,
Buffer,
&BufferLen))
return CR_FAILURE;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Device_ID_ListA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_ListA(
_In_ PCSTR pszFilter,
_Out_writes_(BufferLen) PCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_ID_ListA(%p %p %lu %lx)\n",
pszFilter, Buffer, BufferLen, ulFlags);
return CM_Get_Device_ID_List_ExA(pszFilter, Buffer, BufferLen,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_ID_ListW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_ListW(
_In_ PCWSTR pszFilter,
_Out_writes_(BufferLen) PWCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_ID_ListW(%p %p %lu %lx)\n",
pszFilter, Buffer, BufferLen, ulFlags);
return CM_Get_Device_ID_List_ExW(pszFilter, Buffer, BufferLen,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_ID_List_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_List_ExA(
_In_ PCSTR pszFilter,
_Out_writes_(BufferLen) PCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR BufferW = NULL;
LPWSTR pszFilterW = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_ID_List_ExA(%p %p %lu %lx %p)\n",
pszFilter, Buffer, BufferLen, ulFlags, hMachine);
BufferW = MyMalloc(BufferLen * sizeof(WCHAR));
if (BufferW == NULL)
return CR_OUT_OF_MEMORY;
if (pszFilter == NULL)
{
ret = CM_Get_Device_ID_List_ExW(NULL,
BufferW,
BufferLen,
ulFlags,
hMachine);
}
else
{
if (pSetupCaptureAndConvertAnsiArg(pszFilter, &pszFilterW))
{
ret = CR_INVALID_DEVICE_ID;
goto Done;
}
ret = CM_Get_Device_ID_List_ExW(pszFilterW,
BufferW,
BufferLen,
ulFlags,
hMachine);
MyFree(pszFilterW);
}
if (WideCharToMultiByte(CP_ACP,
0,
BufferW,
BufferLen,
Buffer,
BufferLen,
NULL,
NULL) == 0)
ret = CR_FAILURE;
Done:
MyFree(BufferW);
return ret;
}
/***********************************************************************
* CM_Get_Device_ID_List_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_List_ExW(
_In_ PCWSTR pszFilter,
_Out_writes_(BufferLen) PWCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Get_Device_ID_List_ExW(%p %p %lu %lx %p)\n",
pszFilter, Buffer, BufferLen, ulFlags, hMachine);
if (Buffer == NULL || BufferLen == 0)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_GETIDLIST_FILTER_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
*Buffer = 0;
RpcTryExcept
{
ret = PNP_GetDeviceList(BindingHandle,
(LPWSTR)pszFilter,
Buffer,
&BufferLen,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Device_ID_List_SizeA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_List_SizeA(
_Out_ PULONG pulLen,
_In_opt_ PCSTR pszFilter,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_ID_List_SizeA(%p %s %lx)\n",
pulLen, debugstr_a(pszFilter), ulFlags);
return CM_Get_Device_ID_List_Size_ExA(pulLen, pszFilter, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_ID_List_SizeW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_List_SizeW(
_Out_ PULONG pulLen,
_In_opt_ PCWSTR pszFilter,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_ID_List_SizeW(%p %s %lx)\n",
pulLen, debugstr_w(pszFilter), ulFlags);
return CM_Get_Device_ID_List_Size_ExW(pulLen, pszFilter, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_ID_List_Size_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_List_Size_ExA(
_Out_ PULONG pulLen,
_In_opt_ PCSTR pszFilter,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR pszFilterW = NULL;
CONFIGRET ret = CR_SUCCESS;
FIXME("CM_Get_Device_ID_List_Size_ExA(%p %s %lx %p)\n",
pulLen, debugstr_a(pszFilter), ulFlags, hMachine);
if (pszFilter == NULL)
{
ret = CM_Get_Device_ID_List_Size_ExW(pulLen,
NULL,
ulFlags,
hMachine);
}
else
{
if (pSetupCaptureAndConvertAnsiArg(pszFilter, &pszFilterW))
return CR_INVALID_DEVICE_ID;
ret = CM_Get_Device_ID_List_Size_ExW(pulLen,
pszFilterW,
ulFlags,
hMachine);
MyFree(pszFilterW);
}
return ret;
}
/***********************************************************************
* CM_Get_Device_ID_List_Size_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_List_Size_ExW(
_Out_ PULONG pulLen,
_In_opt_ PCWSTR pszFilter,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
FIXME("CM_Get_Device_ID_List_Size_ExW(%p %s %lx %p)\n",
pulLen, debugstr_w(pszFilter), ulFlags, hMachine);
if (pulLen == NULL)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_GETIDLIST_FILTER_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
*pulLen = 0;
RpcTryExcept
{
ret = PNP_GetDeviceListSize(BindingHandle,
(LPWSTR)pszFilter,
pulLen,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Device_ID_Size [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_Size(
_Out_ PULONG pulLen,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_ID_Size(%p %lx %lx)\n",
pulLen, dnDevInst, ulFlags);
return CM_Get_Device_ID_Size_Ex(pulLen, dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_ID_Size_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_ID_Size_Ex(
_Out_ PULONG pulLen,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
HSTRING_TABLE StringTable = NULL;
LPWSTR DeviceId;
TRACE("CM_Get_Device_ID_Size_Ex(%p %lx %lx %p)\n",
pulLen, dnDevInst, ulFlags, hMachine);
if (pulLen == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(NULL, &StringTable))
return CR_FAILURE;
}
DeviceId = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (DeviceId == NULL)
{
*pulLen = 0;
return CR_SUCCESS;
}
*pulLen = lstrlenW(DeviceId);
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Device_Interface_AliasA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_AliasA(
_In_ LPCSTR pszDeviceInterface,
_In_ LPGUID AliasInterfaceGuid,
_Out_writes_(*pulLength) LPSTR pszAliasDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_Interface_AliasA(%p %p %p %p %lx)\n",
pszDeviceInterface, AliasInterfaceGuid,
pszAliasDeviceInterface, pulLength, ulFlags);
return CM_Get_Device_Interface_Alias_ExA(pszDeviceInterface,
AliasInterfaceGuid, pszAliasDeviceInterface, pulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_Interface_AliasW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_AliasW(
_In_ LPCWSTR pszDeviceInterface,
_In_ LPGUID AliasInterfaceGuid,
_Out_writes_(*pulLength) LPWSTR pszAliasDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_Interface_AliasW(%p %p %p %p %lx)\n",
pszDeviceInterface, AliasInterfaceGuid,
pszAliasDeviceInterface, pulLength, ulFlags);
return CM_Get_Device_Interface_Alias_ExW(pszDeviceInterface,
AliasInterfaceGuid, pszAliasDeviceInterface, pulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_Interface_Alias_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_Alias_ExA(
_In_ LPCSTR pszDeviceInterface,
_In_ LPGUID AliasInterfaceGuid,
_Out_writes_(*pulLength) LPSTR pszAliasDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Get_Device_Interface_Alias_ExA(%p %p %p %p %lx %p)\n",
pszDeviceInterface, AliasInterfaceGuid,
pszAliasDeviceInterface, pulLength, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Get_Device_Interface_Alias_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_Alias_ExW(
_In_ LPCWSTR pszDeviceInterface,
_In_ LPGUID AliasInterfaceGuid,
_Out_writes_(*pulLength) LPWSTR pszAliasDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
ULONG ulTransferLength;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_Interface_Alias_ExW(%p %p %p %p %lx %p)\n",
pszDeviceInterface, AliasInterfaceGuid,
pszAliasDeviceInterface, pulLength, ulFlags, hMachine);
if (pszDeviceInterface == NULL ||
AliasInterfaceGuid == NULL ||
pszAliasDeviceInterface == NULL ||
pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
ulTransferLength = *pulLength;
RpcTryExcept
{
ret = PNP_GetInterfaceDeviceAlias(BindingHandle,
(LPWSTR)pszDeviceInterface,
AliasInterfaceGuid,
pszAliasDeviceInterface,
pulLength,
&ulTransferLength,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Device_Interface_ListA (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_ListA(
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_A pDeviceID,
_Out_writes_(BufferLen) PCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_Interface_ListA(%s %s %p %lu 0x%08lx)\n",
debugstr_guid(InterfaceClassGuid), debugstr_a(pDeviceID),
Buffer, BufferLen, ulFlags);
return CM_Get_Device_Interface_List_ExA(InterfaceClassGuid, pDeviceID,
Buffer, BufferLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_Interface_ListW (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_ListW(
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_W pDeviceID,
_Out_writes_(BufferLen) PWCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_Interface_ListW(%s %s %p %lu 0x%08lx)\n",
debugstr_guid(InterfaceClassGuid), debugstr_w(pDeviceID),
Buffer, BufferLen, ulFlags);
return CM_Get_Device_Interface_List_ExW(InterfaceClassGuid, pDeviceID,
Buffer, BufferLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_Interface_List_ExA (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_List_ExA(
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_A pDeviceID,
_Out_writes_(BufferLen) PCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
DEVINSTID_W pDeviceIdW = NULL;
PWCHAR BufferW = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_Interface_List_ExA(%s %s %p %lu 0x%08lx %p)\n",
debugstr_guid(InterfaceClassGuid), debugstr_a(pDeviceID),
Buffer, BufferLen, ulFlags, hMachine);
if (Buffer == NULL ||
BufferLen == 0)
return CR_INVALID_POINTER;
if (pDeviceID != NULL)
{
if (!pSetupCaptureAndConvertAnsiArg(pDeviceID, &pDeviceIdW))
return CR_INVALID_DEVICE_ID;
}
BufferW = MyMalloc(BufferLen * sizeof(WCHAR));
if (BufferW == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto Done;
}
ret = CM_Get_Device_Interface_List_ExW(InterfaceClassGuid, pDeviceIdW,
BufferW, BufferLen, ulFlags,
hMachine);
if (ret != CR_SUCCESS)
goto Done;
if (WideCharToMultiByte(CP_ACP,
0,
BufferW,
BufferLen,
Buffer,
BufferLen,
NULL,
NULL) == 0)
ret = CR_FAILURE;
Done:
if (BufferW != NULL)
MyFree(BufferW);
if (pDeviceIdW != NULL)
MyFree(pDeviceIdW);
return ret;
}
/***********************************************************************
* CM_Get_Device_Interface_List_ExW (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_List_ExW(
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_W pDeviceID,
_Out_writes_(BufferLen) PWCHAR Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
PNP_RPC_BUFFER_SIZE BufferSize = 0;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_Interface_List_ExW(%s %s %p %lu 0x%08lx %p)\n",
debugstr_guid(InterfaceClassGuid), debugstr_w(pDeviceID),
Buffer, BufferLen, ulFlags, hMachine);
if (Buffer == NULL ||
BufferLen == 0)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_GET_DEVICE_INTERFACE_LIST_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
*Buffer = 0;
BufferSize = BufferLen;
RpcTryExcept
{
ret = PNP_GetInterfaceDeviceList(BindingHandle,
InterfaceClassGuid,
pDeviceID,
(LPBYTE)Buffer,
&BufferSize,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Device_Interface_List_SizeA (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_List_SizeA(
_Out_ PULONG pulLen,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_A pDeviceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_Interface_List_SizeA(%p %p %s 0x%08lx)\n",
pulLen, InterfaceClassGuid, debugstr_a(pDeviceID), ulFlags);
return CM_Get_Device_Interface_List_Size_ExA(pulLen, InterfaceClassGuid,
pDeviceID, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_Interface_List_SizeW (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_List_SizeW(
_Out_ PULONG pulLen,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_W pDeviceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Device_Interface_List_SizeW(%p %p %s 0x%08lx)\n",
pulLen, InterfaceClassGuid, debugstr_w(pDeviceID), ulFlags);
return CM_Get_Device_Interface_List_Size_ExW(pulLen, InterfaceClassGuid,
pDeviceID, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Device_Interface_List_Size_ExA (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_List_Size_ExA(
_Out_ PULONG pulLen,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_A pDeviceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
DEVINSTID_W pDeviceIdW = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_Interface_List_Size_ExA(%p %p %s 0x%08lx %p)\n",
pulLen, InterfaceClassGuid, debugstr_a(pDeviceID), ulFlags, hMachine);
if (pulLen == NULL)
return CR_INVALID_POINTER;
if (pDeviceID != NULL)
{
if (!pSetupCaptureAndConvertAnsiArg(pDeviceID, &pDeviceIdW))
return CR_INVALID_DEVICE_ID;
}
*pulLen = 0;
ret = CM_Get_Device_Interface_List_Size_ExW(pulLen, InterfaceClassGuid,
pDeviceIdW, ulFlags, hMachine);
if (pDeviceIdW != NULL)
MyFree(pDeviceIdW);
return ret;
}
/***********************************************************************
* CM_Get_Device_Interface_List_Size_ExW (SETUPAPI.@)
*/
CONFIGRET
WINAPI
CM_Get_Device_Interface_List_Size_ExW(
_Out_ PULONG pulLen,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ DEVINSTID_W pDeviceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_Device_Interface_List_Size_ExW(%p %p %s 0x%08lx %p)\n",
pulLen, InterfaceClassGuid, debugstr_w(pDeviceID), ulFlags, hMachine);
if (pulLen == NULL)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_GET_DEVICE_INTERFACE_LIST_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
*pulLen = 0;
RpcTryExcept
{
ret = PNP_GetInterfaceDeviceListSize(BindingHandle,
pulLen,
InterfaceClassGuid,
pDeviceID,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_First_Log_Conf [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_First_Log_Conf(
_Out_opt_ PLOG_CONF plcLogConf,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_First_Log_Conf(%p %lx %lx)\n",
plcLogConf, dnDevInst, ulFlags);
return CM_Get_First_Log_Conf_Ex(plcLogConf, dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_First_Log_Conf_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_First_Log_Conf_Ex(
_Out_opt_ PLOG_CONF plcLogConf,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst = NULL;
CONFIGRET ret = CR_SUCCESS;
ULONG ulTag;
PLOG_CONF_INFO pLogConfInfo;
FIXME("CM_Get_First_Log_Conf_Ex(%p %lx %lx %p)\n",
plcLogConf, dnDevInst, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags & ~LOG_CONF_BITS)
return CR_INVALID_FLAG;
if (plcLogConf)
*plcLogConf = 0;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetFirstLogConf(BindingHandle,
lpDevInst,
ulFlags,
&ulTag,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return ret;
if (plcLogConf)
{
pLogConfInfo = HeapAlloc(GetProcessHeap(), 0, sizeof(LOG_CONF_INFO));
if (pLogConfInfo == NULL)
return CR_OUT_OF_MEMORY;
pLogConfInfo->ulMagic = LOG_CONF_MAGIC;
pLogConfInfo->dnDevInst = dnDevInst;
pLogConfInfo->ulType = ulFlags;
pLogConfInfo->ulTag = ulTag;
*plcLogConf = (LOG_CONF)pLogConfInfo;
}
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Global_State [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Global_State(
_Out_ PULONG pulState,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Global_State(%p %lx)\n",
pulState, ulFlags);
return CM_Get_Global_State_Ex(pulState, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Global_State_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Global_State_Ex(
_Out_ PULONG pulState,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Get_Global_State_Ex(%p %lx %p)\n",
pulState, ulFlags, hMachine);
if (pulState == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_GetGlobalState(BindingHandle, pulState, ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_HW_Prof_FlagsA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_HW_Prof_FlagsA(
_In_ DEVINSTID_A szDevInstName,
_In_ ULONG ulHardwareProfile,
_Out_ PULONG pulValue,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_HW_Prof_FlagsA(%s %lu %p %lx)\n",
debugstr_a(szDevInstName), ulHardwareProfile, pulValue, ulFlags);
return CM_Get_HW_Prof_Flags_ExA(szDevInstName, ulHardwareProfile,
pulValue, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_HW_Prof_FlagsW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_HW_Prof_FlagsW(
_In_ DEVINSTID_W szDevInstName,
_In_ ULONG ulHardwareProfile,
_Out_ PULONG pulValue,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_HW_Prof_FlagsW(%s %lu %p %lx)\n",
debugstr_w(szDevInstName), ulHardwareProfile, pulValue, ulFlags);
return CM_Get_HW_Prof_Flags_ExW(szDevInstName, ulHardwareProfile,
pulValue, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_HW_Prof_Flags_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_HW_Prof_Flags_ExA(
_In_ DEVINSTID_A szDevInstName,
_In_ ULONG ulHardwareProfile,
_Out_ PULONG pulValue,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
DEVINSTID_W pszDevIdW = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Get_HW_Prof_Flags_ExA(%s %lu %p %lx %p)\n",
debugstr_a(szDevInstName), ulHardwareProfile, pulValue, ulFlags, hMachine);
if (szDevInstName != NULL)
{
if (pSetupCaptureAndConvertAnsiArg(szDevInstName, &pszDevIdW))
return CR_INVALID_DEVICE_ID;
}
ret = CM_Get_HW_Prof_Flags_ExW(pszDevIdW, ulHardwareProfile,
pulValue, ulFlags, hMachine);
if (pszDevIdW != NULL)
MyFree(pszDevIdW);
return ret;
}
/***********************************************************************
* CM_Get_HW_Prof_Flags_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_HW_Prof_Flags_ExW(
_In_ DEVINSTID_W szDevInstName,
_In_ ULONG ulHardwareProfile,
_Out_ PULONG pulValue,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
FIXME("CM_Get_HW_Prof_Flags_ExW(%s %lu %p %lx %p)\n",
debugstr_w(szDevInstName), ulHardwareProfile, pulValue, ulFlags, hMachine);
if ((szDevInstName == NULL) || (pulValue == NULL))
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
/* FIXME: Check whether szDevInstName is valid */
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_HwProfFlags(BindingHandle, PNP_GET_HWPROFFLAGS, szDevInstName,
ulHardwareProfile, pulValue, NULL, NULL, 0, 0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Hardware_Profile_InfoA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Hardware_Profile_InfoA(
_In_ ULONG ulIndex,
_Out_ PHWPROFILEINFO_A pHWProfileInfo,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Hardware_Profile_InfoA(%lu %p %lx)\n",
ulIndex, pHWProfileInfo, ulFlags);
return CM_Get_Hardware_Profile_Info_ExA(ulIndex, pHWProfileInfo,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Hardware_Profile_InfoW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Hardware_Profile_InfoW(
_In_ ULONG ulIndex,
_Out_ PHWPROFILEINFO_W pHWProfileInfo,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Hardware_Profile_InfoW(%lu %p %lx)\n",
ulIndex, pHWProfileInfo, ulFlags);
return CM_Get_Hardware_Profile_Info_ExW(ulIndex, pHWProfileInfo,
ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Hardware_Profile_Info_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Hardware_Profile_Info_ExA(
_In_ ULONG ulIndex,
_Out_ PHWPROFILEINFO_A pHWProfileInfo,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
HWPROFILEINFO_W LocalProfileInfo;
CONFIGRET ret;
TRACE("CM_Get_Hardware_Profile_Info_ExA(%lu %p %lx %p)\n",
ulIndex, pHWProfileInfo, ulFlags, hMachine);
if (pHWProfileInfo == NULL)
return CR_INVALID_POINTER;
ret = CM_Get_Hardware_Profile_Info_ExW(ulIndex, &LocalProfileInfo,
ulFlags, hMachine);
if (ret == CR_SUCCESS)
{
pHWProfileInfo->HWPI_ulHWProfile = LocalProfileInfo.HWPI_ulHWProfile;
pHWProfileInfo->HWPI_dwFlags = LocalProfileInfo.HWPI_dwFlags;
if (WideCharToMultiByte(CP_ACP,
0,
LocalProfileInfo.HWPI_szFriendlyName,
lstrlenW(LocalProfileInfo.HWPI_szFriendlyName) + 1,
pHWProfileInfo->HWPI_szFriendlyName,
MAX_PROFILE_LEN,
NULL,
NULL) == 0)
ret = CR_FAILURE;
}
return ret;
}
/***********************************************************************
* CM_Get_Hardware_Profile_Info_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Hardware_Profile_Info_ExW(
_In_ ULONG ulIndex,
_Out_ PHWPROFILEINFO_W pHWProfileInfo,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Get_Hardware_Profile_Info_ExW(%lu %p %lx %p)\n",
ulIndex, pHWProfileInfo, ulFlags, hMachine);
if (pHWProfileInfo == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_GetHwProfInfo(BindingHandle, ulIndex, pHWProfileInfo,
sizeof(HWPROFILEINFO_W), 0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Log_Conf_Priority [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Log_Conf_Priority(
_In_ LOG_CONF lcLogConf,
_Out_ PPRIORITY pPriority,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Log_Conf_Priority(%p %p %lx)\n",
lcLogConf, pPriority, ulFlags);
return CM_Get_Log_Conf_Priority_Ex(lcLogConf, pPriority, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Log_Conf_Priority_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Log_Conf_Priority_Ex(
_In_ LOG_CONF lcLogConf,
_Out_ PPRIORITY pPriority,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
PLOG_CONF_INFO pLogConfInfo;
LPWSTR lpDevInst;
CONFIGRET ret;
FIXME("CM_Get_Log_Conf_Priority_Ex(%p %p %lx %p)\n",
lcLogConf, pPriority, ulFlags, hMachine);
pLogConfInfo = (PLOG_CONF_INFO)lcLogConf;
if (!IsValidLogConf(pLogConfInfo))
return CR_INVALID_LOG_CONF;
if (pPriority == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, pLogConfInfo->dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetLogConfPriority(BindingHandle,
lpDevInst,
pLogConfInfo->ulType,
pLogConfInfo->ulTag,
pPriority,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Get_Next_Log_Conf [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Next_Log_Conf(
_Out_opt_ PLOG_CONF plcLogConf,
_In_ LOG_CONF lcLogConf,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Next_Log_Conf(%p %p %lx)\n",
plcLogConf, lcLogConf, ulFlags);
return CM_Get_Next_Log_Conf_Ex(plcLogConf, lcLogConf, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Next_Log_Conf_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Next_Log_Conf_Ex(
_Out_opt_ PLOG_CONF plcLogConf,
_In_ LOG_CONF lcLogConf,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
PLOG_CONF_INFO pLogConfInfo;
PLOG_CONF_INFO pNewLogConfInfo;
ULONG ulNewTag;
LPWSTR lpDevInst;
CONFIGRET ret;
FIXME("CM_Get_Next_Log_Conf_Ex(%p %p %lx %p)\n",
plcLogConf, lcLogConf, ulFlags, hMachine);
if (plcLogConf)
*plcLogConf = 0;
pLogConfInfo = (PLOG_CONF_INFO)lcLogConf;
if (!IsValidLogConf(pLogConfInfo))
return CR_INVALID_LOG_CONF;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, pLogConfInfo->dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetNextLogConf(BindingHandle,
lpDevInst,
pLogConfInfo->ulType,
pLogConfInfo->ulTag,
&ulNewTag,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return ret;
if (plcLogConf)
{
pNewLogConfInfo = HeapAlloc(GetProcessHeap(), 0, sizeof(LOG_CONF_INFO));
if (pNewLogConfInfo == NULL)
return CR_OUT_OF_MEMORY;
pNewLogConfInfo->ulMagic = LOG_CONF_MAGIC;
pNewLogConfInfo->dnDevInst = pLogConfInfo->dnDevInst;
pNewLogConfInfo->ulType = pLogConfInfo->ulType;
pNewLogConfInfo->ulTag = ulNewTag;
*plcLogConf = (LOG_CONF)pNewLogConfInfo;
}
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Next_Re_Des [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Next_Res_Des(
_Out_ PRES_DES prdResDes,
_In_ RES_DES rdResDes,
_In_ RESOURCEID ForResource,
_Out_opt_ PRESOURCEID pResourceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Next_Res_Des(%p %p %lu %p %lx)\n",
prdResDes, rdResDes, ForResource, pResourceID, ulFlags);
return CM_Get_Next_Res_Des_Ex(prdResDes, rdResDes, ForResource,
pResourceID, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Next_Re_Des_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Next_Res_Des_Ex(
_Out_ PRES_DES prdResDes,
_In_ RES_DES rdResDes,
_In_ RESOURCEID ForResource,
_Out_opt_ PRESOURCEID pResourceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
ULONG ulInTag, ulOutTag = 0;
ULONG ulInType, ulOutType = 0;
LPWSTR lpDevInst;
DEVINST dnDevInst;
CONFIGRET ret;
FIXME("CM_Get_Next_Res_Des_Ex(%p %p %lu %p %lx %p)\n",
prdResDes, rdResDes, ForResource, pResourceID, ulFlags, hMachine);
if (prdResDes == NULL)
return CR_INVALID_POINTER;
if (IsValidLogConf((PLOG_CONF_INFO)rdResDes))
{
FIXME("LogConf found!\n");
dnDevInst = ((PLOG_CONF_INFO)rdResDes)->dnDevInst;
ulInTag = ((PLOG_CONF_INFO)rdResDes)->ulTag;
ulInType = ((PLOG_CONF_INFO)rdResDes)->ulType;
}
#if 0
else if (IsValidResDes((PRES_DES_INFO)rdResDes))
{
FIXME("ResDes found!\n");
dnDevInst = ((PRES_DES_INFO)rdResDes)->dnDevInst;
ulInTag = ((PRES_DES_INFO)rdResDes)->ulTag;
ulInType = ((PRES_DES_INFO)rdResDes)->ulType;
}
#endif
else
{
return CR_INVALID_RES_DES;
}
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetNextResDes(BindingHandle,
lpDevInst,
ulInTag,
ulInType,
ForResource,
0, /* unsigned long ulResourceTag, */
&ulOutTag,
&ulOutType,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return ret;
/* FIXME: Create the ResDes handle */
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Parent [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Parent(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Parent(%p %p %lx)\n",
pdnDevInst, dnDevInst, ulFlags);
return CM_Get_Parent_Ex(pdnDevInst, dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Parent_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Parent_Ex(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szRelatedDevInst[MAX_DEVICE_ID_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
DWORD dwIndex, dwLength = MAX_DEVICE_ID_LEN;
CONFIGRET ret;
TRACE("CM_Get_Parent_Ex(%p %lx %lx %p)\n",
pdnDevInst, dnDevInst, ulFlags, hMachine);
if (pdnDevInst == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
*pdnDevInst = -1;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetRelatedDeviceInstance(BindingHandle,
PNP_GET_PARENT_DEVICE_INSTANCE,
lpDevInst,
szRelatedDevInst,
&dwLength,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return ret;
TRACE("szRelatedDevInst: %s\n", debugstr_w(szRelatedDevInst));
dwIndex = pSetupStringTableAddString(StringTable, szRelatedDevInst, 1);
if (dwIndex == -1)
return CR_FAILURE;
*pdnDevInst = dwIndex;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Res_Des_Data [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Res_Des_Data(
_In_ RES_DES rdResDes,
_Out_writes_bytes_(BufferLen) PVOID Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Res_Des_Data(%p %p %lu %lx)\n",
rdResDes, Buffer, BufferLen, ulFlags);
return CM_Get_Res_Des_Data_Ex(rdResDes, Buffer, BufferLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Res_Des_Data_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Res_Des_Data_Ex(
_In_ RES_DES rdResDes,
_Out_writes_bytes_(BufferLen) PVOID Buffer,
_In_ ULONG BufferLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Get_Res_Des_Data_Ex(%p %p %lu %lx %p)\n",
rdResDes, Buffer, BufferLen, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Get_Res_Des_Size [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Res_Des_Data_Size(
_Out_ PULONG pulSize,
_In_ RES_DES rdResDes,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Res_Des_Data_Size(%p %p %lx)\n",
pulSize, rdResDes, ulFlags);
return CM_Get_Res_Des_Data_Size_Ex(pulSize, rdResDes, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Res_Des_Size_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Res_Des_Data_Size_Ex(
_Out_ PULONG pulSize,
_In_ RES_DES rdResDes,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
TRACE("CM_Get_Res_Des_Data_Size_Ex(%p %p %lx %p)\n",
pulSize, rdResDes, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Get_Resource_Conflict_Count [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Resource_Conflict_Count(
_In_ CONFLICT_LIST clConflictList,
_Out_ PULONG pulCount)
{
PCONFLICT_DATA pConflictData;
FIXME("CM_Get_Resource_Conflict_Count(%p %p)\n",
clConflictList, pulCount);
pConflictData = (PCONFLICT_DATA)clConflictList;
if (!IsValidConflictData(pConflictData))
return CR_INVALID_CONFLICT_LIST;
if (pulCount == NULL)
return CR_INVALID_POINTER;
*pulCount = pConflictData->pConflictList->ConflictsListed;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Resource_Conflict_DetailsA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Resource_Conflict_DetailsA(
_In_ CONFLICT_LIST clConflictList,
_In_ ULONG ulIndex,
_Inout_ PCONFLICT_DETAILS_A pConflictDetails)
{
FIXME("CM_Get_Resource_Conflict_CountA(%p %lu %p)\n",
clConflictList, ulIndex, pConflictDetails);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Get_Resource_Conflict_DetailsW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Resource_Conflict_DetailsW(
_In_ CONFLICT_LIST clConflictList,
_In_ ULONG ulIndex,
_Inout_ PCONFLICT_DETAILS_W pConflictDetails)
{
FIXME("CM_Get_Resource_Conflict_CountW(%p %lu %p)\n",
clConflictList, ulIndex, pConflictDetails);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Get_Sibling [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Sibling(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Get_Sibling(%p %p %lx)\n",
pdnDevInst, dnDevInst, ulFlags);
return CM_Get_Sibling_Ex(pdnDevInst, dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Get_Sibling_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Get_Sibling_Ex(
_Out_ PDEVINST pdnDevInst,
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szRelatedDevInst[MAX_DEVICE_ID_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
DWORD dwIndex, dwLength = MAX_DEVICE_ID_LEN;
CONFIGRET ret;
TRACE("CM_Get_Sibling_Ex(%p %lx %lx %p)\n",
pdnDevInst, dnDevInst, ulFlags, hMachine);
if (pdnDevInst == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags != 0)
return CR_INVALID_FLAG;
*pdnDevInst = -1;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_GetRelatedDeviceInstance(BindingHandle,
PNP_GET_SIBLING_DEVICE_INSTANCE,
lpDevInst,
szRelatedDevInst,
&dwLength,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return ret;
TRACE("szRelatedDevInst: %s\n", debugstr_w(szRelatedDevInst));
dwIndex = pSetupStringTableAddString(StringTable, szRelatedDevInst, 1);
if (dwIndex == -1)
return CR_FAILURE;
*pdnDevInst = dwIndex;
return CR_SUCCESS;
}
/***********************************************************************
* CM_Get_Version [SETUPAPI.@]
*/
WORD
WINAPI
CM_Get_Version(VOID)
{
TRACE("CM_Get_Version()\n");
return CM_Get_Version_Ex(NULL);
}
/***********************************************************************
* CM_Get_Version_Ex [SETUPAPI.@]
*/
WORD
WINAPI
CM_Get_Version_Ex(
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
WORD Version = 0;
CONFIGRET ret;
TRACE("CM_Get_Version_Ex(%p)\n", hMachine);
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return 0;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_GetVersion(BindingHandle, &Version);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return 0;
return Version;
}
/***********************************************************************
* CM_Intersect_Range_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Intersect_Range_List(
_In_ RANGE_LIST rlhOld1,
_In_ RANGE_LIST rlhOld2,
_In_ RANGE_LIST rlhNew,
_In_ ULONG ulFlags)
{
FIXME("CM_Intersect_Range_List(%p %p %p %lx)\n",
rlhOld1, rlhOld2, rlhNew, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Invert_Range_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Invert_Range_List(
_In_ RANGE_LIST rlhOld,
_In_ RANGE_LIST rlhNew,
_In_ DWORDLONG ullMaxValue,
_In_ ULONG ulFlags)
{
FIXME("CM_Invert_Range_List(%p %p %I64u %lx)\n",
rlhOld, rlhNew, ullMaxValue, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Is_Dock_Station_Present [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Is_Dock_Station_Present(
_Out_ PBOOL pbPresent)
{
TRACE("CM_Is_Dock_Station_Present(%p)\n",
pbPresent);
return CM_Is_Dock_Station_Present_Ex(pbPresent, NULL);
}
/***********************************************************************
* CM_Is_Dock_Station_Present_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Is_Dock_Station_Present_Ex(
_Out_ PBOOL pbPresent,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Is_Dock_Station_Present_Ex(%p %p)\n",
pbPresent, hMachine);
if (pbPresent == NULL)
return CR_INVALID_POINTER;
*pbPresent = FALSE;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_IsDockStationPresent(BindingHandle,
pbPresent);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Is_Version_Available_Ex [SETUPAPI.@]
*/
BOOL
WINAPI
CM_Is_Version_Available(
_In_ WORD wVersion)
{
TRACE("CM_Is_Version_Available(%hu)\n",
wVersion);
return CM_Is_Version_Available_Ex(wVersion, NULL);
}
/***********************************************************************
* CM_Is_Version_Available_Ex [SETUPAPI.@]
*/
BOOL
WINAPI
CM_Is_Version_Available_Ex(
_In_ WORD wVersion,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
WORD wServerVersion;
CONFIGRET ret;
TRACE("CM_Is_Version_Available_Ex(%hu %p)\n",
wVersion, hMachine);
if (wVersion <= 0x400)
return TRUE;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return FALSE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return FALSE;
}
RpcTryExcept
{
ret = PNP_GetVersion(BindingHandle, &wServerVersion);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
return FALSE;
return (wServerVersion >= wVersion);
}
/***********************************************************************
* CM_Locate_DevNodeA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Locate_DevNodeA(
_Out_ PDEVINST pdnDevInst,
_In_opt_ DEVINSTID_A pDeviceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Locate_DevNodeA(%p %s %lx)\n",
pdnDevInst, debugstr_a(pDeviceID), ulFlags);
return CM_Locate_DevNode_ExA(pdnDevInst, pDeviceID, ulFlags, NULL);
}
/***********************************************************************
* CM_Locate_DevNodeW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Locate_DevNodeW(
_Out_ PDEVINST pdnDevInst,
_In_opt_ DEVINSTID_W pDeviceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Locate_DevNodeW(%p %s %lx)\n",
pdnDevInst, debugstr_w(pDeviceID), ulFlags);
return CM_Locate_DevNode_ExW(pdnDevInst, pDeviceID, ulFlags, NULL);
}
/***********************************************************************
* CM_Locate_DevNode_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Locate_DevNode_ExA(
_Out_ PDEVINST pdnDevInst,
_In_opt_ DEVINSTID_A pDeviceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
DEVINSTID_W pDevIdW = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Locate_DevNode_ExA(%p %s %lx %p)\n",
pdnDevInst, debugstr_a(pDeviceID), ulFlags, hMachine);
if (pDeviceID != NULL)
{
if (pSetupCaptureAndConvertAnsiArg(pDeviceID, &pDevIdW))
return CR_INVALID_DEVICE_ID;
}
ret = CM_Locate_DevNode_ExW(pdnDevInst, pDevIdW, ulFlags, hMachine);
if (pDevIdW != NULL)
MyFree(pDevIdW);
return ret;
}
/***********************************************************************
* CM_Locate_DevNode_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Locate_DevNode_ExW(
_Out_ PDEVINST pdnDevInst,
_In_opt_ DEVINSTID_W pDeviceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR DeviceIdBuffer[MAX_DEVICE_ID_LEN];
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Locate_DevNode_ExW(%p %s %lx %p)\n",
pdnDevInst, debugstr_w(pDeviceID), ulFlags, hMachine);
if (pdnDevInst == NULL)
return CR_INVALID_POINTER;
if (ulFlags & ~CM_LOCATE_DEVNODE_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
if (pDeviceID != NULL && lstrlenW(pDeviceID) != 0)
{
lstrcpyW(DeviceIdBuffer, pDeviceID);
RpcTryExcept
{
/* Validate the device ID */
ret = PNP_ValidateDeviceInstance(BindingHandle,
DeviceIdBuffer,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
}
else
{
RpcTryExcept
{
/* Get the root device ID */
ret = PNP_GetRootDeviceInstance(BindingHandle,
DeviceIdBuffer,
MAX_DEVICE_ID_LEN);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
}
TRACE("DeviceIdBuffer: %s\n", debugstr_w(DeviceIdBuffer));
if (ret == CR_SUCCESS)
{
*pdnDevInst = pSetupStringTableAddString(StringTable, DeviceIdBuffer, 1);
if (*pdnDevInst == -1)
ret = CR_FAILURE;
}
return ret;
}
/***********************************************************************
* CM_Merge_Range_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Merge_Range_List(
_In_ RANGE_LIST rlhOld1,
_In_ RANGE_LIST rlhOld2,
_In_ RANGE_LIST rlhNew,
_In_ ULONG ulFlags)
{
FIXME("CM_Merge_Range_List(%p %p %p %lx)\n",
rlhOld1, rlhOld2, rlhNew, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Modify_Res_Des [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Modify_Res_Des(
_Out_ PRES_DES prdResDes,
_In_ RES_DES rdResDes,
_In_ RESOURCEID ResourceID,
_In_reads_bytes_(ResourceLen) PCVOID ResourceData,
_In_ ULONG ResourceLen,
_In_ ULONG ulFlags)
{
TRACE("CM_Modify_Res_Des(%p %p %lx %p %lu %lx)\n",
prdResDes, rdResDes, ResourceID, ResourceData,
ResourceLen, ulFlags);
return CM_Modify_Res_Des_Ex(prdResDes, rdResDes, ResourceID, ResourceData,
ResourceLen, ulFlags, NULL);
}
/***********************************************************************
* CM_Modify_Res_Des_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Modify_Res_Des_Ex(
_Out_ PRES_DES prdResDes,
_In_ RES_DES rdResDes,
_In_ RESOURCEID ResourceID,
_In_reads_bytes_(ResourceLen) PCVOID ResourceData,
_In_ ULONG ResourceLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Modify_Res_Des_Ex(%p %p %lx %p %lu %lx %p)\n",
prdResDes, rdResDes, ResourceID, ResourceData,
ResourceLen, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Move_DevNode [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Move_DevNode(
_In_ DEVINST dnFromDevInst,
_In_ DEVINST dnToDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Move_DevNode(%lx %lx %lx)\n",
dnFromDevInst, dnToDevInst, ulFlags);
return CM_Move_DevNode_Ex(dnFromDevInst, dnToDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Move_DevNode_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Move_DevNode_Ex(
_In_ DEVINST dnFromDevInst,
_In_ DEVINST dnToDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpFromDevInst;
LPWSTR lpToDevInst;
CONFIGRET ret;
FIXME("CM_Move_DevNode_Ex(%lx %lx %lx %p)\n",
dnFromDevInst, dnToDevInst, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (dnFromDevInst == 0 || dnToDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpFromDevInst = pSetupStringTableStringFromId(StringTable, dnFromDevInst);
if (lpFromDevInst == NULL)
return CR_INVALID_DEVNODE;
lpToDevInst = pSetupStringTableStringFromId(StringTable, dnToDevInst);
if (lpToDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_DeviceInstanceAction(BindingHandle,
PNP_DEVINST_MOVE,
ulFlags,
lpFromDevInst,
lpToDevInst);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Next_Range [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Next_Range(
_Inout_ PRANGE_ELEMENT preElement,
_Out_ PDWORDLONG pullStart,
_Out_ PDWORDLONG pullEnd,
_In_ ULONG ulFlags)
{
PINTERNAL_RANGE_LIST pRangeList;
PINTERNAL_RANGE pRange;
PLIST_ENTRY ListEntry;
CONFIGRET ret = CR_SUCCESS;
FIXME("CM_Next_Range(%p %p %p %lx)\n",
preElement, pullStart, pullEnd, ulFlags);
pRange = (PINTERNAL_RANGE)preElement;
if (pRange == NULL || pRange->pRangeList == NULL)
return CR_FAILURE;
if (pullStart == NULL || pullEnd == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
pRangeList = pRange->pRangeList;
/* Lock the range list */
WaitForSingleObject(pRangeList->hMutex, INFINITE);
/* Fail, if we reached the end of the list */
if (pRange->ListEntry.Flink == &pRangeList->ListHead)
{
ret = CR_FAILURE;
goto done;
}
/* Get the next range */
ListEntry = pRangeList->ListHead.Flink;
pRange = CONTAINING_RECORD(ListEntry, INTERNAL_RANGE, ListEntry);
/* Return the range data */
*pullStart = pRange->ullStart;
*pullEnd = pRange->ullEnd;
*preElement = (RANGE_ELEMENT)pRange;
done:
/* Unlock the range list */
ReleaseMutex(pRangeList->hMutex);
return ret;
}
/***********************************************************************
* CM_Open_Class_KeyA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Open_Class_KeyA(
_In_opt_ LPGUID pClassGuid,
_In_opt_ LPCSTR pszClassName,
_In_ REGSAM samDesired,
_In_ REGDISPOSITION Disposition,
_Out_ PHKEY phkClass,
_In_ ULONG ulFlags)
{
TRACE("CM_Open_Class_KeyA(%p %s %lx %lx %p %lx)\n",
debugstr_guid(pClassGuid), debugstr_a(pszClassName),
samDesired, Disposition, phkClass, ulFlags);
return CM_Open_Class_Key_ExA(pClassGuid, pszClassName, samDesired,
Disposition, phkClass, ulFlags, NULL);
}
/***********************************************************************
* CM_Open_Class_KeyW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Open_Class_KeyW(
_In_opt_ LPGUID pClassGuid,
_In_opt_ LPCWSTR pszClassName,
_In_ REGSAM samDesired,
_In_ REGDISPOSITION Disposition,
_Out_ PHKEY phkClass,
_In_ ULONG ulFlags)
{
TRACE("CM_Open_Class_KeyW(%p %s %lx %lx %p %lx)\n",
debugstr_guid(pClassGuid), debugstr_w(pszClassName),
samDesired, Disposition, phkClass, ulFlags);
return CM_Open_Class_Key_ExW(pClassGuid, pszClassName, samDesired,
Disposition, phkClass, ulFlags, NULL);
}
/***********************************************************************
* CM_Open_Class_Key_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Open_Class_Key_ExA(
_In_opt_ LPGUID pClassGuid,
_In_opt_ LPCSTR pszClassName,
_In_ REGSAM samDesired,
_In_ REGDISPOSITION Disposition,
_Out_ PHKEY phkClass,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR pszClassNameW = NULL;
CONFIGRET ret;
TRACE("CM_Open_Class_Key_ExA(%p %s %lx %lx %p %lx %p)\n",
debugstr_guid(pClassGuid), debugstr_a(pszClassName),
samDesired, Disposition, phkClass, ulFlags, hMachine);
if (pszClassName != NULL)
{
if (pSetupCaptureAndConvertAnsiArg(pszClassName, &pszClassNameW))
return CR_INVALID_DATA;
}
ret = CM_Open_Class_Key_ExW(pClassGuid, pszClassNameW, samDesired,
Disposition, phkClass, ulFlags, hMachine);
if (pszClassNameW != NULL)
MyFree(pszClassNameW);
return ret;
}
/***********************************************************************
* CM_Open_Class_Key_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Open_Class_Key_ExW(
_In_opt_ LPGUID pClassGuid,
_In_opt_ LPCWSTR pszClassName,
_In_ REGSAM samDesired,
_In_ REGDISPOSITION Disposition,
_Out_ PHKEY phkClass,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
WCHAR szKeyName[MAX_PATH];
LPWSTR lpGuidString;
DWORD dwDisposition;
DWORD dwError;
HKEY hKey;
TRACE("CM_Open_Class_Key_ExW(%p %s %lx %lx %p %lx %p)\n",
debugstr_guid(pClassGuid), debugstr_w(pszClassName),
samDesired, Disposition, phkClass, ulFlags, hMachine);
/* Check Disposition and ulFlags */
if ((Disposition & ~RegDisposition_Bits) ||
(ulFlags & ~CM_OPEN_CLASS_KEY_BITS))
return CR_INVALID_FLAG;
/* Check phkClass */
if (phkClass == NULL)
return CR_INVALID_POINTER;
*phkClass = NULL;
if (ulFlags == CM_OPEN_CLASS_KEY_INTERFACE &&
pszClassName != NULL)
return CR_INVALID_DATA;
if (hMachine == NULL)
{
hKey = HKEY_LOCAL_MACHINE;
}
else
{
if (RegConnectRegistryW(((PMACHINE_INFO)hMachine)->szMachineName,
HKEY_LOCAL_MACHINE,
&hKey))
return CR_REGISTRY_ERROR;
}
if (ulFlags & CM_OPEN_CLASS_KEY_INTERFACE)
{
lstrcpyW(szKeyName, DeviceClasses);
}
else
{
lstrcpyW(szKeyName, ControlClass);
}
if (pClassGuid != NULL)
{
if (UuidToStringW((UUID*)pClassGuid, &lpGuidString) != RPC_S_OK)
{
RegCloseKey(hKey);
return CR_INVALID_DATA;
}
lstrcatW(szKeyName, BackslashOpenBrace);
lstrcatW(szKeyName, lpGuidString);
lstrcatW(szKeyName, CloseBrace);
}
if (Disposition == RegDisposition_OpenAlways)
{
dwError = RegCreateKeyExW(hKey, szKeyName, 0, NULL, 0, samDesired,
NULL, phkClass, &dwDisposition);
}
else
{
dwError = RegOpenKeyExW(hKey, szKeyName, 0, samDesired, phkClass);
}
RegCloseKey(hKey);
if (pClassGuid != NULL)
RpcStringFreeW(&lpGuidString);
if (dwError != ERROR_SUCCESS)
{
*phkClass = NULL;
return CR_NO_SUCH_REGISTRY_KEY;
}
if (pszClassName != NULL)
{
RegSetValueExW(*phkClass, Class, 0, REG_SZ, (LPBYTE)pszClassName,
(lstrlenW(pszClassName) + 1) * sizeof(WCHAR));
}
return CR_SUCCESS;
}
/***********************************************************************
* CM_Open_DevNode_Key [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Open_DevNode_Key(
_In_ DEVINST dnDevNode,
_In_ REGSAM samDesired,
_In_ ULONG ulHardwareProfile,
_In_ REGDISPOSITION Disposition,
_Out_ PHKEY phkDevice,
_In_ ULONG ulFlags)
{
TRACE("CM_Open_DevNode_Key(%lx %lx %lu %lx %p %lx)\n",
dnDevNode, samDesired, ulHardwareProfile, Disposition, phkDevice, ulFlags);
return CM_Open_DevNode_Key_Ex(dnDevNode, samDesired, ulHardwareProfile,
Disposition, phkDevice, ulFlags, NULL);
}
/***********************************************************************
* CM_Open_DevNode_Key_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Open_DevNode_Key_Ex(
_In_ DEVINST dnDevNode,
_In_ REGSAM samDesired,
_In_ ULONG ulHardwareProfile,
_In_ REGDISPOSITION Disposition,
_Out_ PHKEY phkDevice,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR pszDevInst, pszKeyPath = NULL, pszInstancePath = NULL;
LONG lError;
DWORD dwDisposition;
HKEY hRootKey = NULL;
CONFIGRET ret = CR_CALL_NOT_IMPLEMENTED;
TRACE("CM_Open_DevNode_Key_Ex(%lx %lx %lu %lx %p %lx %p)\n",
dnDevNode, samDesired, ulHardwareProfile, Disposition, phkDevice, ulFlags, hMachine);
if (phkDevice == NULL)
return CR_INVALID_POINTER;
*phkDevice = NULL;
if (dnDevNode == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_REGISTRY_BITS)
return CR_INVALID_FLAG;
if (Disposition & ~RegDisposition_Bits)
return CR_INVALID_DATA;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
pszDevInst = pSetupStringTableStringFromId(StringTable, dnDevNode);
if (pszDevInst == NULL)
return CR_INVALID_DEVNODE;
TRACE("pszDevInst: %S\n", pszDevInst);
pszKeyPath = MyMalloc(512 * sizeof(WCHAR));
if (pszKeyPath == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
pszInstancePath = MyMalloc(512 * sizeof(WCHAR));
if (pszInstancePath == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
ret = GetDeviceInstanceKeyPath(BindingHandle,
pszDevInst,
pszKeyPath,
pszInstancePath,
ulHardwareProfile,
ulFlags);
if (ret != CR_SUCCESS)
goto done;
TRACE("pszKeyPath: %S\n", pszKeyPath);
TRACE("pszInstancePath: %S\n", pszInstancePath);
wcscat(pszKeyPath, L"\\");
wcscat(pszKeyPath, pszInstancePath);
TRACE("pszKeyPath: %S\n", pszKeyPath);
if (hMachine == NULL)
{
hRootKey = HKEY_LOCAL_MACHINE;
}
else
{
if (RegConnectRegistryW(((PMACHINE_INFO)hMachine)->szMachineName,
HKEY_LOCAL_MACHINE,
&hRootKey))
{
ret = CR_REGISTRY_ERROR;
goto done;
}
}
if (Disposition == RegDisposition_OpenAlways)
{
lError = RegCreateKeyExW(hRootKey,
pszKeyPath,
0,
NULL,
0,
samDesired,
NULL,
phkDevice,
&dwDisposition);
}
else
{
lError = RegOpenKeyExW(hRootKey,
pszKeyPath,
0,
samDesired,
phkDevice);
}
if (lError != ERROR_SUCCESS)
{
*phkDevice = NULL;
ret = CR_NO_SUCH_REGISTRY_KEY;
}
done:
if ((hRootKey != NULL) && (hRootKey != HKEY_LOCAL_MACHINE))
RegCloseKey(hRootKey);
if (pszInstancePath != NULL)
MyFree(pszInstancePath);
if (pszKeyPath != NULL)
MyFree(pszKeyPath);
return ret;
}
/***********************************************************************
* CM_Query_And_Remove_SubTreeA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_And_Remove_SubTreeA(
_In_ DEVINST dnAncestor,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Query_And_Remove_SubTreeA(%lx %p %p %lu %lx)\n",
dnAncestor, pVetoType, pszVetoName, ulNameLength, ulFlags);
return CM_Query_And_Remove_SubTree_ExA(dnAncestor, pVetoType, pszVetoName,
ulNameLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Query_And_Remove_SubTreeW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_And_Remove_SubTreeW(
_In_ DEVINST dnAncestor,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPWSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Query_And_Remove_SubTreeW(%lx %p %p %lu %lx)\n",
dnAncestor, pVetoType, pszVetoName, ulNameLength, ulFlags);
return CM_Query_And_Remove_SubTree_ExW(dnAncestor, pVetoType, pszVetoName,
ulNameLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Query_And_Remove_SubTree_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_And_Remove_SubTree_ExA(
_In_ DEVINST dnAncestor,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR lpLocalVetoName;
CONFIGRET ret;
TRACE("CM_Query_And_Remove_SubTree_ExA(%lx %p %p %lu %lx %p)\n",
dnAncestor, pVetoType, pszVetoName, ulNameLength,
ulFlags, hMachine);
if (pszVetoName == NULL && ulNameLength == 0)
return CR_INVALID_POINTER;
lpLocalVetoName = HeapAlloc(GetProcessHeap(), 0, ulNameLength * sizeof(WCHAR));
if (lpLocalVetoName == NULL)
return CR_OUT_OF_MEMORY;
ret = CM_Query_And_Remove_SubTree_ExW(dnAncestor, pVetoType, lpLocalVetoName,
ulNameLength, ulFlags, hMachine);
if (ret == CR_REMOVE_VETOED)
{
if (WideCharToMultiByte(CP_ACP,
0,
lpLocalVetoName,
ulNameLength,
pszVetoName,
ulNameLength,
NULL,
NULL) == 0)
ret = CR_FAILURE;
}
HeapFree(GetProcessHeap(), 0, lpLocalVetoName);
return ret;
}
/***********************************************************************
* CM_Query_And_Remove_SubTree_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_And_Remove_SubTree_ExW(
_In_ DEVINST dnAncestor,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPWSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Query_And_Remove_SubTree_ExW(%lx %p %p %lu %lx %p)\n",
dnAncestor, pVetoType, pszVetoName, ulNameLength,
ulFlags, hMachine);
if (dnAncestor == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_REMOVE_BITS)
return CR_INVALID_FLAG;
if (pszVetoName == NULL && ulNameLength == 0)
return CR_INVALID_POINTER;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnAncestor);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_QueryRemove(BindingHandle,
lpDevInst,
pVetoType,
pszVetoName,
ulNameLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Query_Arbitrator_Free_Data [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_Arbitrator_Free_Data(
_Out_writes_bytes_(DataLen) PVOID pData,
_In_ ULONG DataLen,
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Query_Arbitrator_Free_Data(%p %lu %lx %lu 0x%08lx)\n",
pData, DataLen, dnDevInst, ResourceID, ulFlags);
return CM_Query_Arbitrator_Free_Data_Ex(pData, DataLen, dnDevInst,
ResourceID, ulFlags, NULL);
}
/***********************************************************************
* CM_Query_Arbitrator_Free_Data_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_Arbitrator_Free_Data_Ex(
_Out_writes_bytes_(DataLen) PVOID pData,
_In_ ULONG DataLen,
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Query_Arbitrator_Free_Data_Ex(%p %lu %lx %lu 0x%08lx %p)\n",
pData, DataLen, dnDevInst, ResourceID, ulFlags, hMachine);
if (pData == NULL || DataLen == 0)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags & ~CM_QUERY_ARBITRATOR_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_QueryArbitratorFreeData(BindingHandle,
pData,
DataLen,
lpDevInst,
ResourceID,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Query_Arbitrator_Free_Size [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_Arbitrator_Free_Size(
_Out_ PULONG pulSize,
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_ ULONG ulFlags)
{
TRACE("CM_Query_Arbitrator_Free_Size(%p %lu %lx 0x%08lx)\n",
pulSize, dnDevInst,ResourceID, ulFlags);
return CM_Query_Arbitrator_Free_Size_Ex(pulSize, dnDevInst, ResourceID,
ulFlags, NULL);
}
/***********************************************************************
* CM_Query_Arbitrator_Free_Size_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_Arbitrator_Free_Size_Ex(
_Out_ PULONG pulSize,
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Query_Arbitrator_Free_Size_Ex(%p %lu %lx 0x%08lx %p)\n",
pulSize, dnDevInst,ResourceID, ulFlags, hMachine);
if (pulSize == NULL)
return CR_INVALID_POINTER;
if (dnDevInst == 0)
return CR_INVALID_DEVINST;
if (ulFlags & ~CM_QUERY_ARBITRATOR_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_QueryArbitratorFreeSize(BindingHandle,
pulSize,
lpDevInst,
ResourceID,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Query_Remove_SubTree [SETUPAPI.@]
*
* This function is obsolete in Windows XP and above.
*/
CONFIGRET
WINAPI
CM_Query_Remove_SubTree(
_In_ DEVINST dnAncestor,
_In_ ULONG ulFlags)
{
TRACE("CM_Query_Remove_SubTree(%lx %lx)\n",
dnAncestor, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Query_Remove_SubTree_Ex [SETUPAPI.@]
*
* This function is obsolete in Windows XP and above.
*/
CONFIGRET
WINAPI
CM_Query_Remove_SubTree_Ex(
_In_ DEVINST dnAncestor,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
TRACE("CM_Query_Remove_SubTree_Ex(%lx %lx %p)\n",
dnAncestor, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Query_Resource_Conflict_List [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Query_Resource_Conflict_List(
_Out_ PCONFLICT_LIST pclConflictList,
_In_ DEVINST dnDevInst,
_In_ RESOURCEID ResourceID,
_In_ PCVOID ResourceData,
_In_ ULONG ResourceLen,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
PPNP_CONFLICT_LIST pConflictBuffer = NULL;
PCONFLICT_DATA pConflictData = NULL;
ULONG ulBufferLength;
LPWSTR lpDevInst;
CONFIGRET ret;
FIXME("CM_Query_Resource_Conflict_List(%p %lx %lu %p %lu %lx %p)\n",
pclConflictList, dnDevInst, ResourceID, ResourceData,
ResourceLen, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_RESDES_WIDTH_BITS)
return CR_INVALID_FLAG;
if (pclConflictList == NULL ||
ResourceData == NULL ||
ResourceLen == 0)
return CR_INVALID_POINTER;
if (ResourceID == 0)
return CR_INVALID_RESOURCEID;
*pclConflictList = 0;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
pConflictData = MyMalloc(sizeof(CONFLICT_DATA));
if (pConflictData == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
ulBufferLength = sizeof(PNP_CONFLICT_LIST) +
sizeof(PNP_CONFLICT_STRINGS) +
(sizeof(wchar_t) * 200);
pConflictBuffer = MyMalloc(ulBufferLength);
if (pConflictBuffer == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto done;
}
RpcTryExcept
{
ret = PNP_QueryResConfList(BindingHandle,
lpDevInst,
ResourceID,
(PBYTE)ResourceData,
ResourceLen,
(PBYTE)pConflictBuffer,
ulBufferLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
if (ret != CR_SUCCESS)
goto done;
pConflictData->ulMagic = CONFLICT_MAGIC;
pConflictData->pConflictList = pConflictBuffer;
*pclConflictList = (CONFLICT_LIST)pConflictData;
done:
if (ret != CR_SUCCESS)
{
if (pConflictBuffer != NULL)
MyFree(pConflictBuffer);
if (pConflictData != NULL)
MyFree(pConflictData);
}
return ret;
}
/***********************************************************************
* CM_Reenumerate_DevNode [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Reenumerate_DevNode(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Reenumerate_DevNode(%lx %lx)\n",
dnDevInst, ulFlags);
return CM_Reenumerate_DevNode_Ex(dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Reenumerate_DevNode_Ex [SETUPAPI.@]
*/
CONFIGRET WINAPI
CM_Reenumerate_DevNode_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
FIXME("CM_Reenumerate_DevNode_Ex(%lx %lx %p)\n",
dnDevInst, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_REENUMERATE_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_DeviceInstanceAction(BindingHandle,
PNP_DEVINST_REENUMERATE,
ulFlags,
lpDevInst,
NULL);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Register_Device_Driver [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Register_Device_Driver(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Register_Device_Driver(%lx 0x%08lx)\n",
dnDevInst, ulFlags);
return CM_Register_Device_Driver_Ex(dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Register_Device_Driver_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Register_Device_Driver_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Register_Device_Driver_Ex(%lx 0x%08lx %p)\n",
dnDevInst, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_REGISTER_DEVICE_DRIVER_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_RegisterDriver(BindingHandle,
lpDevInst,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Register_Device_InterfaceA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Register_Device_InterfaceA(
_In_ DEVINST dnDevInst,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ LPCSTR pszReference,
_Out_writes_(*pulLength) LPSTR pszDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Register_Device_InterfaceA(%lx %s %s %p %p %lx)\n",
dnDevInst, debugstr_guid(InterfaceClassGuid),
pszReference, pszDeviceInterface, pulLength, ulFlags);
return CM_Register_Device_Interface_ExA(dnDevInst, InterfaceClassGuid,
pszReference, pszDeviceInterface,
pulLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Register_Device_InterfaceW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Register_Device_InterfaceW(
_In_ DEVINST dnDevInst,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ LPCWSTR pszReference,
_Out_writes_(*pulLength) LPWSTR pszDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Register_Device_InterfaceW(%lx %s %s %p %p %lx)\n",
dnDevInst, debugstr_guid(InterfaceClassGuid),
debugstr_w(pszReference), pszDeviceInterface, pulLength, ulFlags);
return CM_Register_Device_Interface_ExW(dnDevInst, InterfaceClassGuid,
pszReference, pszDeviceInterface,
pulLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Register_Device_Interface_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Register_Device_Interface_ExA(
_In_ DEVINST dnDevInst,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ LPCSTR pszReference,
_Out_writes_(*pulLength) LPSTR pszDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR pszReferenceW = NULL;
LPWSTR pszDeviceInterfaceW;
ULONG ulLength;
CONFIGRET ret;
TRACE("CM_Register_Device_Interface_ExA(%lx %s %s %p %p %lx %p)\n",
dnDevInst, debugstr_guid(InterfaceClassGuid), debugstr_a(pszReference),
pszDeviceInterface, pulLength, ulFlags, hMachine);
if (pulLength == NULL || pszDeviceInterface == NULL)
return CR_INVALID_POINTER;
if (pszReference != NULL)
{
if (pSetupCaptureAndConvertAnsiArg(pszReference, &pszReferenceW))
return CR_INVALID_DATA;
}
ulLength = *pulLength;
pszDeviceInterfaceW = HeapAlloc(GetProcessHeap(), 0, ulLength * sizeof(WCHAR));
if (pszDeviceInterfaceW == NULL)
{
ret = CR_OUT_OF_MEMORY;
goto Done;
}
ret = CM_Register_Device_Interface_ExW(dnDevInst,
InterfaceClassGuid,
pszReferenceW,
pszDeviceInterfaceW,
&ulLength,
ulFlags,
hMachine);
if (ret == CR_SUCCESS)
{
if (WideCharToMultiByte(CP_ACP,
0,
pszDeviceInterfaceW,
ulLength,
pszDeviceInterface,
*pulLength,
NULL,
NULL) == 0)
ret = CR_FAILURE;
}
*pulLength = ulLength;
Done:
if (pszDeviceInterfaceW != NULL)
HeapFree(GetProcessHeap(), 0, pszDeviceInterfaceW);
if (pszReferenceW != NULL)
MyFree(pszReferenceW);
return ret;
}
/***********************************************************************
* CM_Register_Device_Interface_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Register_Device_Interface_ExW(
_In_ DEVINST dnDevInst,
_In_ LPGUID InterfaceClassGuid,
_In_opt_ LPCWSTR pszReference,
_Out_writes_(*pulLength) LPWSTR pszDeviceInterface,
_Inout_ PULONG pulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
ULONG ulTransferLength;
CONFIGRET ret;
TRACE("CM_Register_Device_Interface_ExW(%lx %s %s %p %p %lx %p)\n",
dnDevInst, debugstr_guid(InterfaceClassGuid), debugstr_w(pszReference),
pszDeviceInterface, pulLength, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (InterfaceClassGuid == NULL ||
pszDeviceInterface == NULL ||
pulLength == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
ulTransferLength = *pulLength;
RpcTryExcept
{
ret = PNP_RegisterDeviceClassAssociation(BindingHandle,
lpDevInst,
InterfaceClassGuid,
(LPWSTR)pszReference,
pszDeviceInterface,
pulLength,
&ulTransferLength,
0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Remove_SubTree [SETUPAPI.@]
*
* This function is obsolete in Windows XP and above.
*/
CONFIGRET
WINAPI
CM_Remove_SubTree(
_In_ DEVINST dnAncestor,
_In_ ULONG ulFlags)
{
TRACE("CM_Remove_SubTree(%lx %lx)\n",
dnAncestor, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Remove_SubTree_Ex [SETUPAPI.@]
*
* This function is obsolete in Windows XP and above.
*/
CONFIGRET
WINAPI
CM_Remove_SubTree_Ex(
_In_ DEVINST dnAncestor,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
TRACE("CM_Remove_SubTree_Ex(%lx %lx %p)\n",
dnAncestor, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Request_Device_EjectA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Request_Device_EjectA(
_In_ DEVINST dnDevInst,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Request_Device_EjectA(%lx %p %p %lu %lx)\n",
dnDevInst, pVetoType, pszVetoName, ulNameLength, ulFlags);
return CM_Request_Device_Eject_ExA(dnDevInst, pVetoType, pszVetoName,
ulNameLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Request_Device_EjectW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Request_Device_EjectW(
_In_ DEVINST dnDevInst,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPWSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Request_Device_EjectW(%lx %p %p %lu %lx)\n",
dnDevInst, pVetoType, pszVetoName, ulNameLength, ulFlags);
return CM_Request_Device_Eject_ExW(dnDevInst, pVetoType, pszVetoName,
ulNameLength, ulFlags, NULL);
}
/***********************************************************************
* CM_Request_Device_Eject_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Request_Device_Eject_ExA(
_In_ DEVINST dnDevInst,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR lpLocalVetoName = NULL;
CONFIGRET ret;
TRACE("CM_Request_Device_Eject_ExA(%lx %p %p %lu %lx %p)\n",
dnDevInst, pVetoType, pszVetoName, ulNameLength, ulFlags, hMachine);
if (ulNameLength != 0)
{
if (pszVetoName == NULL)
return CR_INVALID_POINTER;
lpLocalVetoName = HeapAlloc(GetProcessHeap(), 0, ulNameLength * sizeof(WCHAR));
if (lpLocalVetoName == NULL)
return CR_OUT_OF_MEMORY;
}
ret = CM_Request_Device_Eject_ExW(dnDevInst, pVetoType, lpLocalVetoName,
ulNameLength, ulFlags, hMachine);
if (ret == CR_REMOVE_VETOED && ulNameLength != 0)
{
if (WideCharToMultiByte(CP_ACP,
0,
lpLocalVetoName,
ulNameLength,
pszVetoName,
ulNameLength,
NULL,
NULL) == 0)
ret = CR_FAILURE;
}
HeapFree(GetProcessHeap(), 0, lpLocalVetoName);
return ret;
}
/***********************************************************************
* CM_Request_Device_Eject_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Request_Device_Eject_ExW(
_In_ DEVINST dnDevInst,
_Out_opt_ PPNP_VETO_TYPE pVetoType,
_Out_writes_opt_(ulNameLength) LPWSTR pszVetoName,
_In_ ULONG ulNameLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Request_Device_Eject_ExW(%lx %p %p %lu %lx %p)\n",
dnDevInst, pVetoType, pszVetoName, ulNameLength, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (pszVetoName == NULL && ulNameLength != 0)
return CR_INVALID_POINTER;
/* Windows 2003 SP2 ignores pszVetoName when ulNameLength is zero
* and behaves like when pszVetoName is NULL */
if (ulNameLength == 0)
pszVetoName = NULL;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_RequestDeviceEject(BindingHandle,
lpDevInst,
pVetoType,
pszVetoName,
ulNameLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Request_Eject_PC [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Request_Eject_PC(VOID)
{
TRACE("CM_Request_Eject_PC()\n");
return CM_Request_Eject_PC_Ex(NULL);
}
/***********************************************************************
* CM_Request_Eject_PC_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Request_Eject_PC_Ex(
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Request_Eject_PC_Ex(%p)\n", hMachine);
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_RequestEjectPC(BindingHandle);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Run_Detection [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Run_Detection(
_In_ ULONG ulFlags)
{
TRACE("CM_Run_Detection(%lx)\n", ulFlags);
return CM_Run_Detection_Ex(ulFlags, NULL);
}
/***********************************************************************
* CM_Run_Detection_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Run_Detection_Ex(
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Run_Detection_Ex(%lx %p)\n",
ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (ulFlags & ~CM_DETECT_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_RunDetection(BindingHandle,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Set_Class_Registry_PropertyA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_Class_Registry_PropertyA(
_In_ LPGUID ClassGuid,
_In_ ULONG ulProperty,
_In_reads_bytes_opt_(ulLength) PCVOID Buffer,
_In_ ULONG ulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Set_Class_Registry_PropertyA(%p %lx %p %lu %lx %p)\n",
ClassGuid, ulProperty, Buffer, ulLength, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Set_Class_Registry_PropertyW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_Class_Registry_PropertyW(
_In_ LPGUID ClassGuid,
_In_ ULONG ulProperty,
_In_reads_bytes_opt_(ulLength) PCVOID Buffer,
_In_ ULONG ulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
FIXME("CM_Set_Class_Registry_PropertyW(%p %lx %p %lu %lx %p)\n",
ClassGuid, ulProperty, Buffer, ulLength, ulFlags, hMachine);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Set_DevNode_Problem [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_DevNode_Problem(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProblem,
_In_ ULONG ulFlags)
{
TRACE("CM_Set_DevNode_Problem(%lx %lx %lx)\n",
dnDevInst, ulProblem, ulFlags);
return CM_Set_DevNode_Problem_Ex(dnDevInst, ulProblem, ulFlags, NULL);
}
/***********************************************************************
* CM_Set_DevNode_Problem_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_DevNode_Problem_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProblem,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Set_DevNode_Problem_Ex(%lx %lx %lx %p)\n",
dnDevInst, ulProblem, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_SET_DEVNODE_PROBLEM_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_SetDeviceProblem(BindingHandle,
lpDevInst,
ulProblem,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Set_DevNode_Registry_PropertyA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_DevNode_Registry_PropertyA(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_In_reads_bytes_opt_(ulLength) PCVOID Buffer,
_In_ ULONG ulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Set_DevNode_Registry_PropertyA(%lx %lu %p %lx %lx)\n",
dnDevInst, ulProperty, Buffer, ulLength, ulFlags);
return CM_Set_DevNode_Registry_Property_ExA(dnDevInst, ulProperty,
Buffer, ulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Set_DevNode_Registry_PropertyW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_DevNode_Registry_PropertyW(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_In_reads_bytes_opt_(ulLength) PCVOID Buffer,
_In_ ULONG ulLength,
_In_ ULONG ulFlags)
{
TRACE("CM_Set_DevNode_Registry_PropertyW(%lx %lu %p %lx %lx)\n",
dnDevInst, ulProperty, Buffer, ulLength, ulFlags);
return CM_Set_DevNode_Registry_Property_ExW(dnDevInst, ulProperty,
Buffer, ulLength,
ulFlags, NULL);
}
/***********************************************************************
* CM_Set_DevNode_Registry_Property_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_DevNode_Registry_Property_ExA(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_In_reads_bytes_opt_(ulLength) PCVOID Buffer,
_In_ ULONG ulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
CONFIGRET ret = CR_SUCCESS;
LPWSTR lpBuffer;
ULONG ulType;
FIXME("CM_Set_DevNode_Registry_Property_ExA(%lx %lu %p %lx %lx %p)\n",
dnDevInst, ulProperty, Buffer, ulLength, ulFlags, hMachine);
if (Buffer == NULL && ulLength != 0)
return CR_INVALID_POINTER;
if (ulProperty < CM_DRP_MIN || ulProperty > CM_DRP_MAX)
return CR_INVALID_PROPERTY;
if (Buffer == NULL)
{
ret = CM_Set_DevNode_Registry_Property_ExW(dnDevInst,
ulProperty,
NULL,
0,
ulFlags,
hMachine);
}
else
{
/* Get property type */
ulType = GetRegistryPropertyType(ulProperty);
/* Allocate buffer if needed */
if (ulType == REG_SZ ||
ulType == REG_MULTI_SZ)
{
lpBuffer = MyMalloc(ulLength * sizeof(WCHAR));
if (lpBuffer == NULL)
{
ret = CR_OUT_OF_MEMORY;
}
else
{
if (!MultiByteToWideChar(CP_ACP, 0, Buffer,
ulLength, lpBuffer, ulLength))
{
MyFree(lpBuffer);
ret = CR_FAILURE;
}
else
{
ret = CM_Set_DevNode_Registry_Property_ExW(dnDevInst,
ulProperty,
lpBuffer,
ulLength * sizeof(WCHAR),
ulFlags,
hMachine);
MyFree(lpBuffer);
}
}
}
else
{
ret = CM_Set_DevNode_Registry_Property_ExW(dnDevInst,
ulProperty,
Buffer,
ulLength,
ulFlags,
hMachine);
}
ret = CR_CALL_NOT_IMPLEMENTED;
}
return ret;
}
/***********************************************************************
* CM_Set_DevNode_Registry_Property_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_DevNode_Registry_Property_ExW(
_In_ DEVINST dnDevInst,
_In_ ULONG ulProperty,
_In_reads_bytes_opt_(ulLength) PCVOID Buffer,
_In_ ULONG ulLength,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
ULONG ulType;
CONFIGRET ret;
TRACE("CM_Set_DevNode_Registry_Property_ExW(%lx %lu %p %lx %lx %p)\n",
dnDevInst, ulProperty, Buffer, ulLength, ulFlags, hMachine);
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulProperty < CM_DRP_MIN || ulProperty > CM_DRP_MAX)
return CR_INVALID_PROPERTY;
if (Buffer != NULL && ulLength == 0)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
/* Get property type */
ulType = GetRegistryPropertyType(ulProperty);
RpcTryExcept
{
ret = PNP_SetDeviceRegProp(BindingHandle,
lpDevInst,
ulProperty,
ulType,
(BYTE *)Buffer,
ulLength,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Set_HW_Prof [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_HW_Prof(
_In_ ULONG ulHardwareProfile,
_In_ ULONG ulFlags)
{
TRACE("CM_Set_HW_Prof(%lu %lx)\n",
ulHardwareProfile, ulFlags);
return CM_Set_HW_Prof_Ex(ulHardwareProfile, ulFlags, NULL);
}
/***********************************************************************
* CM_Set_HW_Prof_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_HW_Prof_Ex(
_In_ ULONG ulHardwareProfile,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Set_HW_Prof_Ex(%lu %lx %p)\n",
ulHardwareProfile, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_SetHwProf(BindingHandle, ulHardwareProfile, ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Set_HW_Prof_FlagsA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_HW_Prof_FlagsA(
_In_ DEVINSTID_A szDevInstName,
_In_ ULONG ulConfig,
_In_ ULONG ulValue,
_In_ ULONG ulFlags)
{
TRACE("CM_Set_HW_Prof_FlagsA(%s %lu %lu %lx)\n",
debugstr_a(szDevInstName), ulConfig, ulValue, ulFlags);
return CM_Set_HW_Prof_Flags_ExA(szDevInstName, ulConfig, ulValue,
ulFlags, NULL);
}
/***********************************************************************
* CM_Set_HW_Prof_FlagsW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_HW_Prof_FlagsW(
_In_ DEVINSTID_W szDevInstName,
_In_ ULONG ulConfig,
_In_ ULONG ulValue,
_In_ ULONG ulFlags)
{
TRACE("CM_Set_HW_Prof_FlagsW(%s %lu %lu %lx)\n",
debugstr_w(szDevInstName), ulConfig, ulValue, ulFlags);
return CM_Set_HW_Prof_Flags_ExW(szDevInstName, ulConfig, ulValue,
ulFlags, NULL);
}
/***********************************************************************
* CM_Set_HW_Prof_Flags_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_HW_Prof_Flags_ExA(
_In_ DEVINSTID_A szDevInstName,
_In_ ULONG ulConfig,
_In_ ULONG ulValue,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
DEVINSTID_W pszDevIdW = NULL;
CONFIGRET ret = CR_SUCCESS;
TRACE("CM_Set_HW_Prof_Flags_ExA(%s %lu %lu %lx %p)\n",
debugstr_a(szDevInstName), ulConfig, ulValue, ulFlags, hMachine);
if (szDevInstName != NULL)
{
if (pSetupCaptureAndConvertAnsiArg(szDevInstName, &pszDevIdW))
return CR_INVALID_DEVICE_ID;
}
ret = CM_Set_HW_Prof_Flags_ExW(pszDevIdW, ulConfig, ulValue,
ulFlags, hMachine);
if (pszDevIdW != NULL)
MyFree(pszDevIdW);
return ret;
}
/***********************************************************************
* CM_Set_HW_Prof_Flags_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Set_HW_Prof_Flags_ExW(
_In_ DEVINSTID_W szDevInstName,
_In_ ULONG ulConfig,
_In_ ULONG ulValue,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
FIXME("CM_Set_HW_Prof_Flags_ExW(%s %lu %lu %lx %p)\n",
debugstr_w(szDevInstName), ulConfig, ulValue, ulFlags, hMachine);
if (szDevInstName == NULL)
return CR_INVALID_POINTER;
if (ulFlags & ~ CM_SET_HW_PROF_FLAGS_BITS)
return CR_INVALID_FLAG;
/* FIXME: Check whether szDevInstName is valid */
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_HwProfFlags(BindingHandle, PNP_SET_HWPROFFLAGS, szDevInstName,
ulConfig, &ulValue, NULL, NULL, 0, 0);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Setup_DevNode [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Setup_DevNode(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags)
{
TRACE("CM_Setup_DevNode(%lx %lx)\n",
dnDevInst, ulFlags);
return CM_Setup_DevNode_Ex(dnDevInst, ulFlags, NULL);
}
/***********************************************************************
* CM_Setup_DevNode_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Setup_DevNode_Ex(
_In_ DEVINST dnDevInst,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
FIXME("CM_Setup_DevNode_Ex(%lx %lx %p)\n",
dnDevInst, ulFlags, hMachine);
if (!pSetupIsUserAdmin())
return CR_ACCESS_DENIED;
if (dnDevInst == 0)
return CR_INVALID_DEVNODE;
if (ulFlags & ~CM_SETUP_BITS)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnDevInst);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_DeviceInstanceAction(BindingHandle,
PNP_DEVINST_SETUP,
ulFlags,
lpDevInst,
NULL);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Test_Range_Available [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Test_Range_Available(
_In_ DWORDLONG ullStartValue,
_In_ DWORDLONG ullEndValue,
_In_ RANGE_LIST rlh,
_In_ ULONG ulFlags)
{
FIXME("CM_Test_Range_Available(%I64u %I64u %p %lx)\n",
ullStartValue, ullEndValue, rlh, ulFlags);
return CR_CALL_NOT_IMPLEMENTED;
}
/***********************************************************************
* CM_Uninstall_DevNode [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Uninstall_DevNode(
_In_ DEVINST dnPhantom,
_In_ ULONG ulFlags)
{
TRACE("CM_Uninstall_DevNode(%lx %lx)\n",
dnPhantom, ulFlags);
return CM_Uninstall_DevNode_Ex(dnPhantom, ulFlags, NULL);
}
/***********************************************************************
* CM_Uninstall_DevNode_Ex [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Uninstall_DevNode_Ex(
_In_ DEVINST dnPhantom,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
HSTRING_TABLE StringTable = NULL;
LPWSTR lpDevInst;
CONFIGRET ret;
TRACE("CM_Uninstall_DevNode_Ex(%lx %lx %p)\n",
dnPhantom, ulFlags, hMachine);
if (dnPhantom == 0)
return CR_INVALID_DEVNODE;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
StringTable = ((PMACHINE_INFO)hMachine)->StringTable;
if (StringTable == 0)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, &StringTable))
return CR_FAILURE;
}
lpDevInst = pSetupStringTableStringFromId(StringTable, dnPhantom);
if (lpDevInst == NULL)
return CR_INVALID_DEVNODE;
RpcTryExcept
{
ret = PNP_UninstallDevInst(BindingHandle,
lpDevInst,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}
/***********************************************************************
* CM_Unregister_Device_InterfaceA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Unregister_Device_InterfaceA(
_In_ LPCSTR pszDeviceInterface,
_In_ ULONG ulFlags)
{
TRACE("CM_Unregister_Device_InterfaceA(%s %lx)\n",
debugstr_a(pszDeviceInterface), ulFlags);
return CM_Unregister_Device_Interface_ExA(pszDeviceInterface,
ulFlags, NULL);
}
/***********************************************************************
* CM_Unregister_Device_InterfaceW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Unregister_Device_InterfaceW(
_In_ LPCWSTR pszDeviceInterface,
_In_ ULONG ulFlags)
{
TRACE("CM_Unregister_Device_InterfaceW(%s %lx)\n",
debugstr_w(pszDeviceInterface), ulFlags);
return CM_Unregister_Device_Interface_ExW(pszDeviceInterface,
ulFlags, NULL);
}
/***********************************************************************
* CM_Unregister_Device_Interface_ExA [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Unregister_Device_Interface_ExA(
_In_ LPCSTR pszDeviceInterface,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
LPWSTR pszDeviceInterfaceW = NULL;
CONFIGRET ret;
TRACE("CM_Unregister_Device_Interface_ExA(%s %lx %p)\n",
debugstr_a(pszDeviceInterface), ulFlags, hMachine);
if (pszDeviceInterface == NULL)
return CR_INVALID_POINTER;
if (pSetupCaptureAndConvertAnsiArg(pszDeviceInterface, &pszDeviceInterfaceW))
return CR_INVALID_DATA;
ret = CM_Unregister_Device_Interface_ExW(pszDeviceInterfaceW,
ulFlags, hMachine);
if (pszDeviceInterfaceW != NULL)
MyFree(pszDeviceInterfaceW);
return ret;
}
/***********************************************************************
* CM_Unregister_Device_Interface_ExW [SETUPAPI.@]
*/
CONFIGRET
WINAPI
CM_Unregister_Device_Interface_ExW(
_In_ LPCWSTR pszDeviceInterface,
_In_ ULONG ulFlags,
_In_opt_ HMACHINE hMachine)
{
RPC_BINDING_HANDLE BindingHandle = NULL;
CONFIGRET ret;
TRACE("CM_Unregister_Device_Interface_ExW(%s %lx %p)\n",
debugstr_w(pszDeviceInterface), ulFlags, hMachine);
if (pszDeviceInterface == NULL)
return CR_INVALID_POINTER;
if (ulFlags != 0)
return CR_INVALID_FLAG;
if (hMachine != NULL)
{
BindingHandle = ((PMACHINE_INFO)hMachine)->BindingHandle;
if (BindingHandle == NULL)
return CR_FAILURE;
}
else
{
if (!PnpGetLocalHandles(&BindingHandle, NULL))
return CR_FAILURE;
}
RpcTryExcept
{
ret = PNP_UnregisterDeviceClassAssociation(BindingHandle,
(LPWSTR)pszDeviceInterface,
ulFlags);
}
RpcExcept(EXCEPTION_EXECUTE_HANDLER)
{
ret = RpcStatusToCmStatus(RpcExceptionCode());
}
RpcEndExcept;
return ret;
}