Source code of Windows XP (NT5)
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.

4822 lines
122 KiB

  1. /*++
  2. Copyright (c) 1996-2000 Microsoft Corporation
  3. Module Name:
  4. utils.c
  5. Abstract:
  6. This module contains assorted utility functions for PCI.SYS.
  7. Author:
  8. Peter Johnston (peterj) 20-Nov-1996
  9. Revision History:
  10. --*/
  11. #include "pcip.h"
  12. typedef struct _LIST_CONTEXT {
  13. PCM_PARTIAL_RESOURCE_LIST List;
  14. CM_RESOURCE_TYPE DesiredType;
  15. ULONG Remaining;
  16. PCM_PARTIAL_RESOURCE_DESCRIPTOR Next;
  17. CM_PARTIAL_RESOURCE_DESCRIPTOR Alias;
  18. } LIST_CONTEXT, *PLIST_CONTEXT;
  19. extern PPCI_IRQ_ROUTING_TABLE PciIrqRoutingTable;
  20. VOID
  21. PcipInitializePartialListContext(
  22. IN PLIST_CONTEXT ListContext,
  23. IN PCM_PARTIAL_RESOURCE_LIST PartialList,
  24. IN CM_RESOURCE_TYPE DesiredType
  25. );
  26. PCM_PARTIAL_RESOURCE_DESCRIPTOR
  27. PcipGetNextRangeFromList(
  28. PLIST_CONTEXT ListContext
  29. );
  30. NTSTATUS
  31. PciGetDeviceCapabilities(
  32. IN PDEVICE_OBJECT DeviceObject,
  33. IN PDEVICE_CAPABILITIES DeviceCapabilities
  34. );
  35. #ifdef ALLOC_PRAGMA
  36. #pragma alloc_text(PAGE, PcipDestroySecondaryExtension)
  37. #pragma alloc_text(PAGE, PciFindDescriptorInCmResourceList)
  38. #pragma alloc_text(PAGE, PciFindParentPciFdoExtension)
  39. #pragma alloc_text(PAGE, PciGetDeviceCapabilities)
  40. #pragma alloc_text(PAGE, PciGetDeviceProperty)
  41. #pragma alloc_text(PAGE, PcipGetNextRangeFromList)
  42. #pragma alloc_text(PAGE, PciGetRegistryValue)
  43. #pragma alloc_text(PAGE, PcipInitializePartialListContext)
  44. #pragma alloc_text(PAGE, PciInsertEntryAtHead)
  45. #pragma alloc_text(PAGE, PciInsertEntryAtTail)
  46. #pragma alloc_text(PAGE, PcipLinkSecondaryExtension)
  47. #pragma alloc_text(PAGE, PciOpenKey)
  48. #pragma alloc_text(PAGE, PciQueryBusInformation)
  49. #pragma alloc_text(PAGE, PciQueryLegacyBusInformation)
  50. #pragma alloc_text(PAGE, PciQueryCapabilities)
  51. #pragma alloc_text(PAGE, PciRangeListFromResourceList)
  52. #pragma alloc_text(PAGE, PciSaveBiosConfig)
  53. #pragma alloc_text(PAGE, PciGetBiosConfig)
  54. #pragma alloc_text(PAGE, PciStringToUSHORT)
  55. #pragma alloc_text(PAGE, PciSendIoctl)
  56. #pragma alloc_text(INIT, PciBuildDefaultExclusionLists)
  57. #pragma alloc_text(PAGE, PciIsDeviceOnDebugPath)
  58. #endif
  59. //
  60. // Range lists indicating the ranges excluded from decode when the ISA and/or
  61. // VGA bits are set on a bridge. Initialized by PciBuildDefaultExclusionLists
  62. // from DriverEntry.
  63. //
  64. RTL_RANGE_LIST PciIsaBitExclusionList;
  65. RTL_RANGE_LIST PciVgaAndIsaBitExclusionList;
  66. PCM_PARTIAL_RESOURCE_DESCRIPTOR
  67. PciFindDescriptorInCmResourceList(
  68. IN CM_RESOURCE_TYPE DescriptorType,
  69. IN PCM_RESOURCE_LIST ResourceList,
  70. IN PCM_PARTIAL_RESOURCE_DESCRIPTOR PreviousHit
  71. )
  72. {
  73. ULONG numlists;
  74. PCM_FULL_RESOURCE_DESCRIPTOR full;
  75. PCM_PARTIAL_RESOURCE_DESCRIPTOR descriptor;
  76. if (ResourceList == NULL) {
  77. return NULL;
  78. }
  79. numlists = ResourceList->Count;
  80. full = ResourceList->List;
  81. while (numlists--) {
  82. PCM_PARTIAL_RESOURCE_LIST partial = &full->PartialResourceList;
  83. ULONG count = partial->Count;
  84. descriptor = partial->PartialDescriptors;
  85. while (count--) {
  86. if (descriptor->Type == DescriptorType) {
  87. //
  88. // We have a hit on the type. If we we are doing a
  89. // find next, check to see if we're back where we got
  90. // to last time yet.
  91. //
  92. if (PreviousHit != NULL) {
  93. if (PreviousHit == descriptor) {
  94. //
  95. // We found it again, now we can search for real.
  96. //
  97. PreviousHit = NULL;
  98. }
  99. } else {
  100. //
  101. // It's the one.
  102. //
  103. return descriptor;
  104. }
  105. }
  106. descriptor = PciNextPartialDescriptor(descriptor);
  107. }
  108. full = (PCM_FULL_RESOURCE_DESCRIPTOR)descriptor;
  109. }
  110. return NULL;
  111. }
  112. PVOID
  113. PciFindNextSecondaryExtension(
  114. IN PSINGLE_LIST_ENTRY ListEntry,
  115. IN PCI_SIGNATURE DesiredType
  116. )
  117. {
  118. PPCI_SECONDARY_EXTENSION extension;
  119. while (ListEntry != NULL) {
  120. extension = CONTAINING_RECORD(ListEntry,
  121. PCI_SECONDARY_EXTENSION,
  122. List);
  123. if (extension->ExtensionType == DesiredType) {
  124. //
  125. // This extension is the right type, get out.
  126. //
  127. return extension;
  128. }
  129. ListEntry = extension->List.Next;
  130. }
  131. //
  132. // Didn't find it, fail.
  133. //
  134. return NULL;
  135. }
  136. VOID
  137. PcipLinkSecondaryExtension(
  138. IN PSINGLE_LIST_ENTRY ListHead,
  139. IN PFAST_MUTEX Mutex,
  140. IN PVOID NewExtension,
  141. IN PCI_SIGNATURE Type,
  142. IN PSECONDARYEXTENSIONDESTRUCTOR Destructor
  143. )
  144. /*++
  145. Routine Description:
  146. Add a secondary extension to the secondary extension list for
  147. a PDO/FDO and fill in the header fields.
  148. NOTE: Use the macro PciLinkSecondaryExtension which takes a
  149. PDO extension or FDO extension instead of the list header and
  150. mutex fields.
  151. Arguments:
  152. ListHead &SecondaryExtension.Next from the FDO/PDO extension.
  153. Mutex FDO/PDO Mutex.
  154. NewExtension Extension being added to the list.
  155. Type Member of the enum PCI_SIGNATURE.
  156. Destructor Routine to call when this entry is being torn down.
  157. (Optional).
  158. Return Value:
  159. None.
  160. --*/
  161. {
  162. PPCI_SECONDARY_EXTENSION Header;
  163. PAGED_CODE();
  164. Header = (PPCI_SECONDARY_EXTENSION)NewExtension;
  165. Header->ExtensionType = Type;
  166. Header->Destructor = Destructor;
  167. PciInsertEntryAtHead(ListHead, &Header->List, Mutex);
  168. }
  169. VOID
  170. PcipDestroySecondaryExtension(
  171. IN PSINGLE_LIST_ENTRY ListHead,
  172. IN PFAST_MUTEX Mutex,
  173. IN PVOID Extension
  174. )
  175. /*++
  176. Routine Description:
  177. Remove this secondary extension from the list of secondary
  178. extensions, call its destructor routine and free the memory
  179. allocated to it. The destructor is responsible for deleting
  180. any associated allocations.
  181. Failure is not an option.
  182. Note: Use the macro PciDestroySecondaryExtension instead of
  183. calling this routine directly.
  184. Arguments:
  185. ListHead Pointer to the list this extension is on.
  186. Mutex Mutex for synchronization of list manipulation.
  187. Extension The Secondary extension being destroyed.
  188. Return Value:
  189. None.
  190. --*/
  191. {
  192. PPCI_SECONDARY_EXTENSION Header;
  193. PAGED_CODE();
  194. Header = (PPCI_SECONDARY_EXTENSION)Extension;
  195. PciRemoveEntryFromList(ListHead, &Header->List, Mutex);
  196. //
  197. // Call the extension's destructor if one was specified.
  198. //
  199. if (Header->Destructor != NULL) {
  200. Header->Destructor(Extension);
  201. }
  202. //
  203. // Free the memory allocated for this extension.
  204. //
  205. ExFreePool(Extension);
  206. }
  207. VOID
  208. PciInsertEntryAtTail(
  209. IN PSINGLE_LIST_ENTRY ListHead,
  210. IN PSINGLE_LIST_ENTRY NewEntry,
  211. IN PFAST_MUTEX Mutex
  212. )
  213. {
  214. PSINGLE_LIST_ENTRY Previous;
  215. PAGED_CODE();
  216. if (Mutex) {
  217. ExAcquireFastMutex(Mutex);
  218. }
  219. //
  220. // Find the end of the list.
  221. //
  222. Previous = ListHead;
  223. while (Previous->Next) {
  224. Previous = Previous->Next;
  225. }
  226. //
  227. // Append the entry.
  228. //
  229. Previous->Next = NewEntry;
  230. if (Mutex) {
  231. ExReleaseFastMutex(Mutex);
  232. }
  233. }
  234. VOID
  235. PciInsertEntryAtHead(
  236. IN PSINGLE_LIST_ENTRY ListHead,
  237. IN PSINGLE_LIST_ENTRY NewEntry,
  238. IN PFAST_MUTEX Mutex
  239. )
  240. {
  241. PAGED_CODE();
  242. if (Mutex) {
  243. ExAcquireFastMutex(Mutex);
  244. }
  245. NewEntry->Next = ListHead->Next;
  246. ListHead->Next = NewEntry;
  247. if (Mutex) {
  248. ExReleaseFastMutex(Mutex);
  249. }
  250. }
  251. VOID
  252. PciRemoveEntryFromList(
  253. IN PSINGLE_LIST_ENTRY ListHead,
  254. IN PSINGLE_LIST_ENTRY OldEntry,
  255. IN PFAST_MUTEX Mutex
  256. )
  257. /*++
  258. Routine Description:
  259. Remove an entry from a singly linked list.
  260. It is the caller's responsibility to have locked the list if
  261. there is danger of multiple updates.
  262. Arguments:
  263. ListHead - Address of the first entry in the list.
  264. OldEntry - Address of the entry to be removed from the
  265. list.
  266. Return Value:
  267. None.
  268. --*/
  269. {
  270. PSINGLE_LIST_ENTRY Previous;
  271. //
  272. // Sanity check, the list head can't be removed.
  273. //
  274. ASSERT(ListHead != OldEntry);
  275. if (Mutex) {
  276. ExAcquireFastMutex(Mutex);
  277. }
  278. //
  279. // Locate the entry that points to this entry.
  280. //
  281. for (Previous = ListHead; Previous; Previous = Previous->Next) {
  282. if (Previous->Next == OldEntry) {
  283. break;
  284. }
  285. }
  286. //
  287. // The entry is not in the list - this is bad but fail gracefully...
  288. //
  289. if (!Previous) {
  290. ASSERT(Previous);
  291. goto exit;
  292. }
  293. //
  294. // Pull it off the list.
  295. //
  296. Previous->Next = OldEntry->Next;
  297. OldEntry->Next = NULL;
  298. exit:
  299. if (Mutex) {
  300. ExReleaseFastMutex(Mutex);
  301. }
  302. }
  303. PCM_PARTIAL_RESOURCE_DESCRIPTOR
  304. PciNextPartialDescriptor(
  305. PCM_PARTIAL_RESOURCE_DESCRIPTOR Descriptor
  306. )
  307. /*++
  308. Routine Description:
  309. Given a pointer to a CmPartialResourceDescriptor, return a pointer
  310. to the next descriptor in the same list.
  311. This is only done in a routine (rather than a simple descriptor++)
  312. because if the variable length resource CmResourceTypeDeviceSpecific.
  313. Arguments:
  314. Descriptor - Pointer to the descriptor being advanced over.
  315. Return Value:
  316. Pointer to the next descriptor in the same list (or byte beyond
  317. end of list).
  318. --*/
  319. {
  320. PCM_PARTIAL_RESOURCE_DESCRIPTOR nextDescriptor;
  321. nextDescriptor = Descriptor + 1;
  322. if (Descriptor->Type == CmResourceTypeDeviceSpecific) {
  323. //
  324. // This (old) descriptor is followed by DataSize bytes
  325. // of device specific data, ie, not immediatelly by the
  326. // next descriptor. Adjust nextDescriptor by this amount.
  327. //
  328. nextDescriptor = (PCM_PARTIAL_RESOURCE_DESCRIPTOR)
  329. ((ULONG_PTR)nextDescriptor + Descriptor->u.DeviceSpecificData.DataSize);
  330. }
  331. return nextDescriptor;
  332. }
  333. VOID
  334. PcipInitializePartialListContext(
  335. IN PLIST_CONTEXT ListContext,
  336. IN PCM_PARTIAL_RESOURCE_LIST PartialList,
  337. IN CM_RESOURCE_TYPE DesiredType
  338. )
  339. {
  340. ASSERT(DesiredType != CmResourceTypeNull);
  341. ListContext->List = PartialList;
  342. ListContext->DesiredType = DesiredType;
  343. ListContext->Remaining = PartialList->Count;
  344. ListContext->Next = PartialList->PartialDescriptors;
  345. ListContext->Alias.Type = CmResourceTypeNull;
  346. }
  347. PCM_PARTIAL_RESOURCE_DESCRIPTOR
  348. PcipGetNextRangeFromList(
  349. PLIST_CONTEXT ListContext
  350. )
  351. {
  352. ULONG Addend;
  353. PCM_PARTIAL_RESOURCE_DESCRIPTOR current;
  354. //
  355. // See if we should be generating an alias to the current
  356. // descriptor.
  357. //
  358. if (ListContext->Alias.Type == ListContext->DesiredType) {
  359. //
  360. // Yes, advance to alias by adding offset to next 10 bit or
  361. // 12 bit alias (only allowable values).
  362. //
  363. if (ListContext->Alias.Flags & CM_RESOURCE_PORT_10_BIT_DECODE) {
  364. Addend = 1 << 10;
  365. } else {
  366. Addend = 1 << 12;
  367. }
  368. Addend += ListContext->Alias.u.Generic.Start.LowPart;
  369. if (Addend < (1 << 16)) {
  370. //
  371. // This is a valid alias, return it.
  372. //
  373. ListContext->Alias.u.Generic.Start.LowPart = Addend;
  374. return &ListContext->Alias;
  375. }
  376. //
  377. // Out of aliases to this resource.
  378. //
  379. ListContext->Alias.Type = CmResourceTypeNull;
  380. }
  381. //
  382. // We get here if there are no aliases or it is time to advance
  383. // to the next descriptor of the desired type.
  384. //
  385. while (ListContext->Remaining != 0) {
  386. current = ListContext->Next;
  387. //
  388. // Advance context to next before examining and possibly
  389. // returning current.
  390. //
  391. ListContext->Next = PciNextPartialDescriptor(current);
  392. ListContext->Remaining--;
  393. //
  394. // Is this current descriptor a candidate?
  395. //
  396. if (current->Type == ListContext->DesiredType) {
  397. //
  398. // Return this one to caller. If this descriptor has
  399. // aliases, setup so the next call will return an alias.
  400. //
  401. if (current->Flags & (CM_RESOURCE_PORT_10_BIT_DECODE |
  402. CM_RESOURCE_PORT_12_BIT_DECODE)) {
  403. ListContext->Alias = *current;
  404. }
  405. return current;
  406. }
  407. }
  408. //
  409. // No aliases and no new descriptors of the desired type.
  410. //
  411. return NULL;
  412. }
  413. NTSTATUS
  414. PciQueryPowerCapabilities(
  415. IN PPCI_PDO_EXTENSION PdoExtension,
  416. IN PDEVICE_CAPABILITIES Capabilities
  417. )
  418. /*++
  419. Routine Description:
  420. determine a device's power capabilites by using its parent capabilities
  421. It should be noted that there two ways that the code calculates the system
  422. and device wake levels. The first method, which is preferred, biases toward
  423. the deepest possible system state, and the second, which gets used if the
  424. first fails to find something legal, is biased towards finding the deepest
  425. possible device wake state
  426. Arguments:
  427. PdoExtension - The PDO whose capabilities we will provide
  428. Capablities - Where we will store the device capabilities
  429. Return Value:
  430. NTSTATUS
  431. --*/
  432. {
  433. NTSTATUS status;
  434. DEVICE_CAPABILITIES parentCapabilities;
  435. DEVICE_POWER_STATE deviceState;
  436. DEVICE_POWER_STATE validDeviceWakeState = PowerDeviceUnspecified;
  437. SYSTEM_POWER_STATE index;
  438. SYSTEM_POWER_STATE highestSupportedSleepState = PowerSystemUnspecified;
  439. SYSTEM_POWER_STATE validSystemWakeState = PowerSystemUnspecified;
  440. //
  441. // Get the device capabilities of the parent
  442. //
  443. status = PciGetDeviceCapabilities(
  444. PdoExtension->ParentFdoExtension->PhysicalDeviceObject,
  445. &parentCapabilities
  446. );
  447. if (!NT_SUCCESS(status)) {
  448. return status;
  449. }
  450. //
  451. // Make sure that we have sane device capabilities to start with...
  452. //
  453. if (parentCapabilities.DeviceState[PowerSystemWorking] == PowerDeviceUnspecified) {
  454. parentCapabilities.DeviceState[PowerSystemWorking] = PowerDeviceD0;
  455. }
  456. if (parentCapabilities.DeviceState[PowerSystemShutdown] == PowerDeviceUnspecified) {
  457. parentCapabilities.DeviceState[PowerSystemShutdown] = PowerDeviceD3;
  458. }
  459. //
  460. // Does the device have any PCI power capabilities?
  461. //
  462. if ( (PdoExtension->HackFlags & PCI_HACK_NO_PM_CAPS)) {
  463. //
  464. // Use the parent's mapping as our own
  465. //
  466. RtlCopyMemory(
  467. Capabilities->DeviceState,
  468. parentCapabilities.DeviceState,
  469. (PowerSystemShutdown + 1) * sizeof(DEVICE_POWER_STATE)
  470. );
  471. //
  472. // As D1 and D2 are not supported here, round down to D3.
  473. //
  474. // This code is not enabled so that a hack becomes available for
  475. // older PCI video cards. Basically, older video cards can do D3 hot
  476. // but not D3 cold (in which case they need reposting). ACPI supplies
  477. // a hack by which all PCI-to-PCI bridges are said to map S1->D1. The
  478. // code below lets the parent's D1 "appear" as a state the child
  479. // supports, regardless of it's real capabilities. Video drivers for
  480. // such cards fail D3 (which may be D3-cold), but succeed D1 (which is
  481. // really D3-hot).
  482. //
  483. // Also note that this is not targetted at video cards but rather is
  484. // targetted at any non-PCI power managed device. That means drivers
  485. // for older devices need to either map D1&D2 to D3 themselves, or
  486. // treat unexpected D1&D2 IRPs as if D3. Folklore says that there is
  487. // also a net card or two that also takes advantage of this hack.
  488. //
  489. #if 0
  490. for (index = PowerSystemWorking; index < PowerSystemMaximum; index++) {
  491. //
  492. // This is the device state that the parent supports
  493. //
  494. deviceState = parentCapabilities.DeviceState[index];
  495. //
  496. // Round down if D1 or D2
  497. //
  498. if ((deviceState == PowerDeviceD1) || (deviceState == PowerDeviceD2)) {
  499. Capabilities->DeviceState[index] = PowerDeviceD3;
  500. }
  501. }
  502. #endif
  503. //
  504. // The device has no wake capabilities
  505. //
  506. Capabilities->DeviceWake = PowerDeviceUnspecified;
  507. Capabilities->SystemWake = PowerSystemUnspecified;
  508. //
  509. // Set these bits explicitly
  510. //
  511. Capabilities->DeviceD1 = FALSE;
  512. Capabilities->DeviceD2 = FALSE;
  513. Capabilities->WakeFromD0 = FALSE;
  514. Capabilities->WakeFromD1 = FALSE;
  515. Capabilities->WakeFromD2 = FALSE;
  516. Capabilities->WakeFromD3 = FALSE;
  517. //
  518. // Done
  519. //
  520. return STATUS_SUCCESS;
  521. }
  522. //
  523. // Set all the capabilities bits
  524. //
  525. Capabilities->DeviceD1 = PdoExtension->PowerCapabilities.Support.D1;
  526. Capabilities->DeviceD2 = PdoExtension->PowerCapabilities.Support.D2;
  527. Capabilities->WakeFromD0 = PdoExtension->PowerCapabilities.Support.PMED0;
  528. Capabilities->WakeFromD1 = PdoExtension->PowerCapabilities.Support.PMED1;
  529. Capabilities->WakeFromD2 = PdoExtension->PowerCapabilities.Support.PMED2;
  530. if (parentCapabilities.DeviceWake == PowerDeviceD3) {
  531. //
  532. // If our parent can wake from the D3 state, than we must support
  533. // PM3 From D3 Cold. The (obvious) exception to this is if the
  534. // parent is a root bus...
  535. //
  536. if (PCI_PDO_ON_ROOT(PdoExtension)) {
  537. Capabilities->WakeFromD3 =
  538. PdoExtension->PowerCapabilities.Support.PMED3Hot;
  539. } else {
  540. Capabilities->WakeFromD3 =
  541. PdoExtension->PowerCapabilities.Support.PMED3Cold;
  542. }
  543. } else {
  544. //
  545. // If our parent cannot wake from the D3 state, then we support
  546. // the D3 state if we support PME3Hot
  547. //
  548. Capabilities->WakeFromD3 =
  549. PdoExtension->PowerCapabilities.Support.PMED3Hot;
  550. }
  551. //
  552. // First step is to make sure that all the S-states that we got from
  553. // out parent map to valid D-states for this device
  554. //
  555. // ADRIAO N.B. 08/18/1999 -
  556. // This algorithm works but it's overly aggressive. It is in fact legal
  557. // for a bridge to be in D2 with a card behind it in D1.
  558. //
  559. for (index = PowerSystemWorking; index < PowerSystemMaximum; index++) {
  560. //
  561. // This is the device state that the parent supports
  562. //
  563. deviceState = parentCapabilities.DeviceState[index];
  564. //
  565. // If the device state is D1 and we don't support D1, then
  566. // consider D2 instead
  567. //
  568. if (deviceState == PowerDeviceD1 &&
  569. PdoExtension->PowerCapabilities.Support.D1 == FALSE) {
  570. deviceState++;
  571. }
  572. //
  573. // If the device state is D2 and we don't support D2, then
  574. // consider D3 instead
  575. //
  576. if (deviceState == PowerDeviceD2 &&
  577. PdoExtension->PowerCapabilities.Support.D2 == FALSE) {
  578. deviceState++;
  579. }
  580. //
  581. // We should be able to support this deviceState
  582. //
  583. Capabilities->DeviceState[index] = deviceState;
  584. //
  585. // If this S-state is less than PowerSystemHibernate, and the
  586. // S-State doesn't map to PowerDeviceUnspecified, then consider
  587. // this to be the highest supported SleepState
  588. //
  589. if (index < PowerSystemHibernate &&
  590. Capabilities->DeviceState[index] != PowerDeviceUnspecified) {
  591. highestSupportedSleepState = index;
  592. }
  593. //
  594. // Can we support this as a wake state?
  595. //
  596. if (index < parentCapabilities.SystemWake &&
  597. deviceState >= parentCapabilities.DeviceState[index] &&
  598. parentCapabilities.DeviceState[index] != PowerDeviceUnspecified) {
  599. //
  600. // Consider using this as a valid wake state
  601. //
  602. if ( (deviceState == PowerDeviceD0 && Capabilities->WakeFromD0) ||
  603. (deviceState == PowerDeviceD1 && Capabilities->WakeFromD1) ||
  604. (deviceState == PowerDeviceD2 && Capabilities->WakeFromD2) ) {
  605. validSystemWakeState = index;
  606. validDeviceWakeState = deviceState;
  607. } else if (deviceState == PowerDeviceD3 &&
  608. PdoExtension->PowerCapabilities.Support.PMED3Hot) {
  609. //
  610. // This is a special case logic (which is why it is seperate from
  611. // the above logic
  612. //
  613. if (parentCapabilities.DeviceState[index] < PowerDeviceD3 ||
  614. PdoExtension->PowerCapabilities.Support.PMED3Cold) {
  615. validSystemWakeState = index;
  616. validDeviceWakeState = deviceState;
  617. }
  618. }
  619. }
  620. }
  621. //
  622. // Does the parent device have power management capabilities?
  623. // Does the device have power management capabilities?
  624. // Can we wake up from the same D-states that our parent can? or better?
  625. //
  626. if (parentCapabilities.SystemWake == PowerSystemUnspecified ||
  627. parentCapabilities.DeviceWake == PowerDeviceUnspecified ||
  628. PdoExtension->PowerState.DeviceWakeLevel == PowerDeviceUnspecified ||
  629. PdoExtension->PowerState.DeviceWakeLevel < parentCapabilities.DeviceWake) {
  630. //
  631. // The device doesn't support any kind of wakeup (that we know about)
  632. // or the device doesn't support wakeup from supported D-states, so
  633. // set the latency and return
  634. //
  635. Capabilities->D1Latency = 0;
  636. Capabilities->D2Latency = 0;
  637. Capabilities->D3Latency = 0;
  638. return STATUS_SUCCESS;
  639. }
  640. //
  641. // We should be able to wake the device from the same state
  642. // that our parent can wake from
  643. //
  644. Capabilities->SystemWake = parentCapabilities.SystemWake;
  645. Capabilities->DeviceWake = PdoExtension->PowerState.DeviceWakeLevel;
  646. //
  647. // Change our device wake level to include a state that we support
  648. //
  649. if (Capabilities->DeviceWake == PowerDeviceD0 && !Capabilities->WakeFromD0) {
  650. Capabilities->DeviceWake++;
  651. }
  652. if (Capabilities->DeviceWake == PowerDeviceD1 && !Capabilities->WakeFromD1) {
  653. Capabilities->DeviceWake++;
  654. }
  655. if (Capabilities->DeviceWake == PowerDeviceD2 && !Capabilities->WakeFromD2) {
  656. Capabilities->DeviceWake++;
  657. }
  658. if (Capabilities->DeviceWake == PowerDeviceD3 && !Capabilities->WakeFromD3) {
  659. Capabilities->DeviceWake = PowerDeviceUnspecified;
  660. Capabilities->SystemWake = PowerSystemUnspecified;
  661. }
  662. //
  663. // This is our fallback position. If we got here and there is no wake
  664. // capability using the above method of calcuation, then we should
  665. // check to see if we noticed a valid wake combination while scanning
  666. // the S to D mapping information
  667. //
  668. if ( (Capabilities->DeviceWake == PowerDeviceUnspecified ||
  669. Capabilities->SystemWake == PowerSystemUnspecified) &&
  670. (validSystemWakeState != PowerSystemUnspecified &&
  671. validDeviceWakeState != PowerSystemUnspecified) ) {
  672. Capabilities->DeviceWake = validDeviceWakeState;
  673. Capabilities->SystemWake = validSystemWakeState;
  674. //
  675. // Note that in this case, we might have set DeviceWake to D3, without
  676. // having set the bit, so "correct" that situation.
  677. //
  678. if (validDeviceWakeState == PowerDeviceD3) {
  679. Capabilities->WakeFromD3 = TRUE;
  680. }
  681. }
  682. //
  683. // We shouldn't allow Wake From S4, S5, unless the supports the D3 state
  684. // Even then, we really shouldn't allow S4, S5 unless the device supports
  685. // the D3Cold PME state
  686. //
  687. if (Capabilities->SystemWake > PowerSystemSleeping3) {
  688. //
  689. // Does the device support wake from D3?
  690. //
  691. if (Capabilities->DeviceWake != PowerDeviceD3) {
  692. //
  693. // Reduce the systemwake level to something more realistic
  694. //
  695. Capabilities->SystemWake = highestSupportedSleepState;
  696. }
  697. //
  698. // This is in a seperate if statement so that the code can be easily
  699. // commented out
  700. //
  701. if (!PdoExtension->PowerCapabilities.Support.PMED3Cold) {
  702. //
  703. // Reduce the systemwake level to something more realistic
  704. //
  705. Capabilities->SystemWake = highestSupportedSleepState;
  706. }
  707. }
  708. //
  709. // From the PCI Power Management spec V1.0, table 18
  710. // "PCI Function State Transition Delays".
  711. //
  712. // D1 -> D0 0
  713. // D2 -> D0 200 us
  714. // D3 -> D0 10 ms
  715. //
  716. // The latency entries are in units of 100 us.
  717. //
  718. Capabilities->D1Latency = 0;
  719. Capabilities->D2Latency = 2;
  720. Capabilities->D3Latency = 100;
  721. //
  722. // Make sure that S0 maps to D0
  723. //
  724. ASSERT( Capabilities->DeviceState[PowerSystemWorking] == PowerDeviceD0);
  725. //
  726. // Done
  727. //
  728. return STATUS_SUCCESS;
  729. }
  730. NTSTATUS
  731. PciDetermineSlotNumber(
  732. IN PPCI_PDO_EXTENSION PdoExtension,
  733. IN OUT PULONG SlotNumber
  734. )
  735. /*++
  736. Description:
  737. Determine the slot number associated with a PCI device (if any)
  738. through use of the PCI IRQ routing table information we may have
  739. stored earlier.
  740. If the previous mechanism fails to retrieve a slot number, see if
  741. we can inherit our parent's slot number.
  742. This result may be filtered further by ACPI and other bus filters..
  743. Arguments:
  744. PdoExtension - PDO extension of device in question.
  745. SlotNumber - Pointer to slot number to update
  746. Return Value:
  747. STATUS_SUCCESS if slot # found
  748. --*/
  749. {
  750. PSLOT_INFO slotInfo, lastSlot;
  751. NTSTATUS status;
  752. ULONG length;
  753. if (PciIrqRoutingTable == NULL) {
  754. return STATUS_UNSUCCESSFUL;
  755. }
  756. //
  757. // Don't return UI numbers if we don't have a parent to check for BaseBus
  758. //
  759. if (PCI_PARENT_FDOX(PdoExtension) == NULL) {
  760. return STATUS_UNSUCCESSFUL;
  761. }
  762. slotInfo = (PSLOT_INFO)((PUCHAR) PciIrqRoutingTable +
  763. sizeof(PCI_IRQ_ROUTING_TABLE));
  764. lastSlot = (PSLOT_INFO)((PUCHAR) PciIrqRoutingTable +
  765. PciIrqRoutingTable->TableSize);
  766. // Search for a entry in the routing table that matches this device
  767. while (slotInfo < lastSlot) {
  768. if ((PCI_PARENT_FDOX(PdoExtension)->BaseBus == slotInfo->BusNumber) &&
  769. ((UCHAR)PdoExtension->Slot.u.bits.DeviceNumber == (slotInfo->DeviceNumber >> 3)) &&
  770. (slotInfo->SlotNumber != 0)) {
  771. *SlotNumber = slotInfo->SlotNumber;
  772. return STATUS_SUCCESS;
  773. }
  774. slotInfo++;
  775. }
  776. //
  777. // Maybe our parent has a UI Number that we could 'inherit'.
  778. // but only if we're not a PDO off a root bus otherwise we pick up
  779. // the UI number from the PNPA03 node (likely 0)
  780. //
  781. if (PCI_PDO_ON_ROOT(PdoExtension)) {
  782. return STATUS_UNSUCCESSFUL;
  783. }
  784. status = IoGetDeviceProperty(PCI_PARENT_PDO(PdoExtension),
  785. DevicePropertyUINumber,
  786. sizeof(*SlotNumber),
  787. SlotNumber,
  788. &length);
  789. return status;
  790. }
  791. NTSTATUS
  792. PciQueryCapabilities(
  793. IN PPCI_PDO_EXTENSION PdoExtension,
  794. IN PDEVICE_CAPABILITIES Capabilities
  795. )
  796. /*++
  797. Routine Description:
  798. return a subset of our parent's capabilities.
  799. Arguments:
  800. Capabilities - pointer to a DEVICE_CAPABILITIES structured supplied
  801. by the caller.
  802. Return Value:
  803. Status.
  804. --*/
  805. {
  806. NTSTATUS status = STATUS_SUCCESS;
  807. ULONG slot;
  808. #ifndef HANDLE_BOGUS_CAPS
  809. if (Capabilities->Version < 1) {
  810. //
  811. // do not touch irp!
  812. //
  813. return STATUS_NOT_SUPPORTED;
  814. }
  815. #endif
  816. //
  817. // For PCI devices, the Capabilities Address field contains
  818. // the Device Number in the upper 16 bits and the function
  819. // number in the lower.
  820. //
  821. Capabilities->Address =
  822. PdoExtension->Slot.u.bits.DeviceNumber << 16 |
  823. PdoExtension->Slot.u.bits.FunctionNumber;
  824. //
  825. // The PCI bus driver does not generate Unique IDs for its children.
  826. //
  827. Capabilities->UniqueID = FALSE;
  828. //
  829. // If this PDO is for a HOST BRIDGE, claim that it supports
  830. // being handled Raw. This is so the device controller will
  831. // allow installation of the NULL device on this puppy.
  832. //
  833. if ((PdoExtension->BaseClass == PCI_CLASS_BRIDGE_DEV) &&
  834. (PdoExtension->SubClass == PCI_SUBCLASS_BR_HOST)) {
  835. Capabilities->RawDeviceOK = TRUE;
  836. } else {
  837. Capabilities->RawDeviceOK = FALSE;
  838. }
  839. //
  840. // The following values should be fixed by filters or function
  841. // drivers that actually know the answer.
  842. //
  843. Capabilities->LockSupported = FALSE;
  844. Capabilities->EjectSupported = FALSE;
  845. Capabilities->Removable = FALSE;
  846. Capabilities->DockDevice = FALSE;
  847. PciDetermineSlotNumber(PdoExtension, &Capabilities->UINumber);
  848. //
  849. // Get the device power capabilities
  850. //
  851. status = PciQueryPowerCapabilities( PdoExtension, Capabilities );
  852. if (!NT_SUCCESS(status)) {
  853. return status;
  854. }
  855. #if DBG
  856. if (PciDebug & PciDbgQueryCap) {
  857. PciDebugDumpQueryCapabilities(Capabilities);
  858. }
  859. #endif
  860. //
  861. // Done
  862. //
  863. return status;
  864. }
  865. NTSTATUS
  866. PciQueryBusInformation(
  867. IN PPCI_PDO_EXTENSION PdoExtension,
  868. IN PPNP_BUS_INFORMATION *BusInformation
  869. )
  870. /*++
  871. Routine Description:
  872. Tell PnP that it's talking to a PCI bus.
  873. Arguments:
  874. BusInformation - Pointer to a PPNP_BUS_INFORMATION. We create
  875. a PNP_BUS_INFORMATION and pass its address
  876. back thru here.
  877. Return Value:
  878. Status.
  879. --*/
  880. {
  881. PPNP_BUS_INFORMATION information;
  882. information = ExAllocatePool(PagedPool | POOL_COLD_ALLOCATION, sizeof(PNP_BUS_INFORMATION));
  883. if (information == NULL) {
  884. return STATUS_INSUFFICIENT_RESOURCES;
  885. }
  886. RtlCopyMemory(&information->BusTypeGuid, &GUID_BUS_TYPE_PCI, sizeof(GUID));
  887. information->LegacyBusType = PCIBus;
  888. information->BusNumber = PCI_PARENT_FDOX(PdoExtension)->BaseBus;
  889. *BusInformation = information;
  890. return STATUS_SUCCESS;
  891. }
  892. NTSTATUS
  893. PciQueryLegacyBusInformation(
  894. IN PPCI_FDO_EXTENSION FdoExtension,
  895. IN PLEGACY_BUS_INFORMATION *BusInformation
  896. )
  897. /*++
  898. Routine Description:
  899. Tell PnP that it's talking to a PCI bus.
  900. Arguments:
  901. BusInformation - Pointer to a PLEGACY_BUS_INFORMATION. We create
  902. a LEGACY_BUS_INFORMATION and pass its address
  903. back thru here.
  904. Return Value:
  905. Status.
  906. --*/
  907. {
  908. PLEGACY_BUS_INFORMATION information;
  909. information = ExAllocatePool(PagedPool | POOL_COLD_ALLOCATION, sizeof(LEGACY_BUS_INFORMATION));
  910. if (information == NULL) {
  911. ASSERT(information != NULL);
  912. return STATUS_INSUFFICIENT_RESOURCES;
  913. }
  914. RtlCopyMemory(&information->BusTypeGuid, &GUID_BUS_TYPE_PCI, sizeof(GUID));
  915. information->LegacyBusType = PCIBus;
  916. information->BusNumber = FdoExtension->BaseBus;
  917. *BusInformation = information;
  918. return STATUS_SUCCESS;
  919. }
  920. NTSTATUS
  921. PciGetInterruptAssignment(
  922. IN PPCI_PDO_EXTENSION PdoExtension,
  923. OUT ULONG *Minimum,
  924. OUT ULONG *Maximum
  925. )
  926. {
  927. UCHAR pin = PdoExtension->InterruptPin;
  928. UCHAR line;
  929. //
  930. // Using HAL for interrupts.
  931. //
  932. PIO_RESOURCE_REQUIREMENTS_LIST reqList;
  933. PIO_RESOURCE_DESCRIPTOR resource;
  934. PCI_SLOT_NUMBER slot;
  935. NTSTATUS status = STATUS_RESOURCE_TYPE_NOT_FOUND;
  936. if (pin != 0) {
  937. //
  938. // This hardware uses an interrupt.
  939. //
  940. // Depend on the HAL to understand how IRQ routing is
  941. // really done.
  942. //
  943. reqList = PciAllocateIoRequirementsList(
  944. 1, // number of resources
  945. PCI_PARENT_FDOX(PdoExtension)->BaseBus,
  946. PdoExtension->Slot.u.AsULONG
  947. );
  948. if (reqList == NULL) {
  949. //
  950. // Out of system resources? Bad things are happening.
  951. //
  952. return STATUS_INSUFFICIENT_RESOURCES;
  953. }
  954. resource = reqList->List[0].Descriptors;
  955. resource->Type = CmResourceTypeInterrupt;
  956. resource->ShareDisposition = CmResourceShareShared;
  957. resource->Option = 0;
  958. resource->Flags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
  959. resource->u.Interrupt.MinimumVector = 0x00;
  960. resource->u.Interrupt.MaximumVector = 0xff;
  961. #if defined(NO_LEGACY_DRIVERS)
  962. *Minimum = 0;
  963. *Maximum = 0xFF;
  964. status = STATUS_SUCCESS;
  965. #else
  966. status = HalAdjustResourceList(&reqList);
  967. //
  968. // If the HAL succeeded it will have reallocated the list.
  969. //
  970. resource = reqList->List[0].Descriptors;
  971. if (!NT_SUCCESS(status)) {
  972. PciDebugPrint(
  973. PciDbgInformative,
  974. " PIN %02x, HAL FAILED Interrupt Assignment, status %08x\n",
  975. pin,
  976. status
  977. );
  978. status = STATUS_UNSUCCESSFUL;
  979. } else if (resource->u.Interrupt.MinimumVector >
  980. resource->u.Interrupt.MaximumVector) {
  981. //
  982. // The HAL succeeded but returned an invalid range. This
  983. // is the HALs way of telling us that, sorry, it doesn't
  984. // know either.
  985. //
  986. //
  987. // We have a bug in that we restore the interrupt line to
  988. // config space before we power up the device and thus if
  989. // the device is in D>0 and the interrupt line register
  990. // isn't sticky it doesn't stick. It doesn't matter unless
  991. // we are on a machine that doesn't support interrupt
  992. // routing in which case we are toast. The correct fix is
  993. // to move the restore code after we power managed the device
  994. // but that changes things too much for Whistler Beta2 and this
  995. // is totally rewritten for Blackcomb so, now that you know
  996. // the right way to fix this, the hack is if the HAL fails
  997. // the call use what we would have restored into the interrupt
  998. // line.
  999. //
  1000. //
  1001. // Get the current int line (this is in the same place for all header types)
  1002. //
  1003. PciReadDeviceConfig(PdoExtension,
  1004. &line,
  1005. FIELD_OFFSET(PCI_COMMON_CONFIG, u.type0.InterruptLine),
  1006. sizeof(line)
  1007. );
  1008. //
  1009. // If this is 0 and it was something when we first saw the device then use
  1010. // what we first saw
  1011. //
  1012. if (line == 0 && PdoExtension->RawInterruptLine != 0) {
  1013. *Minimum = *Maximum = (ULONG)PdoExtension->RawInterruptLine;
  1014. status = STATUS_SUCCESS;
  1015. } else {
  1016. PciDebugPrint(
  1017. PciDbgInformative,
  1018. " PIN %02x, HAL could not assign interrupt.\n",
  1019. pin
  1020. );
  1021. status = STATUS_UNSUCCESSFUL;
  1022. }
  1023. } else {
  1024. *Minimum = resource->u.Interrupt.MinimumVector;
  1025. *Maximum = resource->u.Interrupt.MaximumVector;
  1026. PciDebugPrint(
  1027. PciDbgObnoxious,
  1028. " Interrupt assigned = 0x%x through 0x%x\n",
  1029. *Minimum,
  1030. *Maximum
  1031. );
  1032. status = STATUS_SUCCESS;
  1033. }
  1034. ExFreePool(reqList);
  1035. #endif // NO_LEGACY_DRIVERS
  1036. } else {
  1037. #if MSI_SUPPORTED
  1038. if (PdoExtension->CapableMSI) {
  1039. //
  1040. // MSI Only device - we need to return a success here so that
  1041. // this device gets resource requests passed to a (hopefully)
  1042. // MSI-aware arbiter. If the arbiter is not MSI aware, we will
  1043. // simply get extraneous/unusable resources allocated for this
  1044. // device - not to mention the fact that the device will not work.
  1045. //
  1046. // The below could be anything, they are only limited by message
  1047. // size and the available APIC ranges which only the arbiter
  1048. // knows about.
  1049. //
  1050. *Minimum = 0x00;
  1051. *Maximum = 0xFF;
  1052. status = STATUS_SUCCESS;
  1053. }
  1054. #endif // MSI_SUPPORTED
  1055. }
  1056. return status;
  1057. }
  1058. PPCI_PDO_EXTENSION
  1059. PciFindPdoByFunction(
  1060. IN PPCI_FDO_EXTENSION FdoExtension,
  1061. IN PCI_SLOT_NUMBER Slot,
  1062. IN PPCI_COMMON_CONFIG Config
  1063. )
  1064. {
  1065. PPCI_PDO_EXTENSION pdoExtension;
  1066. KIRQL currentIrql;
  1067. //
  1068. // This can be called at >= DISPATCH_LEVEL when we scan the bus on returning
  1069. // from hibernate. Don't try to acquire the locks because (1) it'll crash
  1070. // and (2) it is guaranteed to be single threaded
  1071. //
  1072. currentIrql = KeGetCurrentIrql();
  1073. if (currentIrql < DISPATCH_LEVEL) {
  1074. ExAcquireFastMutex(&FdoExtension->ChildListMutex);
  1075. };
  1076. //
  1077. // Seach each PDO hanging off of the given FDO until we find a matching
  1078. // PCI function or fall off the end of the list.
  1079. //
  1080. for (pdoExtension = FdoExtension->ChildPdoList;
  1081. pdoExtension;
  1082. pdoExtension = pdoExtension->Next) {
  1083. if ((!pdoExtension->ReportedMissing) &&
  1084. (pdoExtension->Slot.u.bits.DeviceNumber == Slot.u.bits.DeviceNumber) &&
  1085. (pdoExtension->Slot.u.bits.FunctionNumber == Slot.u.bits.FunctionNumber)) {
  1086. //
  1087. // Check that the device in this slot hasn't changed. (as best
  1088. // we can).
  1089. //
  1090. if ( (pdoExtension->VendorId == Config->VendorID)
  1091. && (pdoExtension->DeviceId == Config->DeviceID)
  1092. && (pdoExtension->RevisionId == Config->RevisionID)
  1093. #if 0
  1094. //
  1095. // NTRAID #62668 - 4/25/2000
  1096. //
  1097. // These do not contribute towards the device ID itself, and
  1098. // as they are unfortunately volatile on some cards (SubClass
  1099. // changes on the ATIRage, Programming interface on IDE cards).
  1100. // Therefore a change in these fields does not mean a change in
  1101. // the presence of the card.
  1102. //
  1103. // What about the SSVID?
  1104. //
  1105. && (pdoExtension->ProgIf == Config->ProgIf)
  1106. && (pdoExtension->SubClass == Config->SubClass)
  1107. && (pdoExtension->BaseClass == Config->BaseClass)
  1108. #endif
  1109. ) {
  1110. break;
  1111. }
  1112. }
  1113. }
  1114. if (currentIrql < DISPATCH_LEVEL) {
  1115. ExReleaseFastMutex(&FdoExtension->ChildListMutex);
  1116. }
  1117. return pdoExtension;
  1118. }
  1119. PPCI_FDO_EXTENSION
  1120. PciFindParentPciFdoExtension(
  1121. PDEVICE_OBJECT PhysicalDeviceObject,
  1122. IN PFAST_MUTEX Mutex
  1123. )
  1124. /*++
  1125. Routine Description:
  1126. For each Parent PCI FDO, search the child Pdo lists for the supplied
  1127. PhysicalDeviceObject.
  1128. Arguments:
  1129. PhysicalDeviceObject Pdo to find.
  1130. Mutex Mutex list is protected by.
  1131. Return Value:
  1132. If Pdo is found as a child, returns a pointer to the root Fdo's
  1133. device extension, otherwise returns NULL.
  1134. --*/
  1135. {
  1136. PPCI_FDO_EXTENSION fdoExtension;
  1137. PPCI_PDO_EXTENSION pdoExtension;
  1138. PPCI_PDO_EXTENSION target;
  1139. PSINGLE_LIST_ENTRY nextEntry;
  1140. if (Mutex) {
  1141. ExAcquireFastMutex(Mutex);
  1142. }
  1143. target = (PPCI_PDO_EXTENSION)PhysicalDeviceObject->DeviceExtension;
  1144. //
  1145. // For each root
  1146. //
  1147. for ( nextEntry = PciFdoExtensionListHead.Next;
  1148. nextEntry != NULL;
  1149. nextEntry = nextEntry->Next ) {
  1150. fdoExtension = CONTAINING_RECORD(nextEntry,
  1151. PCI_FDO_EXTENSION,
  1152. List);
  1153. //
  1154. // Search the child Pdo list.
  1155. //
  1156. ExAcquireFastMutex(&fdoExtension->ChildListMutex);
  1157. for ( pdoExtension = fdoExtension->ChildPdoList;
  1158. pdoExtension;
  1159. pdoExtension = pdoExtension->Next ) {
  1160. //
  1161. // Is this the one we're looking for?
  1162. //
  1163. if ( pdoExtension == target ) {
  1164. ExReleaseFastMutex(&fdoExtension->ChildListMutex);
  1165. //
  1166. // Yes, return it.
  1167. //
  1168. if (Mutex) {
  1169. ExReleaseFastMutex(Mutex);
  1170. }
  1171. return fdoExtension;
  1172. }
  1173. }
  1174. ExReleaseFastMutex(&fdoExtension->ChildListMutex);
  1175. }
  1176. //
  1177. // Did not find match.
  1178. //
  1179. if (Mutex) {
  1180. ExReleaseFastMutex(Mutex);
  1181. }
  1182. return NULL;
  1183. }
  1184. PCI_OBJECT_TYPE
  1185. PciClassifyDeviceType(
  1186. PPCI_PDO_EXTENSION PdoExtension
  1187. )
  1188. /*++
  1189. Routine Description:
  1190. Examine the Configuration Header BaseClass and SubClass fields
  1191. and classify the device into a simple enumerated type.
  1192. Arguments:
  1193. PdoExtension Pointer to the Physical Device Object extension
  1194. into which the above fields have been previously
  1195. been copied from PCI config space.
  1196. Return Value:
  1197. Returns a device type from the PCI_OBJECT_TYPE enumeration.
  1198. --*/
  1199. {
  1200. ASSERT_PCI_PDO_EXTENSION(PdoExtension);
  1201. switch (PdoExtension->BaseClass) {
  1202. case PCI_CLASS_BRIDGE_DEV:
  1203. //
  1204. // It's a bridge, subdivide it into the kind of bridge.
  1205. //
  1206. switch (PdoExtension->SubClass) {
  1207. case PCI_SUBCLASS_BR_HOST:
  1208. return PciTypeHostBridge;
  1209. case PCI_SUBCLASS_BR_PCI_TO_PCI:
  1210. return PciTypePciBridge;
  1211. case PCI_SUBCLASS_BR_CARDBUS:
  1212. return PciTypeCardbusBridge;
  1213. default:
  1214. //
  1215. // Anything else is just a device.
  1216. //
  1217. break;
  1218. }
  1219. default:
  1220. //
  1221. // Anything else is just another device.
  1222. //
  1223. break;
  1224. }
  1225. return PciTypeDevice;
  1226. }
  1227. ULONG
  1228. PciGetLengthFromBar(
  1229. ULONG BaseAddressRegister
  1230. )
  1231. /*++
  1232. Routine Description:
  1233. Given the contents of a PCI Base Address Register, after it
  1234. has been written with all ones, this routine calculates the
  1235. length (and alignment) requirement for this BAR.
  1236. This method for determining requirements is described in
  1237. section 6.2.5.1 of the PCI Specification (Rev 2.1).
  1238. NTRAID #62631 - 4/25/2000 - andrewth
  1239. The length is a power of two, given only a ULONG to
  1240. contain it, we are restricted to a maximum resource size of
  1241. 2GB.
  1242. Arguments:
  1243. BaseAddressRegister contains something.
  1244. Return Value:
  1245. Returns the length of the resource requirement. This will be a number
  1246. in the range 0 thru 0x80000000.
  1247. --*/
  1248. {
  1249. ULONG Length;
  1250. //
  1251. // A number of least significant bits should be ignored in the
  1252. // determination of the length. These are flag bits, the number
  1253. // of bits is dependent on the type of the resource.
  1254. //
  1255. if (BaseAddressRegister & PCI_ADDRESS_IO_SPACE) {
  1256. //
  1257. // PCI IO space.
  1258. //
  1259. BaseAddressRegister &= PCI_ADDRESS_IO_ADDRESS_MASK;
  1260. } else {
  1261. //
  1262. // PCI Memory space.
  1263. //
  1264. BaseAddressRegister &= PCI_ADDRESS_MEMORY_ADDRESS_MASK;
  1265. }
  1266. //
  1267. // BaseAddressRegister now contains the maximum base address
  1268. // this device can reside at and still exist below the top of
  1269. // memory.
  1270. //
  1271. // The value 0xffffffff was written to the BAR. The device will
  1272. // have adjusted this value to the maximum it can really use.
  1273. //
  1274. // Length MUST be a power of 2.
  1275. //
  1276. // For most devices, h/w will simply have cleared bits from the
  1277. // least significant bit positions so that the address 0xffffffff
  1278. // is adjusted to accomodate the length. eg: if the new value is
  1279. // 0xffffff00, the device requires 256 bytes.
  1280. //
  1281. // The difference between the original and new values is the length (-1).
  1282. //
  1283. // For example, if the value fead back from the BAR is 0xffff0000,
  1284. // the length of this resource is
  1285. //
  1286. // 0xffffffff - 0xffff0000 + 1
  1287. // = 0x0000ffff + 1
  1288. // = 0x00010000
  1289. //
  1290. // ie 16KB.
  1291. //
  1292. // Some devices cannot reside at the top of PCI address space. These
  1293. // devices will have adjusted the value such that length bytes are
  1294. // accomodated below the highest address. For example, if a device
  1295. // must reside below 1MB, and occupies 256 bytes, the value will now
  1296. // be 0x000fff00.
  1297. //
  1298. // In the first case, length can be calculated as-
  1299. //
  1300. Length = (0xffffffff - BaseAddressRegister) + 1;
  1301. if (((Length - 1) & Length) != 0) {
  1302. //
  1303. // We didn't end up with a power of two, must be the latter
  1304. // case, we will have to scan for it.
  1305. //
  1306. Length = 4; // start with minimum possible
  1307. while ((Length | BaseAddressRegister) != BaseAddressRegister) {
  1308. //
  1309. // Length *= 2, note we will eventually drop out of this
  1310. // loop for one of two reasons (a) because we found the
  1311. // length, or (b) because Length left shifted off the end
  1312. // and became 0.
  1313. //
  1314. Length <<= 1;
  1315. }
  1316. }
  1317. //
  1318. // Check that we got something - if this is a 64bit bar then nothing is ok as
  1319. // we might be asking for a range >= 4GB (not that that's going to work any time soon)
  1320. //
  1321. if (!((BaseAddressRegister & PCI_ADDRESS_MEMORY_TYPE_MASK) == PCI_TYPE_64BIT)) {
  1322. ASSERT(Length);
  1323. }
  1324. return Length;
  1325. }
  1326. BOOLEAN
  1327. PciCreateIoDescriptorFromBarLimit(
  1328. IN PIO_RESOURCE_DESCRIPTOR Descriptor,
  1329. IN PULONG BaseAddress,
  1330. IN BOOLEAN Rom
  1331. )
  1332. /*++
  1333. Description:
  1334. Generate an IO resource descriptor to describe the settings
  1335. a Base Address Register can take.
  1336. Arguments:
  1337. Descriptor -
  1338. BaseAddress - Pointer to the value read from a base address register
  1339. immediately after writing all ones to it.
  1340. Rom - If true, this is a base address register for ROM.
  1341. Return Value:
  1342. Returns TRUE if this address register was a 64 bit address register,
  1343. FALSE otherwise.
  1344. --*/
  1345. {
  1346. ULONG bar = *BaseAddress;
  1347. ULONG length;
  1348. ULONG addressMask;
  1349. BOOLEAN returnValue = FALSE;
  1350. //
  1351. // If the Base Address Register contains zero after being written
  1352. // with all ones, it is not implemented. Set the resource type to
  1353. // NULL, no further processing is required.
  1354. //
  1355. // Note: We ignore the I/O bit in the BAR due to HARDWARE BUGS
  1356. // in some people's hardware.
  1357. //
  1358. if ((bar & ~1) == 0) {
  1359. Descriptor->Type = CmResourceTypeNull;
  1360. return FALSE;
  1361. }
  1362. //
  1363. // Default to ordinary (32 bit) memory.
  1364. //
  1365. Descriptor->Flags = CM_RESOURCE_MEMORY_READ_WRITE;
  1366. Descriptor->u.Memory.MaximumAddress.HighPart = 0;
  1367. Descriptor->u.Memory.MinimumAddress.QuadPart = 0;
  1368. if (Rom == TRUE) {
  1369. //
  1370. // Mask out unused bits and indicate in the descriptor that
  1371. // this entry describes ROM.
  1372. //
  1373. bar &= PCI_ADDRESS_ROM_ADDRESS_MASK;
  1374. Descriptor->Flags = CM_RESOURCE_MEMORY_READ_ONLY;
  1375. }
  1376. //
  1377. // Ranges described by PCI Base Address Registers must be a
  1378. // power of 2 in length and naturally aligned. Get the length
  1379. // and set the length and alignment in the descriptor.
  1380. //
  1381. length = PciGetLengthFromBar(bar);
  1382. Descriptor->u.Generic.Length = length;
  1383. Descriptor->u.Generic.Alignment = length;
  1384. if ((bar & PCI_ADDRESS_IO_SPACE) != 0) {
  1385. //
  1386. // This BAR describes I/O space.
  1387. //
  1388. addressMask = PCI_ADDRESS_IO_ADDRESS_MASK;
  1389. Descriptor->Type = CmResourceTypePort;
  1390. Descriptor->Flags = CM_RESOURCE_PORT_IO;
  1391. } else {
  1392. //
  1393. // This BAR describes PCI memory space.
  1394. //
  1395. addressMask = PCI_ADDRESS_MEMORY_ADDRESS_MASK;
  1396. Descriptor->Type = CmResourceTypeMemory;
  1397. if ((bar & PCI_ADDRESS_MEMORY_TYPE_MASK) == PCI_TYPE_64BIT) {
  1398. //
  1399. // This is a 64 bit PCI device. Get the high 32 bits
  1400. // from the next BAR.
  1401. //
  1402. Descriptor->u.Memory.MaximumAddress.HighPart = *(BaseAddress+1);
  1403. returnValue = TRUE;
  1404. } else if ((bar & PCI_ADDRESS_MEMORY_TYPE_MASK) == PCI_TYPE_20BIT) {
  1405. //
  1406. // This device must locate below 1MB, the BAR shouldn't
  1407. // have any top bits set but this isn't clear from the
  1408. // spec. Enforce it by clearing the top bits.
  1409. //
  1410. addressMask &= 0x000fffff;
  1411. }
  1412. if (bar & PCI_ADDRESS_MEMORY_PREFETCHABLE) {
  1413. Descriptor->Flags |= CM_RESOURCE_MEMORY_PREFETCHABLE;
  1414. }
  1415. }
  1416. Descriptor->u.Generic.MaximumAddress.LowPart = bar & addressMask;
  1417. Descriptor->u.Generic.MaximumAddress.QuadPart += (length - 1);
  1418. return returnValue;
  1419. }
  1420. VOID
  1421. PciInvalidateResourceInfoCache(
  1422. IN PPCI_PDO_EXTENSION PdoExtension
  1423. )
  1424. /*++
  1425. Description:
  1426. Delete any cached resource information as it is (potentially)
  1427. no longer valid.
  1428. Currently this doesn't actually have anything to do.
  1429. Arguments:
  1430. PdoExtension PDO cached info may be associated with.
  1431. Return Value:
  1432. None.
  1433. --*/
  1434. {
  1435. }
  1436. BOOLEAN
  1437. PciOpenKey(
  1438. IN PWSTR KeyName,
  1439. IN HANDLE ParentHandle,
  1440. OUT PHANDLE Handle,
  1441. OUT PNTSTATUS Status
  1442. )
  1443. /*++
  1444. Description:
  1445. Open a registry key.
  1446. Arguments:
  1447. KeyName Name of the key to be opened.
  1448. ParentHandle Pointer to the parent handle (OPTIONAL)
  1449. Handle Pointer to a handle to recieve the opened key.
  1450. Return Value:
  1451. TRUE is key successfully opened, FALSE otherwise.
  1452. --*/
  1453. {
  1454. UNICODE_STRING nameString;
  1455. OBJECT_ATTRIBUTES nameAttributes;
  1456. NTSTATUS localStatus;
  1457. PAGED_CODE();
  1458. RtlInitUnicodeString(&nameString, KeyName);
  1459. InitializeObjectAttributes(&nameAttributes,
  1460. &nameString,
  1461. OBJ_CASE_INSENSITIVE,
  1462. ParentHandle,
  1463. (PSECURITY_DESCRIPTOR)NULL
  1464. );
  1465. localStatus = ZwOpenKey(Handle,
  1466. KEY_READ,
  1467. &nameAttributes
  1468. );
  1469. if (Status != NULL) {
  1470. //
  1471. // Caller wants underlying status.
  1472. //
  1473. *Status = localStatus;
  1474. }
  1475. //
  1476. // Return status converted to a boolean, TRUE if
  1477. // successful.
  1478. //
  1479. return NT_SUCCESS(localStatus);
  1480. }
  1481. NTSTATUS
  1482. PciGetRegistryValue(
  1483. IN PWSTR ValueName,
  1484. IN PWSTR KeyName,
  1485. IN HANDLE ParentHandle,
  1486. OUT PVOID *Buffer,
  1487. OUT ULONG *Length
  1488. )
  1489. {
  1490. NTSTATUS status;
  1491. HANDLE keyHandle;
  1492. ULONG neededLength;
  1493. ULONG actualLength;
  1494. UNICODE_STRING unicodeValueName;
  1495. PKEY_VALUE_PARTIAL_INFORMATION info;
  1496. if (!PciOpenKey(KeyName, ParentHandle, &keyHandle, &status)) {
  1497. return status;
  1498. }
  1499. unicodeValueName.Buffer = ValueName;
  1500. unicodeValueName.MaximumLength = (wcslen(ValueName) + 1) * sizeof(WCHAR);
  1501. unicodeValueName.Length = unicodeValueName.MaximumLength - sizeof(WCHAR);
  1502. //
  1503. // Find out how much memory we need for this.
  1504. //
  1505. status = ZwQueryValueKey(
  1506. keyHandle,
  1507. &unicodeValueName,
  1508. KeyValuePartialInformation,
  1509. NULL,
  1510. 0,
  1511. &neededLength
  1512. );
  1513. if (status == STATUS_BUFFER_TOO_SMALL) {
  1514. ASSERT(neededLength != 0);
  1515. //
  1516. // Get memory to return the data in. Note this includes
  1517. // a header that we really don't want.
  1518. //
  1519. info = ExAllocatePool(
  1520. PagedPool | POOL_COLD_ALLOCATION,
  1521. neededLength);
  1522. if (info == NULL) {
  1523. ZwClose(keyHandle);
  1524. return STATUS_INSUFFICIENT_RESOURCES;
  1525. }
  1526. //
  1527. // Get the data.
  1528. //
  1529. status = ZwQueryValueKey(
  1530. keyHandle,
  1531. &unicodeValueName,
  1532. KeyValuePartialInformation,
  1533. info,
  1534. neededLength,
  1535. &actualLength
  1536. );
  1537. if (!NT_SUCCESS(status)) {
  1538. ASSERT(NT_SUCCESS(status));
  1539. ExFreePool(info);
  1540. ZwClose(keyHandle);
  1541. return status;
  1542. }
  1543. ASSERT(neededLength == actualLength);
  1544. //
  1545. // Subtract out the header size and get memory for just
  1546. // the data we want.
  1547. //
  1548. neededLength -= FIELD_OFFSET(KEY_VALUE_PARTIAL_INFORMATION, Data);
  1549. *Buffer = ExAllocatePool(
  1550. PagedPool | POOL_COLD_ALLOCATION,
  1551. neededLength
  1552. );
  1553. if (*Buffer == NULL) {
  1554. ExFreePool(info);
  1555. ZwClose(keyHandle);
  1556. return STATUS_INSUFFICIENT_RESOURCES;
  1557. }
  1558. //
  1559. // Copy data sans header.
  1560. //
  1561. RtlCopyMemory(*Buffer, info->Data, neededLength);
  1562. ExFreePool(info);
  1563. if (Length) {
  1564. *Length = neededLength;
  1565. }
  1566. } else {
  1567. #if DBG
  1568. PciDebugPrint(
  1569. PciDbgInformative,
  1570. "PCI - Unexpected status %08x from ZwQueryValueKey, expected\n",
  1571. status
  1572. );
  1573. PciDebugPrint(
  1574. PciDbgInformative,
  1575. " STATUS_BUFFER_TOO_SMALL (%08x).\n",
  1576. STATUS_BUFFER_TOO_SMALL
  1577. );
  1578. #endif
  1579. if (NT_SUCCESS(status)) {
  1580. //
  1581. // We don't want to report success when this happens.
  1582. //
  1583. status = STATUS_UNSUCCESSFUL;
  1584. }
  1585. }
  1586. ZwClose(keyHandle);
  1587. return status;
  1588. }
  1589. NTSTATUS
  1590. PciGetDeviceCapabilities(
  1591. IN PDEVICE_OBJECT DeviceObject,
  1592. IN PDEVICE_CAPABILITIES DeviceCapabilities
  1593. )
  1594. /*++
  1595. Routine Description:
  1596. This routine sends the get capabilities irp to the given stack
  1597. Arguments:
  1598. DeviceObject A device object in the stack whose capabilities we want
  1599. DeviceCapabilites Where to store the answer
  1600. Return Value:
  1601. NTSTATUS
  1602. --*/
  1603. {
  1604. IO_STATUS_BLOCK ioStatus;
  1605. KEVENT pnpEvent;
  1606. NTSTATUS status;
  1607. PDEVICE_OBJECT targetObject;
  1608. PIO_STACK_LOCATION irpStack;
  1609. PIRP pnpIrp;
  1610. PAGED_CODE();
  1611. //
  1612. // Initialize the capabilities that we will send down
  1613. //
  1614. RtlZeroMemory( DeviceCapabilities, sizeof(DEVICE_CAPABILITIES) );
  1615. DeviceCapabilities->Size = sizeof(DEVICE_CAPABILITIES);
  1616. DeviceCapabilities->Version = 1;
  1617. DeviceCapabilities->Address = -1;
  1618. DeviceCapabilities->UINumber = -1;
  1619. //
  1620. // Initialize the event
  1621. //
  1622. KeInitializeEvent( &pnpEvent, SynchronizationEvent, FALSE );
  1623. //
  1624. // Get the irp that we will send the request to
  1625. //
  1626. targetObject = IoGetAttachedDeviceReference( DeviceObject );
  1627. //
  1628. // Build an Irp
  1629. //
  1630. pnpIrp = IoBuildSynchronousFsdRequest(
  1631. IRP_MJ_PNP,
  1632. targetObject,
  1633. NULL,
  1634. 0,
  1635. NULL,
  1636. &pnpEvent,
  1637. &ioStatus
  1638. );
  1639. if (pnpIrp == NULL) {
  1640. status = STATUS_INSUFFICIENT_RESOURCES;
  1641. goto PciGetDeviceCapabilitiesExit;
  1642. }
  1643. //
  1644. // Pnp Irps all begin life as STATUS_NOT_SUPPORTED;
  1645. //
  1646. pnpIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;
  1647. pnpIrp->IoStatus.Information = 0;
  1648. //
  1649. // Get the top of stack
  1650. //
  1651. irpStack = IoGetNextIrpStackLocation( pnpIrp );
  1652. if (irpStack == NULL) {
  1653. status = STATUS_INVALID_PARAMETER;
  1654. goto PciGetDeviceCapabilitiesExit;
  1655. }
  1656. //
  1657. // Set the top of stack
  1658. //
  1659. RtlZeroMemory( irpStack, sizeof(IO_STACK_LOCATION ) );
  1660. irpStack->MajorFunction = IRP_MJ_PNP;
  1661. irpStack->MinorFunction = IRP_MN_QUERY_CAPABILITIES;
  1662. irpStack->Parameters.DeviceCapabilities.Capabilities = DeviceCapabilities;
  1663. //
  1664. // Make sure that there are no completion routines set
  1665. //
  1666. IoSetCompletionRoutine(
  1667. pnpIrp,
  1668. NULL,
  1669. NULL,
  1670. FALSE,
  1671. FALSE,
  1672. FALSE
  1673. );
  1674. //
  1675. // Call the driver
  1676. //
  1677. status = IoCallDriver( targetObject, pnpIrp );
  1678. if (status == STATUS_PENDING) {
  1679. //
  1680. // Block until the irp comes back
  1681. //
  1682. KeWaitForSingleObject(
  1683. &pnpEvent,
  1684. Executive,
  1685. KernelMode,
  1686. FALSE,
  1687. NULL
  1688. );
  1689. status = ioStatus.Status;
  1690. }
  1691. PciGetDeviceCapabilitiesExit:
  1692. //
  1693. // Done with reference
  1694. //
  1695. ObDereferenceObject( targetObject );
  1696. //
  1697. // Done
  1698. //
  1699. return status;
  1700. }
  1701. ULONGLONG
  1702. PciGetHackFlags(
  1703. IN USHORT VendorID,
  1704. IN USHORT DeviceID,
  1705. IN USHORT SubVendorID,
  1706. IN USHORT SubSystemID,
  1707. IN UCHAR RevisionID
  1708. )
  1709. /*++
  1710. Description:
  1711. Look in the registry for any flags for this VendorId/DeviceId.
  1712. Arguments:
  1713. VendorId PCI Vendor ID (16 bits) of the manufacturer of the
  1714. device.
  1715. DeviceId PCI Device ID (16 bits) of the device.
  1716. SubVendorID PCI SubVendorID representing the manufacturer of the
  1717. subsystem
  1718. SubSystemID PCI SubSystemID representing subsystem
  1719. RevisionID PCI Revision denoting the revision of the device
  1720. Return Value:
  1721. 64 bit flags value or 0 if not found.
  1722. --*/
  1723. {
  1724. PPCI_HACK_TABLE_ENTRY current;
  1725. ULONGLONG hackFlags = 0;
  1726. ULONG match, bestMatch = 0;
  1727. ASSERT(PciHackTable);
  1728. //
  1729. // We want to do a best-case match:
  1730. // VVVVDDDDSSSSssssRR
  1731. // VVVVDDDDSSSSssss
  1732. // VVVVDDDDRR
  1733. // VVVVDDDD
  1734. //
  1735. // List is currently unsorted, so keep updating current best match.
  1736. //
  1737. for (current = PciHackTable; current->VendorID != 0xFFFF; current++) {
  1738. match = 0;
  1739. //
  1740. // Must at least match vendor/dev
  1741. //
  1742. if ((current->DeviceID != DeviceID) ||
  1743. (current->VendorID != VendorID)) {
  1744. continue;
  1745. }
  1746. match = 1;
  1747. //
  1748. // If this entry specifies a revision, check that it is consistent.
  1749. //
  1750. if (current->Flags & PCI_HACK_FLAG_REVISION) {
  1751. if (current->RevisionID == RevisionID) {
  1752. match += 2;
  1753. } else {
  1754. continue;
  1755. }
  1756. }
  1757. //
  1758. // If this entry specifies subsystems, check that they are consistent
  1759. //
  1760. if (current->Flags & PCI_HACK_FLAG_SUBSYSTEM) {
  1761. if (current->SubVendorID == SubVendorID &&
  1762. current->SubSystemID == SubSystemID) {
  1763. match += 4;
  1764. } else {
  1765. continue;
  1766. }
  1767. }
  1768. if (match > bestMatch) {
  1769. bestMatch = match;
  1770. hackFlags = current->HackFlags;
  1771. }
  1772. }
  1773. return hackFlags;
  1774. }
  1775. NTSTATUS
  1776. PciGetDeviceProperty(
  1777. IN PDEVICE_OBJECT PhysicalDeviceObject,
  1778. IN DEVICE_REGISTRY_PROPERTY DeviceProperty,
  1779. OUT PVOID *PropertyBuffer
  1780. )
  1781. {
  1782. NTSTATUS status;
  1783. NTSTATUS expected;
  1784. ULONG length;
  1785. ULONG length2;
  1786. PVOID buffer;
  1787. //
  1788. // Two passes, first pass, find out what size buffer
  1789. // is needed.
  1790. //
  1791. status = IoGetDeviceProperty(
  1792. PhysicalDeviceObject,
  1793. DeviceProperty,
  1794. 0,
  1795. NULL,
  1796. &length
  1797. );
  1798. expected = STATUS_BUFFER_TOO_SMALL;
  1799. if (status == expected) {
  1800. //
  1801. // Good, now get a buffer.
  1802. //
  1803. buffer = ExAllocatePool(PagedPool | POOL_COLD_ALLOCATION, length);
  1804. if (buffer == NULL) {
  1805. PciDebugPrint(
  1806. PciDbgAlways,
  1807. "PCI - Failed to allocate DeviceProperty buffer (%d bytes).\n",
  1808. length
  1809. );
  1810. status = STATUS_INSUFFICIENT_RESOURCES;
  1811. } else {
  1812. //
  1813. // This time, do it for real.
  1814. //
  1815. status = IoGetDeviceProperty(
  1816. PhysicalDeviceObject,
  1817. DeviceProperty,
  1818. length,
  1819. buffer,
  1820. &length2
  1821. );
  1822. if (NT_SUCCESS(status)) {
  1823. ASSERT(length == length2);
  1824. //
  1825. // Return the buffer containing the requested device
  1826. // property to the caller.
  1827. //
  1828. *PropertyBuffer = buffer;
  1829. return STATUS_SUCCESS;
  1830. }
  1831. expected = STATUS_SUCCESS;
  1832. }
  1833. }
  1834. PciDebugPrint(
  1835. PciDbgAlways,
  1836. "PCI - Unexpected status from GetDeviceProperty, saw %08X, expected %08X.\n",
  1837. status,
  1838. expected
  1839. );
  1840. //
  1841. // Clear the caller's buffer pointer, and, if the unexpected status
  1842. // is success (from the first call to IoGetDeviceProperty) change it
  1843. // to STATUS_UNSUCCESSFUL (N.B. This is if course impossible).
  1844. //
  1845. *PropertyBuffer = NULL;
  1846. if (status == STATUS_SUCCESS) {
  1847. ASSERTMSG("PCI Successfully did the impossible!", 0);
  1848. status = STATUS_UNSUCCESSFUL;
  1849. }
  1850. return status;
  1851. }
  1852. NTSTATUS
  1853. PciRangeListFromResourceList(
  1854. IN PPCI_FDO_EXTENSION FdoExtension,
  1855. IN PCM_RESOURCE_LIST ResourceList,
  1856. IN CM_RESOURCE_TYPE DesiredType,
  1857. IN BOOLEAN Complement,
  1858. IN PRTL_RANGE_LIST ResultRange
  1859. )
  1860. /*++
  1861. Description:
  1862. Generates a range list for the resources of a given type
  1863. from a resource list.
  1864. Note: This routine supports only Memory or Io resources.
  1865. Overlapping ranges in the incoming list will be combined.
  1866. Arguments:
  1867. FdoExtension Bus particulars. NOTE: This is only needed for the
  1868. gross X86 hack for A0000 due to buggy MPS BIOS
  1869. implementations. Otherwise this routine is more
  1870. generalized.
  1871. ResourceList Incoming CM Resource List.
  1872. DesiredType Type of resource to be included in the range list.
  1873. Complement Specifies wether or not the range list should be
  1874. the "complement" of the incoming data.
  1875. ResultRange Output range list.
  1876. Return Value:
  1877. TRUE is key successfully opened, FALSE otherwise.
  1878. --*/
  1879. {
  1880. #define EXIT_IF_ERROR(status) \
  1881. if (!NT_SUCCESS(status)) { \
  1882. ASSERT(NT_SUCCESS(status)); \
  1883. goto exitPoint; \
  1884. }
  1885. #if DBG
  1886. #define ADD_RANGE(range, start, end, status) \
  1887. PciDebugPrint( \
  1888. PciDbgObnoxious, \
  1889. " Adding to RtlRange %I64x thru %I64x\n", \
  1890. (ULONGLONG)start, \
  1891. (ULONGLONG)end \
  1892. ); \
  1893. status = RtlAddRange(range, start, end, 0, 0, NULL, NULL); \
  1894. if (!NT_SUCCESS(status)) { \
  1895. ASSERT(NT_SUCCESS(status)); \
  1896. goto exitPoint; \
  1897. }
  1898. #else
  1899. #define ADD_RANGE(range, start, end, status) \
  1900. status = RtlAddRange(range, start, end, 0, 0, NULL, NULL); \
  1901. if (!NT_SUCCESS(status)) { \
  1902. ASSERT(NT_SUCCESS(status)); \
  1903. goto exitPoint; \
  1904. }
  1905. #endif
  1906. NTSTATUS status;
  1907. ULONG elementCount;
  1908. ULONG count;
  1909. ULONG numlists;
  1910. PCM_FULL_RESOURCE_DESCRIPTOR full;
  1911. PCM_PARTIAL_RESOURCE_LIST partial;
  1912. PCM_PARTIAL_RESOURCE_DESCRIPTOR descriptor;
  1913. typedef struct {
  1914. LIST_ENTRY list;
  1915. ULONGLONG start;
  1916. ULONGLONG end;
  1917. BOOLEAN valid;
  1918. } PCI_RANGE_LIST_ELEMENT, *PPCI_RANGE_LIST_ELEMENT;
  1919. PPCI_RANGE_LIST_ELEMENT elementBuffer;
  1920. PPCI_RANGE_LIST_ELEMENT upper;
  1921. PPCI_RANGE_LIST_ELEMENT lower;
  1922. PPCI_RANGE_LIST_ELEMENT current;
  1923. ULONG allocatedElement;
  1924. ULONGLONG start;
  1925. ULONGLONG end;
  1926. #if defined(_X86_) && defined(PCI_NT50_BETA1_HACKS)
  1927. //
  1928. // BETA1_HACKS - Remove this when the problem is fixed.
  1929. //
  1930. // HACK HACK some MPS BIOS implementations don't report the
  1931. // memory range 0xA0000 thru 0xBFFFF. They should. HACK
  1932. // them into the memory list. Gross.
  1933. // Even grosser, assume this applies only to bus 0.
  1934. //
  1935. // The 400 hack is because some cards (Matrox MGA) want access
  1936. // to the SYSTEM BIOS DATA area which is in memory at address
  1937. // 0x400 thru 0x4ff. It's not on the BUS so why are we making
  1938. // it appear here?
  1939. //
  1940. // Note, there is TWO hacks here but we do both under the
  1941. // exact same condition so we have only one boolean. If the
  1942. // two are seperated (or one removed) this needs to be split.
  1943. //
  1944. BOOLEAN doA0000Hack = (DesiredType == CmResourceTypeMemory) &&
  1945. (FdoExtension && (FdoExtension->BaseBus == 0));
  1946. #endif
  1947. PAGED_CODE();
  1948. ASSERT((DesiredType == CmResourceTypeMemory) ||
  1949. (DesiredType == CmResourceTypePort));
  1950. //
  1951. // First, get a count of the number of resources of the desired
  1952. // type in the list. This gives us the maximum number of entries
  1953. // in the resulting list.
  1954. //
  1955. // Plus 1 in case we're complementing it. 2 actually, we start
  1956. // with a beginning and end entry.
  1957. //
  1958. elementCount = 2;
  1959. numlists = 0;
  1960. if (ResourceList != NULL) {
  1961. numlists = ResourceList->Count;
  1962. full = ResourceList->List;
  1963. }
  1964. while (numlists--) {
  1965. partial = &full->PartialResourceList;
  1966. count = partial->Count;
  1967. descriptor = partial->PartialDescriptors;
  1968. while (count--) {
  1969. if (descriptor->Type == DesiredType) {
  1970. if (DesiredType == CmResourceTypePort) {
  1971. if (descriptor->Flags & CM_RESOURCE_PORT_10_BIT_DECODE) {
  1972. elementCount += ((1 << 16) / (1 << 10)) - 1;
  1973. } else if (descriptor->Flags & CM_RESOURCE_PORT_12_BIT_DECODE) {
  1974. elementCount += ((1 << 16) / (1 << 12)) - 1;
  1975. }
  1976. }
  1977. elementCount++;
  1978. }
  1979. descriptor = PciNextPartialDescriptor(descriptor);
  1980. }
  1981. full = (PCM_FULL_RESOURCE_DESCRIPTOR)descriptor;
  1982. }
  1983. PciDebugPrint(
  1984. PciDbgObnoxious,
  1985. "PCI - PciRangeListFromResourceList processing %d elements.\n",
  1986. elementCount - 2
  1987. );
  1988. #if defined(_X86_) && defined(PCI_NT50_BETA1_HACKS)
  1989. if (doA0000Hack) {
  1990. elementCount += 3; // one for A0000 hack, one for 400 hack. + 1 for 70
  1991. }
  1992. #endif
  1993. //
  1994. // Allocate list entries and initialize the list.
  1995. //
  1996. elementBuffer = ExAllocatePool(
  1997. PagedPool | POOL_COLD_ALLOCATION,
  1998. elementCount * sizeof(PCI_RANGE_LIST_ELEMENT)
  1999. );
  2000. if (elementBuffer == NULL) {
  2001. return STATUS_INSUFFICIENT_RESOURCES;
  2002. }
  2003. //
  2004. // Take the first two entries and set them to the absolute minimum
  2005. // and absolute maximum possible values. Everything else will
  2006. // either end up between these or be combined with them.
  2007. //
  2008. // Setting the terminators this way should avoid us having to check
  2009. // for end conditions.
  2010. //
  2011. allocatedElement = 2;
  2012. current = &elementBuffer[1];
  2013. // first element (list min terminator)
  2014. elementBuffer[1].start = elementBuffer[1].end = 0;
  2015. elementBuffer[1].list.Flink = &elementBuffer[0].list;
  2016. elementBuffer[1].list.Blink = &elementBuffer[0].list;
  2017. elementBuffer[1].valid = FALSE;
  2018. // last element (list max terminator)
  2019. elementBuffer[0].start = elementBuffer[0].end = ~0;
  2020. elementBuffer[0].list.Flink = &elementBuffer[1].list;
  2021. elementBuffer[0].list.Blink = &elementBuffer[1].list;
  2022. elementBuffer[0].valid = FALSE;
  2023. #if defined(_X86_) && defined(PCI_NT50_BETA1_HACKS)
  2024. if (doA0000Hack) {
  2025. //
  2026. // Hack in A0000 thru FFFFF by just adding an entry for it
  2027. // to the otherwise empty list.
  2028. //
  2029. // Hack in 400 thru 4ff too.
  2030. //
  2031. PLIST_ENTRY minEntry = &elementBuffer[1].list;
  2032. PLIST_ENTRY maxEntry = &elementBuffer[0].list;
  2033. PPCI_RANGE_LIST_ELEMENT tempElement;
  2034. allocatedElement = 5;
  2035. elementBuffer[2].start = 0x70; // HACK Trident
  2036. elementBuffer[2].end = 0x70;
  2037. elementBuffer[2].valid = TRUE;
  2038. elementBuffer[3].start = 0x400; // HACK Matrox MGA
  2039. elementBuffer[3].end = 0x4FF;
  2040. elementBuffer[3].valid = TRUE;
  2041. elementBuffer[4].start = 0xA0000; // HACK broken MPS BIOS
  2042. elementBuffer[4].end = 0xBFFFF;
  2043. elementBuffer[4].valid = TRUE;
  2044. // set the flinks
  2045. minEntry->Flink = &elementBuffer[2].list;
  2046. elementBuffer[2].list.Flink = &elementBuffer[3].list;
  2047. elementBuffer[3].list.Flink = &elementBuffer[4].list;
  2048. elementBuffer[4].list.Flink = maxEntry;
  2049. // set the blinks
  2050. elementBuffer[2].list.Blink = minEntry;
  2051. elementBuffer[3].list.Blink = &elementBuffer[2].list;
  2052. elementBuffer[4].list.Blink = &elementBuffer[3].list;
  2053. maxEntry->Blink = &elementBuffer[4].list;
  2054. #if DBG
  2055. tempElement = CONTAINING_RECORD(
  2056. minEntry,
  2057. PCI_RANGE_LIST_ELEMENT,
  2058. list
  2059. );
  2060. PciDebugPrint(
  2061. PciDbgObnoxious,
  2062. " === PCI added default initial ranges ===\n"
  2063. );
  2064. do {
  2065. //
  2066. // Print this entry if it is valid.
  2067. //
  2068. if (tempElement->valid == TRUE) {
  2069. PciDebugPrint(
  2070. PciDbgObnoxious,
  2071. " %I64x .. %I64x\n",
  2072. tempElement->start,
  2073. tempElement->end
  2074. );
  2075. }
  2076. //
  2077. // Next entry.
  2078. //
  2079. if (tempElement->list.Flink == minEntry) {
  2080. break;
  2081. }
  2082. tempElement = CONTAINING_RECORD(
  2083. tempElement->list.Flink,
  2084. PCI_RANGE_LIST_ELEMENT,
  2085. list
  2086. );
  2087. } while (TRUE);
  2088. PciDebugPrint(
  2089. PciDbgObnoxious,
  2090. " === end added default initial ranges ===\n"
  2091. );
  2092. #endif
  2093. }
  2094. #endif
  2095. //
  2096. // Starting again at the beginning of the resource list, extract
  2097. // the desired resources and insert them in our new list.
  2098. //
  2099. numlists = 0;
  2100. if (ResourceList != NULL) {
  2101. full = ResourceList->List;
  2102. numlists = ResourceList->Count;
  2103. }
  2104. while (numlists--) {
  2105. LIST_CONTEXT listContext;
  2106. PcipInitializePartialListContext(
  2107. &listContext,
  2108. &full->PartialResourceList,
  2109. DesiredType
  2110. );
  2111. while ((descriptor = PcipGetNextRangeFromList(&listContext)) != NULL) {
  2112. ASSERT(descriptor->Type == DesiredType);
  2113. //
  2114. // insert this element into the list.
  2115. //
  2116. start = (ULONGLONG)descriptor->u.Generic.Start.QuadPart;
  2117. end = start - 1 + descriptor->u.Generic.Length;
  2118. //
  2119. // First find the element to the left of this one
  2120. // (below it).
  2121. //
  2122. lower = current;
  2123. //
  2124. // Just in case we actually need to go right,...
  2125. //
  2126. while (start > lower->end) {
  2127. lower = CONTAINING_RECORD(
  2128. lower->list.Flink,
  2129. PCI_RANGE_LIST_ELEMENT,
  2130. list
  2131. );
  2132. }
  2133. //
  2134. // Search left.
  2135. //
  2136. while (start <= lower->end) {
  2137. if (start >= lower->start) {
  2138. break;
  2139. }
  2140. //
  2141. // Go left.
  2142. //
  2143. lower = CONTAINING_RECORD(
  2144. lower->list.Blink,
  2145. PCI_RANGE_LIST_ELEMENT,
  2146. list
  2147. );
  2148. }
  2149. //
  2150. // Early out if the lower entry completely
  2151. // covers the new entry.
  2152. //
  2153. if ((start >= lower->start) && (end <= lower->end)) {
  2154. //
  2155. // It does, just skip it.
  2156. //
  2157. PciDebugPrint(
  2158. PciDbgObnoxious,
  2159. " -- (%I64x .. %I64x) swallows (%I64x .. %I64x)\n",
  2160. lower->start,
  2161. lower->end,
  2162. start,
  2163. end
  2164. );
  2165. current = lower;
  2166. current->valid = TRUE;
  2167. continue;
  2168. }
  2169. //
  2170. // Then, the one above it.
  2171. //
  2172. upper = lower;
  2173. while (end > upper->start) {
  2174. if (end <= upper->end) {
  2175. break;
  2176. }
  2177. //
  2178. // Go right.
  2179. //
  2180. upper = CONTAINING_RECORD(
  2181. upper->list.Flink,
  2182. PCI_RANGE_LIST_ELEMENT,
  2183. list
  2184. );
  2185. }
  2186. current = &elementBuffer[allocatedElement++];
  2187. current->start = start;
  2188. current->end = end;
  2189. current->valid = TRUE;
  2190. PciDebugPrint(
  2191. PciDbgObnoxious,
  2192. " (%I64x .. %I64x) <= (%I64x .. %I64x) <= (%I64x .. %I64x)\n",
  2193. lower->start,
  2194. lower->end,
  2195. start,
  2196. end,
  2197. upper->start,
  2198. upper->end
  2199. );
  2200. //
  2201. // We now have, the element below this one, possibly
  2202. // overlapping, the element above this one, possibly
  2203. // overlapping, and a new one.
  2204. //
  2205. // The easiest way to deal with this is to create
  2206. // the new entry, link it in, then unlink the overlaps
  2207. // if they exist.
  2208. //
  2209. //
  2210. // Note: The new entry may overlap several entries,
  2211. // these are orphaned.
  2212. //
  2213. // Link it in.
  2214. //
  2215. current->list.Flink = &upper->list;
  2216. current->list.Blink = &lower->list;
  2217. upper->list.Blink = &current->list;
  2218. lower->list.Flink = &current->list;
  2219. //
  2220. // Check for lower overlap.
  2221. //
  2222. if ((lower->valid == TRUE) && (start > 0)) {
  2223. start--;
  2224. }
  2225. if (lower->end >= start) {
  2226. //
  2227. // Overlaps from below,...
  2228. //
  2229. // Merge lower into current.
  2230. //
  2231. current->start = lower->start;
  2232. current->list.Blink = lower->list.Blink;
  2233. //
  2234. //
  2235. // lower is being orphaned, reuse it to get to
  2236. // our new lower neighbor.
  2237. //
  2238. lower = CONTAINING_RECORD(
  2239. lower->list.Blink,
  2240. PCI_RANGE_LIST_ELEMENT,
  2241. list
  2242. );
  2243. lower->list.Flink = &current->list;
  2244. PciDebugPrint(
  2245. PciDbgObnoxious,
  2246. " -- Overlaps lower, merged to (%I64x .. %I64x)\n",
  2247. current->start,
  2248. current->end
  2249. );
  2250. }
  2251. //
  2252. // Check for upper overlap.
  2253. //
  2254. if ((upper->valid == TRUE) && (end < ~0)) {
  2255. end++;
  2256. }
  2257. if ((end >= upper->start) && (current != upper)) {
  2258. //
  2259. // Overlaps above,... merge upper into current.
  2260. //
  2261. current->end = upper->end;
  2262. current->list.Flink = upper->list.Flink;
  2263. //
  2264. // upper is being orphaned, reuse it to get to
  2265. // our new upper neighbor.
  2266. //
  2267. upper = CONTAINING_RECORD(
  2268. upper->list.Flink,
  2269. PCI_RANGE_LIST_ELEMENT,
  2270. list
  2271. );
  2272. upper->list.Blink = &current->list;
  2273. PciDebugPrint(
  2274. PciDbgObnoxious,
  2275. " -- Overlaps upper, merged to (%I64x .. %I64x)\n",
  2276. current->start,
  2277. current->end
  2278. );
  2279. }
  2280. }
  2281. full = (PCM_FULL_RESOURCE_DESCRIPTOR)listContext.Next;
  2282. }
  2283. //
  2284. // Find the lowest value.
  2285. //
  2286. while (current->valid == TRUE) {
  2287. lower = CONTAINING_RECORD(
  2288. current->list.Blink,
  2289. PCI_RANGE_LIST_ELEMENT,
  2290. list
  2291. );
  2292. if ((lower->valid == FALSE) ||
  2293. (lower->start > current->start)) {
  2294. break;
  2295. }
  2296. current = lower;
  2297. }
  2298. #if DBG
  2299. lower = current;
  2300. if (current->valid == FALSE) {
  2301. PciDebugPrint(
  2302. PciDbgObnoxious,
  2303. " ==== No ranges in results list. ====\n"
  2304. );
  2305. } else {
  2306. PciDebugPrint(
  2307. PciDbgObnoxious,
  2308. " === ranges ===\n"
  2309. );
  2310. do {
  2311. if (current->valid == TRUE) {
  2312. PciDebugPrint(
  2313. PciDbgObnoxious,
  2314. " %I64x .. %I64x\n",
  2315. current->start,
  2316. current->end
  2317. );
  2318. }
  2319. //
  2320. // Next entry.
  2321. //
  2322. current = CONTAINING_RECORD(
  2323. current->list.Flink,
  2324. PCI_RANGE_LIST_ELEMENT,
  2325. list
  2326. );
  2327. } while (current != lower);
  2328. }
  2329. #endif
  2330. if (Complement == TRUE) {
  2331. //
  2332. // Invert the list.
  2333. //
  2334. // The generation of the list always results in the orphaning
  2335. // of elementBuffer[1] (which was the original start point),
  2336. // we can use that one for the first element of the new
  2337. // inverted list.
  2338. //
  2339. if (current->valid == FALSE) {
  2340. //
  2341. // Empty list, complement it you get everything.
  2342. //
  2343. ADD_RANGE(ResultRange, 0, ~0, status);
  2344. } else {
  2345. //
  2346. // If the original range doesn't start at zero we must
  2347. // generate an entry from 0 to the start of that range.
  2348. //
  2349. if (current->start != 0) {
  2350. ADD_RANGE(ResultRange, 0, current->start - 1, status);
  2351. }
  2352. //
  2353. // Run the list greating range list entries for the
  2354. // gaps between entries in this list.
  2355. //
  2356. do {
  2357. PPCI_RANGE_LIST_ELEMENT next = CONTAINING_RECORD(
  2358. current->list.Flink,
  2359. PCI_RANGE_LIST_ELEMENT,
  2360. list
  2361. );
  2362. if (current->valid == TRUE) {
  2363. start = current->end + 1;
  2364. end = next->start - 1;
  2365. if ((end < start) || (next == elementBuffer)) {
  2366. end = ~0;
  2367. }
  2368. ADD_RANGE(ResultRange, start, end, status);
  2369. }
  2370. //
  2371. // Next entry.
  2372. //
  2373. current = next;
  2374. } while (current != lower);
  2375. }
  2376. } else {
  2377. //
  2378. // Not complementing,... add a range for each member of the
  2379. // list.
  2380. //
  2381. if (current->valid == TRUE) {
  2382. do {
  2383. ADD_RANGE(ResultRange, current->start, current->end, status);
  2384. //
  2385. // Next entry.
  2386. //
  2387. current = CONTAINING_RECORD(
  2388. current->list.Flink,
  2389. PCI_RANGE_LIST_ELEMENT,
  2390. list
  2391. );
  2392. } while (current != lower);
  2393. }
  2394. }
  2395. status = STATUS_SUCCESS;
  2396. exitPoint:
  2397. ExFreePool(elementBuffer);
  2398. return status;
  2399. #undef EXIT_IF_ERROR
  2400. }
  2401. UCHAR
  2402. PciReadDeviceCapability(
  2403. IN PPCI_PDO_EXTENSION PdoExtension,
  2404. IN UCHAR Offset,
  2405. IN UCHAR Id,
  2406. IN OUT PVOID Buffer,
  2407. IN ULONG Length
  2408. )
  2409. /*++
  2410. Description:
  2411. Searches configuration space for the PCI Capabilities structure
  2412. identified by Id. Begins at offset Offset in PCI config space.
  2413. Arguments:
  2414. PdoExtension Pointer to the PDO Extension for this device.
  2415. Offset Offset into PCI config space to begin traversing
  2416. the capabilities list.
  2417. Id Capabilities ID. (0 if want to match any).
  2418. Buffer Pointer to the buffer where the capabilities
  2419. structure is to be returned (includes capabilities
  2420. header).
  2421. Length Number of bytes wanted (must be at least large
  2422. enough to contain the header).
  2423. Return Value:
  2424. Returns the Offset in PCI config space at which the capability
  2425. was found or 0 if not found.
  2426. --*/
  2427. {
  2428. NTSTATUS status;
  2429. PPCI_CAPABILITIES_HEADER capHeader;
  2430. UCHAR loopCount = 0;
  2431. capHeader = (PPCI_CAPABILITIES_HEADER)Buffer;
  2432. //
  2433. // In case the caller is running the list, check if we got
  2434. // handed the list end.
  2435. //
  2436. if (Offset == 0) {
  2437. return 0;
  2438. }
  2439. ASSERT_PCI_PDO_EXTENSION(PdoExtension);
  2440. ASSERT(PdoExtension->CapabilitiesPtr != 0);
  2441. ASSERT(Buffer);
  2442. ASSERT(Length >= sizeof(PCI_CAPABILITIES_HEADER));
  2443. do {
  2444. //
  2445. // Catch case where the device has been powered off. (Reads
  2446. // from a powered off device return FF,... allowing also for
  2447. // the case where the device is just broken).
  2448. //
  2449. if ((Offset < PCI_COMMON_HDR_LENGTH) ||
  2450. ((Offset & 0x3) != 0)) {
  2451. ASSERT((Offset >= PCI_COMMON_HDR_LENGTH) && ((Offset & 0x3) == 0));
  2452. return 0;
  2453. }
  2454. PciReadDeviceConfig(
  2455. PdoExtension,
  2456. Buffer,
  2457. Offset,
  2458. sizeof(PCI_CAPABILITIES_HEADER)
  2459. );
  2460. //
  2461. // Check if this capability is the one we want (or if we want
  2462. // ALL capability structures).
  2463. //
  2464. // NOTE: Intel 21554 non-transparent P2P bridge has a VPD
  2465. // capability that has the Chassis capability id. Needs to be
  2466. // handled here in the future. Maybe fixed in later revisions.
  2467. //
  2468. if ((capHeader->CapabilityID == Id) || (Id == 0)) {
  2469. break;
  2470. }
  2471. Offset = capHeader->Next;
  2472. //
  2473. // One more check for broken h/w. Make sure we're not
  2474. // traversing a circular list. A Capabilities header
  2475. // cannot be in the common header and must be DWORD aligned
  2476. // in config space so there can only be (256-64)/4 of them.
  2477. //
  2478. if (++loopCount > ((256-64)/4)) {
  2479. PciDebugPrint(
  2480. PciDbgAlways,
  2481. "PCI device %p capabilities list is broken.\n",
  2482. PdoExtension
  2483. );
  2484. return 0;
  2485. }
  2486. } while (Offset != 0);
  2487. //
  2488. // If we found a match and we haven't read all the data, get the
  2489. // remainder.
  2490. //
  2491. if ((Offset != 0) && (Length > sizeof(PCI_CAPABILITIES_HEADER))) {
  2492. if (Length > (sizeof(PCI_COMMON_CONFIG) - Offset)) {
  2493. //
  2494. // If we are too close to the end of config space to
  2495. // return the amount of data the caller requested,
  2496. // truncate.
  2497. //
  2498. // Worst case truncation will be to 4 bytes so no need
  2499. // to check we have data to read (again).
  2500. //
  2501. ASSERT(Length <= (sizeof(PCI_COMMON_CONFIG) - Offset));
  2502. Length = sizeof(PCI_COMMON_CONFIG) - Offset;
  2503. }
  2504. //
  2505. // Read remainder.
  2506. //
  2507. Length -= sizeof(PCI_CAPABILITIES_HEADER);
  2508. PciReadDeviceConfig(
  2509. PdoExtension,
  2510. capHeader + 1,
  2511. Offset + sizeof(PCI_CAPABILITIES_HEADER),
  2512. Length
  2513. );
  2514. }
  2515. return Offset;
  2516. }
  2517. BOOLEAN
  2518. PciCanDisableDecodes(
  2519. IN PPCI_PDO_EXTENSION PdoExtension OPTIONAL,
  2520. IN PPCI_COMMON_CONFIG Config OPTIONAL,
  2521. IN ULONGLONG HackFlags,
  2522. IN ULONG Flags
  2523. )
  2524. // N.B. - not paged so we can power down at dispatch level
  2525. {
  2526. UCHAR baseClass;
  2527. UCHAR subClass;
  2528. BOOLEAN canDisableVideoDecodes;
  2529. canDisableVideoDecodes = (Flags & PCI_CAN_DISABLE_VIDEO_DECODES) == PCI_CAN_DISABLE_VIDEO_DECODES;
  2530. if (ARGUMENT_PRESENT(PdoExtension)) {
  2531. ASSERT(HackFlags == 0);
  2532. HackFlags = PdoExtension->HackFlags;
  2533. baseClass = PdoExtension->BaseClass;
  2534. subClass = PdoExtension->SubClass;
  2535. } else {
  2536. ASSERT(ARGUMENT_PRESENT(Config));
  2537. baseClass = Config->BaseClass;
  2538. subClass = Config->SubClass;
  2539. }
  2540. if (HackFlags & PCI_HACK_PRESERVE_COMMAND) {
  2541. //
  2542. // Bad things happen if we touch this device's command
  2543. // register, leave it alone.
  2544. //
  2545. return FALSE;
  2546. }
  2547. if (HackFlags & PCI_HACK_CB_SHARE_CMD_BITS) {
  2548. //
  2549. // This is a multifunction cardbus controller with a shared
  2550. // command register. Never turn of any of the functions because it has
  2551. // the unfortunate side effect of turning of all of them!
  2552. //
  2553. // NTRAID #62672 - 4/25/2000 - andrewth
  2554. // We should probably ensure that the windows for all functions
  2555. // are closed on all functions before enabling any of them...
  2556. //
  2557. //
  2558. return FALSE;
  2559. }
  2560. if (HackFlags & PCI_HACK_NO_DISABLE_DECODES) {
  2561. //
  2562. // If we disable the decodes on this device bad things happen
  2563. //
  2564. return FALSE;
  2565. }
  2566. //
  2567. // If this is a video device then don't allow the decodes to be disabled unless
  2568. // we are allowed to...
  2569. //
  2570. if ((baseClass == PCI_CLASS_DISPLAY_CTLR && subClass == PCI_SUBCLASS_VID_VGA_CTLR)
  2571. || (baseClass == PCI_CLASS_PRE_20 && subClass == PCI_SUBCLASS_PRE_20_VGA)) {
  2572. return canDisableVideoDecodes;
  2573. }
  2574. //
  2575. // There are various things in the world we shouldn't turn off.
  2576. // The system is quite possibly unable to recover if we do, so
  2577. // don't (just pretend).
  2578. //
  2579. switch (baseClass) {
  2580. case PCI_CLASS_BRIDGE_DEV:
  2581. //
  2582. // Bad things happen if we turn off the HOST bridge (the
  2583. // system doesn't understand that this device, which is
  2584. // on a PCI bus, is actually the parent of that PCI bus),
  2585. // or ISA/EISA/MCA bridges under which are devices we still
  2586. // need to have working but are legacy detected so not in
  2587. // the heirachy in any way we understand.
  2588. //
  2589. if ((subClass == PCI_SUBCLASS_BR_ISA ) ||
  2590. (subClass == PCI_SUBCLASS_BR_EISA) ||
  2591. (subClass == PCI_SUBCLASS_BR_MCA) ||
  2592. (subClass == PCI_SUBCLASS_BR_HOST) ||
  2593. (subClass == PCI_SUBCLASS_BR_OTHER)) {
  2594. return FALSE;
  2595. }
  2596. //
  2597. // We don't want to turn off bridges that might have the VGA card behind
  2598. // then otherwise video stops working. Seeing as we can't actually tell
  2599. // where the VGA card is use the hint that if the bridge is passing VGA
  2600. // ranges the video card is probably somewhere down there.
  2601. //
  2602. if (subClass == PCI_SUBCLASS_BR_PCI_TO_PCI
  2603. || subClass == PCI_SUBCLASS_BR_CARDBUS) {
  2604. BOOLEAN vgaBitSet;
  2605. if (ARGUMENT_PRESENT(PdoExtension)) {
  2606. vgaBitSet = PdoExtension->Dependent.type1.VgaBitSet;
  2607. } else {
  2608. vgaBitSet = (Config->u.type1.BridgeControl & PCI_ENABLE_BRIDGE_VGA) != 0;
  2609. }
  2610. if (vgaBitSet) {
  2611. //
  2612. // We can disable the video path if we are powering down the machine
  2613. //
  2614. return canDisableVideoDecodes;
  2615. }
  2616. }
  2617. break;
  2618. case PCI_CLASS_DISPLAY_CTLR:
  2619. //
  2620. // If a video driver fails to start, the device reverts back to being
  2621. // VGA if it is the VGA device. Don't disable the decodes on VGA
  2622. // devices.
  2623. //
  2624. if (subClass == PCI_SUBCLASS_VID_VGA_CTLR) {
  2625. //
  2626. // We can disable the video path if we are powering down the machine
  2627. //
  2628. return canDisableVideoDecodes;
  2629. }
  2630. break;
  2631. case PCI_CLASS_PRE_20:
  2632. //
  2633. // Same as above.
  2634. //
  2635. if (subClass == PCI_SUBCLASS_PRE_20_VGA) {
  2636. //
  2637. // We can disable the video path if we are powering down the machine
  2638. //
  2639. return canDisableVideoDecodes;
  2640. }
  2641. break;
  2642. }
  2643. return TRUE;
  2644. }
  2645. VOID
  2646. PciDecodeEnable(
  2647. IN PPCI_PDO_EXTENSION PdoExtension,
  2648. IN BOOLEAN Enable,
  2649. IN PUSHORT ExistingCommand OPTIONAL
  2650. )
  2651. /*++
  2652. Description:
  2653. Either sets the decodes to match the extension (misnomered Enable) or zeros
  2654. the decodes entirely.
  2655. N.B. - not paged so we can power down at dispatch level
  2656. Arguments:
  2657. PdoExtension Pointer to the PDO Extension for this device.
  2658. Enable If TRUE, decodes are set to match the extension (on or off).
  2659. If FALSE, decodes are disabled.
  2660. ExistingCommand Optional saved command to prevent a reread of the config
  2661. space command field.
  2662. Return Value:
  2663. Nothing.
  2664. --*/
  2665. {
  2666. USHORT cmd;
  2667. ULONG length = sizeof(cmd);
  2668. //
  2669. // Can we disable it if so ordered?
  2670. //
  2671. if (!Enable && !PciCanDisableDecodes(PdoExtension, NULL, 0, 0)) {
  2672. return;
  2673. }
  2674. if (PdoExtension->HackFlags & PCI_HACK_PRESERVE_COMMAND) {
  2675. //
  2676. // Bad things happen if we touch this device's command
  2677. // register, leave it alone.
  2678. //
  2679. return;
  2680. }
  2681. if (ARGUMENT_PRESENT(ExistingCommand)) {
  2682. //
  2683. // The caller has supplied the current contents of the
  2684. // device's config space.
  2685. //
  2686. cmd = *ExistingCommand;
  2687. } else {
  2688. //
  2689. // Get the current command register from the device.
  2690. //
  2691. PciGetCommandRegister(PdoExtension, &cmd);
  2692. }
  2693. cmd &= ~(PCI_ENABLE_IO_SPACE |
  2694. PCI_ENABLE_MEMORY_SPACE |
  2695. PCI_ENABLE_BUS_MASTER);
  2696. if (Enable) {
  2697. //
  2698. // Set enables
  2699. //
  2700. cmd |= PdoExtension->CommandEnables & (PCI_ENABLE_IO_SPACE
  2701. | PCI_ENABLE_MEMORY_SPACE
  2702. | PCI_ENABLE_BUS_MASTER);
  2703. }
  2704. //
  2705. // Set the new command register into the device.
  2706. //
  2707. PciSetCommandRegister(PdoExtension, cmd);
  2708. }
  2709. NTSTATUS
  2710. PciExcludeRangesFromWindow(
  2711. IN ULONGLONG Start,
  2712. IN ULONGLONG End,
  2713. IN PRTL_RANGE_LIST ArbiterRanges,
  2714. IN PRTL_RANGE_LIST ExclusionRanges
  2715. )
  2716. {
  2717. NTSTATUS status;
  2718. RTL_RANGE_LIST_ITERATOR iterator;
  2719. PRTL_RANGE current;
  2720. FOR_ALL_RANGES(ExclusionRanges, &iterator, current) {
  2721. if (current->Owner == NULL
  2722. && INTERSECT(current->Start, current->End, Start, End)) {
  2723. status = RtlAddRange(ArbiterRanges,
  2724. current->Start,
  2725. current->End,
  2726. 0,
  2727. RTL_RANGE_LIST_ADD_IF_CONFLICT,
  2728. NULL,
  2729. NULL // this range is not on the bus
  2730. );
  2731. if (!NT_SUCCESS(status)) {
  2732. ASSERT(NT_SUCCESS(status));
  2733. return status;
  2734. }
  2735. }
  2736. }
  2737. return STATUS_SUCCESS;
  2738. }
  2739. NTSTATUS
  2740. PciBuildDefaultExclusionLists(
  2741. VOID
  2742. )
  2743. {
  2744. NTSTATUS status;
  2745. ULONG windowBase;
  2746. ASSERT(PciIsaBitExclusionList.Count == 0);
  2747. ASSERT(PciVgaAndIsaBitExclusionList.Count == 0);
  2748. RtlInitializeRangeList(&PciIsaBitExclusionList);
  2749. RtlInitializeRangeList(&PciVgaAndIsaBitExclusionList);
  2750. for (windowBase = 0; windowBase <= 0xFFFF; windowBase += 0x400) {
  2751. //
  2752. // Add the x100-x3ff range to the ISA list
  2753. //
  2754. status = RtlAddRange(&PciIsaBitExclusionList,
  2755. windowBase + 0x100,
  2756. windowBase + 0x3FF,
  2757. 0,
  2758. RTL_RANGE_LIST_ADD_IF_CONFLICT,
  2759. NULL,
  2760. NULL // this range is not on the bus
  2761. );
  2762. if (!NT_SUCCESS(status)) {
  2763. goto cleanup;
  2764. }
  2765. //
  2766. // Add the x100-x3af, x3bc-x3bf and x3e0-x3ff ranges to the VGA/ISA list
  2767. //
  2768. status = RtlAddRange(&PciVgaAndIsaBitExclusionList,
  2769. windowBase + 0x100,
  2770. windowBase + 0x3AF,
  2771. 0,
  2772. RTL_RANGE_LIST_ADD_IF_CONFLICT,
  2773. NULL,
  2774. NULL // this range is not on the bus
  2775. );
  2776. if (!NT_SUCCESS(status)) {
  2777. goto cleanup;
  2778. }
  2779. status = RtlAddRange(&PciVgaAndIsaBitExclusionList,
  2780. windowBase + 0x3BC,
  2781. windowBase + 0x3BF,
  2782. 0,
  2783. RTL_RANGE_LIST_ADD_IF_CONFLICT,
  2784. NULL,
  2785. NULL // this range is not on the bus
  2786. );
  2787. if (!NT_SUCCESS(status)) {
  2788. goto cleanup;
  2789. }
  2790. status = RtlAddRange(&PciVgaAndIsaBitExclusionList,
  2791. windowBase + 0x3E0,
  2792. windowBase + 0x3FF,
  2793. 0,
  2794. RTL_RANGE_LIST_ADD_IF_CONFLICT,
  2795. NULL,
  2796. NULL // this range is not on the bus
  2797. );
  2798. if (!NT_SUCCESS(status)) {
  2799. goto cleanup;
  2800. }
  2801. }
  2802. return STATUS_SUCCESS;
  2803. cleanup:
  2804. RtlFreeRangeList(&PciIsaBitExclusionList);
  2805. RtlFreeRangeList(&PciVgaAndIsaBitExclusionList);
  2806. return status;
  2807. }
  2808. NTSTATUS
  2809. PciSaveBiosConfig(
  2810. IN PPCI_PDO_EXTENSION PdoExtension,
  2811. IN PPCI_COMMON_CONFIG Config
  2812. )
  2813. /*++
  2814. Description:
  2815. This saves the original configuration of a device in the registry
  2816. Arguments:
  2817. PdoExtension Pointer to the PDO Extension for this device.
  2818. Config The config space as the BIOS initialized it
  2819. Return Value:
  2820. Status
  2821. --*/
  2822. {
  2823. NTSTATUS status;
  2824. OBJECT_ATTRIBUTES attributes;
  2825. UNICODE_STRING unicodeString;
  2826. HANDLE deviceHandle, configHandle;
  2827. WCHAR buffer[sizeof(L"DEV_xx&FUN_xx")];
  2828. PAGED_CODE();
  2829. status = IoOpenDeviceRegistryKey(PCI_PARENT_PDO(PdoExtension),
  2830. PLUGPLAY_REGKEY_DEVICE,
  2831. KEY_ALL_ACCESS,
  2832. &deviceHandle
  2833. );
  2834. if (!NT_SUCCESS(status)) {
  2835. goto cleanup;
  2836. }
  2837. PciConstStringToUnicodeString(&unicodeString, BIOS_CONFIG_KEY_NAME);
  2838. InitializeObjectAttributes(&attributes,
  2839. &unicodeString,
  2840. OBJ_KERNEL_HANDLE,
  2841. deviceHandle,
  2842. NULL
  2843. );
  2844. status = ZwCreateKey(&configHandle,
  2845. KEY_ALL_ACCESS,
  2846. &attributes,
  2847. 0,
  2848. NULL,
  2849. REG_OPTION_VOLATILE,
  2850. NULL
  2851. );
  2852. ZwClose(deviceHandle);
  2853. if (!NT_SUCCESS(status)) {
  2854. goto cleanup;
  2855. }
  2856. unicodeString.Length =
  2857. (USHORT)_snwprintf(buffer,
  2858. sizeof(buffer)/sizeof(WCHAR),
  2859. L"DEV_%02x&FUN_%02x",
  2860. PdoExtension->Slot.u.bits.DeviceNumber,
  2861. PdoExtension->Slot.u.bits.FunctionNumber
  2862. );
  2863. unicodeString.Length *= sizeof(WCHAR); // Length is in bytes
  2864. unicodeString.MaximumLength = unicodeString.Length;
  2865. unicodeString.Buffer = buffer;
  2866. status = ZwSetValueKey(configHandle,
  2867. &unicodeString,
  2868. 0,
  2869. REG_BINARY,
  2870. Config,
  2871. PCI_COMMON_HDR_LENGTH
  2872. );
  2873. ZwClose(configHandle);
  2874. return status;
  2875. cleanup:
  2876. return status;
  2877. }
  2878. NTSTATUS
  2879. PciGetBiosConfig(
  2880. IN PPCI_PDO_EXTENSION PdoExtension,
  2881. IN PPCI_COMMON_CONFIG Config
  2882. )
  2883. /*++
  2884. Description:
  2885. This retrieves the original configuration of a device from the registry
  2886. Arguments:
  2887. PdoExtension Pointer to the PDO Extension for this device.
  2888. Config The config space as the BIOS initialized it
  2889. Return Value:
  2890. Status
  2891. --*/
  2892. {
  2893. NTSTATUS status;
  2894. OBJECT_ATTRIBUTES attributes;
  2895. UNICODE_STRING unicodeString;
  2896. HANDLE deviceHandle, configHandle;
  2897. WCHAR buffer[sizeof(L"DEV_xx&FUN_xx")];
  2898. CHAR returnBuffer[sizeof(KEY_VALUE_PARTIAL_INFORMATION) + PCI_COMMON_HDR_LENGTH - 1];
  2899. PKEY_VALUE_PARTIAL_INFORMATION info;
  2900. ULONG resultLength;
  2901. PAGED_CODE();
  2902. status = IoOpenDeviceRegistryKey(PCI_PARENT_PDO(PdoExtension),
  2903. PLUGPLAY_REGKEY_DEVICE,
  2904. KEY_READ | KEY_WRITE,
  2905. &deviceHandle
  2906. );
  2907. if (!NT_SUCCESS(status)) {
  2908. goto cleanup;
  2909. }
  2910. PciConstStringToUnicodeString(&unicodeString, BIOS_CONFIG_KEY_NAME);
  2911. InitializeObjectAttributes(&attributes,
  2912. &unicodeString,
  2913. OBJ_KERNEL_HANDLE,
  2914. deviceHandle,
  2915. NULL
  2916. );
  2917. status = ZwOpenKey(&configHandle,
  2918. KEY_READ,
  2919. &attributes
  2920. );
  2921. ZwClose(deviceHandle);
  2922. if (!NT_SUCCESS(status)) {
  2923. goto cleanup;
  2924. }
  2925. unicodeString.Length =
  2926. (USHORT)_snwprintf(buffer,
  2927. sizeof(buffer)/sizeof(WCHAR),
  2928. L"DEV_%02x&FUN_%02x",
  2929. PdoExtension->Slot.u.bits.DeviceNumber,
  2930. PdoExtension->Slot.u.bits.FunctionNumber
  2931. );
  2932. unicodeString.Length *= sizeof(WCHAR); // Length is in bytes
  2933. unicodeString.MaximumLength = unicodeString.Length;
  2934. unicodeString.Buffer = buffer;
  2935. status = ZwQueryValueKey(configHandle,
  2936. &unicodeString,
  2937. KeyValuePartialInformation,
  2938. &returnBuffer,
  2939. sizeof(returnBuffer),
  2940. &resultLength
  2941. );
  2942. ZwClose(configHandle);
  2943. if (NT_SUCCESS(status)) {
  2944. info = (PKEY_VALUE_PARTIAL_INFORMATION) returnBuffer;
  2945. ASSERT(info->DataLength == PCI_COMMON_HDR_LENGTH);
  2946. RtlCopyMemory(Config, info->Data, PCI_COMMON_HDR_LENGTH);
  2947. }
  2948. return status;
  2949. cleanup:
  2950. return status;
  2951. }
  2952. #if 0
  2953. BOOLEAN
  2954. PciPresenceCheck(
  2955. IN PPCI_PDO_EXTENSION PdoExtension
  2956. )
  2957. {
  2958. UCHAR configSpaceBuffer[PCI_COMMON_HDR_LENGTH];
  2959. PPCI_COMMON_CONFIG cardConfig = (PPCI_COMMON_CONFIG) configSpaceBuffer;
  2960. PAGED_CODE();
  2961. //
  2962. // If the card is already missing, don't bother reexamining it.
  2963. //
  2964. if (PdoExtension->NotPresent) {
  2965. return FALSE;
  2966. }
  2967. if (PciIsSameDevice(PdoExtension)) {
  2968. //
  2969. // Still here.
  2970. //
  2971. return TRUE;
  2972. }
  2973. //
  2974. // Mark it not present, then tell the OS it's gone.
  2975. //
  2976. PdoExtension->NotPresent = 1;
  2977. IoInvalidateDeviceState(PdoExtension->PhysicalDeviceObject);
  2978. return FALSE;
  2979. }
  2980. #endif
  2981. BOOLEAN
  2982. PciStringToUSHORT(
  2983. IN PWCHAR String,
  2984. OUT PUSHORT Result
  2985. )
  2986. /*++
  2987. Description:
  2988. Takes a 4 character hexidecimal sting and converts it into a USHORT.
  2989. Arguments:
  2990. String - the string
  2991. Result - the USHORT
  2992. Return Value:
  2993. TRUE is success, FASLE otherwise
  2994. --*/
  2995. {
  2996. ULONG count;
  2997. USHORT number = 0;
  2998. PWCHAR current;
  2999. current = String;
  3000. for (count = 0; count < 4; count++) {
  3001. number <<= 4;
  3002. if (*current >= L'0' && *current <= L'9') {
  3003. number |= *current - L'0';
  3004. } else if (*current >= L'A' && *current <= L'F') {
  3005. number |= *current + 10 - L'A';
  3006. } else if (*current >= L'a' && *current <= L'f') {
  3007. number |= *current + 10 - L'a';
  3008. } else {
  3009. return FALSE;
  3010. }
  3011. current++;
  3012. }
  3013. *Result = number;
  3014. return TRUE;
  3015. }
  3016. NTSTATUS
  3017. PciSendIoctl(
  3018. IN PDEVICE_OBJECT Device,
  3019. IN ULONG IoctlCode,
  3020. IN PVOID InputBuffer OPTIONAL,
  3021. IN ULONG InputBufferLength,
  3022. IN PVOID OutputBuffer OPTIONAL,
  3023. IN ULONG OutputBufferLength
  3024. )
  3025. /*++
  3026. Description:
  3027. Builds and send an IOCTL to a device and return the results
  3028. Arguments:
  3029. Device - a device on the device stack to receive the IOCTL - the
  3030. irp is always sent to the top of the stack
  3031. IoctlCode - the IOCTL to run
  3032. InputBuffer - arguments to the IOCTL
  3033. InputBufferLength - length in bytes of the InputBuffer
  3034. OutputBuffer - data returned by the IOCTL
  3035. OnputBufferLength - the size in bytes of the OutputBuffer
  3036. Return Value:
  3037. Status
  3038. --*/
  3039. {
  3040. NTSTATUS status;
  3041. IO_STATUS_BLOCK ioStatus;
  3042. KEVENT event;
  3043. PIRP irp;
  3044. PDEVICE_OBJECT targetDevice = NULL;
  3045. PAGED_CODE();
  3046. KeInitializeEvent(&event, SynchronizationEvent, FALSE);
  3047. //
  3048. // Get the top of the stack to send the IRP to
  3049. //
  3050. targetDevice = IoGetAttachedDeviceReference(Device);
  3051. if (!targetDevice) {
  3052. status = STATUS_INVALID_PARAMETER;
  3053. goto exit;
  3054. }
  3055. //
  3056. // Get Io to build the IRP for us
  3057. //
  3058. irp = IoBuildDeviceIoControlRequest(IoctlCode,
  3059. targetDevice,
  3060. InputBuffer,
  3061. InputBufferLength,
  3062. OutputBuffer,
  3063. OutputBufferLength,
  3064. FALSE, // InternalDeviceIoControl
  3065. &event,
  3066. &ioStatus
  3067. );
  3068. if (!irp) {
  3069. status = STATUS_INSUFFICIENT_RESOURCES;
  3070. goto exit;
  3071. }
  3072. //
  3073. // Send the IRP and wait for it to complete
  3074. //
  3075. status = IoCallDriver(targetDevice, irp);
  3076. if (status == STATUS_PENDING) {
  3077. KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL);
  3078. status = ioStatus.Status;
  3079. }
  3080. exit:
  3081. if (targetDevice) {
  3082. ObDereferenceObject(targetDevice);
  3083. }
  3084. return status;
  3085. }
  3086. BOOLEAN
  3087. PciIsOnVGAPath(
  3088. IN PPCI_PDO_EXTENSION Pdo
  3089. )
  3090. /*++
  3091. Description:
  3092. Guesses if we are on the VGA path or not!
  3093. Arguments:
  3094. Pdo - The PDO for the device in question
  3095. Return Value:
  3096. TRUE if we are on the VGA path, TRUE otherwise
  3097. --*/
  3098. {
  3099. switch (Pdo->BaseClass) {
  3100. case PCI_CLASS_BRIDGE_DEV:
  3101. //
  3102. // We don't want to turn off bridges that might have the VGA card behind
  3103. // then otherwise video stops working. Seeing as we can't actually tell
  3104. // where the VGA card is use the hint that if the bridge is passing VGA
  3105. // ranges the video card is probably somewhere down there.
  3106. //
  3107. if (Pdo->SubClass == PCI_SUBCLASS_BR_PCI_TO_PCI
  3108. || Pdo->SubClass == PCI_SUBCLASS_BR_CARDBUS) {
  3109. if (Pdo->Dependent.type1.VgaBitSet) {
  3110. return TRUE;
  3111. }
  3112. }
  3113. break;
  3114. case PCI_CLASS_DISPLAY_CTLR:
  3115. if (Pdo->SubClass == PCI_SUBCLASS_VID_VGA_CTLR) {
  3116. return TRUE;
  3117. }
  3118. break;
  3119. case PCI_CLASS_PRE_20:
  3120. if (Pdo->SubClass == PCI_SUBCLASS_PRE_20_VGA) {
  3121. return TRUE;
  3122. }
  3123. break;
  3124. }
  3125. return FALSE;
  3126. }
  3127. BOOLEAN
  3128. PciIsSlotPresentInParentMethod(
  3129. IN PPCI_PDO_EXTENSION Pdo,
  3130. IN ULONG Method
  3131. )
  3132. /*++
  3133. Description:
  3134. This function checks if the slot this device is in is present in a
  3135. Method named package on the parent of this device.
  3136. Arguments:
  3137. Pdo - The PDO extension for the device
  3138. Method - The Parents method to examine
  3139. Return Value:
  3140. TRUE if present, FALSE otherwise
  3141. --*/
  3142. {
  3143. NTSTATUS status;
  3144. ACPI_EVAL_INPUT_BUFFER input;
  3145. PACPI_EVAL_OUTPUT_BUFFER output = NULL;
  3146. ULONG count, adr;
  3147. PACPI_METHOD_ARGUMENT argument;
  3148. BOOLEAN result = FALSE;
  3149. //
  3150. // Allocate a buffer big enough for all possible slots
  3151. //
  3152. ULONG outputSize = sizeof(ACPI_EVAL_OUTPUT_BUFFER) + sizeof(ACPI_METHOD_ARGUMENT) * (PCI_MAX_DEVICES * PCI_MAX_FUNCTION);
  3153. PAGED_CODE();
  3154. output = ExAllocatePool(PagedPool | POOL_COLD_ALLOCATION, outputSize);
  3155. if (!output) {
  3156. status = STATUS_INSUFFICIENT_RESOURCES;
  3157. goto exit;
  3158. }
  3159. RtlZeroMemory(&input, sizeof(ACPI_EVAL_INPUT_BUFFER));
  3160. RtlZeroMemory(output, outputSize);
  3161. //
  3162. // Send a IOCTL to ACPI to request it to run the method on this device's
  3163. // parent if the method it is present
  3164. //
  3165. input.Signature = ACPI_EVAL_INPUT_BUFFER_SIGNATURE;
  3166. input.MethodNameAsUlong = Method;
  3167. status = PciSendIoctl(PCI_PARENT_FDOX(Pdo)->PhysicalDeviceObject,
  3168. IOCTL_ACPI_EVAL_METHOD,
  3169. &input,
  3170. sizeof(ACPI_EVAL_INPUT_BUFFER),
  3171. output,
  3172. outputSize
  3173. );
  3174. if (!NT_SUCCESS(status)) {
  3175. goto exit;
  3176. }
  3177. //
  3178. // Format my slot number as an _ADR style integer
  3179. //
  3180. adr = (Pdo->Slot.u.bits.DeviceNumber << 16) | Pdo->Slot.u.bits.FunctionNumber;
  3181. for (count = 0; count < output->Count; count++) {
  3182. //
  3183. // Walking the arguments works like this because we are a package of
  3184. // integers
  3185. //
  3186. argument = &output->Argument[count];
  3187. if (argument->Type != ACPI_METHOD_ARGUMENT_INTEGER) {
  3188. status = STATUS_INVALID_PARAMETER;
  3189. goto exit;
  3190. }
  3191. if (argument->Argument == adr) {
  3192. //
  3193. // Jackpot!
  3194. //
  3195. result = TRUE;
  3196. break;
  3197. }
  3198. }
  3199. exit:
  3200. if (output) {
  3201. ExFreePool(output);
  3202. }
  3203. return result;
  3204. }
  3205. BOOLEAN
  3206. PciIsDeviceOnDebugPath(
  3207. IN PPCI_PDO_EXTENSION Pdo
  3208. )
  3209. /*++
  3210. Description:
  3211. This function checks if device is on the path to the debugging device
  3212. NOTE: PDO is only partially initialized at this point. Take care to insure
  3213. that fields touched here are valid.
  3214. Arguments:
  3215. Pdo - The PDO extension for the device
  3216. Return Value:
  3217. TRUE if on the debug path, FALSE otherwise
  3218. --*/
  3219. {
  3220. NTSTATUS status;
  3221. PPCI_DEBUG_PORT current;
  3222. PCI_COMMON_HEADER header;
  3223. PPCI_COMMON_CONFIG config = (PPCI_COMMON_CONFIG) &header;
  3224. PAGED_CODE();
  3225. ASSERT(PciDebugPortsCount <= MAX_DEBUGGING_DEVICES_SUPPORTED);
  3226. //
  3227. // We can't be on the debug path if we aren't using a PCI debug port!
  3228. //
  3229. if (PciDebugPortsCount == 0) {
  3230. return FALSE;
  3231. }
  3232. RtlZeroMemory(&header, sizeof(header));
  3233. //
  3234. // If its a bridge check if one of its subordinate buses has the debugger
  3235. // port on it
  3236. //
  3237. if (Pdo->HeaderType == PCI_BRIDGE_TYPE
  3238. || Pdo->HeaderType == PCI_CARDBUS_BRIDGE_TYPE) {
  3239. //
  3240. // Use the configuration that the firmware left the device in
  3241. //
  3242. status = PciGetBiosConfig(Pdo, config);
  3243. ASSERT(NT_SUCCESS(status));
  3244. FOR_ALL_IN_ARRAY(PciDebugPorts, PciDebugPortsCount, current) {
  3245. if (current->Bus >= config->u.type1.SecondaryBus
  3246. && current->Bus <= config->u.type1.SubordinateBus
  3247. && config->u.type1.SecondaryBus != 0
  3248. && config->u.type1.SubordinateBus != 0) {
  3249. return TRUE;
  3250. }
  3251. }
  3252. } else {
  3253. UCHAR parentBus;
  3254. if (PCI_PDO_ON_ROOT(Pdo)) {
  3255. parentBus = PCI_PARENT_FDOX(Pdo)->BaseBus;
  3256. } else {
  3257. //
  3258. // Get the BIOS config of the parent so we can get its initial bus
  3259. // number
  3260. //
  3261. status = PciGetBiosConfig(PCI_BRIDGE_PDO(PCI_PARENT_FDOX(Pdo)),
  3262. config
  3263. );
  3264. ASSERT(NT_SUCCESS(status));
  3265. if (config->u.type1.SecondaryBus == 0
  3266. || config->u.type1.SubordinateBus == 0) {
  3267. //
  3268. // This is a bridge that wasn't configured by the firmware so this
  3269. // child can't be on the debug path.
  3270. //
  3271. return FALSE;
  3272. } else {
  3273. parentBus = config->u.type1.SecondaryBus;
  3274. }
  3275. }
  3276. //
  3277. // Check if we are the device on the correct bus in the correct slot
  3278. //
  3279. FOR_ALL_IN_ARRAY(PciDebugPorts, PciDebugPortsCount, current) {
  3280. if (current->Bus == parentBus
  3281. && current->Slot.u.AsULONG == Pdo->Slot.u.AsULONG) {
  3282. return TRUE;
  3283. }
  3284. }
  3285. }
  3286. return FALSE;
  3287. }
  3288. NTSTATUS
  3289. PciUpdateLegacyHardwareDescription(
  3290. IN PPCI_FDO_EXTENSION Fdo
  3291. )
  3292. {
  3293. NTSTATUS status;
  3294. HANDLE multifunctionHandle = NULL, indexHandle = NULL;
  3295. WCHAR indexStringBuffer[10];
  3296. UNICODE_STRING indexString, tempString;
  3297. OBJECT_ATTRIBUTES attributes;
  3298. UCHAR infoBuffer[sizeof(KEY_VALUE_PARTIAL_INFORMATION) + 50];
  3299. PKEY_VALUE_PARTIAL_INFORMATION info = (PKEY_VALUE_PARTIAL_INFORMATION) infoBuffer;
  3300. ULONG infoLength;
  3301. ULONG disposition;
  3302. CM_FULL_RESOURCE_DESCRIPTOR descriptor;
  3303. PCM_FULL_RESOURCE_DESCRIPTOR full;
  3304. CONFIGURATION_COMPONENT component;
  3305. ULONG index;
  3306. BOOLEAN createdNewKey = FALSE;
  3307. if (!PciOpenKey(L"\\REGISTRY\\MACHINE\\HARDWARE\\DESCRIPTION\\System\\MultifunctionAdapter",
  3308. NULL,
  3309. &multifunctionHandle,
  3310. &status)) {
  3311. goto exit;
  3312. }
  3313. //
  3314. // HKML\Hardware\Description\System\MultifunctionAdapter is structured as
  3315. // a set of 0 base consecutive numbered keys.
  3316. // Run through all the subkeys and check that we haven't already reported
  3317. // this bus
  3318. //
  3319. indexString.Buffer = indexStringBuffer;
  3320. indexString.MaximumLength = sizeof(indexStringBuffer);
  3321. indexString.Length = 0;
  3322. for (index = 0;;index++) {
  3323. status = RtlIntegerToUnicodeString(index, 10, &indexString);
  3324. if (!NT_SUCCESS(status)) {
  3325. goto exit;
  3326. }
  3327. InitializeObjectAttributes(&attributes,
  3328. &indexString,
  3329. OBJ_CASE_INSENSITIVE,
  3330. multifunctionHandle,
  3331. NULL
  3332. );
  3333. status = ZwCreateKey(&indexHandle,
  3334. KEY_ALL_ACCESS,
  3335. &attributes,
  3336. 0,
  3337. NULL,
  3338. REG_OPTION_VOLATILE,
  3339. &disposition
  3340. );
  3341. if (!NT_SUCCESS(status)) {
  3342. goto exit;
  3343. }
  3344. //
  3345. // As the keys are all consecutive then if we created this key we have
  3346. // enumerated then all and we can get on with registering out data
  3347. //
  3348. if (disposition == REG_CREATED_NEW_KEY) {
  3349. createdNewKey = TRUE;
  3350. break;
  3351. }
  3352. PciConstStringToUnicodeString(&tempString, L"Identifier");
  3353. status = ZwQueryValueKey(indexHandle,
  3354. &tempString,
  3355. KeyValuePartialInformation,
  3356. info,
  3357. sizeof(infoBuffer),
  3358. &infoLength
  3359. );
  3360. if (NT_SUCCESS(status)) {
  3361. if (info->Type == REG_SZ &&
  3362. (wcscmp(L"PCI", (PWSTR)&info->Data) == 0)) {
  3363. //
  3364. // This is a PCI bus, now check if its our bus number
  3365. //
  3366. PciConstStringToUnicodeString(&tempString, L"Configuration Data");
  3367. status = ZwQueryValueKey(indexHandle,
  3368. &tempString,
  3369. KeyValuePartialInformation,
  3370. info,
  3371. sizeof(infoBuffer),
  3372. &infoLength
  3373. );
  3374. if (NT_SUCCESS(status)) {
  3375. if (info->Type == REG_FULL_RESOURCE_DESCRIPTOR) {
  3376. full = (PCM_FULL_RESOURCE_DESCRIPTOR) &info->Data;
  3377. ASSERT(full->InterfaceType == PCIBus);
  3378. if (full->BusNumber == Fdo->BaseBus) {
  3379. //
  3380. // We're already reported this so we don't need to
  3381. // do anything.
  3382. //
  3383. status = STATUS_SUCCESS;
  3384. //
  3385. // indexHandle will be closed by the exit path.
  3386. //
  3387. goto exit;
  3388. }
  3389. }
  3390. }
  3391. }
  3392. }
  3393. ZwClose(indexHandle);
  3394. indexHandle = NULL;
  3395. }
  3396. //
  3397. // if we created a new key then indexHandle is it
  3398. //
  3399. if (createdNewKey) {
  3400. //
  3401. // Fill in the Identifier entry. This is a PCI bus.
  3402. //
  3403. PciConstStringToUnicodeString(&tempString, L"Identifier");
  3404. status = ZwSetValueKey(indexHandle,
  3405. &tempString,
  3406. 0,
  3407. REG_SZ,
  3408. L"PCI",
  3409. sizeof(L"PCI")
  3410. );
  3411. if (!NT_SUCCESS(status)) {
  3412. goto exit;
  3413. }
  3414. //
  3415. // Fill in the Configuration Data entry.
  3416. //
  3417. // Note that the complete descriptor is not written to the registry just
  3418. // enough data to indicate that this is an empty list (the first 16 bytes).
  3419. // This is a bit gross but it is what happens on x86 machines today and
  3420. // after all we're only doing this for backward compatibility.
  3421. //
  3422. RtlZeroMemory(&descriptor, sizeof(CM_FULL_RESOURCE_DESCRIPTOR));
  3423. descriptor.InterfaceType = PCIBus;
  3424. descriptor.BusNumber = Fdo->BaseBus;
  3425. PciConstStringToUnicodeString(&tempString, L"Configuration Data");
  3426. status = ZwSetValueKey(indexHandle,
  3427. &tempString,
  3428. 0,
  3429. REG_FULL_RESOURCE_DESCRIPTOR,
  3430. &descriptor,
  3431. 16
  3432. );
  3433. if (!NT_SUCCESS(status)) {
  3434. goto exit;
  3435. }
  3436. //
  3437. // Fill in the Component Information entry. This is the Flags, Revision, Version,
  3438. // Key and AffinityMask members from the CONFIGURATION_COMPONENT structure.
  3439. //
  3440. // For PCI buses the affinity is set to all processors (0xFFFFFFFF) and
  3441. // everything else is 0.
  3442. //
  3443. RtlZeroMemory(&component, sizeof(CONFIGURATION_COMPONENT));
  3444. component.AffinityMask = 0xFFFFFFFF;
  3445. PciConstStringToUnicodeString(&tempString, L"Component Information");
  3446. status = ZwSetValueKey(indexHandle,
  3447. &tempString,
  3448. 0,
  3449. REG_BINARY,
  3450. &component.Flags,
  3451. FIELD_OFFSET(CONFIGURATION_COMPONENT, ConfigurationDataLength) -
  3452. FIELD_OFFSET(CONFIGURATION_COMPONENT, Flags)
  3453. );
  3454. if (!NT_SUCCESS(status)) {
  3455. goto exit;
  3456. }
  3457. }
  3458. status = STATUS_SUCCESS;
  3459. exit:
  3460. if (indexHandle) {
  3461. //
  3462. // If we are failing attempt to cleanup by deleting the key we tried
  3463. // to create.
  3464. //
  3465. if (!NT_SUCCESS(status) && createdNewKey) {
  3466. ZwDeleteKey(indexHandle);
  3467. }
  3468. ZwClose(indexHandle);
  3469. }
  3470. if (multifunctionHandle) {
  3471. ZwClose(multifunctionHandle);
  3472. }
  3473. return status;
  3474. }
  3475. NTSTATUS
  3476. PciReadDeviceSpace(
  3477. IN PPCI_PDO_EXTENSION PdoExtension,
  3478. IN ULONG WhichSpace,
  3479. IN PVOID Buffer,
  3480. IN ULONG Offset,
  3481. IN ULONG Length,
  3482. OUT PULONG LengthRead
  3483. )
  3484. /*++
  3485. Routine Description:
  3486. This function handles reading from PCI device spaces and is called for both
  3487. the IRP_MN_READ_CONFIG and the BUS_INTERFACE_STANDARD.GetBusData cases.
  3488. Arguments:
  3489. PdoExtension - the PDO for the device we want to read from
  3490. WhichSpace - what type of space we want to read - of the form PCI_WHICHSPACE_*
  3491. Buffer - Supplies a pointer to where the data is to be returned
  3492. Offset - Indicates the offset into the space where the reading should begin.
  3493. Length - Indicates the count of bytes which should be read.
  3494. LengthRead - Indicates the count of bytes which was actually read.
  3495. Return Value:
  3496. Status
  3497. --*/
  3498. {
  3499. // NOT PAGED
  3500. NTSTATUS status;
  3501. PVERIFIER_DATA verifierData;
  3502. *LengthRead = 0;
  3503. switch (WhichSpace) {
  3504. default:
  3505. //
  3506. // Many people hand in the wrong WhichSpace parameters slap them around if we are verifing...
  3507. //
  3508. verifierData = PciVerifierRetrieveFailureData(PCI_VERIFIER_INVALID_WHICHSPACE);
  3509. ASSERT(verifierData);
  3510. VfFailDeviceNode(
  3511. PdoExtension->PhysicalDeviceObject,
  3512. PCI_VERIFIER_DETECTED_VIOLATION,
  3513. PCI_VERIFIER_INVALID_WHICHSPACE,
  3514. verifierData->FailureClass,
  3515. &verifierData->Flags,
  3516. verifierData->FailureText,
  3517. "%DevObj%Ulong",
  3518. PdoExtension->PhysicalDeviceObject,
  3519. WhichSpace
  3520. );
  3521. // fall through
  3522. case PCI_WHICHSPACE_CONFIG:
  3523. status = PciExternalReadDeviceConfig(
  3524. PdoExtension,
  3525. Buffer,
  3526. Offset,
  3527. Length
  3528. );
  3529. if(NT_SUCCESS(status)){
  3530. *LengthRead = Length;
  3531. }
  3532. break;
  3533. case PCI_WHICHSPACE_ROM:
  3534. //
  3535. // Read ROM.
  3536. //
  3537. *LengthRead = Length;
  3538. status = PciReadRomImage(
  3539. PdoExtension,
  3540. WhichSpace,
  3541. Buffer,
  3542. Offset,
  3543. LengthRead
  3544. );
  3545. break;
  3546. }
  3547. return status;
  3548. }
  3549. NTSTATUS
  3550. PciWriteDeviceSpace(
  3551. IN PPCI_PDO_EXTENSION PdoExtension,
  3552. IN ULONG WhichSpace,
  3553. IN PVOID Buffer,
  3554. IN ULONG Offset,
  3555. IN ULONG Length,
  3556. OUT PULONG LengthWritten
  3557. )
  3558. /*++
  3559. Routine Description:
  3560. This function handles reading from PCI device spaces and is called for both
  3561. the IRP_MN_WRITE_CONFIG and the BUS_INTERFACE_STANDARD.SetBusData cases.
  3562. Arguments:
  3563. PdoExtension - the PDO for the device we want to write to
  3564. WhichSpace - what type of space we want to write - of the form PCI_WHICHSPACE_*
  3565. Buffer - Supplies a pointer to where the data is to be written resides
  3566. Offset - Indicates the offset into the space where the writing should begin.
  3567. Length - Indicates the count of bytes which should be written.
  3568. LengthWritten - Indicates the count of bytes which was actually written.
  3569. Return Value:
  3570. Status
  3571. --*/
  3572. {
  3573. NTSTATUS status;
  3574. PVERIFIER_DATA verifierData;
  3575. *LengthWritten = 0;
  3576. //
  3577. // Any config space write could mean the resource requirements
  3578. // list or resource list we have cached are no longer valid.
  3579. //
  3580. PciInvalidateResourceInfoCache(PdoExtension);
  3581. switch (WhichSpace) {
  3582. default:
  3583. //
  3584. // Many people hand in the wrong WhichSpace parameters slap them around if we are verifing...
  3585. //
  3586. verifierData = PciVerifierRetrieveFailureData(PCI_VERIFIER_INVALID_WHICHSPACE);
  3587. ASSERT(verifierData);
  3588. VfFailDeviceNode(
  3589. PdoExtension->PhysicalDeviceObject,
  3590. PCI_VERIFIER_DETECTED_VIOLATION,
  3591. PCI_VERIFIER_INVALID_WHICHSPACE,
  3592. verifierData->FailureClass,
  3593. &verifierData->Flags,
  3594. verifierData->FailureText,
  3595. "%DevObj%Ulong",
  3596. PdoExtension->PhysicalDeviceObject,
  3597. WhichSpace
  3598. );
  3599. // fall through
  3600. case PCI_WHICHSPACE_CONFIG:
  3601. status = PciExternalWriteDeviceConfig(
  3602. PdoExtension,
  3603. Buffer,
  3604. Offset,
  3605. Length
  3606. );
  3607. if( NT_SUCCESS(status)){
  3608. *LengthWritten = Length;
  3609. }
  3610. break;
  3611. case PCI_WHICHSPACE_ROM:
  3612. //
  3613. // You can't write ROM
  3614. //
  3615. PciDebugPrint(
  3616. PciDbgAlways,
  3617. "PCI (%08x) WRITE_CONFIG IRP for ROM, failing.\n",
  3618. PdoExtension
  3619. );
  3620. status = STATUS_INVALID_DEVICE_REQUEST;
  3621. *LengthWritten = 0;
  3622. break;
  3623. }
  3624. return status;
  3625. }