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.

2884 lines
96 KiB

  1. /*--------------------------------------------------------------------------
  2. *
  3. * Copyright (C) Cyclades Corporation, 1999-2001.
  4. * All rights reserved.
  5. *
  6. * Cyclom-Y Enumerator Driver
  7. *
  8. * This file: pnp.c
  9. *
  10. * Description: This module contains contains the plugplay calls
  11. * PNP / WDM BUS driver.
  12. *
  13. * Notes: This code supports Windows 2000 and Windows XP,
  14. * x86 and ia64 processors.
  15. *
  16. * Complies with Cyclades SW Coding Standard rev 1.3.
  17. *
  18. *--------------------------------------------------------------------------
  19. */
  20. /*-------------------------------------------------------------------------
  21. *
  22. * Change History
  23. *
  24. *--------------------------------------------------------------------------
  25. * Initial implementation based on Microsoft sample code.
  26. *
  27. *--------------------------------------------------------------------------
  28. */
  29. #include "pch.h"
  30. static const PHYSICAL_ADDRESS CyyPhysicalZero = {0};
  31. // FANNY_ADDPAGABLE_LATER
  32. //#ifdef ALLOC_PRAGMA
  33. //#pragma alloc_text (PAGE, Cyclomy_AddDevice)
  34. //#pragma alloc_text (PAGE, Cyclomy_PnP)
  35. //#pragma alloc_text (PAGE, Cyclomy_FDO_PnP)
  36. //#pragma alloc_text (PAGE, Cyclomy_PDO_PnP)
  37. //#pragma alloc_text (PAGE, Cyclomy_PnPRemove)
  38. //#pragma alloc_text (PAGE, Cyclomy_StartDevice)
  39. ////#pragma alloc_text (PAGE, Cyclomy_Remove)
  40. //#endif
  41. NTSTATUS
  42. Cyclomy_AddDevice(
  43. IN PDRIVER_OBJECT DriverObject,
  44. IN PDEVICE_OBJECT BusPhysicalDeviceObject
  45. )
  46. /*++
  47. Routine Description.
  48. A bus has been found. Attach our FDO to it.
  49. Allocate any required resources. Set things up. And be prepared for the
  50. first ``start device.''
  51. Arguments:
  52. DriverObject - This very self referenced driver.
  53. BusPhysicalDeviceObject - Device object representing the bus. That to which
  54. we attach a new FDO.
  55. --*/
  56. {
  57. NTSTATUS status;
  58. PDEVICE_OBJECT deviceObject;
  59. PFDO_DEVICE_DATA DeviceData;
  60. ULONG nameLength;
  61. ULONG i;
  62. INTERFACE_TYPE interfaceType;
  63. ULONG interfaceTypeLength;
  64. ULONG uiNumber,uiNumberLength;
  65. PAGED_CODE ();
  66. Cyclomy_KdPrint_Def (SER_DBG_PNP_TRACE, ("Add Device: 0x%x\n",
  67. BusPhysicalDeviceObject));
  68. //
  69. // Create our FDO
  70. //
  71. status = IoCreateDevice(DriverObject, sizeof(FDO_DEVICE_DATA), NULL,
  72. FILE_DEVICE_BUS_EXTENDER, 0, TRUE, &deviceObject);
  73. if (NT_SUCCESS (status)) {
  74. DeviceData = (PFDO_DEVICE_DATA) deviceObject->DeviceExtension;
  75. RtlFillMemory (DeviceData, sizeof (FDO_DEVICE_DATA), 0);
  76. DeviceData->IsFDO = TRUE;
  77. DeviceData->DebugLevel = SER_DEFAULT_DEBUG_OUTPUT_LEVEL;
  78. DeviceData->Self = deviceObject;
  79. DeviceData->DriverObject = DriverObject;
  80. for (i=0; i<CYY_MAX_PORTS; i++) {
  81. DeviceData->AttachedPDO[i] = NULL;
  82. }
  83. DeviceData->NumPDOs = 0;
  84. DeviceData->DeviceState = PowerDeviceD0;
  85. DeviceData->SystemState = PowerSystemWorking; // FANNY: This seems to be not needed
  86. DeviceData->SystemWake=PowerSystemUnspecified;
  87. DeviceData->DeviceWake=PowerDeviceUnspecified;
  88. INITIALIZE_PNP_STATE(DeviceData);
  89. // Set the PDO for use with PlugPlay functions
  90. DeviceData->UnderlyingPDO = BusPhysicalDeviceObject;
  91. //
  92. // Attach our filter driver to the device stack.
  93. // the return value of IoAttachDeviceToDeviceStack is the top of the
  94. // attachment chain. This is where all the IRPs should be routed.
  95. //
  96. // Our filter will send IRPs to the top of the stack and use the PDO
  97. // for all PlugPlay functions.
  98. //
  99. DeviceData->TopOfStack
  100. = IoAttachDeviceToDeviceStack(deviceObject, BusPhysicalDeviceObject);
  101. deviceObject->Flags |= DO_BUFFERED_IO;
  102. // Bias outstanding request to 1 so that we can look for a
  103. // transition to zero when processing the remove device PlugPlay IRP.
  104. DeviceData->OutstandingIO = 1;
  105. KeInitializeEvent(&DeviceData->RemoveEvent, SynchronizationEvent,
  106. FALSE);
  107. //
  108. // Tell the PlugPlay system that this device will need an interface
  109. // device class shingle.
  110. //
  111. // It may be that the driver cannot hang the shingle until it starts
  112. // the device itself, so that it can query some of its properties.
  113. // (Aka the shingles guid (or ref string) is based on the properties
  114. // of the device.)
  115. //
  116. status = IoRegisterDeviceInterface (BusPhysicalDeviceObject,
  117. (LPGUID) &GUID_CYCLOMY_BUS_ENUMERATOR,
  118. NULL,
  119. &DeviceData->DevClassAssocName);
  120. if (!NT_SUCCESS (status)) {
  121. CyyLogError(DriverObject, NULL, CyyPhysicalZero, CyyPhysicalZero,
  122. 0, 0, 0, 0, status, CYY_REGISTER_INTERFACE_FAILURE,
  123. 0, NULL, 0, NULL);
  124. Cyclomy_KdPrint_Def (SER_DBG_PNP_ERROR,
  125. ("AddDevice: IoRegisterDCA failed (%x)", status));
  126. goto CyclomyAddDevice_Error;
  127. }
  128. //
  129. // If for any reason you need to save values in a safe location that
  130. // clients of this DeviceClassAssociate might be interested in reading
  131. // here is the time to do so, with the function
  132. // IoOpenDeviceClassRegistryKey
  133. // the symbolic link name used is was returned in
  134. // DeviceData->DevClassAssocName (the same name which is returned by
  135. // IoGetDeviceClassAssociations and the SetupAPI equivs.
  136. //
  137. #if DBG
  138. {
  139. PWCHAR deviceName = NULL;
  140. status = IoGetDeviceProperty (BusPhysicalDeviceObject,
  141. DevicePropertyPhysicalDeviceObjectName,0,
  142. NULL,&nameLength);
  143. if ((nameLength != 0) && (status == STATUS_BUFFER_TOO_SMALL)) {
  144. deviceName = ExAllocatePool (NonPagedPool, nameLength);
  145. if (NULL == deviceName) {
  146. goto someDebugStuffExit;
  147. }
  148. IoGetDeviceProperty (BusPhysicalDeviceObject,
  149. DevicePropertyPhysicalDeviceObjectName,
  150. nameLength, deviceName, &nameLength);
  151. Cyclomy_KdPrint_Def (SER_DBG_PNP_TRACE,
  152. ("AddDevice: %x to %x->%x (%ws) \n",
  153. deviceObject, DeviceData->TopOfStack,
  154. BusPhysicalDeviceObject, deviceName));
  155. }
  156. someDebugStuffExit:;
  157. if (deviceName != NULL) {
  158. ExFreePool(deviceName);
  159. }
  160. }
  161. #endif
  162. status = IoGetDeviceProperty (BusPhysicalDeviceObject,
  163. DevicePropertyLegacyBusType,
  164. sizeof(interfaceType),
  165. &interfaceType,
  166. &interfaceTypeLength);
  167. if (!NT_SUCCESS (status)) {
  168. if (status == STATUS_OBJECT_NAME_NOT_FOUND) {
  169. interfaceType = Isa;
  170. } else {
  171. CyyLogError(DriverObject, NULL, CyyPhysicalZero, CyyPhysicalZero,
  172. 0, 0, 0, 0, status, CYY_GET_BUS_TYPE_FAILURE,
  173. 0, NULL, 0, NULL);
  174. Cyclomy_KdPrint_Def ( SER_DBG_PNP_ERROR,
  175. ("AddDevice: IoGetDeviceProperty LegacyBusType failed (%x)",
  176. status));
  177. goto CyclomyAddDevice_Error;
  178. }
  179. }
  180. if (interfaceType == PCIBus) {
  181. DeviceData->IsPci = 1;
  182. status = IoGetDeviceProperty (BusPhysicalDeviceObject,
  183. DevicePropertyUINumber ,
  184. sizeof(uiNumber),
  185. &uiNumber,
  186. &uiNumberLength);
  187. if (!NT_SUCCESS (status)) {
  188. uiNumber = 0xFFFFFFFF;
  189. Cyclomy_KdPrint_Def (SER_DBG_PNP_ERROR,
  190. ("AddDevice: IoGetDeviceProperty DevicePropertyUINumber failed (%x)",
  191. status));
  192. }
  193. } else {
  194. // ISA board
  195. uiNumber = 0xFFFFFFFF; // What does DevicePropertyUINumber return for ISA board?
  196. }
  197. DeviceData->UINumber = uiNumber;
  198. //
  199. // Turn on the shingle and point it to the given device object.
  200. //
  201. status = IoSetDeviceInterfaceState (
  202. &DeviceData->DevClassAssocName,
  203. TRUE);
  204. if (!NT_SUCCESS (status)) {
  205. CyyLogError(DriverObject, NULL, CyyPhysicalZero, CyyPhysicalZero,
  206. 0, 0, 0, 0, status, CYY_SET_INTERFACE_STATE_FAILURE,
  207. 0, NULL, 0, NULL);
  208. Cyclomy_KdPrint_Def (SER_DBG_PNP_ERROR,
  209. ("AddDevice: IoSetDeviceClass failed (%x)", status));
  210. //return status;
  211. goto CyclomyAddDevice_Error;
  212. }
  213. deviceObject->Flags |= DO_POWER_PAGABLE;
  214. deviceObject->Flags &= ~DO_DEVICE_INITIALIZING;
  215. } else {
  216. CyyLogError(DriverObject, NULL, CyyPhysicalZero, CyyPhysicalZero,
  217. 0, 0, 0, 0, status, CYY_DEVICE_CREATION_FAILURE,
  218. 0, NULL, 0, NULL);
  219. }
  220. return status;
  221. CyclomyAddDevice_Error:
  222. if (DeviceData->DevClassAssocName.Buffer) {
  223. RtlFreeUnicodeString(&DeviceData->DevClassAssocName);
  224. }
  225. if (DeviceData->TopOfStack) {
  226. IoDetachDevice (DeviceData->TopOfStack);
  227. }
  228. if (deviceObject) {
  229. IoDeleteDevice (deviceObject);
  230. }
  231. return status;
  232. }
  233. NTSTATUS
  234. Cyclomy_PnP (IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp)
  235. /*++
  236. Routine Description:
  237. Answer the plethora of Irp Major PnP IRPS.
  238. --*/
  239. {
  240. PIO_STACK_LOCATION irpStack;
  241. NTSTATUS status;
  242. PCOMMON_DEVICE_DATA commonData;
  243. KIRQL oldIrq;
  244. #if DBG
  245. UCHAR MinorFunction;
  246. #endif
  247. PAGED_CODE ();
  248. irpStack = IoGetCurrentIrpStackLocation (Irp);
  249. ASSERT (irpStack->MajorFunction == IRP_MJ_PNP);
  250. #if DBG
  251. MinorFunction = irpStack->MinorFunction;
  252. #endif
  253. commonData = (PCOMMON_DEVICE_DATA) DeviceObject->DeviceExtension;
  254. //
  255. // If removed, fail the request and get out
  256. //
  257. if (commonData->DevicePnPState == Deleted) { // if (commonData->Removed) added in build 2072.
  258. Cyclomy_KdPrint(commonData, SER_DBG_PNP_TRACE,
  259. ("PNP: removed DO: %x got IRP: %x\n", DeviceObject,
  260. Irp));
  261. Irp->IoStatus.Status = status = STATUS_NO_SUCH_DEVICE;
  262. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  263. goto PnPDone;
  264. }
  265. //
  266. // Call either the FDO or PDO Pnp code
  267. //
  268. if (commonData->IsFDO) {
  269. Cyclomy_KdPrint(commonData, SER_DBG_PNP_TRACE,
  270. ("FDO(%x):%s IRP:%x\n", DeviceObject,
  271. PnPMinorFunctionString(irpStack->MinorFunction),Irp));
  272. status = Cyclomy_FDO_PnP(DeviceObject, Irp, irpStack,
  273. (PFDO_DEVICE_DATA) commonData);
  274. goto PnPDone;
  275. }
  276. //
  277. // PDO
  278. //
  279. Cyclomy_KdPrint(commonData, SER_DBG_PNP_TRACE,
  280. ("PDO(%x):%s IRP:%x\n", DeviceObject,
  281. PnPMinorFunctionString(irpStack->MinorFunction),Irp));
  282. status = Cyclomy_PDO_PnP(DeviceObject, Irp, irpStack,
  283. (PPDO_DEVICE_DATA) commonData);
  284. PnPDone:;
  285. return status;
  286. }
  287. NTSTATUS
  288. Cyclomy_FDO_PnP (
  289. IN PDEVICE_OBJECT DeviceObject,
  290. IN PIRP Irp,
  291. IN PIO_STACK_LOCATION IrpStack,
  292. IN PFDO_DEVICE_DATA DeviceData
  293. )
  294. /*++
  295. Routine Description:
  296. Handle requests from the PlugPlay system for the BUS itself
  297. NB: the various Minor functions of the PlugPlay system will not be
  298. overlapped and do not have to be reentrant
  299. --*/
  300. {
  301. NTSTATUS status;
  302. KIRQL oldIrq;
  303. KEVENT event;
  304. ULONG length;
  305. ULONG i;
  306. PLIST_ENTRY entry;
  307. PPDO_DEVICE_DATA pdoData;
  308. PDEVICE_RELATIONS relations;
  309. PIO_STACK_LOCATION stack;
  310. PRTL_QUERY_REGISTRY_TABLE QueryTable = NULL;
  311. ULONG DebugLevelDefault = SER_DEFAULT_DEBUG_OUTPUT_LEVEL;
  312. HANDLE instanceKey;
  313. UNICODE_STRING keyName;
  314. ULONG numOfPorts;
  315. PAGED_CODE ();
  316. status = Cyclomy_IncIoCount (DeviceData);
  317. if (!NT_SUCCESS (status)) {
  318. //Irp->IoStatus.Information = 0; Removed in build 2072
  319. Irp->IoStatus.Status = status;
  320. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  321. return status;
  322. }
  323. stack = IoGetCurrentIrpStackLocation (Irp);
  324. switch (IrpStack->MinorFunction) {
  325. case IRP_MN_FILTER_RESOURCE_REQUIREMENTS: {
  326. PIO_RESOURCE_REQUIREMENTS_LIST pReqList;
  327. PIO_RESOURCE_LIST pResList;
  328. PIO_RESOURCE_DESCRIPTOR pResDesc;
  329. ULONG i, j;
  330. ULONG reqCnt;
  331. ULONG gotPLX;
  332. ULONG gotMemory;
  333. ULONG gotInt;
  334. ULONG listNum;
  335. // FANNY: The serial driver had it as SynchronizationEvent.
  336. KeInitializeEvent(&event, NotificationEvent, FALSE);
  337. IoCopyCurrentIrpStackLocationToNext(Irp);
  338. IoSetCompletionRoutine(Irp,
  339. CyclomySyncCompletion,
  340. &event,
  341. TRUE,
  342. TRUE,
  343. TRUE);
  344. status = IoCallDriver(DeviceData->TopOfStack, Irp);
  345. if (status == STATUS_PENDING) {
  346. KeWaitForSingleObject (&event, Executive, KernelMode, FALSE,
  347. NULL);
  348. }
  349. if (Irp->IoStatus.Information == 0) {
  350. if (stack->Parameters.FilterResourceRequirements
  351. .IoResourceRequirementList == 0) {
  352. Cyclomy_KdPrint(DeviceData, SER_DBG_CYCLADES, ("Can't filter NULL resources!"
  353. "\n"));
  354. status = Irp->IoStatus.Status;
  355. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  356. Cyclomy_DecIoCount (DeviceData);
  357. return status;
  358. }
  359. Irp->IoStatus.Information = (ULONG_PTR)stack->Parameters
  360. .FilterResourceRequirements
  361. .IoResourceRequirementList;
  362. }
  363. //
  364. // Force ISR ports in IO_RES_REQ_LIST to shared status
  365. // Force interrupts to shared status
  366. //
  367. //
  368. // We will only process the first list -- multiport boards
  369. // should not have alternative resources
  370. //
  371. pReqList = (PIO_RESOURCE_REQUIREMENTS_LIST)Irp->IoStatus.Information;
  372. pResList = &pReqList->List[0];
  373. Cyclomy_KdPrint(DeviceData, SER_DBG_CYCLADES, ("------- List has %x lists "
  374. "(including alternatives)\n",
  375. pReqList->AlternativeLists));
  376. for (listNum = 0; listNum < (pReqList->AlternativeLists);
  377. listNum++) {
  378. gotPLX = 0;
  379. gotMemory = 0;
  380. gotInt = 0;
  381. Cyclomy_KdPrint(DeviceData, SER_DBG_CYCLADES, ("------- List has %x resources in it\n",
  382. pResList->Count));
  383. for (j = 0; (j < pResList->Count); j++) {
  384. pResDesc = &pResList->Descriptors[j];
  385. switch (pResDesc->Type) {
  386. case CmResourceTypeMemory:
  387. if (pResDesc->u.Memory.Length == CYY_RUNTIME_LENGTH) {
  388. gotPLX = 1;
  389. //wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww
  390. //pResDesc->ShareDisposition = CmResourceShareShared;
  391. //wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww
  392. //TODO FANNY: Which should be the ShareDisposition for Y?
  393. //pResDesc->ShareDisposition = CmResourceShareDriverExclusive;
  394. //Cyclomy_KdPrint(DeviceData,SER_DBG_CYCLADES,("------- Sharing PLX Memory for "
  395. // "device %x\n", DeviceData->TopOfStack));
  396. } else {
  397. gotMemory = 1;
  398. //wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww
  399. //pResDesc->ShareDisposition = CmResourceShareShared;
  400. //wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww
  401. //TODO FANNY: Which should be the ShareDisposition for Y?
  402. //pResDesc->ShareDisposition = CmResourceShareDriverExclusive;
  403. //Cyclomy_KdPrint(DeviceData,SER_DBG_CYCLADES,("------- Sharing Board Memory for "
  404. // "device %x\n", DeviceData->TopOfStack));
  405. }
  406. break;
  407. case CmResourceTypePort:
  408. Cyclomy_KdPrint(DeviceData,SER_DBG_CYCLADES,
  409. ("------- We should not have Port resource\n"));
  410. break;
  411. case CmResourceTypeInterrupt:
  412. gotInt = 1;
  413. if (DeviceData->IsPci) {
  414. pResDesc->ShareDisposition = CmResourceShareShared;
  415. }
  416. Cyclomy_KdPrint(DeviceData,SER_DBG_CYCLADES,("------- Sharing interrupt for "
  417. "device %x\n", DeviceData->TopOfStack));
  418. break;
  419. default:
  420. break;
  421. }
  422. //
  423. // If we found what we need, we can break out of the loop
  424. //
  425. // FANNY: STRANGE, THERE ARE TWICE FOR EACH TYPE. IT SEEMS THAT
  426. // BOTH RAW AND TRANSLATED ARE LISTED.
  427. // (gotPLX && gotMemory && gotInt)
  428. // break;
  429. //
  430. }
  431. pResList = (PIO_RESOURCE_LIST)((PUCHAR)pResList
  432. + sizeof(IO_RESOURCE_LIST)
  433. + sizeof(IO_RESOURCE_DESCRIPTOR)
  434. * (pResList->Count - 1));
  435. }
  436. status = STATUS_SUCCESS;
  437. break;
  438. }
  439. case IRP_MN_START_DEVICE:
  440. //
  441. // BEFORE you are allowed to ``touch'' the device object to which
  442. // the FDO is attached (that send an irp from the bus to the Device
  443. // object to which the bus is attached). You must first pass down
  444. // the start IRP. It might not be powered on, or able to access or
  445. // something.
  446. //
  447. // FANNY_TODO
  448. // SHOULD I CALL MmLockPagableCodeSection as the serial driver?
  449. // if (DeviceData->Started) {
  450. // Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  451. // ("Device already started\n"));
  452. // status = STATUS_SUCCESS;
  453. // break;
  454. // }
  455. KeInitializeEvent (&event, NotificationEvent, FALSE);
  456. IoCopyCurrentIrpStackLocationToNext (Irp);
  457. IoSetCompletionRoutine (Irp,
  458. CyclomySyncCompletion,
  459. &event,
  460. TRUE,
  461. TRUE,
  462. TRUE);
  463. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  464. if (STATUS_PENDING == status) {
  465. // wait for it...
  466. status = KeWaitForSingleObject (&event,
  467. Executive,
  468. KernelMode,
  469. FALSE, // Not allertable
  470. NULL); // No timeout structure
  471. ASSERT (STATUS_SUCCESS == status);
  472. status = Irp->IoStatus.Status;
  473. }
  474. if (NT_SUCCESS(status)) {
  475. //
  476. // Get the debug level from the registry
  477. //
  478. if (NULL == (QueryTable = ExAllocatePool(
  479. PagedPool,
  480. sizeof(RTL_QUERY_REGISTRY_TABLE)*2
  481. ))) {
  482. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_ERROR,
  483. ("Failed to allocate memory to query registy\n"));
  484. DeviceData->DebugLevel = DebugLevelDefault;
  485. } else {
  486. RtlZeroMemory(
  487. QueryTable,
  488. sizeof(RTL_QUERY_REGISTRY_TABLE)*2
  489. );
  490. QueryTable[0].QueryRoutine = NULL;
  491. QueryTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
  492. QueryTable[0].EntryContext = &DeviceData->DebugLevel;
  493. QueryTable[0].Name = L"DebugLevel";
  494. QueryTable[0].DefaultType = REG_DWORD;
  495. QueryTable[0].DefaultData = &DebugLevelDefault;
  496. QueryTable[0].DefaultLength= sizeof(ULONG);
  497. // CIMEXCIMEX: The rest of the table isn't filled in! Comment changed bld 2128
  498. if (!NT_SUCCESS(RtlQueryRegistryValues(
  499. RTL_REGISTRY_SERVICES,
  500. L"cyclom-y",
  501. QueryTable,
  502. NULL,
  503. NULL))) {
  504. Cyclomy_KdPrint (DeviceData,SER_DBG_PNP_ERROR,
  505. ("Failed to get debug level from registry. "
  506. "Using default\n"));
  507. DeviceData->DebugLevel = DebugLevelDefault;
  508. }
  509. ExFreePool( QueryTable );
  510. }
  511. status = Cyclomy_GetResourceInfo(DeviceObject,
  512. IrpStack->Parameters.StartDevice.AllocatedResources,
  513. IrpStack->Parameters.StartDevice.AllocatedResourcesTranslated);
  514. if (NT_SUCCESS(status)) {
  515. ULONG numberOfResources = CYY_NUMBER_OF_RESOURCES;
  516. if (!DeviceData->IsPci) {
  517. numberOfResources--;
  518. }
  519. status = Cyclomy_BuildResourceList(&DeviceData->PChildResourceList,
  520. &DeviceData->PChildResourceListSize,
  521. IrpStack->Parameters.StartDevice.AllocatedResources,
  522. numberOfResources);
  523. if (!NT_SUCCESS(status)) {
  524. goto CaseSTART_end;
  525. }
  526. status = Cyclomy_BuildResourceList(&DeviceData->PChildResourceListTr,
  527. &DeviceData->PChildResourceListSizeTr,
  528. IrpStack->Parameters.StartDevice.AllocatedResourcesTranslated,
  529. numberOfResources);
  530. if (!NT_SUCCESS(status)) {
  531. goto CaseSTART_end;
  532. }
  533. status = Cyclomy_BuildRequirementsList(&DeviceData->PChildRequiredList,
  534. IrpStack->Parameters.StartDevice.AllocatedResources,
  535. numberOfResources);
  536. if (!NT_SUCCESS(status)) {
  537. goto CaseSTART_end;
  538. }
  539. //
  540. // See if we are in the proper power state.
  541. //
  542. if (DeviceData->DeviceState != PowerDeviceD0) {
  543. status = Cyclomy_GotoPowerState(DeviceData->UnderlyingPDO, DeviceData,
  544. PowerDeviceD0);
  545. if (!NT_SUCCESS(status)) {
  546. goto CaseSTART_end;
  547. }
  548. }
  549. numOfPorts=Cyclomy_DoesBoardExist(DeviceData);
  550. if (!numOfPorts){
  551. Cyclomy_KdPrint_Def(SER_DBG_CYCLADES,("Does Port exist test failed\n"));
  552. status = STATUS_SERIAL_NO_DEVICE_INITED;
  553. goto CaseSTART_end;
  554. }
  555. Cyclomy_KdPrint(DeviceData,SER_DBG_CYCLADES,("Board found!\n"));
  556. Cyclomy_EnableInterruptInPLX(DeviceData); // Enable interrupt in the PLX
  557. // Save number of ports to the Registry, so that Property Page
  558. // code can retrieve it.
  559. IoOpenDeviceRegistryKey(DeviceData->UnderlyingPDO,PLUGPLAY_REGKEY_DEVICE,
  560. STANDARD_RIGHTS_WRITE,&instanceKey);
  561. RtlInitUnicodeString(&keyName,L"NumOfPorts");
  562. ZwSetValueKey(instanceKey,&keyName,0,REG_DWORD,&numOfPorts,sizeof(ULONG));
  563. ZwFlushKey(instanceKey);
  564. ZwClose(instanceKey);
  565. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  566. ("Start Device: Device started successfully\n"));
  567. SET_NEW_PNP_STATE(DeviceData, Started);
  568. // TODO: FOR NOW, LET'S KEEP THIS DEVICE IN POWER D0.
  569. // THE SERIAL DRIVER SEEMS TO POWER DOWN TO D3, AND BECOME D0 DURING OPEN.
  570. // BUT NOT SURE IF THE BOARD NEED TO BE IN D0 WHILE THE CHILD DEVICES
  571. // ARE ENUMARATED.
  572. }
  573. }
  574. CaseSTART_end:
  575. if (!NT_SUCCESS(status)) {
  576. Cyclomy_ReleaseResources(DeviceData);
  577. }
  578. //
  579. // We must now complete the IRP, since we stopped it in the
  580. // completetion routine with MORE_PROCESSING_REQUIRED.
  581. //
  582. //Irp->IoStatus.Information = 0; Removed in build 2072
  583. break;
  584. case IRP_MN_QUERY_STOP_DEVICE:
  585. //
  586. // Test to see if there are any PDO created as children of this FDO
  587. // If there are then conclude the device is busy and fail the
  588. // query stop.
  589. //
  590. // CIMEXCIMEX (BUGBUG replaced by CIMEXCIMEX on build 2128 - Fanny)
  591. // We could do better, by seing if the children PDOs are actually
  592. // currently open. If they are not then we could stop, get new
  593. // resouces, fill in the new resouce values, and then when a new client
  594. // opens the PDO use the new resources. But this works for now.
  595. //
  596. //TODO FANNY: FOR NOW WE WILL ALWAYS ACCEPT TO STOP DEVICE. REVIEW THIS LATER...
  597. // if (DeviceData->AttachedPDO) {
  598. // status = STATUS_UNSUCCESSFUL;
  599. // } else {
  600. // status = STATUS_SUCCESS;
  601. // }
  602. status = STATUS_SUCCESS;
  603. Irp->IoStatus.Status = status;
  604. if (NT_SUCCESS(status)) {
  605. SET_NEW_PNP_STATE(DeviceData, StopPending);
  606. IoSkipCurrentIrpStackLocation (Irp);
  607. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  608. } else {
  609. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  610. }
  611. Cyclomy_DecIoCount (DeviceData);
  612. return status;
  613. case IRP_MN_CANCEL_STOP_DEVICE:
  614. KeInitializeEvent (&event, NotificationEvent, FALSE);
  615. IoCopyCurrentIrpStackLocationToNext (Irp);
  616. IoSetCompletionRoutine (Irp,
  617. CyclomySyncCompletion,
  618. &event,
  619. TRUE,
  620. TRUE,
  621. TRUE);
  622. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  623. if (STATUS_PENDING == status) {
  624. // wait for it...
  625. status = KeWaitForSingleObject (&event,
  626. Executive,
  627. KernelMode,
  628. FALSE, // Not allertable
  629. NULL); // No timeout structure
  630. ASSERT (STATUS_SUCCESS == status);
  631. status = Irp->IoStatus.Status;
  632. }
  633. if(StopPending == DeviceData->DevicePnPState)
  634. {
  635. //
  636. // We did receive a query-stop, so restore.
  637. //
  638. RESTORE_PREVIOUS_PNP_STATE(DeviceData);
  639. ASSERT(DeviceData->DevicePnPState == Started);
  640. }
  641. break;
  642. case IRP_MN_STOP_DEVICE:
  643. //
  644. // After the start IRP has been sent to the lower driver object, the
  645. // bus may NOT send any more IRPS down ``touch'' until another START
  646. // has occured.
  647. // What ever access is required must be done before the Irp is passed
  648. // on.
  649. //
  650. // Stop device means that the resources given durring Start device
  651. // are no revoked. So we need to stop using them
  652. //
  653. Cyclomy_ReleaseResources(DeviceData);
  654. SET_NEW_PNP_STATE(DeviceData, Stopped);
  655. //
  656. // We don't need a completion routine so fire and forget.
  657. //
  658. // Set the current stack location to the next stack location and
  659. // call the next device object.
  660. //
  661. Irp->IoStatus.Status = STATUS_SUCCESS;
  662. IoSkipCurrentIrpStackLocation (Irp);
  663. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  664. Cyclomy_DecIoCount (DeviceData);
  665. return status;
  666. case IRP_MN_QUERY_REMOVE_DEVICE:
  667. //
  668. // If we were to fail this call then we would need to complete the
  669. // IRP here. Since we are not, set the status to SUCCESS and
  670. // call the next driver.
  671. //
  672. SET_NEW_PNP_STATE(DeviceData, RemovePending);
  673. Irp->IoStatus.Status = STATUS_SUCCESS;
  674. IoSkipCurrentIrpStackLocation (Irp);
  675. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  676. Cyclomy_DecIoCount (DeviceData);
  677. return status;
  678. case IRP_MN_CANCEL_REMOVE_DEVICE:
  679. //
  680. // If we were to fail this call then we would need to complete the
  681. // IRP here. Since we are not, set the status to SUCCESS and
  682. // call the next driver.
  683. //
  684. //
  685. // First check to see whether you have received cancel-remove
  686. // without first receiving a query-remove. This could happen if
  687. // someone above us fails a query-remove and passes down the
  688. // subsequent cancel-remove.
  689. //
  690. if(RemovePending == DeviceData->DevicePnPState)
  691. {
  692. //
  693. // We did receive a query-remove, so restore.
  694. //
  695. RESTORE_PREVIOUS_PNP_STATE(DeviceData);
  696. }
  697. IoSkipCurrentIrpStackLocation (Irp);
  698. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  699. Cyclomy_DecIoCount (DeviceData);
  700. return status;
  701. case IRP_MN_SURPRISE_REMOVAL:
  702. SET_NEW_PNP_STATE(DeviceData, SurpriseRemovePending);
  703. Irp->IoStatus.Status = STATUS_SUCCESS;
  704. IoSkipCurrentIrpStackLocation (Irp);
  705. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  706. Cyclomy_DecIoCount (DeviceData);
  707. return status;
  708. case IRP_MN_REMOVE_DEVICE:
  709. //
  710. // The PlugPlay system has detected the removal of this device. We
  711. // have no choice but to detach and delete the device object.
  712. // (If we wanted to express and interest in preventing this removal,
  713. // we should have filtered the query remove and query stop routines.)
  714. //
  715. // Note! we might receive a remove WITHOUT first receiving a stop.
  716. // ASSERT (!DeviceData->Removed);
  717. // We will accept no new requests
  718. //
  719. // DeviceData->Removed = TRUE;
  720. SET_NEW_PNP_STATE(DeviceData, Deleted);
  721. //
  722. // Complete any outstanding IRPs queued by the driver here.
  723. //
  724. //
  725. // Make the DCA go away. Some drivers may choose to remove the DCA
  726. // when they receive a stop or even a query stop. We just don't care.
  727. //
  728. IoSetDeviceInterfaceState (&DeviceData->DevClassAssocName, FALSE);
  729. //
  730. // Here if we had any outstanding requests in a personal queue we should
  731. // complete them all now.
  732. //
  733. // Note, the device is guarenteed stopped, so we cannot send it any non-
  734. // PNP IRPS.
  735. //
  736. //
  737. // Fire and forget
  738. //
  739. Irp->IoStatus.Status = STATUS_SUCCESS;
  740. IoSkipCurrentIrpStackLocation (Irp);
  741. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  742. //
  743. // Wait for all outstanding requests to complete
  744. //
  745. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  746. ("Waiting for outstanding requests\n"));
  747. i = InterlockedDecrement (&DeviceData->OutstandingIO);
  748. ASSERT (0 < i);
  749. if (0 != InterlockedDecrement (&DeviceData->OutstandingIO)) {
  750. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_INFO,
  751. ("Remove Device waiting for request to complete\n"));
  752. KeWaitForSingleObject (&DeviceData->RemoveEvent,
  753. Executive,
  754. KernelMode,
  755. FALSE, // Not Alertable
  756. NULL); // No timeout
  757. }
  758. //
  759. // Free the associated resources
  760. //
  761. //
  762. // Detach from the underlying devices.
  763. //
  764. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_INFO,
  765. ("IoDetachDevice: 0x%x\n", DeviceData->TopOfStack));
  766. IoDetachDevice (DeviceData->TopOfStack);
  767. //
  768. // Clean up any resources here
  769. //
  770. Cyclomy_ReleaseResources(DeviceData);
  771. ExFreePool (DeviceData->DevClassAssocName.Buffer);
  772. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_INFO,
  773. ("IoDeleteDevice: 0x%x\n", DeviceObject));
  774. //
  775. // Remove any PDO's we ejected
  776. //
  777. //FANNY: CHANGED TO SUPPORT MORE THAN ONE CHILD DEVICE
  778. // if (DeviceData->AttachedPDO != NULL) {
  779. // ASSERT(DeviceData->NumPDOs == 1);
  780. //
  781. // Cyclomy_PnPRemove(DeviceData->AttachedPDO, DeviceData->PdoData);
  782. // DeviceData->PdoData = NULL;
  783. // DeviceData->AttachedPDO = NULL;
  784. // DeviceData->NumPDOs = 0;
  785. // }
  786. i=DeviceData->NumPDOs;
  787. while(i--) {
  788. if (DeviceData->AttachedPDO[i] != NULL) {
  789. (DeviceData->PdoData[i])->Attached = FALSE;
  790. if(SurpriseRemovePending != (DeviceData->PdoData[i])->DevicePnPState) {
  791. Cyclomy_PnPRemove(DeviceData->AttachedPDO[i], DeviceData->PdoData[i]);
  792. }
  793. DeviceData->PdoData[i] = NULL;
  794. DeviceData->AttachedPDO[i] = NULL;
  795. }
  796. }
  797. DeviceData->NumPDOs = 0;
  798. IoDeleteDevice(DeviceObject);
  799. return status;
  800. case IRP_MN_QUERY_DEVICE_RELATIONS:
  801. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  802. ("\tQueryDeviceRelation Type: %d\n",
  803. IrpStack->Parameters.QueryDeviceRelations.Type));
  804. if (BusRelations != IrpStack->Parameters.QueryDeviceRelations.Type) {
  805. //
  806. // We don't support this
  807. //
  808. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  809. ("Query Device Relations - Non bus\n"));
  810. goto CYY_FDO_PNP_DEFAULT;
  811. }
  812. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  813. ("\tQuery Bus Relations\n"));
  814. // Check for new devices or if old devices still there.
  815. status = Cyclomy_ReenumerateDevices(Irp, DeviceData );
  816. //
  817. // Tell the plug and play system about all the PDOs.
  818. //
  819. // There might also be device relations below and above this FDO,
  820. // so, be sure to propagate the relations from the upper drivers.
  821. //
  822. // No Completion routine is needed so long as the status is preset
  823. // to success. (PDOs complete plug and play irps with the current
  824. // IoStatus.Status and IoStatus.Information as the default.)
  825. //
  826. //KeAcquireSpinLock (&DeviceData->Spin, &oldIrq);
  827. i = (0 == Irp->IoStatus.Information) ? 0 :
  828. ((PDEVICE_RELATIONS) Irp->IoStatus.Information)->Count;
  829. // The current number of PDOs in the device relations structure
  830. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  831. ("#PDOS = %d + %d\n", i, DeviceData->NumPDOs));
  832. length = sizeof(DEVICE_RELATIONS) +
  833. ((DeviceData->NumPDOs + i) * sizeof (PDEVICE_OBJECT));
  834. relations = (PDEVICE_RELATIONS) ExAllocatePool (NonPagedPool, length);
  835. if (NULL == relations) {
  836. Irp->IoStatus.Status = STATUS_INSUFFICIENT_RESOURCES;
  837. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  838. Cyclomy_DecIoCount(DeviceData);
  839. return STATUS_INSUFFICIENT_RESOURCES;
  840. }
  841. //
  842. // Copy in the device objects so far
  843. //
  844. if (i) {
  845. RtlCopyMemory (
  846. relations->Objects,
  847. ((PDEVICE_RELATIONS) Irp->IoStatus.Information)->Objects,
  848. i * sizeof (PDEVICE_OBJECT));
  849. }
  850. relations->Count = DeviceData->NumPDOs + i;
  851. //
  852. // For each PDO on this bus add a pointer to the device relations
  853. // buffer, being sure to take out a reference to that object.
  854. // The PlugPlay system will dereference the object when it is done with
  855. // it and free the device relations buffer.
  856. //
  857. //FANNY: CHANGED TO SUPPORT ADDITIONAL CHILD DEVICES
  858. // if (DeviceData->NumPDOs) {
  859. // relations->Objects[relations->Count-1] = DeviceData->AttachedPDO;
  860. // ObReferenceObject (DeviceData->AttachedPDO);
  861. // }
  862. for (i=0; i< DeviceData->NumPDOs; i++) {
  863. relations->Objects[relations->Count - DeviceData->NumPDOs + i] =
  864. DeviceData->AttachedPDO[i];
  865. ObReferenceObject (DeviceData->AttachedPDO[i]);
  866. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  867. ("Child PDOS: %x\n", DeviceData->AttachedPDO[i]));
  868. }
  869. //
  870. // Set up and pass the IRP further down the stack
  871. //
  872. Irp->IoStatus.Status = STATUS_SUCCESS;
  873. if (0 != Irp->IoStatus.Information) {
  874. ExFreePool ((PVOID) Irp->IoStatus.Information);
  875. }
  876. Irp->IoStatus.Information = (ULONG_PTR)relations;
  877. IoSkipCurrentIrpStackLocation (Irp);
  878. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  879. Cyclomy_DecIoCount (DeviceData);
  880. return status;
  881. case IRP_MN_QUERY_CAPABILITIES: {
  882. PIO_STACK_LOCATION irpSp;
  883. //
  884. // Send this down to the PDO first
  885. //
  886. KeInitializeEvent (&event, NotificationEvent, FALSE);
  887. IoCopyCurrentIrpStackLocationToNext (Irp);
  888. IoSetCompletionRoutine (Irp,
  889. CyclomySyncCompletion,
  890. &event,
  891. TRUE,
  892. TRUE,
  893. TRUE);
  894. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  895. if (STATUS_PENDING == status) {
  896. // wait for it...
  897. status = KeWaitForSingleObject (&event,
  898. Executive,
  899. KernelMode,
  900. FALSE, // Not allertable
  901. NULL); // No timeout structure
  902. ASSERT (STATUS_SUCCESS == status);
  903. status = Irp->IoStatus.Status;
  904. }
  905. if (NT_SUCCESS(status)) {
  906. irpSp = IoGetCurrentIrpStackLocation(Irp);
  907. DeviceData->SystemWake=irpSp->Parameters.DeviceCapabilities.Capabilities->SystemWake;
  908. DeviceData->DeviceWake=irpSp->Parameters.DeviceCapabilities.Capabilities->DeviceWake;
  909. //#if DBG
  910. // DbgPrint("PowerSystemSleeping1 %d\n",
  911. // irpSp->Parameters.DeviceCapabilities.Capabilities->DeviceState[PowerSystemSleeping1]);
  912. // DbgPrint("PowerSystemSleeping2 %d\n",
  913. // irpSp->Parameters.DeviceCapabilities.Capabilities->DeviceState[PowerSystemSleeping2]);
  914. // DbgPrint("PowerSystemSleeping3 %d\n",
  915. // irpSp->Parameters.DeviceCapabilities.Capabilities->DeviceState[PowerSystemSleeping3]);
  916. // DbgPrint("PowerSystemHibernate %d\n",
  917. // irpSp->Parameters.DeviceCapabilities.Capabilities->DeviceState[PowerSystemHibernate]);
  918. // DbgPrint("PowerSystemShutdown %d\n",
  919. // irpSp->Parameters.DeviceCapabilities.Capabilities->DeviceState[PowerSystemShutdown]);
  920. // In the test with a system that supports standby, the result was:
  921. // D1, Unspecified, Unspecified, D3, D3.
  922. //#endif
  923. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_INFO, ("SystemWake %d\n",DeviceData->SystemWake));
  924. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_INFO, ("DeviceWake %d\n",DeviceData->DeviceWake));
  925. }
  926. break;
  927. }
  928. default:
  929. //
  930. // In the default case we merely call the next driver since
  931. // we don't know what to do.
  932. //
  933. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_TRACE,
  934. ("FDO(%x):%s not handled\n", DeviceObject,
  935. PnPMinorFunctionString(IrpStack->MinorFunction)));
  936. CYY_FDO_PNP_DEFAULT:
  937. //
  938. // Fire and Forget
  939. //
  940. IoSkipCurrentIrpStackLocation (Irp);
  941. //
  942. // Done, do NOT complete the IRP, it will be processed by the lower
  943. // device object, which will complete the IRP
  944. //
  945. status = IoCallDriver (DeviceData->TopOfStack, Irp);
  946. Cyclomy_DecIoCount (DeviceData);
  947. return status;
  948. }
  949. Irp->IoStatus.Status = status;
  950. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  951. Cyclomy_DecIoCount (DeviceData);
  952. return status;
  953. }
  954. NTSTATUS
  955. Cyclomy_PDO_PnP (IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp,
  956. IN PIO_STACK_LOCATION IrpStack, IN PPDO_DEVICE_DATA DeviceData)
  957. /*++
  958. Routine Description:
  959. Handle requests from the PlugPlay system for the devices on the BUS
  960. --*/
  961. {
  962. PDEVICE_CAPABILITIES deviceCapabilities;
  963. ULONG information;
  964. PWCHAR buffer;
  965. ULONG length, i, j;
  966. NTSTATUS status;
  967. KIRQL oldIrq;
  968. HANDLE keyHandle;
  969. UNICODE_STRING keyName;
  970. PWCHAR returnBuffer = NULL;
  971. PAGED_CODE();
  972. status = Irp->IoStatus.Status;
  973. //
  974. // NB: since we are a bus enumerator, we have no one to whom we could
  975. // defer these irps. Therefore we do not pass them down but merely
  976. // return them.
  977. //
  978. switch (IrpStack->MinorFunction) {
  979. case IRP_MN_QUERY_CAPABILITIES:
  980. //
  981. // Get the packet.
  982. //
  983. deviceCapabilities=IrpStack->Parameters.DeviceCapabilities.Capabilities;
  984. //
  985. // Set the capabilities.
  986. //
  987. deviceCapabilities->Version = 1;
  988. deviceCapabilities->Size = sizeof (DEVICE_CAPABILITIES);
  989. //
  990. // We cannot wake the system.
  991. //
  992. deviceCapabilities->SystemWake
  993. = ((PFDO_DEVICE_DATA)DeviceData->ParentFdo->DeviceExtension)
  994. ->SystemWake;
  995. deviceCapabilities->DeviceWake
  996. = ((PFDO_DEVICE_DATA)DeviceData->ParentFdo->DeviceExtension)
  997. ->DeviceWake;
  998. //
  999. // We have no latencies
  1000. //
  1001. deviceCapabilities->D1Latency = 0;
  1002. deviceCapabilities->D2Latency = 0;
  1003. deviceCapabilities->D3Latency = 0;
  1004. deviceCapabilities->UniqueID = FALSE;
  1005. //
  1006. // Initialize supported DeviceState
  1007. //
  1008. deviceCapabilities->DeviceState[PowerSystemWorking] = PowerDeviceD0;
  1009. deviceCapabilities->DeviceState[PowerSystemSleeping1] = PowerDeviceD3;
  1010. deviceCapabilities->DeviceState[PowerSystemSleeping2] = PowerDeviceD3;
  1011. deviceCapabilities->DeviceState[PowerSystemSleeping3] = PowerDeviceD3;
  1012. deviceCapabilities->DeviceState[PowerSystemHibernate] = PowerDeviceD3;
  1013. deviceCapabilities->DeviceState[PowerSystemShutdown] = PowerDeviceD3;
  1014. status = STATUS_SUCCESS;
  1015. break;
  1016. case IRP_MN_QUERY_DEVICE_TEXT: {
  1017. if ((IrpStack->Parameters.QueryDeviceText.DeviceTextType
  1018. != DeviceTextDescription) || DeviceData->DevDesc.Buffer == NULL) {
  1019. break;
  1020. }
  1021. // FANNY - CHANGE TO MaximumLength
  1022. // returnBuffer = ExAllocatePool(PagedPool, DeviceData->DevDesc.Length);
  1023. returnBuffer = ExAllocatePool(PagedPool, DeviceData->DevDesc.MaximumLength);
  1024. if (returnBuffer == NULL) {
  1025. status = STATUS_INSUFFICIENT_RESOURCES;
  1026. break;
  1027. }
  1028. status = STATUS_SUCCESS;
  1029. // FANNY - CHANGE TO MaximumLength
  1030. // RtlCopyMemory(returnBuffer, DeviceData->DevDesc.Buffer,
  1031. // DeviceData->DevDesc.Length);
  1032. RtlCopyMemory(returnBuffer, DeviceData->DevDesc.Buffer,
  1033. DeviceData->DevDesc.MaximumLength);
  1034. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_TRACE,
  1035. ("TextID: buf %ws\n", returnBuffer));
  1036. Irp->IoStatus.Information = (ULONG_PTR)returnBuffer;
  1037. break;
  1038. }
  1039. case IRP_MN_QUERY_ID:
  1040. //
  1041. // Query the IDs of the device
  1042. //
  1043. switch (IrpStack->Parameters.QueryId.IdType) {
  1044. //
  1045. // The other ID's we just copy from the buffers and are done.
  1046. //
  1047. case BusQueryDeviceID:
  1048. case BusQueryHardwareIDs:
  1049. case BusQueryCompatibleIDs:
  1050. case BusQueryInstanceID:
  1051. {
  1052. PUNICODE_STRING pId;
  1053. status = STATUS_SUCCESS;
  1054. switch (IrpStack->Parameters.QueryId.IdType) {
  1055. case BusQueryDeviceID:
  1056. pId = &DeviceData->DeviceIDs;
  1057. break;
  1058. case BusQueryHardwareIDs:
  1059. pId = &DeviceData->HardwareIDs;
  1060. break;
  1061. case BusQueryCompatibleIDs:
  1062. pId = &DeviceData->CompIDs;
  1063. break;
  1064. case BusQueryInstanceID:
  1065. // Build an instance ID. This is what PnP uses to tell if it has
  1066. // seen this thing before or not. Build it from the first hardware
  1067. // id and the port number.
  1068. pId = &DeviceData->InstanceIDs;
  1069. break;
  1070. }
  1071. buffer = pId->Buffer;
  1072. if (buffer != NULL) {
  1073. // FANNY CHANGED
  1074. //length = pId->Length;
  1075. length = pId->MaximumLength;
  1076. returnBuffer = ExAllocatePool(PagedPool, length);
  1077. if (returnBuffer != NULL) {
  1078. #if DBG
  1079. RtlFillMemory(returnBuffer, length, 0xff);
  1080. #endif
  1081. // FANNY CHANGED
  1082. //RtlCopyMemory(returnBuffer, buffer, pId->Length);
  1083. RtlCopyMemory(returnBuffer, buffer, length);
  1084. } else {
  1085. status = STATUS_INSUFFICIENT_RESOURCES;
  1086. }
  1087. } else {
  1088. // FANNY ADDED
  1089. status = STATUS_NOT_FOUND;
  1090. }
  1091. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_TRACE,
  1092. ("ID: Unicode 0x%x\n", pId));
  1093. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_TRACE,
  1094. ("ID: buf 0x%x\n", returnBuffer));
  1095. Irp->IoStatus.Information = (ULONG_PTR)returnBuffer;
  1096. }
  1097. break;
  1098. }
  1099. break;
  1100. case IRP_MN_QUERY_BUS_INFORMATION: {
  1101. PPNP_BUS_INFORMATION pBusInfo;
  1102. PFDO_DEVICE_DATA parentExtension;
  1103. parentExtension = (DeviceData->ParentFdo)->DeviceExtension;
  1104. ASSERTMSG("Cyclomy appears not to be the sole bus?!?",
  1105. Irp->IoStatus.Information == (ULONG_PTR)NULL);
  1106. pBusInfo = ExAllocatePool(PagedPool, sizeof(PNP_BUS_INFORMATION));
  1107. if (pBusInfo == NULL) {
  1108. status = STATUS_INSUFFICIENT_RESOURCES;
  1109. break;
  1110. }
  1111. pBusInfo->BusTypeGuid = GUID_BUS_TYPE_CYCLOMY;
  1112. if (parentExtension->IsPci) {
  1113. pBusInfo->LegacyBusType = PCIBus;
  1114. } else {
  1115. pBusInfo->LegacyBusType = Isa;
  1116. }
  1117. //
  1118. // We really can't track our bus number since we can be torn
  1119. // down with our bus
  1120. //
  1121. //pBusInfo->BusNumber = 0;
  1122. pBusInfo->BusNumber = parentExtension->UINumber;
  1123. Irp->IoStatus.Information = (ULONG_PTR)pBusInfo;
  1124. status = STATUS_SUCCESS;
  1125. break;
  1126. }
  1127. case IRP_MN_QUERY_DEVICE_RELATIONS:
  1128. Cyclomy_KdPrint (DeviceData, SER_DBG_PNP_TRACE,
  1129. ("\tQueryDeviceRelation Type: %d\n",
  1130. IrpStack->Parameters.QueryDeviceRelations.Type));
  1131. switch (IrpStack->Parameters.QueryDeviceRelations.Type) {
  1132. case TargetDeviceRelation: {
  1133. PDEVICE_RELATIONS pDevRel;
  1134. //
  1135. // No one else should respond to this since we are the PDO
  1136. //
  1137. ASSERT(Irp->IoStatus.Information == 0);
  1138. if (Irp->IoStatus.Information != 0) {
  1139. break;
  1140. }
  1141. pDevRel = ExAllocatePool(PagedPool, sizeof(DEVICE_RELATIONS));
  1142. if (pDevRel == NULL) {
  1143. status = STATUS_INSUFFICIENT_RESOURCES;
  1144. break;
  1145. }
  1146. pDevRel->Count = 1;
  1147. pDevRel->Objects[0] = DeviceObject;
  1148. ObReferenceObject(DeviceObject);
  1149. status = STATUS_SUCCESS;
  1150. Irp->IoStatus.Information = (ULONG_PTR)pDevRel;
  1151. break;
  1152. }
  1153. default:
  1154. break;
  1155. }
  1156. break;
  1157. case IRP_MN_START_DEVICE:
  1158. //
  1159. // Set the hw resources in the registry for this device.
  1160. //
  1161. status = IoOpenDeviceRegistryKey(DeviceObject, PLUGPLAY_REGKEY_DEVICE,
  1162. STANDARD_RIGHTS_WRITE, &keyHandle);
  1163. if (!NT_SUCCESS(status)) {
  1164. //
  1165. // This is a fatal error. If we can't get to our registry key,
  1166. // we are sunk.
  1167. //
  1168. Cyclomy_KdPrint(DeviceData, SER_DBG_SS_ERROR,
  1169. ("IoOpenDeviceRegistryKey failed - %x\n", status));
  1170. } else {
  1171. ULONG portIndex;
  1172. PFDO_DEVICE_DATA parentExtension;
  1173. // Set the Port Index in the Registry
  1174. RtlInitUnicodeString(&keyName, L"PortIndex");
  1175. portIndex = DeviceData->PortIndex;
  1176. //
  1177. // Doesn't matter whether this works or not.
  1178. //
  1179. ZwSetValueKey(keyHandle, &keyName, 0, REG_DWORD, &portIndex,
  1180. sizeof(ULONG));
  1181. parentExtension = (DeviceData->ParentFdo)->DeviceExtension;
  1182. RtlInitUnicodeString(&keyName, L"PortResources");
  1183. status = ZwSetValueKey(keyHandle, &keyName, 0, REG_RESOURCE_LIST,
  1184. parentExtension->PChildResourceList,
  1185. parentExtension->PChildResourceListSize);
  1186. RtlInitUnicodeString(&keyName, L"PortResourcesTr");
  1187. status = ZwSetValueKey(keyHandle, &keyName, 0, REG_RESOURCE_LIST,
  1188. parentExtension->PChildResourceListTr,
  1189. parentExtension->PChildResourceListSizeTr);
  1190. ZwFlushKey(keyHandle);
  1191. ZwClose(keyHandle);
  1192. }
  1193. SET_NEW_PNP_STATE(DeviceData, Started);
  1194. status = STATUS_SUCCESS;
  1195. break;
  1196. case IRP_MN_QUERY_STOP_DEVICE:
  1197. //
  1198. // No reason here why we can't stop the device.
  1199. // If there were a reason we should speak now for answering success
  1200. // here may result in a stop device irp.
  1201. //
  1202. SET_NEW_PNP_STATE(DeviceData, StopPending);
  1203. status = STATUS_SUCCESS;
  1204. break;
  1205. case IRP_MN_CANCEL_STOP_DEVICE:
  1206. //
  1207. // The stop was canceled. Whatever state we set, or resources we put
  1208. // on hold in anticipation of the forcoming STOP device IRP should be
  1209. // put back to normal. Someone, in the long list of concerned parties,
  1210. // has failed the stop device query.
  1211. //
  1212. //
  1213. // First check to see whether you have received cancel-stop
  1214. // without first receiving a query-stop. This could happen if someone
  1215. // above us fails a query-stop and passes down the subsequent
  1216. // cancel-stop.
  1217. //
  1218. if(StopPending == DeviceData->DevicePnPState)
  1219. {
  1220. //
  1221. // We did receive a query-stop, so restore.
  1222. //
  1223. RESTORE_PREVIOUS_PNP_STATE(DeviceData);
  1224. }
  1225. status = STATUS_SUCCESS;
  1226. break;
  1227. case IRP_MN_STOP_DEVICE:
  1228. //
  1229. // Here we shut down the device. The opposite of start.
  1230. //
  1231. SET_NEW_PNP_STATE(DeviceData, Stopped);
  1232. status = STATUS_SUCCESS;
  1233. break;
  1234. case IRP_MN_QUERY_REMOVE_DEVICE:
  1235. //
  1236. // Just like Query Stop only now the impending doom is the remove irp
  1237. //
  1238. SET_NEW_PNP_STATE(DeviceData, RemovePending);
  1239. status = STATUS_SUCCESS;
  1240. break;
  1241. case IRP_MN_CANCEL_REMOVE_DEVICE:
  1242. //
  1243. // Clean up a remove that did not go through, just like cancel STOP.
  1244. //
  1245. //
  1246. // First check to see whether you have received cancel-remove
  1247. // without first receiving a query-remove. This could happen if
  1248. // someone above us fails a query-remove and passes down the
  1249. // subsequent cancel-remove.
  1250. //
  1251. if(RemovePending == DeviceData->DevicePnPState)
  1252. {
  1253. //
  1254. // We did receive a query-remove, so restore.
  1255. //
  1256. RESTORE_PREVIOUS_PNP_STATE(DeviceData);
  1257. }
  1258. status = STATUS_SUCCESS;
  1259. break;
  1260. case IRP_MN_SURPRISE_REMOVAL:
  1261. //
  1262. // We should stop all access to the device and relinquish all the
  1263. // resources. Let's just mark that it happened and we will do
  1264. // the cleanup later in IRP_MN_REMOVE_DEVICE.
  1265. //
  1266. SET_NEW_PNP_STATE(DeviceData, SurpriseRemovePending);
  1267. status = STATUS_SUCCESS;
  1268. break;
  1269. case IRP_MN_REMOVE_DEVICE:
  1270. //
  1271. // Attached is only set to FALSE by the enumeration process.
  1272. //
  1273. if (!DeviceData->Attached) {
  1274. SET_NEW_PNP_STATE(DeviceData, Deleted);
  1275. status = Cyclomy_PnPRemove(DeviceObject, DeviceData);
  1276. }
  1277. else { // else added in build 2128 - Fanny
  1278. //
  1279. // Succeed the remove
  1280. ///
  1281. SET_NEW_PNP_STATE(DeviceData, NotStarted);
  1282. status = STATUS_SUCCESS;
  1283. }
  1284. // Changed in build 2072
  1285. // status = STATUS_SUCCESS;
  1286. break;
  1287. case IRP_MN_QUERY_RESOURCES: {
  1288. #if 0
  1289. PCM_RESOURCE_LIST pChildRes, pQueryRes;
  1290. PFDO_DEVICE_DATA parentExtension;
  1291. ULONG listSize;
  1292. parentExtension = (DeviceData->ParentFdo)->DeviceExtension;
  1293. pChildRes = parentExtension->PChildResourceList;
  1294. listSize = parentExtension->PChildResourceListSize;
  1295. if (pChildRes) {
  1296. pQueryRes = ExAllocatePool(PagedPool, listSize);
  1297. if (pQueryRes == NULL) {
  1298. Irp->IoStatus.Information = (ULONG_PTR) NULL;
  1299. status = STATUS_INSUFFICIENT_RESOURCES;
  1300. } else {
  1301. RtlCopyMemory(pQueryRes,pChildRes,listSize);
  1302. Irp->IoStatus.Information = (ULONG_PTR)pQueryRes;
  1303. status = STATUS_SUCCESS;
  1304. }
  1305. }
  1306. #endif
  1307. break;
  1308. }
  1309. case IRP_MN_QUERY_RESOURCE_REQUIREMENTS: {
  1310. #if 0
  1311. PIO_RESOURCE_REQUIREMENTS_LIST pChildReq, pQueryReq;
  1312. PFDO_DEVICE_DATA parentExtension;
  1313. parentExtension = (DeviceData->ParentFdo)->DeviceExtension;
  1314. pChildReq = parentExtension->PChildRequiredList;
  1315. if (pChildReq) {
  1316. pQueryReq = ExAllocatePool(PagedPool, pChildReq->ListSize);
  1317. if (pQueryReq == NULL) {
  1318. Irp->IoStatus.Information = (ULONG_PTR) NULL;
  1319. status = STATUS_INSUFFICIENT_RESOURCES;
  1320. } else {
  1321. RtlCopyMemory(pQueryReq,pChildReq,pChildReq->ListSize);
  1322. Irp->IoStatus.Information = (ULONG_PTR)pQueryReq;
  1323. status = STATUS_SUCCESS;
  1324. }
  1325. }
  1326. #endif
  1327. break;
  1328. }
  1329. case IRP_MN_READ_CONFIG:
  1330. case IRP_MN_WRITE_CONFIG: // we have no config space
  1331. case IRP_MN_EJECT:
  1332. case IRP_MN_SET_LOCK:
  1333. case IRP_MN_QUERY_INTERFACE: // We do not have any non IRP based interfaces.
  1334. default:
  1335. Cyclomy_KdPrint(DeviceData, SER_DBG_PNP_TRACE,
  1336. ("PDO(%x):%s not handled\n", DeviceObject,
  1337. PnPMinorFunctionString(IrpStack->MinorFunction)));
  1338. // For PnP requests to the PDO that we do not understand we should
  1339. // return the IRP WITHOUT setting the status or information fields.
  1340. // They may have already been set by a filter (eg acpi).
  1341. break;
  1342. }
  1343. Irp->IoStatus.Status = status;
  1344. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  1345. return status;
  1346. }
  1347. NTSTATUS
  1348. Cyclomy_PnPRemove (PDEVICE_OBJECT Device, PPDO_DEVICE_DATA PdoData)
  1349. /*++
  1350. Routine Description:
  1351. The PlugPlay subsystem has instructed that this PDO should be removed.
  1352. We should therefore
  1353. - Complete any requests queued in the driver
  1354. - If the device is still attached to the system,
  1355. then complete the request and return.
  1356. - Otherwise, cleanup device specific allocations, memory, events...
  1357. - Call IoDeleteDevice
  1358. - Return from the dispatch routine.
  1359. Note that if the device is still connected to the bus (IE in this case
  1360. the control panel has not yet told us that the serial device has
  1361. disappeared) then the PDO must remain around, and must be returned during
  1362. any query Device relaions IRPS.
  1363. --*/
  1364. {
  1365. Cyclomy_KdPrint(PdoData, SER_DBG_PNP_TRACE,
  1366. ("Cyclomy_PnPRemove: 0x%x\n", Device));
  1367. //
  1368. // Complete any outstanding requests with STATUS_DELETE_PENDING.
  1369. //
  1370. // Serenum does not queue any irps at this time so we have nothing to do.
  1371. //
  1372. //REMOVED BY FANNY. THIS CHECK IS ALREADY DONE AT IRP_MN_REMOVE_DEVICE PDO.
  1373. //if (PdoData->Attached) {
  1374. // return STATUS_SUCCESS;
  1375. //}
  1376. //PdoData->Removed = TRUE;
  1377. //
  1378. // Free any resources.
  1379. //
  1380. CyclomyFreeUnicodeString(&PdoData->HardwareIDs);
  1381. //CyclomyFreeUnicodeString(&PdoData->CompIDs); We never allocate CompIDs.
  1382. RtlFreeUnicodeString(&PdoData->DeviceIDs);
  1383. RtlFreeUnicodeString(&PdoData->InstanceIDs);
  1384. RtlFreeUnicodeString(&PdoData->DevDesc);
  1385. Cyclomy_KdPrint(PdoData, SER_DBG_PNP_INFO,
  1386. ("IoDeleteDevice: 0x%x\n", Device));
  1387. IoDeleteDevice(Device);
  1388. return STATUS_SUCCESS;
  1389. }
  1390. NTSTATUS
  1391. Cyclomy_GetResourceInfo(IN PDEVICE_OBJECT PDevObj,
  1392. IN PCM_RESOURCE_LIST PResList,
  1393. IN PCM_RESOURCE_LIST PTrResList)
  1394. /*++
  1395. Routine Description:
  1396. This routine gets the resources that PnP allocated to this device.
  1397. Arguments:
  1398. PDevObj - Pointer to the devobj that is starting
  1399. PResList - Pointer to the untranslated resources needed by this device
  1400. PTrResList - Pointer to the translated resources needed by this device
  1401. Return Value:
  1402. STATUS_SUCCESS on success, something else appropriate on failure
  1403. --*/
  1404. {
  1405. PFDO_DEVICE_DATA pDevExt = PDevObj->DeviceExtension;
  1406. NTSTATUS status = STATUS_SUCCESS;
  1407. ULONG count;
  1408. ULONG i;
  1409. PCM_PARTIAL_RESOURCE_LIST pPartialResourceList, pPartialTrResourceList;
  1410. PCM_PARTIAL_RESOURCE_DESCRIPTOR pPartialResourceDesc, pPartialTrResourceDesc;
  1411. PCM_FULL_RESOURCE_DESCRIPTOR pFullResourceDesc = NULL,
  1412. pFullTrResourceDesc = NULL;
  1413. KAFFINITY Affinity;
  1414. KINTERRUPT_MODE InterruptMode;
  1415. ULONG zero = 0;
  1416. PAGED_CODE();
  1417. // Let's get our resources
  1418. pFullResourceDesc = &PResList->List[0];
  1419. pFullTrResourceDesc = &PTrResList->List[0];
  1420. if (pFullResourceDesc) {
  1421. pPartialResourceList = &pFullResourceDesc->PartialResourceList;
  1422. pPartialResourceDesc = pPartialResourceList->PartialDescriptors;
  1423. count = pPartialResourceList->Count;
  1424. pDevExt->InterfaceType = pFullResourceDesc->InterfaceType;
  1425. pDevExt->BusNumber = pFullResourceDesc->BusNumber;
  1426. for (i = 0; i < count; i++, pPartialResourceDesc++) {
  1427. switch (pPartialResourceDesc->Type) {
  1428. case CmResourceTypeMemory: {
  1429. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypeMemory\n"));
  1430. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Memory.Start = %x\n",
  1431. pPartialResourceDesc->u.Memory.Start));
  1432. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Memory.Length = %x\n",
  1433. pPartialResourceDesc->u.Memory.Length));
  1434. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Flags = %x\n",
  1435. pPartialResourceDesc->Flags));
  1436. if (pPartialResourceDesc->u.Memory.Length == CYY_RUNTIME_LENGTH) {
  1437. pDevExt->PhysicalRuntime = pPartialResourceDesc->u.Memory.Start;
  1438. pDevExt->RuntimeLength = pPartialResourceDesc->u.Memory.Length;
  1439. } else {
  1440. pDevExt->PhysicalBoardMemory = pPartialResourceDesc->u.Memory.Start;
  1441. pDevExt->BoardMemoryLength = pPartialResourceDesc->u.Memory.Length;
  1442. }
  1443. break;
  1444. }
  1445. case CmResourceTypePort: {
  1446. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypePort\n"));
  1447. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Port.Start = %x\n",
  1448. pPartialResourceDesc->u.Port.Start));
  1449. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Port.Length = %x\n",
  1450. pPartialResourceDesc->u.Port.Length));
  1451. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Flags = %x\n",
  1452. pPartialResourceDesc->Flags));
  1453. break;
  1454. }
  1455. case CmResourceTypeInterrupt: {
  1456. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypeInterrupt\n"));
  1457. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Interrupt.Level = %x\n",
  1458. pPartialResourceDesc->u.Interrupt.Level));
  1459. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Interrupt.Vector = %x\n",
  1460. pPartialResourceDesc->u.Interrupt.Vector));
  1461. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Interrupt.Affinity = %x\n",
  1462. pPartialResourceDesc->u.Interrupt.Affinity));
  1463. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Flags = %x\n",
  1464. pPartialResourceDesc->Flags));
  1465. pDevExt->OriginalIrql = pPartialResourceDesc->u.Interrupt.Level;
  1466. pDevExt->OriginalVector =pPartialResourceDesc->u.Interrupt.Vector;
  1467. Affinity = pPartialResourceDesc->u.Interrupt.Affinity;
  1468. if (pPartialResourceDesc->Flags & CM_RESOURCE_INTERRUPT_LATCHED) {
  1469. InterruptMode = Latched;
  1470. } else {
  1471. InterruptMode = LevelSensitive;
  1472. }
  1473. break;
  1474. }
  1475. case CmResourceTypeDeviceSpecific: {
  1476. PCM_SERIAL_DEVICE_DATA sDeviceData;
  1477. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypeDeviceSpecific\n"));
  1478. break;
  1479. }
  1480. default: {
  1481. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceType = %x\n",
  1482. pPartialResourceDesc->Type));
  1483. break;
  1484. }
  1485. } // switch (pPartialResourceDesc->Type)
  1486. } // for (i = 0; i < count; i++, pPartialResourceDesc++)
  1487. } // if (pFullResourceDesc)
  1488. //SEE_LATER_IF_IT_SHOULD_BE_ADDED
  1489. // //
  1490. // // Do the same for the translated resources
  1491. // //
  1492. //
  1493. // gotInt = 0;
  1494. // gotISR = 0;
  1495. // gotIO = 0;
  1496. // curIoIndex = 0;
  1497. //
  1498. if (pFullTrResourceDesc) {
  1499. pPartialTrResourceList = &pFullTrResourceDesc->PartialResourceList;
  1500. pPartialTrResourceDesc = pPartialTrResourceList->PartialDescriptors;
  1501. count = pPartialTrResourceList->Count;
  1502. //
  1503. // Reload PConfig with the translated values for later use
  1504. //
  1505. pDevExt->InterfaceType = pFullTrResourceDesc->InterfaceType;
  1506. pDevExt->BusNumber = pFullTrResourceDesc->BusNumber;
  1507. //FANNY
  1508. // pDevExt->TrInterruptStatus = SerialPhysicalZero;
  1509. for (i = 0; i < count; i++, pPartialTrResourceDesc++) {
  1510. switch (pPartialTrResourceDesc->Type) {
  1511. case CmResourceTypeMemory: {
  1512. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypeMemory translated\n"));
  1513. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Memory.Start = %x\n",
  1514. pPartialTrResourceDesc->u.Memory.Start));
  1515. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Memory.Length = %x\n",
  1516. pPartialTrResourceDesc->u.Memory.Length));
  1517. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Flags = %x\n",
  1518. pPartialTrResourceDesc->Flags));
  1519. if (pPartialTrResourceDesc->u.Memory.Length == CYY_RUNTIME_LENGTH) {
  1520. pDevExt->TranslatedRuntime = pPartialTrResourceDesc->u.Memory.Start;
  1521. pDevExt->RuntimeLength = pPartialTrResourceDesc->u.Memory.Length;
  1522. } else {
  1523. pDevExt->TranslatedBoardMemory = pPartialTrResourceDesc->u.Memory.Start;
  1524. pDevExt->BoardMemoryLength = pPartialTrResourceDesc->u.Memory.Length;
  1525. }
  1526. break;
  1527. }
  1528. case CmResourceTypePort: {
  1529. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypePort translated\n"));
  1530. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Port.Start = %x\n",
  1531. pPartialTrResourceDesc->u.Port.Start));
  1532. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Port.Length = %x\n",
  1533. pPartialTrResourceDesc->u.Port.Length));
  1534. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Flags = %x\n",
  1535. pPartialTrResourceDesc->Flags));
  1536. break;
  1537. }
  1538. case CmResourceTypeInterrupt: {
  1539. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("CmResourceTypeInterrupt translated\n"));
  1540. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Interrupt.Level = %x\n",
  1541. pPartialTrResourceDesc->u.Interrupt.Level));
  1542. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Interrupt.Vector = %x\n",
  1543. pPartialTrResourceDesc->u.Interrupt.Vector));
  1544. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("u.Interrupt.Affinity = %x\n",
  1545. pPartialTrResourceDesc->u.Interrupt.Affinity));
  1546. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Flags = %x\n",
  1547. pPartialTrResourceDesc->Flags));
  1548. pDevExt->Vector = pPartialTrResourceDesc->u.Interrupt.Vector;
  1549. pDevExt->Irql = (KIRQL) pPartialTrResourceDesc->u.Interrupt.Level;
  1550. Affinity = pPartialTrResourceDesc->u.Interrupt.Affinity;
  1551. break;
  1552. }
  1553. default: {
  1554. break;
  1555. }
  1556. } // switch (pPartialTrResourceDesc->Type)
  1557. } // for (i = 0; i < count; i++, pPartialTrResourceDesc++)
  1558. } // if (pFullTrResourceDesc)
  1559. //
  1560. // Do some error checking on the configuration info we have.
  1561. //
  1562. // Make sure that the interrupt is non zero (which we defaulted
  1563. // it to).
  1564. //
  1565. // Make sure that the portaddress is non zero (which we defaulted
  1566. // it to).
  1567. //
  1568. // Make sure that the DosDevices is not NULL (which we defaulted
  1569. // it to).
  1570. //
  1571. // We need to make sure that if an interrupt status
  1572. // was specified, that a port index was also specfied,
  1573. // and if so that the port index is <= maximum ports
  1574. // on a board.
  1575. //
  1576. // We should also validate that the bus type and number
  1577. // are correct.
  1578. //
  1579. // We will also validate that the interrupt mode makes
  1580. // sense for the bus.
  1581. //
  1582. if (!pDevExt->TranslatedRuntime.LowPart && pDevExt->IsPci) {
  1583. CyyLogError(
  1584. pDevExt->DriverObject,
  1585. NULL,
  1586. pDevExt->PhysicalBoardMemory,
  1587. CyyPhysicalZero,
  1588. 0,
  1589. 0,
  1590. 0,
  1591. 0,
  1592. STATUS_SUCCESS,
  1593. CYY_INVALID_RUNTIME_REGISTERS,
  1594. 0,
  1595. NULL,
  1596. 0,
  1597. NULL
  1598. );
  1599. Cyclomy_KdPrint (pDevExt,SER_DBG_CYCLADES,
  1600. ("Bogus PLX address %x\n",
  1601. pDevExt->TranslatedRuntime.LowPart));
  1602. status = STATUS_INSUFFICIENT_RESOURCES;
  1603. goto GetResourceInfo_Cleanup;
  1604. }
  1605. if (!pDevExt->TranslatedBoardMemory.LowPart) {
  1606. CyyLogError(
  1607. pDevExt->DriverObject,
  1608. NULL,
  1609. pDevExt->PhysicalBoardMemory,
  1610. CyyPhysicalZero,
  1611. 0,
  1612. 0,
  1613. 0,
  1614. 0,
  1615. STATUS_SUCCESS,
  1616. CYY_INVALID_BOARD_MEMORY,
  1617. 0,
  1618. NULL,
  1619. 0,
  1620. NULL
  1621. );
  1622. Cyclomy_KdPrint (pDevExt,SER_DBG_CYCLADES,
  1623. ("Bogus TranslatedBoardMemory address %x\n",
  1624. pDevExt->TranslatedBoardMemory.LowPart));
  1625. status = STATUS_INSUFFICIENT_RESOURCES;
  1626. goto GetResourceInfo_Cleanup;
  1627. }
  1628. if (!pDevExt->OriginalVector) {
  1629. CyyLogError(
  1630. pDevExt->DriverObject,
  1631. NULL,
  1632. pDevExt->PhysicalBoardMemory,
  1633. CyyPhysicalZero,
  1634. 0,
  1635. 0,
  1636. 0,
  1637. 0,
  1638. STATUS_SUCCESS,
  1639. CYY_INVALID_INTERRUPT,
  1640. 0,
  1641. NULL,
  1642. 0,
  1643. NULL
  1644. );
  1645. Cyclomy_KdPrint (pDevExt,SER_DBG_CYCLADES,("Bogus vector %x\n",
  1646. pDevExt->OriginalVector));
  1647. status = STATUS_INSUFFICIENT_RESOURCES;
  1648. goto GetResourceInfo_Cleanup;
  1649. }
  1650. //
  1651. // We don't want to cause the hal to have a bad day,
  1652. // so let's check the interface type and bus number.
  1653. //
  1654. // We only need to check the registry if they aren't
  1655. // equal to the defaults.
  1656. //
  1657. if (pDevExt->BusNumber != 0) {
  1658. BOOLEAN foundIt = 0;
  1659. if (pDevExt->InterfaceType >= MaximumInterfaceType) {
  1660. CyyLogError(
  1661. pDevExt->DriverObject,
  1662. NULL,
  1663. pDevExt->PhysicalBoardMemory,
  1664. CyyPhysicalZero,
  1665. 0,
  1666. 0,
  1667. 0,
  1668. pDevExt->InterfaceType,
  1669. STATUS_SUCCESS,
  1670. CYY_UNKNOWN_BUS,
  1671. 0,
  1672. NULL,
  1673. 0,
  1674. NULL
  1675. );
  1676. Cyclomy_KdPrint (pDevExt,SER_DBG_CYCLADES,
  1677. ("Invalid Bus type %x\n", pDevExt->BusNumber));
  1678. //status = SERIAL_UNKNOWN_BUS;
  1679. status = STATUS_INSUFFICIENT_RESOURCES;
  1680. goto GetResourceInfo_Cleanup;
  1681. }
  1682. IoQueryDeviceDescription(
  1683. (INTERFACE_TYPE *)&pDevExt->InterfaceType,
  1684. &zero,
  1685. NULL,
  1686. NULL,
  1687. NULL,
  1688. NULL,
  1689. Cyclomy_ItemCallBack,
  1690. &foundIt
  1691. );
  1692. if (!foundIt) {
  1693. CyyLogError(
  1694. pDevExt->DriverObject,
  1695. NULL,
  1696. pDevExt->PhysicalBoardMemory,
  1697. CyyPhysicalZero,
  1698. 0,
  1699. 0,
  1700. 0,
  1701. pDevExt->InterfaceType,
  1702. STATUS_SUCCESS,
  1703. CYY_BUS_NOT_PRESENT,
  1704. 0,
  1705. NULL,
  1706. 0,
  1707. NULL
  1708. );
  1709. Cyclomy_KdPrint(
  1710. pDevExt,
  1711. SER_DBG_CYCLADES,
  1712. ("There aren't that many of those\n"
  1713. "------- busses on this system,%x\n",
  1714. pDevExt->BusNumber)
  1715. );
  1716. //status = SERIAL_BUS_NOT_PRESENT;
  1717. status = STATUS_INSUFFICIENT_RESOURCES;
  1718. goto GetResourceInfo_Cleanup;
  1719. }
  1720. }
  1721. //
  1722. // Dump out the board configuration.
  1723. //
  1724. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES, ("PhysicalRuntime: %x\n",
  1725. pDevExt->PhysicalRuntime.LowPart));
  1726. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES, ("TranslatedRuntime: %x\n",
  1727. pDevExt->TranslatedRuntime.LowPart));
  1728. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES, ("RuntimeLength: %x\n",
  1729. pDevExt->RuntimeLength));
  1730. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES, ("PhysicalBoardMemory: %x\n",
  1731. pDevExt->PhysicalBoardMemory.LowPart));
  1732. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES, ("TranslatedBoardMemory: %x\n",
  1733. pDevExt->TranslatedBoardMemory.LowPart));
  1734. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES, ("BoardMemoryLength: %x\n",
  1735. pDevExt->BoardMemoryLength));
  1736. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("OriginalIrql = %x\n",
  1737. pDevExt->OriginalIrql));
  1738. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("OriginalVector = %x\n",
  1739. pDevExt->OriginalVector));
  1740. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Irql = %x\n",
  1741. pDevExt->Irql));
  1742. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Vector = %x\n",
  1743. pDevExt->Vector));
  1744. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Affinity = %x\n",
  1745. Affinity));
  1746. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("InterfaceType = %x\n",
  1747. pDevExt->InterfaceType));
  1748. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("BusNumber = %x\n",
  1749. pDevExt->BusNumber));
  1750. // ABOVE: COPIED FROM SerialGetPortInfo
  1751. // ------------------------------------
  1752. // BELOW: COPIED FROM SerialInitController
  1753. if (pDevExt->IsPci) {
  1754. pDevExt->Runtime = MmMapIoSpace(pDevExt->TranslatedRuntime,
  1755. pDevExt->RuntimeLength,
  1756. FALSE);
  1757. if (!pDevExt->Runtime){
  1758. CyyLogError(
  1759. pDevExt->DriverObject,
  1760. NULL,
  1761. pDevExt->PhysicalBoardMemory,
  1762. CyyPhysicalZero,
  1763. 0,
  1764. 0,
  1765. 0,
  1766. 0,
  1767. STATUS_SUCCESS,
  1768. CYY_RUNTIME_NOT_MAPPED,
  1769. 0,
  1770. NULL,
  1771. 0,
  1772. NULL
  1773. );
  1774. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Could not map memory for PLX\n"));
  1775. status = STATUS_NONE_MAPPED;
  1776. goto GetResourceInfo_Cleanup;
  1777. }
  1778. }
  1779. pDevExt->BoardMemory = MmMapIoSpace(pDevExt->TranslatedBoardMemory,
  1780. pDevExt->BoardMemoryLength,
  1781. FALSE);
  1782. if (!pDevExt->BoardMemory){
  1783. CyyLogError(
  1784. pDevExt->DriverObject,
  1785. NULL,
  1786. pDevExt->PhysicalBoardMemory,
  1787. CyyPhysicalZero,
  1788. 0,
  1789. 0,
  1790. 0,
  1791. 0,
  1792. STATUS_SUCCESS,
  1793. CYY_BOARD_NOT_MAPPED,
  1794. 0,
  1795. NULL,
  1796. 0,
  1797. NULL
  1798. );
  1799. Cyclomy_KdPrint(pDevExt,SER_DBG_CYCLADES,("Could not map memory for CD1400\n"));
  1800. status = STATUS_NONE_MAPPED;
  1801. goto GetResourceInfo_Cleanup;
  1802. }
  1803. GetResourceInfo_Cleanup:
  1804. if (!NT_SUCCESS(status)) {
  1805. if (pDevExt->Runtime) {
  1806. MmUnmapIoSpace(pDevExt->Runtime, pDevExt->RuntimeLength);
  1807. pDevExt->Runtime = NULL;
  1808. }
  1809. if (pDevExt->BoardMemory) {
  1810. MmUnmapIoSpace(pDevExt->BoardMemory, pDevExt->BoardMemoryLength);
  1811. pDevExt->BoardMemory = NULL;
  1812. }
  1813. }
  1814. Cyclomy_KdPrint (pDevExt,SER_DBG_CYCLADES, ("leaving Cyclomy_GetResourceInfo\n"));
  1815. return status;
  1816. }
  1817. VOID
  1818. Cyclomy_ReleaseResources(IN PFDO_DEVICE_DATA PDevExt)
  1819. {
  1820. Cyclomy_KdPrint (PDevExt,SER_DBG_CYCLADES, ("entering Cyclomy_ReleaseResources\n"));
  1821. if (PDevExt->PChildRequiredList) {
  1822. ExFreePool(PDevExt->PChildRequiredList);
  1823. PDevExt->PChildRequiredList = NULL;
  1824. }
  1825. if (PDevExt->PChildResourceList) {
  1826. ExFreePool(PDevExt->PChildResourceList);
  1827. PDevExt->PChildResourceList = NULL;
  1828. }
  1829. if (PDevExt->PChildResourceListTr) {
  1830. ExFreePool(PDevExt->PChildResourceListTr);
  1831. PDevExt->PChildResourceListTr = NULL;
  1832. }
  1833. if (PDevExt->Runtime) {
  1834. MmUnmapIoSpace(PDevExt->Runtime, PDevExt->RuntimeLength);
  1835. PDevExt->Runtime = NULL;
  1836. }
  1837. if (PDevExt->BoardMemory) {
  1838. MmUnmapIoSpace(PDevExt->BoardMemory, PDevExt->BoardMemoryLength);
  1839. PDevExt->BoardMemory = NULL;
  1840. }
  1841. Cyclomy_KdPrint (PDevExt,SER_DBG_CYCLADES, ("leaving Cyclomy_ReleaseResources\n"));
  1842. }
  1843. NTSTATUS
  1844. Cyclomy_ItemCallBack(
  1845. IN PVOID Context,
  1846. IN PUNICODE_STRING PathName,
  1847. IN INTERFACE_TYPE BusType,
  1848. IN ULONG BusNumber,
  1849. IN PKEY_VALUE_FULL_INFORMATION *BusInformation,
  1850. IN CONFIGURATION_TYPE ControllerType,
  1851. IN ULONG ControllerNumber,
  1852. IN PKEY_VALUE_FULL_INFORMATION *ControllerInformation,
  1853. IN CONFIGURATION_TYPE PeripheralType,
  1854. IN ULONG PeripheralNumber,
  1855. IN PKEY_VALUE_FULL_INFORMATION *PeripheralInformation
  1856. )
  1857. /*++
  1858. Routine Description:
  1859. This routine is called to check if a particular item
  1860. is present in the registry.
  1861. Arguments:
  1862. Context - Pointer to a boolean.
  1863. PathName - unicode registry path. Not Used.
  1864. BusType - Internal, Isa, ...
  1865. BusNumber - Which bus if we are on a multibus system.
  1866. BusInformation - Configuration information about the bus. Not Used.
  1867. ControllerType - Controller type.
  1868. ControllerNumber - Which controller if there is more than one
  1869. controller in the system.
  1870. ControllerInformation - Array of pointers to the three pieces of
  1871. registry information.
  1872. PeripheralType - Should be a peripheral.
  1873. PeripheralNumber - Which peripheral - not used..
  1874. PeripheralInformation - Configuration information. Not Used.
  1875. Return Value:
  1876. STATUS_SUCCESS
  1877. --*/
  1878. {
  1879. UNREFERENCED_PARAMETER (PathName);
  1880. UNREFERENCED_PARAMETER (BusType);
  1881. UNREFERENCED_PARAMETER (BusNumber);
  1882. UNREFERENCED_PARAMETER (BusInformation);
  1883. UNREFERENCED_PARAMETER (ControllerType);
  1884. UNREFERENCED_PARAMETER (ControllerNumber);
  1885. UNREFERENCED_PARAMETER (ControllerInformation);
  1886. UNREFERENCED_PARAMETER (PeripheralType);
  1887. UNREFERENCED_PARAMETER (PeripheralNumber);
  1888. UNREFERENCED_PARAMETER (PeripheralInformation);
  1889. PAGED_CODE();
  1890. *((BOOLEAN *)Context) = TRUE;
  1891. return STATUS_SUCCESS;
  1892. }
  1893. NTSTATUS
  1894. Cyclomy_BuildRequirementsList(
  1895. OUT PIO_RESOURCE_REQUIREMENTS_LIST *PChildRequiredList_Pointer,
  1896. IN PCM_RESOURCE_LIST PResourceList, IN ULONG NumberOfResources
  1897. )
  1898. {
  1899. NTSTATUS status = STATUS_SUCCESS;
  1900. ULONG count;
  1901. ULONG i,j;
  1902. PCM_FULL_RESOURCE_DESCRIPTOR pFullResourceDesc = NULL;
  1903. PCM_PARTIAL_RESOURCE_LIST pPartialResourceList;
  1904. PCM_PARTIAL_RESOURCE_DESCRIPTOR pPartialResourceDesc;
  1905. ULONG requiredLength;
  1906. PIO_RESOURCE_REQUIREMENTS_LIST requiredList;
  1907. PIO_RESOURCE_LIST requiredResList;
  1908. PIO_RESOURCE_DESCRIPTOR requiredResDesc;
  1909. *PChildRequiredList_Pointer = NULL;
  1910. // Validate input parameter
  1911. if (PResourceList == NULL) {
  1912. status = STATUS_INSUFFICIENT_RESOURCES;
  1913. goto CyclomyBuildRequirementList_Error;
  1914. }
  1915. ASSERT(PResourceList->Count == 1);
  1916. // Initialize requiredList
  1917. requiredLength = sizeof(IO_RESOURCE_REQUIREMENTS_LIST)
  1918. + sizeof(IO_RESOURCE_DESCRIPTOR) * (NumberOfResources - 1);
  1919. requiredList = ExAllocatePool(PagedPool, requiredLength);
  1920. if (requiredList == NULL) {
  1921. status = STATUS_INSUFFICIENT_RESOURCES;
  1922. goto CyclomyBuildRequirementList_Error;
  1923. }
  1924. RtlZeroMemory(requiredList, requiredLength);
  1925. // Get information from PResourceList and build requiredList
  1926. pFullResourceDesc = &PResourceList->List[0];
  1927. if (pFullResourceDesc) {
  1928. pPartialResourceList = &pFullResourceDesc->PartialResourceList;
  1929. pPartialResourceDesc = pPartialResourceList->PartialDescriptors;
  1930. count = pPartialResourceList->Count;
  1931. if (count < NumberOfResources) {
  1932. ExFreePool(requiredList);
  1933. return STATUS_INSUFFICIENT_RESOURCES;
  1934. }
  1935. requiredList->ListSize = requiredLength;
  1936. requiredList->InterfaceType = pFullResourceDesc->InterfaceType;
  1937. requiredList->BusNumber = pFullResourceDesc->BusNumber;
  1938. requiredList->SlotNumber = 0; //?????? There's no SlotNumber in the Resource List
  1939. requiredList->AlternativeLists = 1;
  1940. requiredResList = &requiredList->List[0];
  1941. requiredResList->Count = NumberOfResources;
  1942. requiredResDesc = &requiredResList->Descriptors[0];
  1943. for (i=0,j=0; i<count && j<NumberOfResources; i++,pPartialResourceDesc++) {
  1944. switch (pPartialResourceDesc->Type) {
  1945. case CmResourceTypeMemory: {
  1946. requiredResDesc->Type = pPartialResourceDesc->Type;
  1947. //requiredResDesc->ShareDisposition = pPartialResourceDesc->ShareDisposition;
  1948. requiredResDesc->ShareDisposition = CmResourceShareShared;
  1949. requiredResDesc->Flags = pPartialResourceDesc->Flags;
  1950. requiredResDesc->u.Memory.Length = pPartialResourceDesc->u.Memory.Length;
  1951. requiredResDesc->u.Memory.Alignment = 4;
  1952. requiredResDesc->u.Memory.MinimumAddress = pPartialResourceDesc->u.Memory.Start;
  1953. requiredResDesc->u.Memory.MaximumAddress.QuadPart
  1954. = pPartialResourceDesc->u.Memory.Start.QuadPart
  1955. + pPartialResourceDesc->u.Memory.Length - 1;
  1956. requiredResDesc++;
  1957. j++;
  1958. break;
  1959. }
  1960. case CmResourceTypePort: {
  1961. break;
  1962. }
  1963. case CmResourceTypeInterrupt: {
  1964. requiredResDesc->Type = pPartialResourceDesc->Type;
  1965. requiredResDesc->ShareDisposition = CmResourceShareShared;
  1966. requiredResDesc->Flags = pPartialResourceDesc->Flags;
  1967. requiredResDesc->u.Interrupt.MinimumVector
  1968. = pPartialResourceDesc->u.Interrupt.Vector;
  1969. requiredResDesc->u.Interrupt.MaximumVector
  1970. = pPartialResourceDesc->u.Interrupt.Vector;
  1971. requiredResDesc++;
  1972. j++;
  1973. break;
  1974. }
  1975. default:
  1976. break;
  1977. } // end switch
  1978. } // end for
  1979. } // end if (pFullResourceDesc)
  1980. *PChildRequiredList_Pointer = requiredList;
  1981. CyclomyBuildRequirementList_Error:
  1982. return status;
  1983. }
  1984. NTSTATUS
  1985. Cyclomy_BuildResourceList(
  1986. OUT PCM_RESOURCE_LIST *POutList_Pointer,
  1987. OUT ULONG *ListSize_Pointer,
  1988. IN PCM_RESOURCE_LIST PInList,
  1989. IN ULONG NumberOfResources
  1990. )
  1991. {
  1992. NTSTATUS status = STATUS_SUCCESS;
  1993. ULONG i,j;
  1994. PCM_PARTIAL_RESOURCE_DESCRIPTOR pPartialInDesc;
  1995. ULONG length;
  1996. PCM_RESOURCE_LIST pOutList;
  1997. PCM_PARTIAL_RESOURCE_DESCRIPTOR pPartialOutDesc;
  1998. *POutList_Pointer = NULL;
  1999. *ListSize_Pointer =0;
  2000. // Validate input parameter
  2001. if (PInList == NULL) {
  2002. status = STATUS_INSUFFICIENT_RESOURCES;
  2003. goto CyclomyBuildResourceList_Error;
  2004. }
  2005. ASSERT(PInList->Count == 1);
  2006. if (PInList->List[0].PartialResourceList.Count < NumberOfResources) {
  2007. status = STATUS_INSUFFICIENT_RESOURCES;
  2008. goto CyclomyBuildResourceList_Error;
  2009. }
  2010. // Initialize pOutList
  2011. length = sizeof(CM_RESOURCE_LIST)
  2012. + sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR) * (NumberOfResources - 1);
  2013. pOutList = ExAllocatePool(PagedPool, length);
  2014. if (pOutList == NULL) {
  2015. status = STATUS_INSUFFICIENT_RESOURCES;
  2016. goto CyclomyBuildResourceList_Error;
  2017. }
  2018. RtlZeroMemory(pOutList, length);
  2019. // Get information from PInList and build pOutList
  2020. pOutList->Count = 1; // not sure if we have to report Translated information too.
  2021. pOutList->List[0].InterfaceType = PInList->List[0].InterfaceType;
  2022. pOutList->List[0].BusNumber = PInList->List[0].BusNumber;
  2023. pOutList->List[0].PartialResourceList.Count = NumberOfResources;
  2024. pPartialOutDesc = &pOutList->List[0].PartialResourceList.PartialDescriptors[0];
  2025. pPartialInDesc = &PInList->List[0].PartialResourceList.PartialDescriptors[0];
  2026. for (i=0,j=0; i < PInList->List[0].PartialResourceList.Count; i++,pPartialInDesc++) {
  2027. if (j==NumberOfResources) {
  2028. break;
  2029. }
  2030. switch(pPartialInDesc->Type) {
  2031. case CmResourceTypeMemory:
  2032. pPartialOutDesc->ShareDisposition = CmResourceShareShared;
  2033. pPartialOutDesc->Type = pPartialInDesc->Type;
  2034. pPartialOutDesc->Flags = pPartialInDesc->Flags;
  2035. pPartialOutDesc->u.Memory.Start = pPartialInDesc->u.Memory.Start;
  2036. pPartialOutDesc->u.Memory.Length = pPartialInDesc->u.Memory.Length;
  2037. pPartialOutDesc++;
  2038. j++;
  2039. break;
  2040. case CmResourceTypeInterrupt:
  2041. pPartialOutDesc->ShareDisposition = CmResourceShareShared;
  2042. pPartialOutDesc->Type = pPartialInDesc->Type;
  2043. pPartialOutDesc->Flags = pPartialInDesc->Flags;
  2044. pPartialOutDesc->u.Interrupt.Level = pPartialInDesc->u.Interrupt.Level;
  2045. pPartialOutDesc->u.Interrupt.Vector = pPartialInDesc->u.Interrupt.Vector;
  2046. pPartialOutDesc->u.Interrupt.Affinity = pPartialInDesc->u.Interrupt.Affinity;
  2047. pPartialOutDesc++;
  2048. j++;
  2049. break;
  2050. default:
  2051. break;
  2052. } // end switch
  2053. } // end for
  2054. *POutList_Pointer = pOutList;
  2055. *ListSize_Pointer = length;
  2056. CyclomyBuildResourceList_Error:
  2057. return status;
  2058. }
  2059. VOID
  2060. Cyclomy_Delay(
  2061. ULONG NumberOfMilliseconds
  2062. )
  2063. /*--------------------------------------------------------------------------
  2064. Cyclomy_Delay()
  2065. Routine Description: Delay routine.
  2066. Arguments:
  2067. NumberOfMilliseconds - Number of milliseconds to be delayed.
  2068. Return Value: none.
  2069. --------------------------------------------------------------------------*/
  2070. {
  2071. LARGE_INTEGER startOfSpin, nextQuery, difference, delayTime;
  2072. delayTime.QuadPart = NumberOfMilliseconds*10*1000; // unit is 100ns
  2073. KeQueryTickCount(&startOfSpin);
  2074. do {
  2075. KeQueryTickCount(&nextQuery);
  2076. difference.QuadPart = nextQuery.QuadPart - startOfSpin.QuadPart;
  2077. ASSERT(KeQueryTimeIncrement() <= MAXLONG);
  2078. if (difference.QuadPart * KeQueryTimeIncrement() >=
  2079. delayTime.QuadPart) {
  2080. break;
  2081. }
  2082. } while (1);
  2083. }
  2084. ULONG
  2085. Cyclomy_DoesBoardExist(
  2086. IN PFDO_DEVICE_DATA Extension
  2087. )
  2088. /*++
  2089. Routine Description:
  2090. This routine examines if the board is present.
  2091. Arguments:
  2092. Extension - A pointer to a serial device extension.
  2093. Return Value:
  2094. Will return number of ports.
  2095. --*/
  2096. {
  2097. ULONG numPorts = 0;
  2098. const ULONG CyyCDOffset[] = { // CD1400 offsets within the board
  2099. 0x00000000,0x00000400,0x00000800,0x00000C00,
  2100. 0x00000200,0x00000600,0x00000A00,0x00000E00
  2101. };
  2102. ULONG i;
  2103. UCHAR dataread;
  2104. ULONG isPci = Extension->IsPci;
  2105. // Reset the board
  2106. CYY_RESET_BOARD(Extension->BoardMemory, isPci);
  2107. Cyclomy_Delay(1);
  2108. CYY_CLEAR_INTERRUPT(Extension->BoardMemory, isPci);
  2109. Cyclomy_Delay(1);
  2110. for (i=0; i < CYY_MAX_CHIPS; i++) {
  2111. Extension->Cd1400Base[i] = Extension->BoardMemory +
  2112. (CyyCDOffset[i] << isPci);
  2113. }
  2114. // Clear all GFRCR's
  2115. for (i=0; i < CYY_MAX_CHIPS; i++) {
  2116. CD1400_WRITE(Extension->Cd1400Base[i],isPci,GFRCR,0x00);
  2117. }
  2118. // Test CD1400 presence
  2119. for (i=0; i < CYY_MAX_CHIPS; i++) {
  2120. dataread = CD1400_READ(Extension->Cd1400Base[i],isPci,GFRCR);
  2121. //**************************
  2122. // Error Injection
  2123. //dataread = 0xff;
  2124. //**************************
  2125. if (dataread != 0x00) {
  2126. if (i==0) {
  2127. CyyLogError(
  2128. Extension->DriverObject,
  2129. Extension->Self,
  2130. Extension->PhysicalBoardMemory,
  2131. CyyPhysicalZero,
  2132. 0,
  2133. 0,
  2134. 0,
  2135. dataread,
  2136. STATUS_SUCCESS,
  2137. CYY_GFRCR_FAILURE,
  2138. 0,
  2139. NULL,
  2140. 0,
  2141. NULL
  2142. );
  2143. }
  2144. break;
  2145. }
  2146. dataread = CD1400_READ(Extension->Cd1400Base[i],isPci,CCR);
  2147. //**************************
  2148. // Error Injection
  2149. //dataread = 0xff;
  2150. //**************************
  2151. if (dataread != 0) {
  2152. if (i==0) {
  2153. CyyLogError(
  2154. Extension->DriverObject,
  2155. Extension->Self,
  2156. Extension->PhysicalBoardMemory,
  2157. CyyPhysicalZero,
  2158. 0,
  2159. 0,
  2160. 0,
  2161. dataread,
  2162. STATUS_SUCCESS,
  2163. CYY_CCR_FAILURE,
  2164. 0,
  2165. NULL,
  2166. 0,
  2167. NULL
  2168. );
  2169. }
  2170. break;
  2171. }
  2172. CD1400_WRITE(Extension->Cd1400Base[i],isPci,CCR,CCR_RESET_CD1400);
  2173. Cyclomy_Delay(1); //KeDelayExecutionThread(KernelMode,FALSE,&d1ms);
  2174. dataread = CD1400_READ(Extension->Cd1400Base[i],isPci,GFRCR);
  2175. if (dataread == 0) {
  2176. Cyclomy_Delay(1); //KeDelayExecutionThread(KernelMode,FALSE,&d1ms);
  2177. dataread = CD1400_READ(Extension->Cd1400Base[i],isPci,GFRCR);
  2178. }
  2179. //**************************
  2180. // Error Injection
  2181. //dataread = 0x55;
  2182. //**************************
  2183. if ((dataread & 0xf0) != 0x40) {
  2184. if (i==0) {
  2185. CyyLogError(
  2186. Extension->DriverObject,
  2187. Extension->Self,
  2188. Extension->PhysicalBoardMemory,
  2189. CyyPhysicalZero,
  2190. 0,
  2191. 0,
  2192. 0,
  2193. dataread,
  2194. STATUS_SUCCESS,
  2195. CYY_BAD_CD1400_REVISION,
  2196. 0,
  2197. NULL,
  2198. 0,
  2199. NULL
  2200. );
  2201. }
  2202. break;
  2203. }
  2204. // Configure channel 0 serial
  2205. CD1400_WRITE(Extension->Cd1400Base[i],isPci,GCR,GCR_CH0_IS_SERIAL);
  2206. // Configure internal clock to 1ms
  2207. if (dataread > REV_G) {
  2208. CD1400_WRITE(Extension->Cd1400Base[i],isPci,PPR,CLOCK_60_1MS);
  2209. } else {
  2210. CD1400_WRITE(Extension->Cd1400Base[i],isPci,PPR,CLOCK_25_1MS);
  2211. }
  2212. }
  2213. numPorts = i*4;
  2214. for (;i < CYY_MAX_CHIPS; i++) {
  2215. Extension->Cd1400Base[i] = NULL;
  2216. }
  2217. return numPorts;
  2218. }
  2219. VOID
  2220. Cyclomy_EnableInterruptInPLX(
  2221. IN PFDO_DEVICE_DATA PDevExt
  2222. )
  2223. {
  2224. // Enable PLX interrupts
  2225. if (PDevExt->IsPci){
  2226. UCHAR plx_ver;
  2227. ULONG original;
  2228. plx_ver = CYY_READ_PCI_TYPE(PDevExt->BoardMemory);
  2229. plx_ver &= 0x0f;
  2230. switch(plx_ver) {
  2231. case CYY_PLX9050:
  2232. original = PLX9050_READ_INTERRUPT_CONTROL(PDevExt->Runtime);
  2233. PLX9050_WRITE_INTERRUPT_CONTROL(PDevExt->Runtime,
  2234. original|PLX9050_INT_ENABLE);
  2235. break;
  2236. case CYY_PLX9060:
  2237. case CYY_PLX9080:
  2238. default:
  2239. original = PLX9060_READ_INTERRUPT_CONTROL(PDevExt->Runtime);
  2240. PLX9060_WRITE_INTERRUPT_CONTROL(PDevExt->Runtime,
  2241. original|PLX9060_INT_ENABLE);
  2242. break;
  2243. }
  2244. }
  2245. }
  2246. PCHAR
  2247. PnPMinorFunctionString (
  2248. UCHAR MinorFunction
  2249. )
  2250. {
  2251. switch (MinorFunction)
  2252. {
  2253. case IRP_MN_START_DEVICE:
  2254. return "IRP_MN_START_DEVICE";
  2255. case IRP_MN_QUERY_REMOVE_DEVICE:
  2256. return "IRP_MN_QUERY_REMOVE_DEVICE";
  2257. case IRP_MN_REMOVE_DEVICE:
  2258. return "IRP_MN_REMOVE_DEVICE";
  2259. case IRP_MN_CANCEL_REMOVE_DEVICE:
  2260. return "IRP_MN_CANCEL_REMOVE_DEVICE";
  2261. case IRP_MN_STOP_DEVICE:
  2262. return "IRP_MN_STOP_DEVICE";
  2263. case IRP_MN_QUERY_STOP_DEVICE:
  2264. return "IRP_MN_QUERY_STOP_DEVICE";
  2265. case IRP_MN_CANCEL_STOP_DEVICE:
  2266. return "IRP_MN_CANCEL_STOP_DEVICE";
  2267. case IRP_MN_QUERY_DEVICE_RELATIONS:
  2268. return "IRP_MN_QUERY_DEVICE_RELATIONS";
  2269. case IRP_MN_QUERY_INTERFACE:
  2270. return "IRP_MN_QUERY_INTERFACE";
  2271. case IRP_MN_QUERY_CAPABILITIES:
  2272. return "IRP_MN_QUERY_CAPABILITIES";
  2273. case IRP_MN_QUERY_RESOURCES:
  2274. return "IRP_MN_QUERY_RESOURCES";
  2275. case IRP_MN_QUERY_RESOURCE_REQUIREMENTS:
  2276. return "IRP_MN_QUERY_RESOURCE_REQUIREMENTS";
  2277. case IRP_MN_QUERY_DEVICE_TEXT:
  2278. return "IRP_MN_QUERY_DEVICE_TEXT";
  2279. case IRP_MN_FILTER_RESOURCE_REQUIREMENTS:
  2280. return "IRP_MN_FILTER_RESOURCE_REQUIREMENTS";
  2281. case IRP_MN_READ_CONFIG:
  2282. return "IRP_MN_READ_CONFIG";
  2283. case IRP_MN_WRITE_CONFIG:
  2284. return "IRP_MN_WRITE_CONFIG";
  2285. case IRP_MN_EJECT:
  2286. return "IRP_MN_EJECT";
  2287. case IRP_MN_SET_LOCK:
  2288. return "IRP_MN_SET_LOCK";
  2289. case IRP_MN_QUERY_ID:
  2290. return "IRP_MN_QUERY_ID";
  2291. case IRP_MN_QUERY_PNP_DEVICE_STATE:
  2292. return "IRP_MN_QUERY_PNP_DEVICE_STATE";
  2293. case IRP_MN_QUERY_BUS_INFORMATION:
  2294. return "IRP_MN_QUERY_BUS_INFORMATION";
  2295. case IRP_MN_DEVICE_USAGE_NOTIFICATION:
  2296. return "IRP_MN_DEVICE_USAGE_NOTIFICATION";
  2297. case IRP_MN_SURPRISE_REMOVAL:
  2298. return "IRP_MN_SURPRISE_REMOVAL";
  2299. case IRP_MN_QUERY_LEGACY_BUS_INFORMATION:
  2300. return "IRP_MN_QUERY_LEGACY_BUS_INFORMATION";
  2301. default:
  2302. return "IRP_MN_?????";
  2303. }
  2304. }