|
|
/*--------------------------------------------------------------------------
* * Copyright (C) Cyclades Corporation, 1999-2001. * All rights reserved. * * Cyclom-Y Enumerator Driver * * This file: enum.c * * Description: This module contains the enumeration code needed * to figure out whether or not a device is attached * to the serial port. If there is one, it will * obtain the PNP COM ID (if the device is PNP) and * parse out the relevant fields. * * 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"
#define MAX_DEVNODE_NAME 256 // Total size of Device ID
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGESENM, Cyclomy_ReenumerateDevices)
//#pragma alloc_text (PAGE, Cyclomy_GetRegistryKeyValue)
#endif
#if !defined(__isascii)
#define __isascii(_c) ( (unsigned)(_c) < 0x80 )
#endif // !defined(__isascii)
NTSTATUS Cyclomy_ReenumerateDevices(IN PIRP Irp, IN PFDO_DEVICE_DATA FdoData) /*++
Routine Description:
This enumerates the cyclom-y bus which is represented by Fdo (a pointer to the device object representing the cyclom-y bus). It creates new PDOs for any new devices which have been discovered since the last enumeration
Arguments:
FdoData - Pointer to the fdo's device extension for the serial bus which needs to be enumerated Irp - Pointer to the Irp which was sent to reenumerate.
Return value:
NTSTATUS
--*/ { PIRP NewIrp; NTSTATUS status = STATUS_SUCCESS; KEVENT event; KTIMER timer;
IO_STATUS_BLOCK IoStatusBlock; UNICODE_STRING pdoUniName; UNICODE_STRING instanceStr; WCHAR instanceNumberBuffer[20]; static ULONG currentInstance = 0; // PDEVICE_OBJECT pdo = FdoData->AttachedPDO;
PDEVICE_OBJECT pdo; PPDO_DEVICE_DATA pdoData;
UNICODE_STRING HardwareIDs; UNICODE_STRING CompIDs; UNICODE_STRING DeviceIDs; UNICODE_STRING DevDesc; UNICODE_STRING InstanceIDs;
ULONG i;
WCHAR pdoName[] = CYY_PDO_NAME_BASE;
ULONG FdoFlags = FdoData->Self->Flags;
ULONG numPorts;
UNREFERENCED_PARAMETER (Irp);
PAGED_CODE();
// Cyclom-Y port enumeration
numPorts = 0; for (i=0; i < CYY_MAX_CHIPS; i++) { if (FdoData->Cd1400Base[i]){ numPorts += 4; } }
//************************************************************************
// HARDCODE NUMBER OF PORTS TO 1
// numPorts = 1;
//************************************************************************
Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("numPorts detected = %d\n",numPorts));
if (numPorts < FdoData->NumPDOs) { for (i=numPorts; i < CYY_MAX_PORTS; i++) { pdo = FdoData->AttachedPDO[i]; if (pdo != NULL) { // Something was there. The device must have been unplugged.
// Remove the PDO.
Cyclomy_PDO_EnumMarkMissing(FdoData, pdo->DeviceExtension); } } goto ExitReenumerate; }
if (numPorts == FdoData->NumPDOs) { // All ports already enumerated.
Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("All ports already enumerated\n",numPorts)); goto ExitReenumerate; }
// New ports that need to be enumerated.
RtlZeroMemory(&pdoUniName,sizeof(UNICODE_STRING)); pdoUniName.MaximumLength = DEVICE_OBJECT_NAME_LENGTH * sizeof(WCHAR); pdoUniName.Buffer = ExAllocatePool(PagedPool,pdoUniName.MaximumLength + sizeof(WCHAR)); if (pdoUniName.Buffer == NULL) { Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Couldn't allocate memory for device name\n")); status = STATUS_INSUFFICIENT_RESOURCES; goto ExitReenumerate; }
for (i=FdoData->NumPDOs; numPorts && (i< CYY_MAX_PORTS); i++) { UCHAR RawString[MAX_DEVICE_ID_LEN]; ANSI_STRING AnsiString;
RtlZeroMemory(pdoUniName.Buffer,pdoUniName.MaximumLength); pdoUniName.Length = 0; RtlAppendUnicodeToString(&pdoUniName,pdoName); RtlInitUnicodeString(&instanceStr, NULL); instanceStr.MaximumLength = sizeof(instanceNumberBuffer); instanceStr.Buffer = instanceNumberBuffer; RtlIntegerToUnicodeString(currentInstance++, 10, &instanceStr); RtlAppendUnicodeStringToString(&pdoUniName, &instanceStr);
//
// Allocate a pdo
//
status = IoCreateDevice(FdoData->Self->DriverObject, sizeof(PDO_DEVICE_DATA), &pdoUniName, FILE_DEVICE_UNKNOWN, FILE_AUTOGENERATED_DEVICE_NAME, FALSE, &pdo);
if (!NT_SUCCESS(status)) { Cyclomy_KdPrint(FdoData, SER_DBG_SS_ERROR, ("Create device failed\n")); ExFreePool(pdoUniName.Buffer); goto ExitReenumerate; }
Cyclomy_KdPrint(FdoData, SER_DBG_SS_TRACE, ("Created PDO on top of filter: %x\n",pdo));
pdoData = pdo->DeviceExtension;
RtlInitUnicodeString(&pdoData->HardwareIDs, NULL); RtlInitUnicodeString(&pdoData->CompIDs, NULL); RtlInitUnicodeString(&pdoData->DeviceIDs, NULL); RtlInitUnicodeString(&pdoData->DevDesc, NULL); RtlInitUnicodeString(&pdoData->InstanceIDs,NULL);
// Hardware ID
sprintf((PCHAR)RawString,"%s%u",CYYPORT_PNP_ID_STR,i+1); // Cyclom-Y\\Port1, etc
Cyclomy_InitMultiString(FdoData, &pdoData->HardwareIDs, RawString, NULL); Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Hardware Id %ws\n",pdoData->HardwareIDs.Buffer));
// That's how ..\parclass\pnppdo.c does. (Fanny)
// Instance ID
sprintf((PCHAR)RawString,"%02u",i+1); RtlInitAnsiString(&AnsiString,(PCHAR)RawString); RtlAnsiStringToUnicodeString(&pdoData->InstanceIDs,&AnsiString,TRUE); Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Instance Id %s\n",AnsiString.Buffer));
// Device ID
sprintf((PCHAR)RawString,CYYPORT_DEV_ID_STR); RtlInitAnsiString(&AnsiString,(PCHAR)RawString); RtlAnsiStringToUnicodeString(&pdoData->DeviceIDs,&AnsiString,TRUE); Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Device Id %s\n",AnsiString.Buffer));
// Device Description
sprintf((PCHAR)RawString,"Cyclom-Y Port %2u",i+1); RtlInitAnsiString(&AnsiString,(PUCHAR)RawString); RtlAnsiStringToUnicodeString(&pdoData->DevDesc,&AnsiString,TRUE); Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Device Description %s\n",AnsiString.Buffer));
Cyclomy_InitPDO(i, pdo, FdoData); numPorts--; }
ExFreePool(pdoUniName.Buffer);
ExitReenumerate:;
return status; }
void Cyclomy_PDO_EnumMarkMissing(PFDO_DEVICE_DATA FdoData, PPDO_DEVICE_DATA PdoData) /*++
Routine Description: Removes the attached pdo from the fdo's list of children.
NOTE: THIS FUNCTION CAN ONLY BE CALLED DURING AN ENUMERATION. If called outside of enumeration, Cyclom-y might delete it's PDO before PnP has been told the PDO is gone.
Arguments: FdoData - Pointer to the fdo's device extension PdoData - Pointer to the pdo's device extension
Return value: none
--*/ { ULONG IndexPDO = PdoData->PortIndex; Cyclomy_KdPrint (FdoData, SER_DBG_SS_TRACE, ("Removing Pdo %x\n", PdoData->Self)); ASSERT(PdoData->Attached); PdoData->Attached = FALSE; FdoData->AttachedPDO[IndexPDO] = NULL; FdoData->PdoData[IndexPDO] = NULL; FdoData->NumPDOs--; }
NTSTATUS Cyclomy_GetRegistryKeyValue(IN HANDLE Handle, IN PWCHAR KeyNameString, IN ULONG KeyNameStringLength, IN PVOID Data, IN ULONG DataLength, OUT PULONG ActualLength) /*++
Routine Description:
Reads a registry key value from an already opened registry key.
Arguments:
Handle Handle to the opened registry key
KeyNameString ANSI string to the desired key
KeyNameStringLength Length of the KeyNameString
Data Buffer to place the key value in
DataLength Length of the data buffer
Return Value:
STATUS_SUCCESS if all works, otherwise status of system call that went wrong.
--*/ { UNICODE_STRING keyName; ULONG length; PKEY_VALUE_FULL_INFORMATION fullInfo;
NTSTATUS ntStatus = STATUS_INSUFFICIENT_RESOURCES;
RtlInitUnicodeString (&keyName, KeyNameString);
length = sizeof(KEY_VALUE_FULL_INFORMATION) + KeyNameStringLength + DataLength; fullInfo = ExAllocatePool(PagedPool, length);
if (ActualLength != NULL) { *ActualLength = 0; }
if (fullInfo) { ntStatus = ZwQueryValueKey (Handle, &keyName, KeyValueFullInformation, fullInfo, length, &length);
if (NT_SUCCESS(ntStatus)) { //
// If there is enough room in the data buffer, copy the output
//
if (DataLength >= fullInfo->DataLength) { RtlCopyMemory(Data, ((PUCHAR)fullInfo) + fullInfo->DataOffset, fullInfo->DataLength); if (ActualLength != NULL) { *ActualLength = fullInfo->DataLength; } } }
ExFreePool(fullInfo); }
if (!NT_SUCCESS(ntStatus) && !NT_ERROR(ntStatus)) { if (ntStatus == STATUS_BUFFER_OVERFLOW) { ntStatus = STATUS_BUFFER_TOO_SMALL; } else { ntStatus = STATUS_UNSUCCESSFUL; } } return ntStatus; }
|