Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

423 lines
11 KiB

/*--------------------------------------------------------------------------
*
* Copyright (C) Cyclades Corporation, 1999-2001.
* All rights reserved.
*
* Cyclom-Y Enumerator Driver
*
* This file: cyclom-y.c
*
* Description: This module contains contains the entry points
* for a standard bus PNP / WDM driver.
*
* Notes: This code supports Windows 2000 and Windows XP,
* x86 and ia64 processors.
*
* Complies with Cyclades SW Coding Standard rev 1.3.
*
*--------------------------------------------------------------------------
*/
/*-------------------------------------------------------------------------
*
* Change History
*
*--------------------------------------------------------------------------
* Initial implementation based on Microsoft sample code.
*
*--------------------------------------------------------------------------
*/
#include "pch.h"
//
// Declare some entry functions as pageable, and make DriverEntry
// discardable
//
NTSTATUS DriverEntry(PDRIVER_OBJECT, PUNICODE_STRING);
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT, DriverEntry)
#pragma alloc_text(PAGE, Cyclomy_DriverUnload)
#endif
NTSTATUS
DriverEntry (
IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING UniRegistryPath
)
/*++
Routine Description:
Initialize the entry points of the driver.
--*/
{
ULONG i;
PRTL_QUERY_REGISTRY_TABLE QueryTable = NULL;
ULONG breakOnEntryDefault = FALSE;
ULONG shouldBreakOnEntry = FALSE;
UNREFERENCED_PARAMETER (UniRegistryPath);
Cyclomy_KdPrint_Def (SER_DBG_SS_TRACE, ("Driver Entry\n"));
Cyclomy_KdPrint_Def (SER_DBG_SS_TRACE, ("RegPath: %x\n", UniRegistryPath));
//
// Get the BreakOnEntry from the registry
//
if (NULL == (QueryTable = ExAllocatePool(
PagedPool,
sizeof(RTL_QUERY_REGISTRY_TABLE)*2
))) {
Cyclomy_KdPrint_Def (SER_DBG_PNP_ERROR,
("Failed to allocate memory to query registry\n"));
} else {
RtlZeroMemory(
QueryTable,
sizeof(RTL_QUERY_REGISTRY_TABLE)*2
);
QueryTable[0].QueryRoutine = NULL;
QueryTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
QueryTable[0].EntryContext = &shouldBreakOnEntry;
QueryTable[0].Name = L"BreakOnEntry";
QueryTable[0].DefaultType = REG_DWORD;
QueryTable[0].DefaultData = &breakOnEntryDefault;
QueryTable[0].DefaultLength= sizeof(ULONG);
// BUGBUG: The rest of the table isn't filled in!
if (!NT_SUCCESS(RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES,
L"cyclom-y",
QueryTable,
NULL,
NULL))) {
Cyclomy_KdPrint_Def (SER_DBG_PNP_ERROR,
("Failed to get BreakOnEntry level from registry. Using default\n"));
shouldBreakOnEntry = breakOnEntryDefault;
}
ExFreePool( QueryTable );
}
if (shouldBreakOnEntry) {
DbgBreakPoint();
}
DriverObject->MajorFunction [IRP_MJ_CREATE] =
DriverObject->MajorFunction [IRP_MJ_CLOSE] = Cyclomy_CreateClose;
DriverObject->MajorFunction [IRP_MJ_PNP] = Cyclomy_PnP;
DriverObject->MajorFunction [IRP_MJ_POWER] = Cyclomy_Power;
DriverObject->MajorFunction [IRP_MJ_DEVICE_CONTROL] = Cyclomy_IoCtl;
DriverObject->MajorFunction [IRP_MJ_SYSTEM_CONTROL] = Cyclomy_DispatchPassThrough;
DriverObject->DriverUnload = Cyclomy_DriverUnload;
DriverObject->DriverExtension->AddDevice = Cyclomy_AddDevice;
return STATUS_SUCCESS;
}
NTSTATUS
CyclomySyncCompletion(IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp,
IN PKEVENT CyclomySyncEvent)
{
UNREFERENCED_PARAMETER(DeviceObject);
UNREFERENCED_PARAMETER(Irp);
KeSetEvent(CyclomySyncEvent, IO_NO_INCREMENT, FALSE);
return STATUS_MORE_PROCESSING_REQUIRED;
}
NTSTATUS
Cyclomy_CreateClose(IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp)
/*++
Routine Description:
Some outside source is trying to create a file against us.
If this is for the FDO (the bus itself) then the caller is trying to
open the propriatary conection to tell us which serial port to enumerate.
If this is for the PDO (an object on the bus) then this is a client that
wishes to use the serial port.
--*/
{
PIO_STACK_LOCATION irpStack;
NTSTATUS status;
PFDO_DEVICE_DATA fdoData;
KEVENT completionEvent;
PDEVICE_OBJECT pNextDevice;
UNREFERENCED_PARAMETER(DeviceObject);
status = STATUS_INVALID_DEVICE_REQUEST;
Irp->IoStatus.Information = 0;
fdoData = DeviceObject->DeviceExtension;
if (fdoData->IsFDO) {
if (fdoData->DevicePnPState == Deleted){
status = STATUS_DELETE_PENDING;
} else {
irpStack = IoGetCurrentIrpStackLocation(Irp);
switch (irpStack->MajorFunction) {
case IRP_MJ_CREATE:
Cyclomy_KdPrint_Def(SER_DBG_SS_TRACE, ("Create"));
if ((fdoData->DevicePnPState == RemovePending) ||
(fdoData->DevicePnPState == SurpriseRemovePending)) {
status = STATUS_DELETE_PENDING;
} else {
status = STATUS_SUCCESS;
}
break;
case IRP_MJ_CLOSE:
Cyclomy_KdPrint_Def (SER_DBG_SS_TRACE, ("Close \n"));
status = STATUS_SUCCESS;
break;
}
}
}
Irp->IoStatus.Status = status;
IoCompleteRequest (Irp, IO_NO_INCREMENT);
return status;
}
NTSTATUS
Cyclomy_IoCtl (
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
--*/
{
PIO_STACK_LOCATION irpStack;
NTSTATUS status;
PCOMMON_DEVICE_DATA commonData;
PFDO_DEVICE_DATA fdoData;
Cyclomy_KdPrint_Def (SER_DBG_IOCTL_TRACE, ("Cyclomy_IoCtl\n"));
status = STATUS_SUCCESS;
irpStack = IoGetCurrentIrpStackLocation (Irp);
ASSERT (IRP_MJ_DEVICE_CONTROL == irpStack->MajorFunction);
commonData = (PCOMMON_DEVICE_DATA) DeviceObject->DeviceExtension;
fdoData = (PFDO_DEVICE_DATA) DeviceObject->DeviceExtension;
//
// We only take Device Control requests for the FDO.
// That is the bus itself.
if (!commonData->IsFDO) {
//
// These commands are only allowed to go to the FDO.
//
status = STATUS_INVALID_DEVICE_REQUEST;
Irp->IoStatus.Status = status;
IoCompleteRequest (Irp, IO_NO_INCREMENT);
return status;
}
status = Cyclomy_IncIoCount (fdoData);
if (!NT_SUCCESS (status)) {
//
// This bus has received the PlugPlay remove IRP. It will no longer
// respond to external requests.
//
Irp->IoStatus.Status = status;
IoCompleteRequest (Irp, IO_NO_INCREMENT);
return status;
}
// Actually, we don't handle any Ioctl.
status = STATUS_INVALID_DEVICE_REQUEST;
Cyclomy_DecIoCount (fdoData);
Irp->IoStatus.Status = status;
IoCompleteRequest (Irp, IO_NO_INCREMENT);
return status;
}
VOID
Cyclomy_DriverUnload (
IN PDRIVER_OBJECT Driver
)
/*++
Routine Description:
Clean up everything we did in driver entry.
--*/
{
UNREFERENCED_PARAMETER (Driver);
PAGED_CODE();
//
// All the device objects should be gone.
//
ASSERT (NULL == Driver->DeviceObject);
//
// Here we free any resources allocated in DriverEntry
//
return;
}
NTSTATUS
Cyclomy_IncIoCount (
PFDO_DEVICE_DATA Data
)
{
InterlockedIncrement (&Data->OutstandingIO);
if (Data->DevicePnPState == Deleted) {
if (0 == InterlockedDecrement (&Data->OutstandingIO)) {
KeSetEvent (&Data->RemoveEvent, 0, FALSE);
}
return STATUS_DELETE_PENDING;
}
return STATUS_SUCCESS;
}
VOID
Cyclomy_DecIoCount (
PFDO_DEVICE_DATA Data
)
{
if (0 == InterlockedDecrement (&Data->OutstandingIO)) {
KeSetEvent (&Data->RemoveEvent, 0, FALSE);
}
}
NTSTATUS
Cyclomy_DispatchPassThrough(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Passes a request on to the lower driver.
--*/
{
PIO_STACK_LOCATION IrpStack =
IoGetCurrentIrpStackLocation( Irp );
#if 1
Cyclomy_KdPrint_Def (SER_DBG_SS_TRACE, (
"[Cyclomy_DispatchPassThrough] "
"IRP: %8x; "
"MajorFunction: %d\n",
Irp,
IrpStack->MajorFunction ));
#endif
//
// Pass the IRP to the target
//
IoSkipCurrentIrpStackLocation (Irp);
if (((PPDO_DEVICE_DATA) DeviceObject->DeviceExtension)->IsFDO) {
return IoCallDriver(
((PFDO_DEVICE_DATA) DeviceObject->DeviceExtension)->TopOfStack,
Irp );
} else {
return IoCallDriver(
((PFDO_DEVICE_DATA) ((PPDO_DEVICE_DATA) DeviceObject->
DeviceExtension)->ParentFdo->DeviceExtension)->TopOfStack,
Irp );
}
}
void
Cyclomy_InitPDO (
ULONG Index,
PDEVICE_OBJECT Pdo,
PFDO_DEVICE_DATA FdoData
)
/*
Description:
Common code to initialize a newly created cyclom-y pdo.
Called either when the control panel exposes a device or when Cyclom-Y senses
a new device was attached.
Parameters:
Pdo - The pdo
FdoData - The fdo's device extension
//Exposed - Was this pdo was found by serenum (FALSE) or was it was EXPOSEd by
// a control panel applet (TRUE)? -> Removed in build 2072
*/
{
ULONG FdoFlags = FdoData->Self->Flags;
PPDO_DEVICE_DATA pdoData = Pdo->DeviceExtension;
HANDLE keyHandle;
NTSTATUS status;
//
// Check the IO style
//
if (FdoFlags & DO_BUFFERED_IO) {
Pdo->Flags |= DO_BUFFERED_IO;
} else if (FdoFlags & DO_DIRECT_IO) {
Pdo->Flags |= DO_DIRECT_IO;
}
//
// Increment the pdo's stacksize so that it can pass irps through
//
Pdo->StackSize += FdoData->Self->StackSize;
//
// Initialize the rest of the device extension
//
pdoData->PortIndex = Index;
pdoData->IsFDO = FALSE;
pdoData->Self = Pdo;
pdoData->ParentFdo = FdoData->Self;
pdoData->Attached = TRUE; // attached to the bus
INITIALIZE_PNP_STATE(pdoData);
pdoData->DebugLevel = FdoData->DebugLevel; // Copy the debug level
pdoData->DeviceState = PowerDeviceD0;
pdoData->SystemState = PowerSystemWorking;
//
// Add the pdo to cyclom-y's list
//
ASSERT(FdoData->AttachedPDO[Index] == NULL);
ASSERT(FdoData->PdoData[Index] == NULL);
// ASSERT(FdoData->NumPDOs == 0); rem because NumPDOs can be > 0 in cyclom-y
FdoData->AttachedPDO[Index] = Pdo;
FdoData->PdoData[Index] = pdoData;
FdoData->NumPDOs++;
Pdo->Flags &= ~DO_DEVICE_INITIALIZING; // Moved to end in DDK final version
Pdo->Flags |= DO_POWER_PAGABLE;
}