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.

345 lines
10 KiB

  1. /*--------------------------------------------------------------------------
  2. *
  3. * Copyright (C) Cyclades Corporation, 1999-2001.
  4. * All rights reserved.
  5. *
  6. * Cyclom-Y Enumerator Driver
  7. *
  8. * This file: enum.c
  9. *
  10. * Description: This module contains the enumeration code needed
  11. * to figure out whether or not a device is attached
  12. * to the serial port. If there is one, it will
  13. * obtain the PNP COM ID (if the device is PNP) and
  14. * parse out the relevant fields.
  15. *
  16. * Notes: This code supports Windows 2000 and Windows XP,
  17. * x86 and ia64 processors.
  18. *
  19. * Complies with Cyclades SW Coding Standard rev 1.3.
  20. *
  21. *--------------------------------------------------------------------------
  22. */
  23. /*-------------------------------------------------------------------------
  24. *
  25. * Change History
  26. *
  27. *--------------------------------------------------------------------------
  28. * Initial implementation based on Microsoft sample code.
  29. *
  30. *--------------------------------------------------------------------------
  31. */
  32. #include "pch.h"
  33. #define MAX_DEVNODE_NAME 256 // Total size of Device ID
  34. #ifdef ALLOC_PRAGMA
  35. #pragma alloc_text(PAGESENM, Cyclomy_ReenumerateDevices)
  36. //#pragma alloc_text (PAGE, Cyclomy_GetRegistryKeyValue)
  37. #endif
  38. #if !defined(__isascii)
  39. #define __isascii(_c) ( (unsigned)(_c) < 0x80 )
  40. #endif // !defined(__isascii)
  41. NTSTATUS
  42. Cyclomy_ReenumerateDevices(IN PIRP Irp, IN PFDO_DEVICE_DATA FdoData)
  43. /*++
  44. Routine Description:
  45. This enumerates the cyclom-y bus which is represented by Fdo (a pointer
  46. to the device object representing the cyclom-y bus). It creates new PDOs
  47. for any new devices which have been discovered since the last enumeration
  48. Arguments:
  49. FdoData - Pointer to the fdo's device extension
  50. for the serial bus which needs to be enumerated
  51. Irp - Pointer to the Irp which was sent to reenumerate.
  52. Return value:
  53. NTSTATUS
  54. --*/
  55. {
  56. PIRP NewIrp;
  57. NTSTATUS status = STATUS_SUCCESS;
  58. KEVENT event;
  59. KTIMER timer;
  60. IO_STATUS_BLOCK IoStatusBlock;
  61. UNICODE_STRING pdoUniName;
  62. UNICODE_STRING instanceStr;
  63. WCHAR instanceNumberBuffer[20];
  64. static ULONG currentInstance = 0;
  65. // PDEVICE_OBJECT pdo = FdoData->AttachedPDO;
  66. PDEVICE_OBJECT pdo;
  67. PPDO_DEVICE_DATA pdoData;
  68. UNICODE_STRING HardwareIDs;
  69. UNICODE_STRING CompIDs;
  70. UNICODE_STRING DeviceIDs;
  71. UNICODE_STRING DevDesc;
  72. UNICODE_STRING InstanceIDs;
  73. ULONG i;
  74. WCHAR pdoName[] = CYY_PDO_NAME_BASE;
  75. ULONG FdoFlags = FdoData->Self->Flags;
  76. ULONG numPorts;
  77. UNREFERENCED_PARAMETER (Irp);
  78. PAGED_CODE();
  79. // Cyclom-Y port enumeration
  80. numPorts = 0;
  81. for (i=0; i < CYY_MAX_CHIPS; i++) {
  82. if (FdoData->Cd1400Base[i]){
  83. numPorts += 4;
  84. }
  85. }
  86. //************************************************************************
  87. // HARDCODE NUMBER OF PORTS TO 1
  88. // numPorts = 1;
  89. //************************************************************************
  90. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("numPorts detected = %d\n",numPorts));
  91. if (numPorts < FdoData->NumPDOs) {
  92. for (i=numPorts; i < CYY_MAX_PORTS; i++) {
  93. pdo = FdoData->AttachedPDO[i];
  94. if (pdo != NULL) {
  95. // Something was there. The device must have been unplugged.
  96. // Remove the PDO.
  97. Cyclomy_PDO_EnumMarkMissing(FdoData, pdo->DeviceExtension);
  98. }
  99. }
  100. goto ExitReenumerate;
  101. }
  102. if (numPorts == FdoData->NumPDOs) {
  103. // All ports already enumerated.
  104. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("All ports already enumerated\n",numPorts));
  105. goto ExitReenumerate;
  106. }
  107. // New ports that need to be enumerated.
  108. RtlZeroMemory(&pdoUniName,sizeof(UNICODE_STRING));
  109. pdoUniName.MaximumLength = DEVICE_OBJECT_NAME_LENGTH * sizeof(WCHAR);
  110. pdoUniName.Buffer = ExAllocatePool(PagedPool,pdoUniName.MaximumLength
  111. + sizeof(WCHAR));
  112. if (pdoUniName.Buffer == NULL) {
  113. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Couldn't allocate memory for device name\n"));
  114. status = STATUS_INSUFFICIENT_RESOURCES;
  115. goto ExitReenumerate;
  116. }
  117. for (i=FdoData->NumPDOs; numPorts && (i< CYY_MAX_PORTS); i++) {
  118. UCHAR RawString[MAX_DEVICE_ID_LEN];
  119. ANSI_STRING AnsiString;
  120. RtlZeroMemory(pdoUniName.Buffer,pdoUniName.MaximumLength);
  121. pdoUniName.Length = 0;
  122. RtlAppendUnicodeToString(&pdoUniName,pdoName);
  123. RtlInitUnicodeString(&instanceStr, NULL);
  124. instanceStr.MaximumLength = sizeof(instanceNumberBuffer);
  125. instanceStr.Buffer = instanceNumberBuffer;
  126. RtlIntegerToUnicodeString(currentInstance++, 10, &instanceStr);
  127. RtlAppendUnicodeStringToString(&pdoUniName, &instanceStr);
  128. //
  129. // Allocate a pdo
  130. //
  131. status = IoCreateDevice(FdoData->Self->DriverObject,
  132. sizeof(PDO_DEVICE_DATA), &pdoUniName,
  133. FILE_DEVICE_UNKNOWN,
  134. FILE_AUTOGENERATED_DEVICE_NAME, FALSE, &pdo);
  135. if (!NT_SUCCESS(status)) {
  136. Cyclomy_KdPrint(FdoData, SER_DBG_SS_ERROR, ("Create device failed\n"));
  137. ExFreePool(pdoUniName.Buffer);
  138. goto ExitReenumerate;
  139. }
  140. Cyclomy_KdPrint(FdoData, SER_DBG_SS_TRACE,
  141. ("Created PDO on top of filter: %x\n",pdo));
  142. pdoData = pdo->DeviceExtension;
  143. RtlInitUnicodeString(&pdoData->HardwareIDs, NULL);
  144. RtlInitUnicodeString(&pdoData->CompIDs, NULL);
  145. RtlInitUnicodeString(&pdoData->DeviceIDs, NULL);
  146. RtlInitUnicodeString(&pdoData->DevDesc, NULL);
  147. RtlInitUnicodeString(&pdoData->InstanceIDs,NULL);
  148. // Hardware ID
  149. sprintf((PCHAR)RawString,"%s%u",CYYPORT_PNP_ID_STR,i+1); // Cyclom-Y\\Port1, etc
  150. Cyclomy_InitMultiString(FdoData, &pdoData->HardwareIDs, RawString, NULL);
  151. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Hardware Id %ws\n",pdoData->HardwareIDs.Buffer));
  152. // That's how ..\parclass\pnppdo.c does. (Fanny)
  153. // Instance ID
  154. sprintf((PCHAR)RawString,"%02u",i+1);
  155. RtlInitAnsiString(&AnsiString,(PCHAR)RawString);
  156. RtlAnsiStringToUnicodeString(&pdoData->InstanceIDs,&AnsiString,TRUE);
  157. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Instance Id %s\n",AnsiString.Buffer));
  158. // Device ID
  159. sprintf((PCHAR)RawString,CYYPORT_DEV_ID_STR);
  160. RtlInitAnsiString(&AnsiString,(PCHAR)RawString);
  161. RtlAnsiStringToUnicodeString(&pdoData->DeviceIDs,&AnsiString,TRUE);
  162. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Device Id %s\n",AnsiString.Buffer));
  163. // Device Description
  164. sprintf((PCHAR)RawString,"Cyclom-Y Port %2u",i+1);
  165. RtlInitAnsiString(&AnsiString,(PUCHAR)RawString);
  166. RtlAnsiStringToUnicodeString(&pdoData->DevDesc,&AnsiString,TRUE);
  167. Cyclomy_KdPrint(FdoData,SER_DBG_CYCLADES,("Device Description %s\n",AnsiString.Buffer));
  168. Cyclomy_InitPDO(i, pdo, FdoData);
  169. numPorts--;
  170. }
  171. ExFreePool(pdoUniName.Buffer);
  172. ExitReenumerate:;
  173. return status;
  174. }
  175. void
  176. Cyclomy_PDO_EnumMarkMissing(PFDO_DEVICE_DATA FdoData, PPDO_DEVICE_DATA PdoData)
  177. /*++
  178. Routine Description:
  179. Removes the attached pdo from the fdo's list of children.
  180. NOTE: THIS FUNCTION CAN ONLY BE CALLED DURING AN ENUMERATION. If called
  181. outside of enumeration, Cyclom-y might delete it's PDO before PnP has
  182. been told the PDO is gone.
  183. Arguments:
  184. FdoData - Pointer to the fdo's device extension
  185. PdoData - Pointer to the pdo's device extension
  186. Return value:
  187. none
  188. --*/
  189. {
  190. ULONG IndexPDO = PdoData->PortIndex;
  191. Cyclomy_KdPrint (FdoData, SER_DBG_SS_TRACE, ("Removing Pdo %x\n",
  192. PdoData->Self));
  193. ASSERT(PdoData->Attached);
  194. PdoData->Attached = FALSE;
  195. FdoData->AttachedPDO[IndexPDO] = NULL;
  196. FdoData->PdoData[IndexPDO] = NULL;
  197. FdoData->NumPDOs--;
  198. }
  199. NTSTATUS
  200. Cyclomy_GetRegistryKeyValue(IN HANDLE Handle, IN PWCHAR KeyNameString,
  201. IN ULONG KeyNameStringLength, IN PVOID Data,
  202. IN ULONG DataLength, OUT PULONG ActualLength)
  203. /*++
  204. Routine Description:
  205. Reads a registry key value from an already opened registry key.
  206. Arguments:
  207. Handle Handle to the opened registry key
  208. KeyNameString ANSI string to the desired key
  209. KeyNameStringLength Length of the KeyNameString
  210. Data Buffer to place the key value in
  211. DataLength Length of the data buffer
  212. Return Value:
  213. STATUS_SUCCESS if all works, otherwise status of system call that
  214. went wrong.
  215. --*/
  216. {
  217. UNICODE_STRING keyName;
  218. ULONG length;
  219. PKEY_VALUE_FULL_INFORMATION fullInfo;
  220. NTSTATUS ntStatus = STATUS_INSUFFICIENT_RESOURCES;
  221. RtlInitUnicodeString (&keyName, KeyNameString);
  222. length = sizeof(KEY_VALUE_FULL_INFORMATION) + KeyNameStringLength
  223. + DataLength;
  224. fullInfo = ExAllocatePool(PagedPool, length);
  225. if (ActualLength != NULL) {
  226. *ActualLength = 0;
  227. }
  228. if (fullInfo) {
  229. ntStatus = ZwQueryValueKey (Handle,
  230. &keyName,
  231. KeyValueFullInformation,
  232. fullInfo,
  233. length,
  234. &length);
  235. if (NT_SUCCESS(ntStatus)) {
  236. //
  237. // If there is enough room in the data buffer, copy the output
  238. //
  239. if (DataLength >= fullInfo->DataLength) {
  240. RtlCopyMemory(Data, ((PUCHAR)fullInfo) + fullInfo->DataOffset,
  241. fullInfo->DataLength);
  242. if (ActualLength != NULL) {
  243. *ActualLength = fullInfo->DataLength;
  244. }
  245. }
  246. }
  247. ExFreePool(fullInfo);
  248. }
  249. if (!NT_SUCCESS(ntStatus) && !NT_ERROR(ntStatus)) {
  250. if (ntStatus == STATUS_BUFFER_OVERFLOW) {
  251. ntStatus = STATUS_BUFFER_TOO_SMALL;
  252. } else {
  253. ntStatus = STATUS_UNSUCCESSFUL;
  254. }
  255. }
  256. return ntStatus;
  257. }