2006-12-10 15:09:46 +00:00
|
|
|
/*
|
2007-03-14 19:50:47 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS Kernel Streaming
|
2009-09-11 06:33:55 +00:00
|
|
|
* FILE: drivers/wdm/audio/backpln/portcls/api.cpp
|
2007-03-14 19:50:47 +00:00
|
|
|
* PURPOSE: Port Class driver / DriverEntry and IRP handlers
|
|
|
|
* PROGRAMMER: Andrew Greenwood
|
2009-09-11 06:33:55 +00:00
|
|
|
* Johannes Anderwald
|
2007-03-14 19:50:47 +00:00
|
|
|
* HISTORY:
|
|
|
|
* 27 Jan 07 Created
|
|
|
|
*/
|
2007-04-20 16:45:18 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
#include "private.hpp"
|
|
|
|
|
2014-01-03 17:10:56 +00:00
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
//
|
|
|
|
// This is called from DriverEntry so that PortCls can take care of some
|
|
|
|
// IRPs and map some others to the main KS driver. In most cases this will
|
|
|
|
// be the first function called by an audio driver.
|
|
|
|
//
|
|
|
|
// First 2 parameters are from DriverEntry.
|
|
|
|
//
|
|
|
|
// The AddDevice parameter is a driver-supplied pointer to a function which
|
|
|
|
// typically then calls PcAddAdapterDevice (see below.)
|
|
|
|
//
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2006-12-10 15:09:46 +00:00
|
|
|
PcInitializeAdapterDriver(
|
|
|
|
IN PDRIVER_OBJECT DriverObject,
|
|
|
|
IN PUNICODE_STRING RegistryPathName,
|
|
|
|
IN PDRIVER_ADD_DEVICE AddDevice)
|
|
|
|
{
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("PcInitializeAdapterDriver\n");
|
2009-09-11 06:33:55 +00:00
|
|
|
PC_ASSERT_IRQL_EQUAL(PASSIVE_LEVEL);
|
2007-03-14 19:50:47 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// Our IRP handlers
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("Setting IRP handlers\n");
|
2007-03-14 19:50:47 +00:00
|
|
|
DriverObject->MajorFunction[IRP_MJ_CREATE] = PcDispatchIrp;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_PNP] = PcDispatchIrp;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_POWER] = PcDispatchIrp;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = PcDispatchIrp;
|
2009-05-29 12:40:09 +00:00
|
|
|
DriverObject->MajorFunction[IRP_MJ_SHUTDOWN] = PcDispatchIrp;
|
2007-03-14 19:50:47 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// The driver-supplied AddDevice
|
2007-03-14 19:50:47 +00:00
|
|
|
DriverObject->DriverExtension->AddDevice = AddDevice;
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// KS handles these
|
2009-04-03 17:06:16 +00:00
|
|
|
DPRINT("Setting KS function handlers\n");
|
2007-03-14 19:50:47 +00:00
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_CLOSE);
|
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_DEVICE_CONTROL);
|
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_FLUSH_BUFFERS);
|
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_QUERY_SECURITY);
|
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_READ);
|
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_SET_SECURITY);
|
|
|
|
KsSetMajorFunctionHandler(DriverObject, IRP_MJ_WRITE);
|
|
|
|
|
2009-04-03 17:06:16 +00:00
|
|
|
DPRINT("PortCls has finished initializing the adapter driver\n");
|
2006-12-10 15:09:46 +00:00
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
//
|
|
|
|
// Typically called by a driver's AddDevice function, which is set when
|
|
|
|
// calling PcInitializeAdapterDriver. This performs some common driver
|
|
|
|
// operations, such as creating a device extension.
|
|
|
|
//
|
|
|
|
// The StartDevice parameter is a driver-supplied function which gets
|
|
|
|
// called in response to IRP_MJ_PNP / IRP_MN_START_DEVICE.
|
|
|
|
//
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2006-12-10 15:09:46 +00:00
|
|
|
PcAddAdapterDevice(
|
|
|
|
IN PDRIVER_OBJECT DriverObject,
|
|
|
|
IN PDEVICE_OBJECT PhysicalDeviceObject,
|
|
|
|
IN PCPFNSTARTDEVICE StartDevice,
|
|
|
|
IN ULONG MaxObjects,
|
|
|
|
IN ULONG DeviceExtensionSize)
|
|
|
|
{
|
2007-03-14 19:50:47 +00:00
|
|
|
NTSTATUS status = STATUS_UNSUCCESSFUL;
|
2009-08-11 08:49:30 +00:00
|
|
|
PDEVICE_OBJECT fdo;
|
2009-01-17 11:19:27 +00:00
|
|
|
PDEVICE_OBJECT PrevDeviceObject;
|
2009-04-14 07:21:05 +00:00
|
|
|
PPCLASS_DEVICE_EXTENSION portcls_ext = NULL;
|
2007-03-14 19:50:47 +00:00
|
|
|
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("PcAddAdapterDevice called\n");
|
2009-09-11 06:33:55 +00:00
|
|
|
PC_ASSERT_IRQL_EQUAL(PASSIVE_LEVEL);
|
2007-03-14 19:50:47 +00:00
|
|
|
|
2009-01-17 11:19:27 +00:00
|
|
|
if (!DriverObject || !PhysicalDeviceObject || !StartDevice)
|
2007-03-14 19:50:47 +00:00
|
|
|
{
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// check if the DeviceExtensionSize is provided
|
2007-03-14 19:50:47 +00:00
|
|
|
if ( DeviceExtensionSize < PORT_CLASS_DEVICE_EXTENSION_SIZE )
|
|
|
|
{
|
2021-06-11 12:29:21 +00:00
|
|
|
// driver does not need a device extension
|
2007-03-14 19:50:47 +00:00
|
|
|
if ( DeviceExtensionSize != 0 )
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// DeviceExtensionSize must be zero
|
2007-03-14 19:50:47 +00:00
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
2009-09-11 06:33:55 +00:00
|
|
|
// set size to our extension size
|
2009-01-17 11:19:27 +00:00
|
|
|
DeviceExtensionSize = PORT_CLASS_DEVICE_EXTENSION_SIZE;
|
2007-03-14 19:50:47 +00:00
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// create the device
|
2007-03-14 19:50:47 +00:00
|
|
|
status = IoCreateDevice(DriverObject,
|
|
|
|
DeviceExtensionSize,
|
|
|
|
NULL,
|
2008-12-11 11:23:14 +00:00
|
|
|
FILE_DEVICE_KS,
|
2009-01-17 11:19:27 +00:00
|
|
|
FILE_AUTOGENERATED_DEVICE_NAME | FILE_DEVICE_SECURE_OPEN,
|
2007-03-14 19:50:47 +00:00
|
|
|
FALSE,
|
|
|
|
&fdo);
|
|
|
|
|
2009-01-17 11:19:27 +00:00
|
|
|
if (!NT_SUCCESS(status))
|
2007-03-14 19:50:47 +00:00
|
|
|
{
|
2007-07-08 19:44:11 +00:00
|
|
|
DPRINT("IoCreateDevice() failed with status 0x%08lx\n", status);
|
2007-03-14 19:50:47 +00:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// Obtain the new device extension
|
2009-01-27 12:43:33 +00:00
|
|
|
portcls_ext = (PPCLASS_DEVICE_EXTENSION) fdo->DeviceExtension;
|
2009-09-11 06:33:55 +00:00
|
|
|
// initialize the device extension
|
2009-01-17 11:19:27 +00:00
|
|
|
RtlZeroMemory(portcls_ext, DeviceExtensionSize);
|
2009-09-11 06:33:55 +00:00
|
|
|
// allocate create item
|
|
|
|
portcls_ext->CreateItems = (PKSOBJECT_CREATE_ITEM)AllocateItem(NonPagedPool, MaxObjects * sizeof(KSOBJECT_CREATE_ITEM), TAG_PORTCLASS);
|
2007-03-14 19:50:47 +00:00
|
|
|
|
2009-04-05 16:39:51 +00:00
|
|
|
if (!portcls_ext->CreateItems)
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// not enough resources
|
2009-04-14 07:21:05 +00:00
|
|
|
status = STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
goto cleanup;
|
2009-04-05 16:39:51 +00:00
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// store max subdevice count
|
2009-07-11 17:44:41 +00:00
|
|
|
portcls_ext->MaxSubDevices = MaxObjects;
|
2009-09-11 06:33:55 +00:00
|
|
|
// store the physical device object
|
2009-01-02 15:05:57 +00:00
|
|
|
portcls_ext->PhysicalDeviceObject = PhysicalDeviceObject;
|
2009-09-11 06:33:55 +00:00
|
|
|
// set up the start device function
|
2007-03-14 19:50:47 +00:00
|
|
|
portcls_ext->StartDevice = StartDevice;
|
2009-09-11 06:33:55 +00:00
|
|
|
// initialize timer lock
|
2009-06-26 12:05:13 +00:00
|
|
|
KeInitializeSpinLock(&portcls_ext->TimerListLock);
|
2009-09-11 06:33:55 +00:00
|
|
|
// initialize timer list
|
2009-06-26 12:05:13 +00:00
|
|
|
InitializeListHead(&portcls_ext->TimerList);
|
2009-09-11 06:33:55 +00:00
|
|
|
// initialize io timer
|
2009-06-26 14:17:12 +00:00
|
|
|
IoInitializeTimer(fdo, PcIoTimerRoutine, NULL);
|
2009-09-11 06:33:55 +00:00
|
|
|
// start the io timer
|
2009-06-26 14:17:12 +00:00
|
|
|
IoStartTimer(fdo);
|
2007-03-14 19:50:47 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// set io flags
|
2009-01-17 11:19:27 +00:00
|
|
|
fdo->Flags |= DO_DIRECT_IO | DO_POWER_PAGABLE;
|
2009-09-11 06:33:55 +00:00
|
|
|
// clear initializing flag
|
2009-01-17 11:19:27 +00:00
|
|
|
fdo->Flags &= ~ DO_DEVICE_INITIALIZING;
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// allocate the device header
|
2009-01-17 11:19:27 +00:00
|
|
|
status = KsAllocateDeviceHeader(&portcls_ext->KsDeviceHeader, MaxObjects, portcls_ext->CreateItems);
|
2009-09-11 06:33:55 +00:00
|
|
|
// did we succeed
|
2008-12-11 11:23:14 +00:00
|
|
|
if (!NT_SUCCESS(status))
|
|
|
|
{
|
2009-04-14 07:21:05 +00:00
|
|
|
goto cleanup;
|
2008-12-11 11:23:14 +00:00
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// attach device to device stack
|
2009-01-17 11:19:27 +00:00
|
|
|
PrevDeviceObject = IoAttachDeviceToDeviceStack(fdo, PhysicalDeviceObject);
|
2009-09-11 06:33:55 +00:00
|
|
|
// did we succeed
|
2009-01-17 11:19:27 +00:00
|
|
|
if (PrevDeviceObject)
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// store the device object in the device header
|
2009-07-22 14:42:51 +00:00
|
|
|
//KsSetDevicePnpBaseObject(portcls_ext->KsDeviceHeader, fdo, PrevDeviceObject);
|
2009-01-17 11:19:27 +00:00
|
|
|
portcls_ext->PrevDeviceObject = PrevDeviceObject;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// return error code
|
2009-04-14 07:21:05 +00:00
|
|
|
status = STATUS_UNSUCCESSFUL;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-05-29 12:40:09 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// register shutdown notification
|
2009-05-29 12:40:09 +00:00
|
|
|
IoRegisterShutdownNotification(PhysicalDeviceObject);
|
|
|
|
|
2009-04-14 07:21:05 +00:00
|
|
|
return status;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
|
2009-08-11 08:49:30 +00:00
|
|
|
if (portcls_ext->KsDeviceHeader)
|
2009-04-14 07:21:05 +00:00
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// free the device header
|
2009-08-11 08:49:30 +00:00
|
|
|
KsFreeDeviceHeader(portcls_ext->KsDeviceHeader);
|
2009-04-14 07:21:05 +00:00
|
|
|
}
|
|
|
|
|
2009-08-11 08:49:30 +00:00
|
|
|
if (portcls_ext->CreateItems)
|
2009-04-14 07:21:05 +00:00
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// free previously allocated create items
|
2009-08-11 08:49:30 +00:00
|
|
|
FreeItem(portcls_ext->CreateItems, TAG_PORTCLASS);
|
2009-01-17 11:19:27 +00:00
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// delete created fdo
|
2009-08-11 08:49:30 +00:00
|
|
|
IoDeleteDevice(fdo);
|
|
|
|
|
2008-12-11 11:23:14 +00:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2009-03-06 19:06:21 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2008-12-11 11:23:14 +00:00
|
|
|
PcRegisterSubdevice(
|
|
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PWCHAR Name,
|
|
|
|
IN PUNKNOWN Unknown)
|
|
|
|
{
|
2009-01-27 12:43:33 +00:00
|
|
|
PPCLASS_DEVICE_EXTENSION DeviceExt;
|
2008-12-11 11:23:14 +00:00
|
|
|
NTSTATUS Status;
|
2009-01-02 15:05:57 +00:00
|
|
|
ISubdevice *SubDevice;
|
|
|
|
UNICODE_STRING SymbolicLinkName;
|
2009-09-27 00:50:06 +00:00
|
|
|
PSUBDEVICE_DESCRIPTOR SubDeviceDescriptor;
|
2009-01-18 23:46:09 +00:00
|
|
|
ULONG Index;
|
2009-05-29 12:40:09 +00:00
|
|
|
UNICODE_STRING RefName;
|
2009-07-11 17:44:41 +00:00
|
|
|
PSYMBOLICLINK_ENTRY SymEntry;
|
2009-01-02 15:05:57 +00:00
|
|
|
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("PcRegisterSubdevice DeviceObject %p Name %S Unknown %p\n", DeviceObject, Name, Unknown);
|
2008-12-11 11:23:14 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
PC_ASSERT_IRQL_EQUAL(PASSIVE_LEVEL);
|
|
|
|
|
|
|
|
// check if all parameters are valid
|
2008-12-11 11:23:14 +00:00
|
|
|
if (!DeviceObject || !Name || !Unknown)
|
2009-01-17 11:19:27 +00:00
|
|
|
{
|
|
|
|
DPRINT("PcRegisterSubdevice invalid parameter\n");
|
2008-12-11 11:23:14 +00:00
|
|
|
return STATUS_INVALID_PARAMETER;
|
2009-01-17 11:19:27 +00:00
|
|
|
}
|
2008-12-11 11:23:14 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// get device extension
|
2009-01-27 12:43:33 +00:00
|
|
|
DeviceExt = (PPCLASS_DEVICE_EXTENSION)DeviceObject->DeviceExtension;
|
2009-03-06 19:06:21 +00:00
|
|
|
|
2008-12-11 11:23:14 +00:00
|
|
|
if (!DeviceExt)
|
2009-03-06 19:06:21 +00:00
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// should not happen
|
2009-10-21 17:52:11 +00:00
|
|
|
DbgBreakPoint();
|
2008-12-11 11:23:14 +00:00
|
|
|
return STATUS_UNSUCCESSFUL;
|
2009-03-06 19:06:21 +00:00
|
|
|
}
|
2008-12-11 11:23:14 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// look up our undocumented interface
|
|
|
|
Status = Unknown->QueryInterface(IID_ISubdevice, (LPVOID*)&SubDevice);
|
2009-01-02 15:05:57 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("No ISubdevice interface\n");
|
2009-09-11 06:33:55 +00:00
|
|
|
// the provided port driver doesnt support ISubdevice
|
2009-01-02 15:05:57 +00:00
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
}
|
2009-03-26 09:59:45 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// get the subdevice descriptor
|
|
|
|
Status = SubDevice->GetDescriptor(&SubDeviceDescriptor);
|
2009-01-02 15:05:57 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("Failed to get subdevice descriptor %x\n", Status);
|
2009-09-11 06:33:55 +00:00
|
|
|
SubDevice->Release();
|
2009-03-26 09:59:45 +00:00
|
|
|
return STATUS_UNSUCCESSFUL;
|
2009-01-02 15:05:57 +00:00
|
|
|
}
|
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// add an create item to the device header
|
2009-03-26 09:59:45 +00:00
|
|
|
Status = KsAddObjectCreateItemToDeviceHeader(DeviceExt->KsDeviceHeader, PcCreateItemDispatch, (PVOID)SubDevice, Name, NULL);
|
2009-01-18 23:46:09 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
2009-01-02 15:05:57 +00:00
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// failed to attach
|
|
|
|
SubDevice->Release();
|
2009-10-19 18:54:01 +00:00
|
|
|
DPRINT("KsAddObjectCreateItemToDeviceHeader failed with %x\n", Status);
|
2009-03-26 09:59:45 +00:00
|
|
|
return Status;
|
2009-01-02 15:05:57 +00:00
|
|
|
}
|
2008-12-11 11:23:14 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// initialize reference string
|
2009-05-29 12:40:09 +00:00
|
|
|
RtlInitUnicodeString(&RefName, Name);
|
2009-09-27 00:50:06 +00:00
|
|
|
RtlInitUnicodeString(&SubDeviceDescriptor->RefString, Name);
|
2009-07-11 17:44:41 +00:00
|
|
|
|
2009-01-18 23:46:09 +00:00
|
|
|
for(Index = 0; Index < SubDeviceDescriptor->InterfaceCount; Index++)
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// FIXME
|
|
|
|
// check if reference string with that name already exists
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2009-01-18 23:46:09 +00:00
|
|
|
Status = IoRegisterDeviceInterface(DeviceExt->PhysicalDeviceObject,
|
|
|
|
&SubDeviceDescriptor->Interfaces[Index],
|
2009-08-21 10:06:29 +00:00
|
|
|
&RefName,
|
2009-01-18 23:46:09 +00:00
|
|
|
&SymbolicLinkName);
|
2009-08-21 10:06:29 +00:00
|
|
|
|
2009-01-18 23:46:09 +00:00
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// activate device interface
|
2009-01-18 23:46:09 +00:00
|
|
|
IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE);
|
2009-09-11 06:33:55 +00:00
|
|
|
// allocate symbolic link entry
|
|
|
|
SymEntry = (PSYMBOLICLINK_ENTRY)AllocateItem(NonPagedPool, sizeof(SYMBOLICLINK_ENTRY), TAG_PORTCLASS);
|
2009-07-11 17:44:41 +00:00
|
|
|
if (SymEntry)
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// initialize symbolic link item
|
2009-07-11 17:44:41 +00:00
|
|
|
RtlInitUnicodeString(&SymEntry->SymbolicLink, SymbolicLinkName.Buffer);
|
2009-09-11 06:33:55 +00:00
|
|
|
// store item
|
2009-09-27 00:50:06 +00:00
|
|
|
InsertTailList(&SubDeviceDescriptor->SymbolicLinkList, &SymEntry->Entry);
|
2009-07-11 17:44:41 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-09-11 06:33:55 +00:00
|
|
|
// allocating failed
|
2009-07-11 17:44:41 +00:00
|
|
|
RtlFreeUnicodeString(&SymbolicLinkName);
|
|
|
|
}
|
2009-01-18 23:46:09 +00:00
|
|
|
}
|
|
|
|
}
|
2008-12-11 11:23:14 +00:00
|
|
|
|
2009-09-11 06:33:55 +00:00
|
|
|
// release SubDevice reference
|
|
|
|
SubDevice->Release();
|
2009-05-29 12:40:09 +00:00
|
|
|
|
2009-01-17 11:19:27 +00:00
|
|
|
return STATUS_SUCCESS;
|
2008-12-11 11:23:14 +00:00
|
|
|
}
|