2005-07-31 12:11:56 +00:00
|
|
|
|
/*
|
|
|
|
|
* ReactOS kernel
|
|
|
|
|
* Copyright (C) 2005 ReactOS Team
|
|
|
|
|
*
|
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
*/
|
|
|
|
|
/*
|
|
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
|
* PROJECT: ReactOS kernel
|
|
|
|
|
* FILE: services/umpnpmgr/umpnpmgr.c
|
|
|
|
|
* PURPOSE: User-mode Plug and Play manager
|
|
|
|
|
* PROGRAMMER: Eric Kohl
|
2005-10-14 18:24:19 +00:00
|
|
|
|
* Herv<EFBFBD> Poussineau (hpoussin@reactos.org)
|
2005-07-31 12:11:56 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
2005-10-19 17:03:38 +00:00
|
|
|
|
#define WIN32_NO_STATUS
|
2005-07-31 12:11:56 +00:00
|
|
|
|
#include <windows.h>
|
2006-01-08 08:17:38 +00:00
|
|
|
|
#include <cmtypes.h>
|
2006-01-08 18:55:24 +00:00
|
|
|
|
#include <cmfuncs.h>
|
|
|
|
|
#include <rtlfuncs.h>
|
2005-11-22 04:57:45 +00:00
|
|
|
|
#include <umpnpmgr/sysguid.h>
|
2005-12-01 21:37:19 +00:00
|
|
|
|
#include <wdmguid.h>
|
|
|
|
|
#include <cfgmgr32.h>
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
#include <rpc.h>
|
|
|
|
|
#include <rpcdce.h>
|
|
|
|
|
|
|
|
|
|
#include "pnp_c.h"
|
|
|
|
|
|
|
|
|
|
#define NDEBUG
|
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
|
|
/* GLOBALS ******************************************************************/
|
|
|
|
|
|
|
|
|
|
static VOID CALLBACK
|
|
|
|
|
ServiceMain(DWORD argc, LPTSTR *argv);
|
|
|
|
|
|
|
|
|
|
static SERVICE_TABLE_ENTRY ServiceTable[2] =
|
|
|
|
|
{
|
|
|
|
|
{TEXT("PlugPlay"), ServiceMain},
|
|
|
|
|
{NULL, NULL}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static WCHAR szRootDeviceId[] = L"HTREE\\ROOT\\0";
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
static HKEY hEnumKey = NULL;
|
|
|
|
|
static HKEY hClassKey = NULL;
|
|
|
|
|
|
2005-12-17 19:58:23 +00:00
|
|
|
|
static HANDLE hUserToken = NULL;
|
|
|
|
|
static HANDLE hInstallEvent = NULL;
|
|
|
|
|
|
|
|
|
|
|
2005-07-31 12:11:56 +00:00
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
|
|
|
|
|
static DWORD WINAPI
|
|
|
|
|
RpcServerThread(LPVOID lpParameter)
|
|
|
|
|
{
|
|
|
|
|
RPC_STATUS Status;
|
|
|
|
|
|
|
|
|
|
DPRINT("RpcServerThread() called\n");
|
|
|
|
|
|
|
|
|
|
Status = RpcServerUseProtseqEpW(L"ncacn_np",
|
|
|
|
|
20,
|
|
|
|
|
L"\\pipe\\umpnpmgr",
|
|
|
|
|
NULL); // Security descriptor
|
|
|
|
|
if (Status != RPC_S_OK)
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("RpcServerUseProtseqEpW() failed (Status %lx)\n", Status);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Status = RpcServerRegisterIf(pnp_v1_0_s_ifspec,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL);
|
|
|
|
|
if (Status != RPC_S_OK)
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("RpcServerRegisterIf() failed (Status %lx)\n", Status);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Status = RpcServerListen(1,
|
|
|
|
|
20,
|
|
|
|
|
FALSE);
|
|
|
|
|
if (Status != RPC_S_OK)
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("RpcServerListen() failed (Status %lx)\n", Status);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT("RpcServerThread() done\n");
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void __RPC_FAR * __RPC_USER midl_user_allocate(size_t len)
|
|
|
|
|
{
|
|
|
|
|
return HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, len);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void __RPC_USER midl_user_free(void __RPC_FAR * ptr)
|
|
|
|
|
{
|
|
|
|
|
HeapFree(GetProcessHeap(), 0, ptr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-10-03 07:06:42 +00:00
|
|
|
|
static CONFIGRET WINAPI
|
|
|
|
|
NtStatusToCrError(NTSTATUS Status)
|
|
|
|
|
{
|
|
|
|
|
switch (Status)
|
|
|
|
|
{
|
2005-12-17 12:23:41 +00:00
|
|
|
|
case STATUS_NO_SUCH_DEVICE:
|
|
|
|
|
return CR_NO_SUCH_DEVINST;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
/* FIXME: add more mappings */
|
|
|
|
|
DPRINT1("Unable to map status 0x%08lx\n", Status);
|
|
|
|
|
return CR_FAILURE;
|
2005-10-03 07:06:42 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 2 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetVersion(handle_t BindingHandle,
|
|
|
|
|
unsigned short *Version)
|
|
|
|
|
{
|
|
|
|
|
*Version = 0x0400;
|
|
|
|
|
return CR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 3 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetGlobalState(handle_t BindingHandle,
|
|
|
|
|
unsigned long *State,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
*State = CM_GLOBAL_STATE_CAN_DO_UI | CM_GLOBAL_STATE_SERVICES_AVAILABLE;
|
|
|
|
|
return CR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 4 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_InitDetection(handle_t BindingHandle)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("PNP_InitDetection() called\n");
|
|
|
|
|
return CR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* Function 5 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_ReportLogOn(handle_t BindingHandle,
|
|
|
|
|
unsigned long Admin,
|
|
|
|
|
unsigned long ProcessId)
|
|
|
|
|
{
|
|
|
|
|
HANDLE hProcess;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_ReportLogOn(%lu, %lu) called\n", Admin, ProcessId);
|
|
|
|
|
|
|
|
|
|
/* Get the users token */
|
|
|
|
|
hProcess = OpenProcess(PROCESS_ALL_ACCESS,
|
|
|
|
|
TRUE,
|
|
|
|
|
ProcessId);
|
|
|
|
|
if (hProcess != NULL)
|
|
|
|
|
{
|
|
|
|
|
if (hUserToken != NULL)
|
|
|
|
|
{
|
|
|
|
|
CloseHandle(hUserToken);
|
|
|
|
|
hUserToken = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OpenProcessToken(hProcess,
|
|
|
|
|
TOKEN_ALL_ACCESS,
|
|
|
|
|
&hUserToken);
|
|
|
|
|
CloseHandle(hProcess);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Trigger the installer thread */
|
|
|
|
|
if (hInstallEvent != NULL)
|
|
|
|
|
SetEvent(hInstallEvent);
|
|
|
|
|
|
|
|
|
|
return CR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 6 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_ValidateDeviceInstance(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
HKEY hEnumKey = NULL;
|
|
|
|
|
HKEY hDeviceKey = NULL;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_ValidateDeviceInstance(%S %lx) called\n",
|
|
|
|
|
DeviceInstance, Flags);
|
|
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
|
L"System\\CurrentControlSet\\Enum",
|
|
|
|
|
0,
|
|
|
|
|
KEY_ALL_ACCESS,
|
|
|
|
|
&hEnumKey))
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Could not open the Enum Key!\n");
|
|
|
|
|
ret = CR_FAILURE;
|
|
|
|
|
goto Done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(hEnumKey,
|
|
|
|
|
DeviceInstance,
|
|
|
|
|
0,
|
|
|
|
|
KEY_READ,
|
|
|
|
|
&hDeviceKey))
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Could not open the Device Key!\n");
|
|
|
|
|
ret = CR_NO_SUCH_DEVNODE;
|
|
|
|
|
goto Done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* FIXME: add more tests */
|
|
|
|
|
|
|
|
|
|
Done:
|
|
|
|
|
if (hDeviceKey != NULL)
|
|
|
|
|
RegCloseKey(hDeviceKey);
|
|
|
|
|
|
|
|
|
|
if (hEnumKey != NULL)
|
|
|
|
|
RegCloseKey(hEnumKey);
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_ValidateDeviceInstance() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 7 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetRootDeviceInstance(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
unsigned long Length)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_GetRootDeviceInstance() called\n");
|
|
|
|
|
|
|
|
|
|
if (Length < lstrlenW(szRootDeviceId) + 1)
|
|
|
|
|
{
|
|
|
|
|
ret = CR_BUFFER_SMALL;
|
|
|
|
|
goto Done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
lstrcpyW(DeviceInstance,
|
|
|
|
|
szRootDeviceId);
|
|
|
|
|
|
|
|
|
|
Done:
|
|
|
|
|
DPRINT("PNP_GetRootDeviceInstance() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 8 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetRelatedDeviceInstance(handle_t BindingHandle,
|
|
|
|
|
unsigned long Relationship,
|
|
|
|
|
wchar_t *DeviceId,
|
|
|
|
|
wchar_t *RelatedDeviceId,
|
|
|
|
|
unsigned long Length,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
PLUGPLAY_CONTROL_RELATED_DEVICE_DATA PlugPlayData;
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_GetRelatedDeviceInstance() called\n");
|
|
|
|
|
DPRINT(" Relationship %ld\n", Relationship);
|
|
|
|
|
DPRINT(" DeviceId %S\n", DeviceId);
|
|
|
|
|
|
|
|
|
|
RtlInitUnicodeString(&PlugPlayData.TargetDeviceInstance,
|
|
|
|
|
DeviceId);
|
|
|
|
|
|
|
|
|
|
PlugPlayData.Relation = Relationship;
|
|
|
|
|
|
2006-05-10 17:47:44 +00:00
|
|
|
|
PlugPlayData.RelatedDeviceInstanceLength = Length;
|
|
|
|
|
PlugPlayData.RelatedDeviceInstance = RelatedDeviceId;
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
Status = NtPlugPlayControl(PlugPlayControlGetRelatedDevice,
|
|
|
|
|
(PVOID)&PlugPlayData,
|
|
|
|
|
sizeof(PLUGPLAY_CONTROL_RELATED_DEVICE_DATA));
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
|
{
|
2005-10-03 07:06:42 +00:00
|
|
|
|
ret = NtStatusToCrError(Status);
|
2005-07-31 12:11:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_GetRelatedDeviceInstance() done (returns %lx)\n", ret);
|
2005-08-01 19:42:19 +00:00
|
|
|
|
if (ret == CR_SUCCESS)
|
2005-07-31 12:11:56 +00:00
|
|
|
|
{
|
|
|
|
|
DPRINT("RelatedDevice: %wZ\n", &PlugPlayData.RelatedDeviceInstance);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 9 */
|
2005-08-01 19:42:19 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_EnumerateSubKeys(handle_t BindingHandle,
|
|
|
|
|
unsigned long Branch,
|
|
|
|
|
unsigned long Index,
|
|
|
|
|
wchar_t *Buffer,
|
|
|
|
|
unsigned long Length,
|
|
|
|
|
unsigned long *RequiredLength,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
HKEY hKey;
|
|
|
|
|
DWORD dwError;
|
|
|
|
|
|
2005-09-10 19:01:44 +00:00
|
|
|
|
DPRINT("PNP_EnumerateSubKeys() called\n");
|
2005-08-01 19:42:19 +00:00
|
|
|
|
|
|
|
|
|
switch (Branch)
|
|
|
|
|
{
|
2005-09-10 19:01:44 +00:00
|
|
|
|
case PNP_BRANCH_ENUM:
|
2005-08-01 19:42:19 +00:00
|
|
|
|
hKey = hEnumKey;
|
|
|
|
|
break;
|
|
|
|
|
|
2005-09-10 19:01:44 +00:00
|
|
|
|
case PNP_BRANCH_CLASS:
|
2005-08-01 19:42:19 +00:00
|
|
|
|
hKey = hClassKey;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return CR_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*RequiredLength = Length;
|
|
|
|
|
dwError = RegEnumKeyExW(hKey,
|
|
|
|
|
Index,
|
|
|
|
|
Buffer,
|
|
|
|
|
RequiredLength,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL);
|
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
|
{
|
2005-09-10 19:01:44 +00:00
|
|
|
|
ret = (dwError == ERROR_NO_MORE_ITEMS) ? CR_NO_SUCH_VALUE : CR_FAILURE;
|
2005-08-01 19:42:19 +00:00
|
|
|
|
}
|
2005-09-10 19:01:44 +00:00
|
|
|
|
else
|
2005-08-01 19:42:19 +00:00
|
|
|
|
{
|
2005-09-11 08:45:02 +00:00
|
|
|
|
(*RequiredLength)++;
|
2005-08-01 19:42:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2005-09-10 19:01:44 +00:00
|
|
|
|
DPRINT("PNP_EnumerateSubKeys() done (returns %lx)\n", ret);
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 11 */
|
2005-09-24 09:30:43 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetDeviceListSize(handle_t BindingHandle,
|
|
|
|
|
wchar_t *Filter,
|
|
|
|
|
unsigned long *Length,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("PNP_GetDeviceListSize() called\n");
|
|
|
|
|
|
|
|
|
|
/* FIXME */
|
|
|
|
|
*Length = 2;
|
|
|
|
|
|
|
|
|
|
return CR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 12 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetDepth(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
unsigned long *Depth,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
PLUGPLAY_CONTROL_DEPTH_DATA PlugPlayData;
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
DPRINT("PNP_GetDepth() called\n");
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
RtlInitUnicodeString(&PlugPlayData.DeviceInstance,
|
|
|
|
|
DeviceInstance);
|
|
|
|
|
|
|
|
|
|
Status = NtPlugPlayControl(PlugPlayControlGetDeviceDepth,
|
|
|
|
|
(PVOID)&PlugPlayData,
|
|
|
|
|
sizeof(PLUGPLAY_CONTROL_DEPTH_DATA));
|
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
*Depth = PlugPlayData.Depth;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2005-10-03 07:06:42 +00:00
|
|
|
|
ret = NtStatusToCrError(Status);
|
2005-07-31 12:11:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
DPRINT("PNP_GetDepth() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 13 */
|
2005-08-01 19:42:19 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetDeviceRegProp(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
unsigned long Property,
|
|
|
|
|
unsigned long *DataType,
|
|
|
|
|
char *Buffer,
|
|
|
|
|
unsigned long *TransferLen,
|
|
|
|
|
unsigned long *Length,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
2005-08-07 09:03:35 +00:00
|
|
|
|
PLUGPLAY_CONTROL_PROPERTY_DATA PlugPlayData;
|
2005-08-01 19:42:19 +00:00
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
2005-08-03 20:10:32 +00:00
|
|
|
|
LPWSTR lpValueName = NULL;
|
2005-08-07 09:03:35 +00:00
|
|
|
|
HKEY hKey = 0;
|
|
|
|
|
NTSTATUS Status;
|
2005-08-01 19:42:19 +00:00
|
|
|
|
|
2005-08-07 09:03:35 +00:00
|
|
|
|
DPRINT("PNP_GetDeviceRegProp() called\n");
|
2005-08-01 19:42:19 +00:00
|
|
|
|
|
2005-08-03 20:10:32 +00:00
|
|
|
|
switch (Property)
|
|
|
|
|
{
|
|
|
|
|
case CM_DRP_DEVICEDESC:
|
|
|
|
|
lpValueName = L"DeviceDesc";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_HARDWAREID:
|
|
|
|
|
lpValueName = L"HardwareID";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_COMPATIBLEIDS:
|
|
|
|
|
lpValueName = L"CompatibleIDs";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_SERVICE:
|
|
|
|
|
lpValueName = L"Service";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CLASS:
|
|
|
|
|
lpValueName = L"Class";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CLASSGUID:
|
|
|
|
|
lpValueName = L"ClassGUID";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_DRIVER:
|
|
|
|
|
lpValueName = L"Driver";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CONFIGFLAGS:
|
|
|
|
|
lpValueName = L"ConfigFlags";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_MFG:
|
|
|
|
|
lpValueName = L"Mfg";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_FRIENDLYNAME:
|
|
|
|
|
lpValueName = L"FriendlyName";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_LOCATION_INFORMATION:
|
|
|
|
|
lpValueName = L"LocationInformation";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_PHYSICAL_DEVICE_OBJECT_NAME:
|
|
|
|
|
lpValueName = NULL;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CAPABILITIES:
|
|
|
|
|
lpValueName = L"Capabilities";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_UI_NUMBER:
|
2005-08-07 09:03:35 +00:00
|
|
|
|
lpValueName = NULL;
|
2005-08-03 20:10:32 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_UPPERFILTERS:
|
|
|
|
|
lpValueName = L"UpperFilters";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_LOWERFILTERS:
|
|
|
|
|
lpValueName = L"LowerFilters";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_BUSTYPEGUID:
|
2005-08-07 09:03:35 +00:00
|
|
|
|
lpValueName = NULL;
|
2005-08-03 20:10:32 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_LEGACYBUSTYPE:
|
2005-08-07 09:03:35 +00:00
|
|
|
|
lpValueName = NULL;
|
2005-08-03 20:10:32 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_BUSNUMBER:
|
2005-08-07 09:03:35 +00:00
|
|
|
|
lpValueName = NULL;
|
2005-08-03 20:10:32 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_ENUMERATOR_NAME:
|
2005-08-07 09:03:35 +00:00
|
|
|
|
lpValueName = NULL;
|
2005-08-03 20:10:32 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return CR_INVALID_PROPERTY;
|
|
|
|
|
}
|
|
|
|
|
|
2005-08-07 09:03:35 +00:00
|
|
|
|
DPRINT("Value name: %S\n", lpValueName);
|
2005-08-03 20:10:32 +00:00
|
|
|
|
|
|
|
|
|
if (lpValueName)
|
|
|
|
|
{
|
|
|
|
|
/* Retrieve information from the Registry */
|
2005-08-07 09:03:35 +00:00
|
|
|
|
if (RegOpenKeyExW(hEnumKey,
|
|
|
|
|
DeviceInstance,
|
|
|
|
|
0,
|
|
|
|
|
KEY_ALL_ACCESS,
|
|
|
|
|
&hKey))
|
|
|
|
|
return CR_INVALID_DEVNODE;
|
|
|
|
|
|
|
|
|
|
if (RegQueryValueExW(hKey,
|
|
|
|
|
lpValueName,
|
|
|
|
|
NULL,
|
|
|
|
|
DataType,
|
|
|
|
|
(LPBYTE)Buffer,
|
|
|
|
|
Length))
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
|
|
|
|
|
/* FIXME: Check buffer size */
|
2005-08-03 20:10:32 +00:00
|
|
|
|
|
2005-08-07 09:03:35 +00:00
|
|
|
|
RegCloseKey(hKey);
|
2005-08-03 20:10:32 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Retrieve information from the Device Node */
|
2005-08-07 09:03:35 +00:00
|
|
|
|
RtlInitUnicodeString(&PlugPlayData.DeviceInstance,
|
|
|
|
|
DeviceInstance);
|
|
|
|
|
PlugPlayData.Buffer = Buffer;
|
|
|
|
|
PlugPlayData.BufferSize = *TransferLen;
|
|
|
|
|
|
|
|
|
|
switch (Property)
|
|
|
|
|
{
|
|
|
|
|
#if 0
|
|
|
|
|
case CM_DRP_PHYSICAL_DEVICE_OBJECT_NAME:
|
|
|
|
|
PlugPlayData.Property = DevicePropertyPhysicalDeviceObjectName;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_UI_NUMBER:
|
|
|
|
|
PlugPlayData.Property = DevicePropertyUINumber;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_BUSTYPEGUID:
|
|
|
|
|
PlugPlayData.Property = DevicePropertyBusTypeGuid;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_LEGACYBUSTYPE:
|
|
|
|
|
PlugPlayData.Property = DevicePropertyLegacyBusType;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_BUSNUMBER:
|
|
|
|
|
PlugPlayData.Property = DevicePropertyBusNumber;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_ENUMERATOR_NAME:
|
|
|
|
|
PlugPlayData.Property = DevicePropertyEnumeratorName;
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return CR_INVALID_PROPERTY;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Status = NtPlugPlayControl(PlugPlayControlProperty,
|
|
|
|
|
(PVOID)&PlugPlayData,
|
|
|
|
|
sizeof(PLUGPLAY_CONTROL_PROPERTY_DATA));
|
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
*Length = PlugPlayData.BufferSize;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2005-10-03 07:06:42 +00:00
|
|
|
|
ret = NtStatusToCrError(Status);
|
2005-08-07 09:03:35 +00:00
|
|
|
|
}
|
2005-08-03 20:10:32 +00:00
|
|
|
|
}
|
|
|
|
|
|
2005-08-07 09:03:35 +00:00
|
|
|
|
DPRINT("PNP_GetDeviceRegProp() done (returns %lx)\n", ret);
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 14 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_SetDeviceRegProp(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceId,
|
|
|
|
|
unsigned long Property,
|
|
|
|
|
unsigned long DataType,
|
|
|
|
|
char *Buffer,
|
|
|
|
|
unsigned long Length,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
LPWSTR lpValueName = NULL;
|
|
|
|
|
HKEY hKey = 0;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_SetDeviceRegProp() called\n");
|
|
|
|
|
|
|
|
|
|
DPRINT("DeviceId: %S\n", DeviceId);
|
|
|
|
|
DPRINT("Property: %lu\n", Property);
|
|
|
|
|
DPRINT("DataType: %lu\n", DataType);
|
|
|
|
|
DPRINT("Length: %lu\n", Length);
|
|
|
|
|
|
|
|
|
|
switch (Property)
|
|
|
|
|
{
|
|
|
|
|
case CM_DRP_DEVICEDESC:
|
|
|
|
|
lpValueName = L"DeviceDesc";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_HARDWAREID:
|
|
|
|
|
lpValueName = L"HardwareID";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_COMPATIBLEIDS:
|
|
|
|
|
lpValueName = L"CompatibleIDs";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_SERVICE:
|
|
|
|
|
lpValueName = L"Service";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CLASS:
|
|
|
|
|
lpValueName = L"Class";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CLASSGUID:
|
|
|
|
|
lpValueName = L"ClassGUID";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_DRIVER:
|
|
|
|
|
lpValueName = L"Driver";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_CONFIGFLAGS:
|
|
|
|
|
lpValueName = L"ConfigFlags";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_MFG:
|
|
|
|
|
lpValueName = L"Mfg";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_FRIENDLYNAME:
|
|
|
|
|
lpValueName = L"FriendlyName";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_LOCATION_INFORMATION:
|
|
|
|
|
lpValueName = L"LocationInformation";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_UPPERFILTERS:
|
|
|
|
|
lpValueName = L"UpperFilters";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CM_DRP_LOWERFILTERS:
|
|
|
|
|
lpValueName = L"LowerFilters";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return CR_INVALID_PROPERTY;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT("Value name: %S\n", lpValueName);
|
|
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(hEnumKey,
|
|
|
|
|
DeviceId,
|
|
|
|
|
0,
|
|
|
|
|
KEY_ALL_ACCESS,
|
|
|
|
|
&hKey))
|
|
|
|
|
return CR_INVALID_DEVNODE;
|
|
|
|
|
|
|
|
|
|
if (Length == 0)
|
|
|
|
|
{
|
|
|
|
|
if (RegDeleteValueW(hKey,
|
|
|
|
|
lpValueName))
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (RegSetValueExW(hKey,
|
|
|
|
|
lpValueName,
|
|
|
|
|
0,
|
|
|
|
|
DataType,
|
|
|
|
|
(const BYTE*)Buffer,
|
|
|
|
|
Length))
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_SetDeviceRegProp() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* Function 15 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetClassInstance(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceId, /* in */
|
|
|
|
|
wchar_t *Buffer, /* out */
|
|
|
|
|
unsigned long Length)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_Get_Class_Instance() called\n");
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_Get_Class_Instance() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 16 */
|
2005-09-24 09:30:43 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_CreateKey(handle_t BindingHandle,
|
|
|
|
|
wchar_t *SubKey,
|
|
|
|
|
unsigned long samDesired,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_CreateKey() called\n");
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_CreateKey() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 17 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_DeleteRegistryKey(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceId,
|
|
|
|
|
wchar_t *ParentKey,
|
|
|
|
|
wchar_t *ChildKey,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_DeleteRegistryKey() called\n");
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_DeleteRegistryKey() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Function 18 */
|
|
|
|
|
#if 0
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetClassCount(handle_t BindingHandle,
|
|
|
|
|
unsigned long *ClassCount,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
HANDLE hKey = NULL;
|
|
|
|
|
DWORD dwError;
|
|
|
|
|
|
|
|
|
|
dwError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
|
pszRegPathClass,
|
|
|
|
|
0,
|
|
|
|
|
KEY_QUERY_VALUE,
|
|
|
|
|
&hKey);
|
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
|
return CR_INVALID_DATA;
|
|
|
|
|
|
|
|
|
|
dwError = RegQueryInfoKeyW(hKey,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
&ClassCount,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL);
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
|
return CR_INVALID_DATA;
|
|
|
|
|
|
|
|
|
|
return CR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Function 19 */
|
2005-08-28 15:07:45 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetClassName(handle_t BindingHandle,
|
2005-09-10 13:02:02 +00:00
|
|
|
|
wchar_t *ClassGuid,
|
|
|
|
|
wchar_t *Buffer,
|
|
|
|
|
unsigned long *Length,
|
2005-08-28 15:07:45 +00:00
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
WCHAR szKeyName[MAX_PATH];
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
HKEY hKey = NULL;
|
|
|
|
|
ULONG ulSize;
|
|
|
|
|
|
2005-09-10 13:02:02 +00:00
|
|
|
|
DPRINT("PNP_GetClassName() called\n");
|
2005-08-28 15:07:45 +00:00
|
|
|
|
|
|
|
|
|
lstrcpyW(szKeyName, L"System\\CurrentControlSet\\Control\\Class");
|
|
|
|
|
lstrcatW(szKeyName, L"\\");
|
2006-06-24 10:37:55 +00:00
|
|
|
|
if(lstrlenW(ClassGuid) < sizeof(szKeyName)/sizeof(WCHAR)-lstrlenW(szKeyName))
|
2006-06-23 14:55:52 +00:00
|
|
|
|
lstrcatW(szKeyName, ClassGuid);
|
|
|
|
|
else return CR_INVALID_DATA;
|
2005-08-28 15:07:45 +00:00
|
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
|
szKeyName,
|
|
|
|
|
0,
|
|
|
|
|
KEY_QUERY_VALUE,
|
|
|
|
|
&hKey))
|
|
|
|
|
return CR_REGISTRY_ERROR;
|
|
|
|
|
|
|
|
|
|
ulSize = *Length * sizeof(WCHAR);
|
|
|
|
|
if (RegQueryValueExW(hKey,
|
|
|
|
|
L"Class",
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
(LPBYTE)Buffer,
|
|
|
|
|
&ulSize))
|
|
|
|
|
{
|
|
|
|
|
*Length = 0;
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*Length = ulSize / sizeof(WCHAR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
|
|
2005-09-10 13:02:02 +00:00
|
|
|
|
DPRINT("PNP_GetClassName() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 20 */
|
2005-09-10 13:02:02 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_DeleteClassKey(handle_t BindingHandle,
|
|
|
|
|
wchar_t *ClassGuid,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_GetClassName(%S, %lx) called\n", ClassGuid, Flags);
|
|
|
|
|
|
|
|
|
|
if (Flags & CM_DELETE_CLASS_SUBKEYS)
|
|
|
|
|
{
|
|
|
|
|
if (RegDeleteTreeW(hClassKey, ClassGuid) != ERROR_SUCCESS)
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (RegDeleteKeyW(hClassKey, ClassGuid) != ERROR_SUCCESS)
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_DeleteClassKey() done (returns %lx)\n", ret);
|
2005-08-28 15:07:45 +00:00
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-01-22 14:33:04 +00:00
|
|
|
|
/* Function 28 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_CreateDevInst(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceId, /* [in, out, string, size_is(Length)] */
|
|
|
|
|
wchar_t *ParentDeviceId, /* [in, string] */
|
|
|
|
|
unsigned long Length, /* [in] */
|
|
|
|
|
unsigned long Flags) /* [in] */
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_CALL_NOT_IMPLEMENTED;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_CreateDevInst() called\n");
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_CreateDevInst() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 29 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_DeviceInstanceAction(handle_t BindingHandle,
|
|
|
|
|
unsigned long MajorAction,
|
|
|
|
|
unsigned long MinorAction,
|
|
|
|
|
wchar_t *DeviceInstance1,
|
|
|
|
|
wchar_t *DeviceInstance2)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_DeviceInstanceAction() called\n");
|
|
|
|
|
|
|
|
|
|
switch (MajorAction)
|
|
|
|
|
{
|
2005-12-17 19:58:23 +00:00
|
|
|
|
case 2:
|
|
|
|
|
DPRINT("Move device instance\n");
|
|
|
|
|
/* FIXME */
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
|
DPRINT("Setup device instance\n");
|
|
|
|
|
/* FIXME */
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED;
|
|
|
|
|
break;
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
case 4:
|
|
|
|
|
DPRINT("Enable device instance\n");
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* FIXME */
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED;
|
2005-12-17 12:23:41 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
DPRINT("Disable device instance\n");
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* FIXME */
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED;
|
2005-12-17 12:23:41 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 7:
|
|
|
|
|
DPRINT("Reenumerate device instance\n");
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* FIXME */
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED;
|
2005-12-17 12:23:41 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
DPRINT1("Unknown function %lu\n", MajorAction);
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT("PNP_DeviceInstanceAction() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Function 30 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetDeviceStatus(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
unsigned long *pStatus,
|
|
|
|
|
unsigned long *pProblem,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
PLUGPLAY_CONTROL_STATUS_DATA PlugPlayData;
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
DPRINT("PNP_GetDeviceStatus() called\n");
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
RtlInitUnicodeString(&PlugPlayData.DeviceInstance,
|
|
|
|
|
DeviceInstance);
|
|
|
|
|
PlugPlayData.Operation = 0; /* Get status */
|
|
|
|
|
|
|
|
|
|
Status = NtPlugPlayControl(PlugPlayControlDeviceStatus,
|
|
|
|
|
(PVOID)&PlugPlayData,
|
|
|
|
|
sizeof(PLUGPLAY_CONTROL_STATUS_DATA));
|
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
*pStatus = PlugPlayData.DeviceStatus;
|
|
|
|
|
*pProblem = PlugPlayData.DeviceProblem;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2005-10-03 07:06:42 +00:00
|
|
|
|
ret = NtStatusToCrError(Status);
|
2005-07-31 12:11:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
DPRINT("PNP_GetDeviceStatus() done (returns %lx)\n", ret);
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* Function 31 */
|
2005-07-31 12:11:56 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_SetDeviceProblem(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
unsigned long Problem,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_SetDeviceProblem() called\n");
|
|
|
|
|
|
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_SetDeviceProblem() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 19:58:23 +00:00
|
|
|
|
/* Function 33 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_UninstallDevInst(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_UninstallDevInst() called\n");
|
|
|
|
|
|
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_UninstallDevInst() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-06-11 13:20:46 +00:00
|
|
|
|
static BOOL
|
|
|
|
|
CheckForDeviceId(LPWSTR lpDeviceIdList,
|
|
|
|
|
LPWSTR lpDeviceId)
|
|
|
|
|
{
|
|
|
|
|
LPWSTR lpPtr;
|
|
|
|
|
DWORD dwLength;
|
|
|
|
|
|
|
|
|
|
lpPtr = lpDeviceIdList;
|
|
|
|
|
while (*lpPtr != 0)
|
|
|
|
|
{
|
|
|
|
|
dwLength = wcslen(lpPtr);
|
|
|
|
|
if (!_wcsicmp(lpPtr, lpDeviceId))
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
lpPtr += (dwLength + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static VOID
|
|
|
|
|
AppendDeviceId(LPWSTR lpDeviceIdList,
|
|
|
|
|
LPDWORD lpDeviceIdListSize,
|
|
|
|
|
LPWSTR lpDeviceId)
|
|
|
|
|
{
|
|
|
|
|
DWORD dwLen;
|
|
|
|
|
DWORD dwPos;
|
|
|
|
|
|
|
|
|
|
dwLen = wcslen(lpDeviceId);
|
|
|
|
|
dwPos = (*lpDeviceIdListSize / sizeof(WCHAR)) - 1;
|
|
|
|
|
|
|
|
|
|
wcscpy(&lpDeviceIdList[dwPos], lpDeviceId);
|
|
|
|
|
|
|
|
|
|
dwPos += (dwLen + 1);
|
|
|
|
|
|
|
|
|
|
lpDeviceIdList[dwPos] = 0;
|
|
|
|
|
|
|
|
|
|
*lpDeviceIdListSize = dwPos * sizeof(WCHAR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-06-04 19:12:58 +00:00
|
|
|
|
/* Function 34 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_AddID(handle_t BindingHandle,
|
2006-06-11 13:20:46 +00:00
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
wchar_t *DeviceId,
|
2006-06-04 19:12:58 +00:00
|
|
|
|
DWORD Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
2006-06-11 13:20:46 +00:00
|
|
|
|
HKEY hDeviceKey;
|
|
|
|
|
LPWSTR pszSubKey;
|
|
|
|
|
DWORD dwDeviceIdListSize;
|
|
|
|
|
WCHAR szDeviceIdList[512];
|
2006-06-04 19:12:58 +00:00
|
|
|
|
|
2006-06-11 13:20:46 +00:00
|
|
|
|
DPRINT("PNP_AddID() called\n");
|
|
|
|
|
DPRINT(" DeviceInstance: %S\n", DeviceInstance);
|
|
|
|
|
DPRINT(" DeviceId: %S\n", DeviceId);
|
|
|
|
|
DPRINT(" Flags: %lx\n", Flags);
|
2006-06-04 19:12:58 +00:00
|
|
|
|
|
2006-06-11 13:20:46 +00:00
|
|
|
|
if (RegOpenKeyExW(hEnumKey,
|
|
|
|
|
DeviceInstance,
|
|
|
|
|
0,
|
|
|
|
|
KEY_QUERY_VALUE | KEY_SET_VALUE,
|
|
|
|
|
&hDeviceKey) != ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Failed to open the device key!\n");
|
|
|
|
|
return CR_INVALID_DEVNODE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pszSubKey = (Flags & CM_ADD_ID_COMPATIBLE) ? L"CompatibleIDs" : L"HardwareID";
|
|
|
|
|
|
|
|
|
|
dwDeviceIdListSize = 512 * sizeof(WCHAR);
|
|
|
|
|
if (RegQueryValueExW(hDeviceKey,
|
|
|
|
|
pszSubKey,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
(LPBYTE)szDeviceIdList,
|
|
|
|
|
&dwDeviceIdListSize) != ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Failed to query the desired ID string!\n");
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
goto Done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check whether the device ID is already in use */
|
|
|
|
|
if (CheckForDeviceId(szDeviceIdList, DeviceId))
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Device ID was found in the ID string!\n");
|
|
|
|
|
ret = CR_SUCCESS;
|
|
|
|
|
goto Done;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Append the Device ID */
|
|
|
|
|
AppendDeviceId(szDeviceIdList, &dwDeviceIdListSize, DeviceId);
|
|
|
|
|
|
|
|
|
|
if (RegSetValueExW(hDeviceKey,
|
|
|
|
|
pszSubKey,
|
|
|
|
|
0,
|
|
|
|
|
REG_MULTI_SZ,
|
|
|
|
|
(LPBYTE)szDeviceIdList,
|
|
|
|
|
dwDeviceIdListSize) != ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Failed to set the desired ID string!\n");
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Done:
|
|
|
|
|
RegCloseKey(hDeviceKey);
|
2006-06-04 19:12:58 +00:00
|
|
|
|
|
2006-06-11 13:20:46 +00:00
|
|
|
|
DPRINT("PNP_AddID() done (returns %lx)\n", ret);
|
2006-06-04 19:12:58 +00:00
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-10-02 10:07:25 +00:00
|
|
|
|
/* Function 38 */
|
2005-09-25 21:02:43 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_IsDockStationPresent(handle_t BindingHandle,
|
|
|
|
|
unsigned long *Present)
|
|
|
|
|
{
|
|
|
|
|
HKEY hKey;
|
|
|
|
|
DWORD dwType;
|
|
|
|
|
DWORD dwValue;
|
|
|
|
|
DWORD dwSize;
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_IsDockStationPresent() called\n");
|
|
|
|
|
|
|
|
|
|
*Present = FALSE;
|
|
|
|
|
|
|
|
|
|
if (RegOpenKeyExW(HKEY_CURRENT_CONFIG,
|
|
|
|
|
L"CurrentDockInfo",
|
|
|
|
|
0,
|
|
|
|
|
KEY_READ,
|
|
|
|
|
&hKey) != ERROR_SUCCESS)
|
|
|
|
|
return CR_REGISTRY_ERROR;
|
|
|
|
|
|
|
|
|
|
dwSize = sizeof(DWORD);
|
|
|
|
|
if (RegQueryValueExW(hKey,
|
|
|
|
|
L"DockingState",
|
|
|
|
|
NULL,
|
|
|
|
|
&dwType,
|
|
|
|
|
(LPBYTE)&dwValue,
|
|
|
|
|
&dwSize) != ERROR_SUCCESS)
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
|
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
|
|
|
|
|
|
if (ret == CR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
if (dwType != REG_DWORD || dwSize != sizeof(DWORD))
|
|
|
|
|
{
|
|
|
|
|
ret = CR_REGISTRY_ERROR;
|
|
|
|
|
}
|
|
|
|
|
else if (dwValue != 0)
|
|
|
|
|
{
|
|
|
|
|
*Present = TRUE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_IsDockStationPresent() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-10-02 10:07:25 +00:00
|
|
|
|
/* Function 39 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_RequestEjectPC(handle_t BindingHandle)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_RequestEjectPC() called\n");
|
|
|
|
|
|
|
|
|
|
ret = CR_FAILURE; /* FIXME */
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_RequestEjectPC() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2006-06-18 14:36:42 +00:00
|
|
|
|
/* Function 40 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_HwProfFlags(handle_t BindingHandle,
|
|
|
|
|
unsigned long Action,
|
|
|
|
|
wchar_t *DeviceId,
|
|
|
|
|
unsigned long ProfileId,
|
|
|
|
|
unsigned long *Value, // out
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_HwProfFlags() called\n");
|
|
|
|
|
|
|
|
|
|
ret = CR_CALL_NOT_IMPLEMENTED; /* FIXME */
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_HwProfFlags() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-07-08 21:37:32 +00:00
|
|
|
|
/* Function 42 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_AddEmptyLogConf(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
ULONG ulPriority,
|
2006-07-13 21:28:01 +00:00
|
|
|
|
ULONG *pulLogConfTag,
|
2006-07-08 21:37:32 +00:00
|
|
|
|
ULONG ulFlags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_AddEmptyLogConf() called\n");
|
|
|
|
|
|
2006-07-13 21:28:01 +00:00
|
|
|
|
*pulLogConfTag = 0; /* FIXME */
|
2006-07-08 21:37:32 +00:00
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_AddEmptyLogConf() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-07-13 21:28:01 +00:00
|
|
|
|
/* Function 43 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_FreeLogConf(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
ULONG ulType,
|
|
|
|
|
ULONG ulLogConfTag,
|
|
|
|
|
ULONG ulFlags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_FreeLogConf() called\n");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_FreeLogConf() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-07-22 14:45:14 +00:00
|
|
|
|
/* Function 44 */
|
2006-07-16 12:11:26 +00:00
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetFirstLogConf(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
ULONG ulPriority,
|
|
|
|
|
ULONG *pulLogConfTag,
|
|
|
|
|
ULONG ulFlags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_GetFirstLogConf() called\n");
|
|
|
|
|
|
|
|
|
|
*pulLogConfTag = 0; /* FIXME */
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_GetFirstLogConf() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-07-22 14:45:14 +00:00
|
|
|
|
/* Function 45 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_GetNextLogConf(handle_t BindingHandle,
|
|
|
|
|
wchar_t *DeviceInstance,
|
|
|
|
|
ULONG ulLogConfType,
|
|
|
|
|
ULONG ulCurrentTag,
|
|
|
|
|
ULONG *pulNextTag,
|
|
|
|
|
ULONG ulFlags)
|
|
|
|
|
{
|
|
|
|
|
CONFIGRET ret = CR_SUCCESS;
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_GetNextLogConf() called\n");
|
|
|
|
|
|
|
|
|
|
*pulNextTag = 0; /* FIXME */
|
|
|
|
|
|
|
|
|
|
DPRINT1("PNP_GetNextLogConf() done (returns %lx)\n", ret);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
/* Function 58 */
|
|
|
|
|
CONFIGRET
|
|
|
|
|
PNP_RunDetection(handle_t BindingHandle,
|
|
|
|
|
unsigned long Flags)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("PNP_RunDetection() called\n");
|
|
|
|
|
return CR_CALL_NOT_IMPLEMENTED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-10-15 16:39:22 +00:00
|
|
|
|
typedef BOOL (WINAPI *PDEV_INSTALL_W)(HWND, HINSTANCE, LPCWSTR, INT);
|
2005-10-14 18:24:19 +00:00
|
|
|
|
|
|
|
|
|
static BOOL
|
2005-11-10 08:37:40 +00:00
|
|
|
|
InstallDevice(PCWSTR DeviceInstance, BOOL SetupIsActive)
|
2005-10-14 18:24:19 +00:00
|
|
|
|
{
|
|
|
|
|
PLUGPLAY_CONTROL_STATUS_DATA PlugPlayData;
|
|
|
|
|
HMODULE hNewDev = NULL;
|
|
|
|
|
PDEV_INSTALL_W DevInstallW;
|
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
BOOL DeviceInstalled = FALSE;
|
|
|
|
|
|
|
|
|
|
RtlInitUnicodeString(&PlugPlayData.DeviceInstance,
|
|
|
|
|
DeviceInstance);
|
|
|
|
|
PlugPlayData.Operation = 0; /* Get status */
|
|
|
|
|
|
|
|
|
|
/* Get device status */
|
|
|
|
|
Status = NtPlugPlayControl(PlugPlayControlDeviceStatus,
|
|
|
|
|
(PVOID)&PlugPlayData,
|
|
|
|
|
sizeof(PLUGPLAY_CONTROL_STATUS_DATA));
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
if (PlugPlayData.DeviceStatus & DNF_STARTED || PlugPlayData.DeviceStatus & DNF_START_FAILED)
|
|
|
|
|
/* Device is already started, or disabled due to some problem. Don't install it */
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
/* Install device */
|
|
|
|
|
SetEnvironmentVariable(L"USERPROFILE", L"."); /* FIXME: why is it needed? */
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-10-14 18:24:19 +00:00
|
|
|
|
hNewDev = LoadLibraryW(L"newdev.dll");
|
|
|
|
|
if (!hNewDev)
|
|
|
|
|
goto cleanup;
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-10-14 18:24:19 +00:00
|
|
|
|
DevInstallW = (PDEV_INSTALL_W)GetProcAddress(hNewDev, (LPCSTR)"DevInstallW");
|
|
|
|
|
if (!DevInstallW)
|
|
|
|
|
goto cleanup;
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-11-10 08:37:40 +00:00
|
|
|
|
if (!DevInstallW(NULL, NULL, DeviceInstance, SetupIsActive ? SW_HIDE : SW_SHOWNOACTIVATE))
|
2005-10-14 18:24:19 +00:00
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
|
|
DeviceInstalled = TRUE;
|
|
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
|
if (hNewDev != NULL)
|
|
|
|
|
FreeLibrary(hNewDev);
|
|
|
|
|
|
|
|
|
|
return DeviceInstalled;
|
|
|
|
|
}
|
2005-10-02 10:07:25 +00:00
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-11-10 08:37:40 +00:00
|
|
|
|
static BOOL
|
|
|
|
|
SetupIsActive(VOID)
|
|
|
|
|
{
|
2006-06-02 14:37:25 +00:00
|
|
|
|
HKEY hKey = NULL;
|
2005-11-10 08:37:40 +00:00
|
|
|
|
DWORD regType, active, size;
|
|
|
|
|
LONG rc;
|
|
|
|
|
BOOL ret = FALSE;
|
|
|
|
|
|
|
|
|
|
rc = RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SYSTEM\\Setup", 0, KEY_QUERY_VALUE, &hKey);
|
|
|
|
|
if (rc != ERROR_SUCCESS)
|
|
|
|
|
goto cleanup;
|
|
|
|
|
|
2005-11-10 08:47:43 +00:00
|
|
|
|
size = sizeof(DWORD);
|
2005-11-10 08:37:40 +00:00
|
|
|
|
rc = RegQueryValueExW(hKey, L"SystemSetupInProgress", NULL, ®Type, (LPBYTE)&active, &size);
|
|
|
|
|
if (rc != ERROR_SUCCESS)
|
|
|
|
|
goto cleanup;
|
2005-11-10 08:47:43 +00:00
|
|
|
|
if (regType != REG_DWORD || size != sizeof(DWORD))
|
2005-11-10 08:37:40 +00:00
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
|
|
ret = (active != 0);
|
|
|
|
|
|
|
|
|
|
cleanup:
|
2006-06-02 14:37:25 +00:00
|
|
|
|
if (hKey != NULL)
|
2005-11-10 08:37:40 +00:00
|
|
|
|
RegCloseKey(hKey);
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-11-10 08:37:40 +00:00
|
|
|
|
DPRINT("System setup in progress? %S\n", ret ? L"YES" : L"NO");
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-11-10 08:37:40 +00:00
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2005-12-17 12:23:41 +00:00
|
|
|
|
|
2005-07-31 12:11:56 +00:00
|
|
|
|
static DWORD WINAPI
|
|
|
|
|
PnpEventThread(LPVOID lpParameter)
|
|
|
|
|
{
|
|
|
|
|
PPLUGPLAY_EVENT_BLOCK PnpEvent;
|
|
|
|
|
ULONG PnpEventSize;
|
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
RPC_STATUS RpcStatus;
|
2005-11-10 08:37:40 +00:00
|
|
|
|
BOOL setupActive;
|
2005-07-31 12:11:56 +00:00
|
|
|
|
|
|
|
|
|
PnpEventSize = 0x1000;
|
|
|
|
|
PnpEvent = HeapAlloc(GetProcessHeap(), 0, PnpEventSize);
|
|
|
|
|
if (PnpEvent == NULL)
|
|
|
|
|
return ERROR_OUTOFMEMORY;
|
|
|
|
|
|
2005-11-10 08:37:40 +00:00
|
|
|
|
setupActive = SetupIsActive();
|
|
|
|
|
|
2005-07-31 12:11:56 +00:00
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("Calling NtGetPlugPlayEvent()\n");
|
|
|
|
|
|
|
|
|
|
/* Wait for the next pnp event */
|
|
|
|
|
Status = NtGetPlugPlayEvent(0, 0, PnpEvent, PnpEventSize);
|
|
|
|
|
|
|
|
|
|
/* Resize the buffer for the PnP event if it's too small. */
|
|
|
|
|
if (Status == STATUS_BUFFER_TOO_SMALL)
|
|
|
|
|
{
|
|
|
|
|
PnpEventSize += 0x400;
|
|
|
|
|
PnpEvent = HeapReAlloc(GetProcessHeap(), 0, PnpEvent, PnpEventSize);
|
|
|
|
|
if (PnpEvent == NULL)
|
|
|
|
|
return ERROR_OUTOFMEMORY;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
DPRINT("NtPlugPlayEvent() failed (Status %lx)\n", Status);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DPRINT("Received PnP Event\n");
|
|
|
|
|
if (UuidEqual(&PnpEvent->EventGuid, (UUID*)&GUID_DEVICE_ARRIVAL, &RpcStatus))
|
|
|
|
|
{
|
2005-09-06 18:07:46 +00:00
|
|
|
|
DPRINT("Device arrival event: %S\n", PnpEvent->TargetDevice.DeviceIds);
|
2005-11-10 08:37:40 +00:00
|
|
|
|
InstallDevice(PnpEvent->TargetDevice.DeviceIds, setupActive);
|
2005-07-31 12:11:56 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("Unknown event\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* FIXME: Process the pnp event */
|
|
|
|
|
|
|
|
|
|
/* Dequeue the current pnp event and signal the next one */
|
|
|
|
|
NtPlugPlayControl(PlugPlayControlUserResponse, NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HeapFree(GetProcessHeap(), 0, PnpEvent);
|
|
|
|
|
|
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static VOID CALLBACK
|
|
|
|
|
ServiceMain(DWORD argc, LPTSTR *argv)
|
|
|
|
|
{
|
|
|
|
|
HANDLE hThread;
|
|
|
|
|
DWORD dwThreadId;
|
|
|
|
|
|
|
|
|
|
DPRINT("ServiceMain() called\n");
|
|
|
|
|
|
|
|
|
|
hThread = CreateThread(NULL,
|
|
|
|
|
0,
|
|
|
|
|
PnpEventThread,
|
|
|
|
|
NULL,
|
|
|
|
|
0,
|
|
|
|
|
&dwThreadId);
|
|
|
|
|
if (hThread != NULL)
|
|
|
|
|
CloseHandle(hThread);
|
|
|
|
|
|
|
|
|
|
hThread = CreateThread(NULL,
|
|
|
|
|
0,
|
|
|
|
|
RpcServerThread,
|
|
|
|
|
NULL,
|
|
|
|
|
0,
|
|
|
|
|
&dwThreadId);
|
|
|
|
|
if (hThread != NULL)
|
|
|
|
|
CloseHandle(hThread);
|
|
|
|
|
|
|
|
|
|
DPRINT("ServiceMain() done\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
main(int argc, char *argv[])
|
|
|
|
|
{
|
2005-08-01 19:42:19 +00:00
|
|
|
|
DWORD dwError;
|
|
|
|
|
|
2005-07-31 12:11:56 +00:00
|
|
|
|
DPRINT("Umpnpmgr: main() started\n");
|
|
|
|
|
|
2005-12-17 19:58:23 +00:00
|
|
|
|
hInstallEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
|
|
|
|
|
if (hInstallEvent == NULL)
|
|
|
|
|
{
|
|
|
|
|
dwError = GetLastError();
|
|
|
|
|
DPRINT1("Could not create the Install Event! (Error %lu)\n", dwError);
|
|
|
|
|
return dwError;
|
|
|
|
|
}
|
|
|
|
|
|
2005-08-01 19:42:19 +00:00
|
|
|
|
dwError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
|
L"System\\CurrentControlSet\\Enum",
|
|
|
|
|
0,
|
|
|
|
|
KEY_ALL_ACCESS,
|
|
|
|
|
&hEnumKey);
|
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("Could not open the Enum Key! (Error %lu)\n", dwError);
|
|
|
|
|
return dwError;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dwError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
|
L"System\\CurrentControlSet\\Control\\Class",
|
|
|
|
|
0,
|
|
|
|
|
KEY_ALL_ACCESS,
|
|
|
|
|
&hClassKey);
|
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("Could not open the Class Key! (Error %lu)\n", dwError);
|
|
|
|
|
return dwError;
|
|
|
|
|
}
|
|
|
|
|
|
2005-07-31 12:11:56 +00:00
|
|
|
|
StartServiceCtrlDispatcher(ServiceTable);
|
|
|
|
|
|
|
|
|
|
DPRINT("Umpnpmgr: main() done\n");
|
|
|
|
|
|
|
|
|
|
ExitThread(0);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* EOF */
|