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

2715 lines
85 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. Module Name:
  4. pmpcibus.c
  5. Abstract:
  6. Implements simplified PCI configuration
  7. read and write functions for use in
  8. an ACPI HAL.
  9. Author:
  10. Jake Oshins (jakeo) 1-Dec-1997
  11. Environment:
  12. Kernel mode only.
  13. Revision History:
  14. --*/
  15. #include "halp.h"
  16. #include "pci.h"
  17. #include "pcip.h"
  18. #include "cardbus.h"
  19. #define MAX(a, b) \
  20. ((a) > (b) ? (a) : (b))
  21. #define MIN(a, b) \
  22. ((a) < (b) ? (a) : (b))
  23. NTSTATUS
  24. HalpSearchForPciDebuggingDevice(
  25. IN OUT PDEBUG_DEVICE_DESCRIPTOR PciDevice,
  26. IN ULONG StartBusNumber,
  27. IN ULONG EndBusNumber,
  28. IN ULONG MinMem,
  29. IN ULONG MaxMem,
  30. IN USHORT MinIo,
  31. IN USHORT MaxIo,
  32. IN BOOLEAN ConfigureBridges
  33. );
  34. PCIPBUSDATA HalpFakePciBusData = {
  35. {
  36. PCI_DATA_TAG,//Tag
  37. PCI_DATA_VERSION,//Version
  38. (PciReadWriteConfig)HalpReadPCIConfig,//ReadConfig
  39. (PciReadWriteConfig) HalpWritePCIConfig,//WriteConfig
  40. (PciPin2Line)HalpPCIPin2ISALine,//Pin2Line
  41. (PciLine2Pin)HalpPCIISALine2Pin,//Line2Pin
  42. {0},//ParentSlot
  43. NULL,NULL,NULL,NULL//Reserved[4]
  44. },
  45. {0},//Config
  46. PCI_MAX_DEVICES,//MaxDevice
  47. };
  48. BUS_HANDLER HalpFakePciBusHandler = {
  49. BUS_HANDLER_VERSION,//Version
  50. PCIBus,//InterfaceType
  51. PCIConfiguration,//ConfigurationType
  52. 0,//BusNumber
  53. NULL,//DeviceObject
  54. NULL,//ParentHandler
  55. (PPCIBUSDATA)&HalpFakePciBusData,//BusData
  56. 0,//DeviceControlExtensionSize
  57. NULL,//BusAddresses
  58. {0},//Reserved[4]
  59. (PGETSETBUSDATA)HalpGetPCIData,//GetBusData
  60. (PGETSETBUSDATA)HalpSetPCIData,//SetBusData
  61. NULL,//AdjustResourceList
  62. (PASSIGNSLOTRESOURCES)HalpAssignPCISlotResources,//AssignSlotResources
  63. NULL,//GetInterruptVector
  64. NULL,//TranslateBusAddress
  65. };
  66. ULONG HalpMinPciBus = 0;
  67. ULONG HalpMaxPciBus = 0;
  68. PCI_TYPE1_CFG_CYCLE_BITS HalpPciDebuggingDevice[MAX_DEBUGGING_DEVICES_SUPPORTED] = {0};
  69. extern BOOLEAN HalpDoingCrashDump;
  70. PVOID
  71. HalpGetAcpiTablePhase0(
  72. IN PLOADER_PARAMETER_BLOCK LoaderBlock,
  73. IN ULONG Signature
  74. );
  75. VOID
  76. HalpFindFreeResourceLimits(
  77. IN ULONG Bus,
  78. IN OUT ULONG *MinIo,
  79. IN OUT ULONG *MaxIo,
  80. IN OUT ULONG *MinMem,
  81. IN OUT ULONG *MaxMem,
  82. IN OUT ULONG *MinBus,
  83. IN OUT ULONG *MaxBus
  84. );
  85. NTSTATUS
  86. HalpSetupUnconfiguredDebuggingDevice(
  87. IN ULONG Bus,
  88. IN ULONG Slot,
  89. IN ULONG IoMin,
  90. IN ULONG IoMax,
  91. IN ULONG MemMin,
  92. IN ULONG MemMax,
  93. IN OUT PDEBUG_DEVICE_DESCRIPTOR PciDevice
  94. );
  95. NTSTATUS
  96. HalpConfigurePciBridge(
  97. IN PDEBUG_DEVICE_DESCRIPTOR PciDevice,
  98. IN ULONG Bus,
  99. IN ULONG Slot,
  100. IN ULONG IoMin,
  101. IN ULONG IoMax,
  102. IN ULONG MemMin,
  103. IN ULONG MemMax,
  104. IN ULONG BusMin,
  105. IN ULONG BusMax,
  106. IN OUT PPCI_COMMON_CONFIG PciData
  107. );
  108. NTSTATUS
  109. HalpConfigureCardBusBridge(
  110. IN PDEBUG_DEVICE_DESCRIPTOR PciDevice,
  111. IN ULONG Bus,
  112. IN ULONG Slot,
  113. IN ULONG IoMin,
  114. IN ULONG IoMax,
  115. IN ULONG MemMin,
  116. IN ULONG MemMax,
  117. IN ULONG BusMin,
  118. IN ULONG BusMax,
  119. IN OUT PPCI_COMMON_CONFIG PciData
  120. );
  121. VOID
  122. HalpUnconfigurePciBridge(
  123. IN ULONG Bus,
  124. IN ULONG Slot
  125. );
  126. VOID
  127. HalpUnconfigureCardBusBridge(
  128. IN ULONG Bus,
  129. IN ULONG Slot
  130. );
  131. ULONG
  132. HalpKdStallExecution(
  133. ULONG LoopCount
  134. );
  135. #ifdef ALLOC_PRAGMA
  136. #pragma alloc_text(INIT,HalpInitializePciStubs)
  137. #pragma alloc_text(INIT,HalpRegisterKdSupportFunctions)
  138. #pragma alloc_text(INIT,HalpRegisterPciDebuggingDeviceInfo)
  139. #pragma alloc_text(PAGEKD,HalpConfigurePciBridge)
  140. #pragma alloc_text(PAGEKD,HalpConfigureCardBusBridge)
  141. #pragma alloc_text(PAGEKD,HalpFindFreeResourceLimits)
  142. #pragma alloc_text(PAGEKD,HalpPhase0GetPciDataByOffset)
  143. #pragma alloc_text(PAGEKD,HalpPhase0SetPciDataByOffset)
  144. #pragma alloc_text(PAGEKD,HalpReleasePciDeviceForDebugging)
  145. #pragma alloc_text(PAGEKD,HalpSearchForPciDebuggingDevice)
  146. #pragma alloc_text(PAGEKD,HalpSetupPciDeviceForDebugging)
  147. #pragma alloc_text(PAGEKD,HalpSetupUnconfiguredDebuggingDevice)
  148. #pragma alloc_text(PAGEKD,HalpUnconfigurePciBridge)
  149. #pragma alloc_text(PAGEKD,HalpUnconfigureCardBusBridge)
  150. #pragma alloc_text(PAGEKD,HalpKdStallExecution)
  151. #endif
  152. VOID
  153. HalpInitializePciStubs (
  154. VOID
  155. )
  156. {
  157. PPCI_REGISTRY_INFO_INTERNAL PCIRegInfo;
  158. PPCIPBUSDATA BusData;
  159. UCHAR iBuffer[PCI_COMMON_HDR_LENGTH];
  160. ULONG HwType;
  161. ULONG BusNo = 0;
  162. ULONG Bytes;
  163. ULONG Slot;
  164. PCI_COMMON_CONFIG PciData;
  165. PCIRegInfo = HalpQueryPciRegistryInfo();
  166. if (PCIRegInfo) {
  167. //
  168. // PCIRegInfo describes the system's PCI support as indicated by the BIOS.
  169. //
  170. HwType = PCIRegInfo->HardwareMechanism & 0xf;
  171. ExFreePool(PCIRegInfo);
  172. } else {
  173. //
  174. // no PCI bus information was gathered by NTDETECT,
  175. // assume type 1 access.
  176. //
  177. HwType = 1;
  178. }
  179. //
  180. // Initialize spinlock for synchronizing access to PCI space
  181. //
  182. KeInitializeSpinLock (&HalpPCIConfigLock);
  183. BusData = (PPCIPBUSDATA) HalpFakePciBusHandler.BusData;
  184. //
  185. // Set defaults
  186. //
  187. switch (HwType) {
  188. case 1:
  189. //
  190. // Initialize access port information for Type1 handlers
  191. //
  192. RtlCopyMemory (&PCIConfigHandler,
  193. &PCIConfigHandlerType1,
  194. sizeof (PCIConfigHandler));
  195. BusData->Config.Type1.Address = (PULONG)PCI_TYPE1_ADDR_PORT;
  196. BusData->Config.Type1.Data = PCI_TYPE1_DATA_PORT;
  197. break;
  198. case 2:
  199. //
  200. // Initialize access port information for Type2 handlers
  201. //
  202. RtlCopyMemory (&PCIConfigHandler,
  203. &PCIConfigHandlerType2,
  204. sizeof (PCIConfigHandler));
  205. BusData->Config.Type2.CSE = PCI_TYPE2_CSE_PORT;
  206. BusData->Config.Type2.Forward = PCI_TYPE2_FORWARD_PORT;
  207. BusData->Config.Type2.Base = PCI_TYPE2_ADDRESS_BASE;
  208. //
  209. // Early PCI machines didn't decode the last bit of
  210. // the device id. Shrink type 2 support max device.
  211. //
  212. BusData->MaxDevice = 0x10;
  213. break;
  214. default:
  215. // unsupport type
  216. DBGMSG ("HAL: Unkown PCI type\n");
  217. }
  218. //
  219. // Make a good guess about how many PCI busses are in the system
  220. // and initialize HalpMaxPciBus with this guess. When the PCI
  221. // driver starts, we will get a better answer here.
  222. //
  223. // Calling HaliPciInterfaceReadConfig has the side effect of
  224. // bumping up HalpMaxPciBus if it hits a populated device. So
  225. // the algorithm here is to just keep searching for devices until
  226. // we have searched 0x10 busses higher than the current maximum.
  227. //
  228. while (BusNo < 0x100) {
  229. //
  230. // Scan across this bus. As soon as we find a device,
  231. // then move on to another bus.
  232. //
  233. for (Slot = 0;
  234. Slot < PCI_MAX_DEVICES;
  235. Slot++) {
  236. Bytes = HaliPciInterfaceReadConfig(NULL,
  237. (UCHAR)BusNo,
  238. Slot,
  239. (PVOID)&PciData,
  240. 0,
  241. 4
  242. );
  243. if ((Bytes != 0) &&
  244. (PciData.VendorID != PCI_INVALID_VENDORID)) {
  245. //
  246. // This was a populated device. Bump up HalpMaxPciBus.
  247. //
  248. HalpMaxPciBus = BusNo;
  249. break;
  250. }
  251. }
  252. BusNo++;
  253. }
  254. }
  255. ULONG
  256. HaliPciInterfaceReadConfig(
  257. IN PVOID Context,
  258. IN UCHAR BusOffset,
  259. IN ULONG Slot,
  260. IN PVOID Buffer,
  261. IN ULONG Offset,
  262. IN ULONG Length
  263. )
  264. {
  265. PCI_SLOT_NUMBER slotNum;
  266. BUS_HANDLER busHand;
  267. UNREFERENCED_PARAMETER(Context);
  268. slotNum.u.AsULONG = Slot;
  269. //
  270. // Fake a bus handler.
  271. //
  272. RtlCopyMemory(&busHand, &HalpFakePciBusHandler, sizeof(BUS_HANDLER));
  273. //
  274. // Calculate the right bus number.
  275. //
  276. busHand.BusNumber = BusOffset;
  277. HalpReadPCIConfig(&busHand,
  278. slotNum,
  279. Buffer,
  280. Offset,
  281. Length
  282. );
  283. //
  284. // This is a hack. The legacy HAL interfaces need to be able
  285. // to distinguish between busses that exist and busses that
  286. // don't. And many users of the legacy interfaces implicitly
  287. // assume that PCI busses are tightly packed. (i.e. All busses
  288. // between the lowest numbered one and the highest numbered one
  289. // exist.) So here we are keeping track of the highest numbered
  290. // bus that we have seen so far.
  291. //
  292. if ((Length >= 2) &&
  293. (((PPCI_COMMON_CONFIG)Buffer)->VendorID != PCI_INVALID_VENDORID)) {
  294. //
  295. // This is a valid device.
  296. //
  297. if (busHand.BusNumber > HalpMaxPciBus) {
  298. //
  299. // This is the highest numbered bus we have
  300. // yet seen.
  301. //
  302. HalpMaxPciBus = busHand.BusNumber;
  303. }
  304. }
  305. return Length;
  306. }
  307. ULONG
  308. HaliPciInterfaceWriteConfig(
  309. IN PVOID Context,
  310. IN UCHAR BusOffset,
  311. IN ULONG Slot,
  312. IN PVOID Buffer,
  313. IN ULONG Offset,
  314. IN ULONG Length
  315. )
  316. {
  317. PCI_SLOT_NUMBER slotNum;
  318. BUS_HANDLER busHand;
  319. UNREFERENCED_PARAMETER(Context);
  320. slotNum.u.AsULONG = Slot;
  321. //
  322. // Fake a bus handler.
  323. //
  324. RtlCopyMemory(&busHand, &HalpFakePciBusHandler, sizeof(BUS_HANDLER));
  325. //
  326. // Calculate the right bus number.
  327. //
  328. busHand.BusNumber = BusOffset;
  329. HalpWritePCIConfig(&busHand,
  330. slotNum,
  331. Buffer,
  332. Offset,
  333. Length
  334. );
  335. return Length;
  336. }
  337. VOID
  338. HaliSetMaxLegacyPciBusNumber(
  339. IN ULONG BusNumber
  340. )
  341. /*++
  342. Routine Description:
  343. This routine bumps the Legacy PCI bus maximum up to whatever
  344. is passed in. This may be necessary because the ACPI driver
  345. needs to run a configuration cycle to a PCI device before the
  346. PCI driver loads. This happens mostly in the context of a
  347. _REG method.
  348. Arguments:
  349. BusNumber - max PCI bus number
  350. Return Value:
  351. none
  352. Notes:
  353. Caller is responsible for acquiring any necessary PCI config
  354. spinlocks.
  355. --*/
  356. {
  357. if (BusNumber > HalpMaxPciBus) {
  358. HalpMaxPciBus = BusNumber;
  359. }
  360. }
  361. ULONG
  362. HalpPhase0SetPciDataByOffset (
  363. ULONG BusNumber,
  364. ULONG SlotNumber,
  365. PVOID Buffer,
  366. ULONG Offset,
  367. ULONG Length
  368. )
  369. /*++
  370. Routine Description:
  371. This routine writes to PCI configuration space prior to bus handler
  372. installation.
  373. Arguments:
  374. BusNumber PCI Bus Number. This is the 8 bit BUS Number which is
  375. bits 23-16 of the Configuration Address. In support of
  376. multiple top level busses, the upper 24 bits of this
  377. argument will supply the index into the table of
  378. configuration address registers.
  379. SlotNumber PCI Slot Number, 8 bits composed of the 5 bit device
  380. number (bits 15-11 of the configuration address) and
  381. the 3 bit function number (10-8).
  382. Buffer Address of source data.
  383. Offset Number of bytes to skip from base of PCI config area.
  384. Length Number of bytes to write
  385. Return Value:
  386. Returns length of data written.
  387. Notes:
  388. Caller is responsible for acquiring any necessary PCI config
  389. spinlocks.
  390. --*/
  391. {
  392. PCI_TYPE1_CFG_BITS ConfigAddress;
  393. ULONG ReturnLength;
  394. PCI_SLOT_NUMBER slot;
  395. PUCHAR Bfr = (PUCHAR)Buffer;
  396. ASSERT(!(Offset & ~0xff));
  397. ASSERT(Length);
  398. ASSERT((Offset + Length) <= 256);
  399. if ( Length + Offset > 256 ) {
  400. if ( Offset > 256 ) {
  401. return 0;
  402. }
  403. Length = 256 - Offset;
  404. }
  405. ReturnLength = Length;
  406. slot.u.AsULONG = SlotNumber;
  407. ConfigAddress.u.bits.BusNumber = BusNumber;
  408. ConfigAddress.u.bits.DeviceNumber = slot.u.bits.DeviceNumber;
  409. ConfigAddress.u.bits.FunctionNumber = slot.u.bits.FunctionNumber;
  410. ConfigAddress.u.bits.RegisterNumber = (Offset & 0xfc) >> 2;
  411. ConfigAddress.u.bits.Enable = TRUE;
  412. if ( Offset & 0x3 ) {
  413. //
  414. // Access begins at a non-register boundary in the config
  415. // space. We need to read the register containing the data
  416. // and rewrite only the changed data. (I wonder if this
  417. // ever really happens?)
  418. //
  419. ULONG SubOffset = Offset & 0x3;
  420. ULONG SubLength = 4 - SubOffset;
  421. union {
  422. ULONG All;
  423. UCHAR Bytes[4];
  424. } Tmp;
  425. if ( SubLength > Length ) {
  426. SubLength = Length;
  427. }
  428. //
  429. // Adjust Length (remaining) and (new) Offset by amount covered
  430. // in this first word.
  431. //
  432. Length -= SubLength;
  433. Offset += SubLength;
  434. //
  435. // Get the first word (register), replace only those bytes that
  436. // need to be changed, then write the whole thing back out again.
  437. //
  438. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDR_PORT, ConfigAddress.u.AsULONG);
  439. Tmp.All = READ_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT);
  440. while ( SubLength-- ) {
  441. Tmp.Bytes[SubOffset++] = *Bfr++;
  442. }
  443. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT, Tmp.All);
  444. //
  445. // Aim ConfigAddressRegister at the next word (register).
  446. //
  447. ConfigAddress.u.bits.RegisterNumber++;
  448. }
  449. //
  450. // Do the majority of the transfer 4 bytes at a time.
  451. //
  452. while ( Length > sizeof(ULONG) ) {
  453. ULONG Tmp = *(UNALIGNED PULONG)Bfr;
  454. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDR_PORT, ConfigAddress.u.AsULONG);
  455. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT, Tmp);
  456. ConfigAddress.u.bits.RegisterNumber++;
  457. Bfr += sizeof(ULONG);
  458. Length -= sizeof(ULONG);
  459. }
  460. //
  461. // Do bytes in last register.
  462. //
  463. if ( Length ) {
  464. union {
  465. ULONG All;
  466. UCHAR Bytes[4];
  467. } Tmp;
  468. ULONG i = 0;
  469. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDR_PORT, ConfigAddress.u.AsULONG);
  470. Tmp.All = READ_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT);
  471. while ( Length-- ) {
  472. Tmp.Bytes[i++] = *(PUCHAR)Bfr++;
  473. }
  474. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT, Tmp.All);
  475. }
  476. return ReturnLength;
  477. }
  478. ULONG
  479. HalpPhase0GetPciDataByOffset (
  480. ULONG BusNumber,
  481. ULONG SlotNumber,
  482. PVOID Buffer,
  483. ULONG Offset,
  484. ULONG Length
  485. )
  486. /*++
  487. Routine Description:
  488. This routine reads PCI config space prior to bus handlder installation.
  489. Arguments:
  490. BusNumber PCI Bus Number. This is the 8 bit BUS Number which is
  491. bits 23-16 of the Configuration Address. In support of
  492. multiple top level busses, the upper 24 bits of this
  493. argument will supply the index into the table of
  494. configuration address registers.
  495. SlotNumber PCI Slot Number, 8 bits composed of the 5 bit device
  496. number (bits 15-11 of the configuration address) and
  497. the 3 bit function number (10-8).
  498. Buffer Address of source data.
  499. Offset Number of bytes to skip from base of PCI config area.
  500. Length Number of bytes to write
  501. Return Value:
  502. Amount of data read.
  503. --*/
  504. {
  505. PCI_TYPE1_CFG_BITS ConfigAddress;
  506. PCI_TYPE1_CFG_BITS ConfigAddressTemp;
  507. ULONG ReturnLength;
  508. ULONG i;
  509. PCI_SLOT_NUMBER slot;
  510. union {
  511. ULONG All;
  512. UCHAR Bytes[4];
  513. } Tmp;
  514. ASSERT(!(Offset & ~0xff));
  515. ASSERT(Length);
  516. ASSERT((Offset + Length) <= 256);
  517. if ( Length + Offset > 256 ) {
  518. if ( Offset > 256 ) {
  519. return 0;
  520. }
  521. Length = 256 - Offset;
  522. }
  523. ReturnLength = Length;
  524. slot.u.AsULONG = SlotNumber;
  525. ConfigAddress.u.bits.BusNumber = BusNumber;
  526. ConfigAddress.u.bits.DeviceNumber = slot.u.bits.DeviceNumber;
  527. ConfigAddress.u.bits.FunctionNumber = slot.u.bits.FunctionNumber;
  528. ConfigAddress.u.bits.RegisterNumber = (Offset & 0xfc) >> 2;
  529. ConfigAddress.u.bits.Enable = TRUE;
  530. //
  531. // If we are being asked to read data when function != 0, check
  532. // first to see if this device decares itself as a multi-function
  533. // device. If it doesn't, don't do this read.
  534. //
  535. if (ConfigAddress.u.bits.FunctionNumber != 0) {
  536. ConfigAddressTemp.u.bits.RegisterNumber = 3; // contains header type
  537. ConfigAddressTemp.u.bits.FunctionNumber = 0; // look at base package
  538. ConfigAddressTemp.u.bits.DeviceNumber = ConfigAddress.u.bits.DeviceNumber;
  539. ConfigAddressTemp.u.bits.BusNumber = ConfigAddress.u.bits.BusNumber;
  540. ConfigAddressTemp.u.bits.Enable = TRUE;
  541. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDR_PORT, ConfigAddressTemp.u.AsULONG);
  542. Tmp.All = READ_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT);
  543. if (!(Tmp.Bytes[2] & 0x80)) { // if the Header type field's multi-function bit is not set
  544. for (i = 0; i < Length; i++) {
  545. *((PUCHAR)Buffer)++ = 0xff; // Make this read as if the device isn't populated
  546. }
  547. return Length;
  548. }
  549. }
  550. i = Offset & 0x3;
  551. while ( Length ) {
  552. WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDR_PORT, ConfigAddress.u.AsULONG);
  553. Tmp.All = READ_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT);
  554. while ( (i < 4) && Length) {
  555. *((PUCHAR)Buffer)++ = Tmp.Bytes[i];
  556. i++;
  557. Length--;
  558. }
  559. i = 0;
  560. ConfigAddress.u.bits.RegisterNumber++;
  561. }
  562. return ReturnLength;
  563. }
  564. NTSTATUS
  565. HalpSetupPciDeviceForDebugging(
  566. IN PLOADER_PARAMETER_BLOCK LoaderBlock, OPTIONAL
  567. IN OUT PDEBUG_DEVICE_DESCRIPTOR PciDevice
  568. )
  569. /*++
  570. Routine Description:
  571. This routine finds and initializes a PCI device to be
  572. used for communicating with a debugger.
  573. The caller fills in as much of DEBUG_DEVICE_DESCRIPTOR
  574. as it cares to, filling unused fields with (-1).
  575. This routine attempts to find a matching PCI device. It
  576. matches first based on Bus and Slot, if the caller has
  577. provided them. Then it matches on VendorID/DeviceID, if
  578. the caller has provided them. Last, it matches on
  579. BaseClass/SubClass.
  580. This routine will fill in any unused fields in the structure
  581. so that the caller can know specifically which PCI
  582. device matched the criteria.
  583. If the matching PCI device is not enabled, or it is
  584. behind a PCI to PCI bridge that is not enabled, this
  585. routine makes a best-effort attempt to find a safe
  586. configuration that allows the device (and possibly bridges)
  587. to function, and enables them.
  588. If the PCI device implements memory mapped Base Address
  589. registers, this function will create a virtual to physical
  590. mapping for the memory ranges implied by the Base Address
  591. Registers and fill in the TranslatedAddress field with
  592. virtual pointers to the bases of the ranges. It will then
  593. fill in the Type field with CmResourceTypeMemory. And
  594. the Valid field with be TRUE.
  595. If the PCI device implements I/O port Base Address registers,
  596. this function will put the translated port address in
  597. TranslatedAddress, setting the Type field to CmResourceTypePort
  598. and the Valid field to TRUE.
  599. If the PCI device does not implement a specific Base Address
  600. Register, the Valid field will be FALSE.
  601. Arguments:
  602. PciDevice - Structure indicating the device
  603. Return Value:
  604. STATUS_SUCCESS if the device is configured and usable.
  605. STATUS_NO_MORE_MATCHES if no device matched the criteria.
  606. STATUS_INSUFFICIENT_RESOURCES if the memory requirements
  607. couldn't be met.
  608. STATUS_UNSUCCESSFUL if the routine failed for other reasons.
  609. --*/
  610. {
  611. NTSTATUS status;
  612. PCI_SLOT_NUMBER slot;
  613. ULONG i, j;
  614. ULONG maxPhys;
  615. status = HalpSearchForPciDebuggingDevice(
  616. PciDevice,
  617. 0,
  618. 0xff,
  619. 0x10000000,
  620. 0xfc000000,
  621. 0x1000,
  622. 0xffff,
  623. FALSE);
  624. if (!NT_SUCCESS(status)) {
  625. //
  626. // We didn't find the device using a conservative
  627. // search. Try a more invasive one.
  628. //
  629. status = HalpSearchForPciDebuggingDevice(
  630. PciDevice,
  631. 0,
  632. 0xff,
  633. 0x10000000,
  634. 0xfc000000,
  635. 0x1000,
  636. 0xffff,
  637. TRUE);
  638. }
  639. //
  640. // Record the Bus/Dev/Func so that we can stuff it in the
  641. // registry later.
  642. //
  643. if (NT_SUCCESS(status)) {
  644. if (PciDevice->Initialized) {
  645. //
  646. // Here we were just asked to reconfigure the bridges since
  647. // we already
  648. //
  649. return status;
  650. }
  651. slot.u.AsULONG = PciDevice->Slot;
  652. for (i = 0;
  653. i < MAX_DEBUGGING_DEVICES_SUPPORTED;
  654. i++) {
  655. if ((HalpPciDebuggingDevice[i].u.bits.Reserved1 == TRUE) &&
  656. (HalpPciDebuggingDevice[i].u.bits.FunctionNumber ==
  657. slot.u.bits.FunctionNumber) &&
  658. (HalpPciDebuggingDevice[i].u.bits.DeviceNumber ==
  659. slot.u.bits.DeviceNumber) &&
  660. (HalpPciDebuggingDevice[i].u.bits.BusNumber ==
  661. PciDevice->Bus)) {
  662. //
  663. // This device has already been set up for
  664. // debugging. Thus we should refuse to set
  665. // it up again.
  666. //
  667. return STATUS_UNSUCCESSFUL;
  668. }
  669. }
  670. for (i = 0;
  671. i < MAX_DEBUGGING_DEVICES_SUPPORTED;
  672. i++) {
  673. if (HalpPciDebuggingDevice[i].u.bits.Reserved1 == FALSE) {
  674. //
  675. // This slot is available.
  676. //
  677. HalpPciDebuggingDevice[i].u.bits.FunctionNumber =
  678. slot.u.bits.FunctionNumber;
  679. HalpPciDebuggingDevice[i].u.bits.DeviceNumber =
  680. slot.u.bits.DeviceNumber;
  681. HalpPciDebuggingDevice[i].u.bits.BusNumber = PciDevice->Bus;
  682. HalpPciDebuggingDevice[i].u.bits.Reserved1 = TRUE;
  683. PciDevice->Initialized = TRUE;
  684. break;
  685. }
  686. }
  687. //
  688. // Check to see if the caller wants any memory.
  689. //
  690. if (PciDevice->Memory.Length != 0) {
  691. if (!LoaderBlock) {
  692. return STATUS_INVALID_PARAMETER_1;
  693. }
  694. if (PciDevice->Memory.MaxEnd.QuadPart == 0) {
  695. PciDevice->Memory.MaxEnd.QuadPart = -1;
  696. }
  697. maxPhys = PciDevice->Memory.MaxEnd.HighPart ? 0xffffffff : PciDevice->Memory.MaxEnd.LowPart;
  698. maxPhys -= PciDevice->Memory.Length;
  699. //
  700. // The HAL APIs will always return page-aligned
  701. // memory. So ignore Aligned for now.
  702. //
  703. maxPhys = PtrToUlong(PAGE_ALIGN(maxPhys));
  704. maxPhys += ADDRESS_AND_SIZE_TO_SPAN_PAGES(maxPhys, PciDevice->Memory.Length);
  705. PciDevice->Memory.Start.HighPart = 0;
  706. PciDevice->Memory.Start.LowPart =
  707. HalpAllocPhysicalMemory(LoaderBlock,
  708. maxPhys,
  709. ADDRESS_AND_SIZE_TO_SPAN_PAGES(maxPhys, PciDevice->Memory.Length),
  710. FALSE);
  711. if (!PciDevice->Memory.Start.LowPart) {
  712. return STATUS_INSUFFICIENT_RESOURCES;
  713. }
  714. PciDevice->Memory.VirtualAddress =
  715. HalpMapPhysicalMemory64(PciDevice->Memory.Start,
  716. ADDRESS_AND_SIZE_TO_SPAN_PAGES(maxPhys, PciDevice->Memory.Length));
  717. }
  718. }
  719. return status;
  720. }
  721. VOID
  722. HalpFindFreeResourceLimits(
  723. IN ULONG Bus,
  724. IN OUT ULONG *MinIo,
  725. IN OUT ULONG *MaxIo,
  726. IN OUT ULONG *MinMem,
  727. IN OUT ULONG *MaxMem,
  728. IN OUT ULONG *MinBus,
  729. IN OUT ULONG *MaxBus
  730. )
  731. {
  732. UCHAR buffer[PCI_COMMON_HDR_LENGTH];
  733. PPCI_COMMON_CONFIG pciData;
  734. UCHAR bus, dev, func, bytesRead;
  735. PCI_SLOT_NUMBER pciSlot, targetSlot;
  736. ULONG newMinMem, newMaxMem;
  737. ULONG newMinIo, newMaxIo;
  738. ULONG newMinBus, newMaxBus;
  739. UCHAR barNo;
  740. pciData = (PPCI_COMMON_CONFIG)buffer;
  741. pciSlot.u.AsULONG = 0;
  742. newMinMem = *MinMem;
  743. newMaxMem = *MaxMem;
  744. newMinIo = *MinIo;
  745. newMaxIo = *MaxIo;
  746. newMinBus = *MinBus;
  747. newMaxBus = *MaxBus;
  748. for (dev = 0; dev < PCI_MAX_DEVICES; dev++) {
  749. for (func = 0; func < PCI_MAX_FUNCTION; func++) {
  750. pciSlot.u.bits.DeviceNumber = dev;
  751. pciSlot.u.bits.FunctionNumber = func;
  752. bytesRead = (UCHAR)HalpPhase0GetPciDataByOffset(Bus,
  753. pciSlot.u.AsULONG,
  754. pciData,
  755. 0,
  756. PCI_COMMON_HDR_LENGTH);
  757. if (bytesRead == 0) continue;
  758. if (pciData->VendorID != PCI_INVALID_VENDORID) {
  759. switch (PCI_CONFIGURATION_TYPE(pciData)) {
  760. case PCI_DEVICE_TYPE:
  761. //
  762. // While we scan across the bus, keep track
  763. // of the minimum decoder values that we've seen.
  764. // This will be used if we have to configure the
  765. // device. This relies on the fact that most BIOSes
  766. // assign addresses from the top down.
  767. //
  768. for (barNo = 0; barNo < PCI_TYPE0_ADDRESSES; barNo++) {
  769. if (pciData->u.type0.BaseAddresses[barNo] &
  770. PCI_ADDRESS_IO_SPACE) {
  771. if (pciData->u.type0.BaseAddresses[barNo] &
  772. PCI_ADDRESS_IO_ADDRESS_MASK) {
  773. //
  774. // This BAR is implemented
  775. //
  776. if ((pciData->u.type0.BaseAddresses[barNo] &
  777. PCI_ADDRESS_IO_ADDRESS_MASK) <
  778. ((newMaxIo / 2) + (newMinIo / 2))) {
  779. //
  780. // This BAR is at the bottom of the range.
  781. // Bump up the min.
  782. //
  783. newMinIo = (USHORT)MAX (newMinIo,
  784. (pciData->u.type0.BaseAddresses[barNo] &
  785. PCI_ADDRESS_IO_ADDRESS_MASK) + 0x100);
  786. } else {
  787. //
  788. // This BAR is not at the bottom of the range.
  789. // Bump down the max.
  790. //
  791. newMaxIo = (USHORT)MIN (newMaxIo,
  792. pciData->u.type0.BaseAddresses[barNo] &
  793. PCI_ADDRESS_IO_ADDRESS_MASK);
  794. }
  795. }
  796. } else {
  797. if (pciData->u.type0.BaseAddresses[barNo] &
  798. PCI_ADDRESS_MEMORY_ADDRESS_MASK) {
  799. //
  800. // The BAR is populated.
  801. //
  802. if ((pciData->u.type0.BaseAddresses[barNo] &
  803. PCI_ADDRESS_MEMORY_ADDRESS_MASK) <
  804. ((newMaxMem / 2) + (newMinMem / 2))) {
  805. //
  806. // This BAR is at the bottom of the range.
  807. // Bump up the min.
  808. //
  809. newMinMem = MAX (newMinMem,
  810. (pciData->u.type0.BaseAddresses[barNo] &
  811. PCI_ADDRESS_MEMORY_ADDRESS_MASK) + 0x10000);
  812. } else {
  813. //
  814. // This BAR is not at the bottom of the range.
  815. // Bump down the max.
  816. //
  817. newMaxMem = MIN (newMaxMem,
  818. (pciData->u.type0.BaseAddresses[barNo] &
  819. PCI_ADDRESS_MEMORY_ADDRESS_MASK));
  820. }
  821. }
  822. }
  823. }
  824. break;
  825. case PCI_CARDBUS_BRIDGE_TYPE:
  826. {
  827. ULONG bridgeMemMin = 0, bridgeMemMax = 0;
  828. USHORT bridgeIoMin, bridgeIoMax;
  829. ULONG LegacyBaseAddress;
  830. UCHAR bytesRead;
  831. bytesRead = (UCHAR) HalpPhase0GetPciDataByOffset(Bus,
  832. pciSlot.u.AsULONG,
  833. &LegacyBaseAddress,
  834. CARDBUS_LEGACY_MODE_BASE_ADDR,
  835. 4);
  836. if (bytesRead != 4) continue;
  837. if ((LegacyBaseAddress & ~1) &&
  838. (pciData->u.type2.SecondaryBus != 0) &&
  839. (pciData->u.type2.SubordinateBus !=0) &&
  840. (pciData->u.type2.Range[0].Base != 0) &&
  841. (pciData->u.type2.SocketRegistersBaseAddress != 0) &&
  842. (pciData->Command & PCI_ENABLE_MEMORY_SPACE) &&
  843. (pciData->Command & PCI_ENABLE_IO_SPACE)) {
  844. bridgeMemMin = pciData->u.type2.Range[0].Base;
  845. bridgeMemMax = pciData->u.type2.Range[0].Limit | 0xfff;
  846. bridgeIoMin = (USHORT)pciData->u.type2.Range[2].Base;
  847. bridgeIoMax = (USHORT)pciData->u.type2.Range[2].Limit | 0x3;
  848. //
  849. // Keep track of address space allocation.
  850. //
  851. if (bridgeIoMin > ((newMaxIo / 2) + (newMinIo / 2))) {
  852. newMaxIo = MIN(newMaxIo, bridgeIoMin);
  853. }
  854. if (bridgeIoMax < ((newMaxIo / 2) + (newMinIo / 2))) {
  855. newMinIo = MAX(newMinIo, bridgeIoMax) + 1;
  856. }
  857. if (bridgeMemMin > ((newMaxMem / 2) + (newMinMem / 2))) {
  858. newMaxMem = MIN(newMaxMem, bridgeMemMin);
  859. }
  860. if (bridgeMemMax < ((newMaxMem / 2) + (newMinMem / 2))) {
  861. newMinMem = MAX(newMinMem, bridgeMemMax) + 1;
  862. }
  863. //
  864. // Keep track of bus numbers.
  865. //
  866. if (pciData->u.type2.PrimaryBus > ((newMaxBus / 2) + (newMinBus / 2))) {
  867. newMaxBus = MIN(newMaxBus, pciData->u.type2.PrimaryBus);
  868. }
  869. if (pciData->u.type2.SubordinateBus < ((newMaxBus / 2) + (newMinBus / 2))) {
  870. newMinBus = MAX(newMinBus, pciData->u.type2.SubordinateBus) + 1;
  871. }
  872. }
  873. }
  874. break;
  875. case PCI_BRIDGE_TYPE:
  876. {
  877. ULONG bridgeMemMin = 0, bridgeMemMax = 0;
  878. USHORT bridgeIoMin, bridgeIoMax;
  879. if ((pciData->u.type1.SecondaryBus != 0) &&
  880. (pciData->u.type1.SubordinateBus !=0) &&
  881. (pciData->Command & PCI_ENABLE_MEMORY_SPACE) &&
  882. (pciData->Command & PCI_ENABLE_IO_SPACE)) {
  883. bridgeMemMin = PciBridgeMemory2Base(pciData->u.type1.MemoryBase);
  884. bridgeMemMax = PciBridgeMemory2Limit(pciData->u.type1.MemoryLimit);
  885. bridgeIoMin = (USHORT)PciBridgeIO2Base(pciData->u.type1.IOBase, 0);
  886. bridgeIoMax = (USHORT)PciBridgeIO2Limit(pciData->u.type1.IOLimit, 0);
  887. //
  888. // Keep track of address space allocation.
  889. //
  890. if (bridgeIoMin > ((newMaxIo / 2) + (newMinIo / 2))) {
  891. newMaxIo = MIN(newMaxIo, bridgeIoMin);
  892. }
  893. if (bridgeIoMax < ((newMaxIo / 2) + (newMinIo / 2))) {
  894. newMinIo = MAX(newMinIo, bridgeIoMax) + 1;
  895. }
  896. if (bridgeMemMin > ((newMaxMem / 2) + (newMinMem / 2))) {
  897. newMaxMem = MIN(newMaxMem, bridgeMemMin);
  898. }
  899. if (bridgeMemMax < ((newMaxMem / 2) + (newMinMem / 2))) {
  900. newMinMem = MAX(newMinMem, bridgeMemMax) + 1;
  901. }
  902. //
  903. // Keep track of bus numbers.
  904. //
  905. if (pciData->u.type1.PrimaryBus > ((newMaxBus / 2) + (newMinBus / 2))) {
  906. newMaxBus = MIN(newMaxBus, pciData->u.type1.PrimaryBus);
  907. }
  908. if (pciData->u.type1.SubordinateBus < ((newMaxBus / 2) + (newMinBus / 2))) {
  909. newMinBus = MAX(newMinBus, pciData->u.type1.SubordinateBus) + 1;
  910. }
  911. }
  912. break;
  913. default:
  914. break;
  915. }
  916. }
  917. if (!PCI_MULTIFUNCTION_DEVICE(pciData) &&
  918. (func == 0)) {
  919. break;
  920. }
  921. }
  922. }
  923. }
  924. *MinMem = newMinMem;
  925. *MaxMem = newMaxMem;
  926. *MinIo = newMinIo;
  927. *MaxIo = newMaxIo;
  928. *MinBus = newMinBus;
  929. *MaxBus = newMaxBus;
  930. }
  931. NTSTATUS
  932. HalpSetupUnconfiguredDebuggingDevice(
  933. IN ULONG Bus,
  934. IN ULONG Slot,
  935. IN ULONG IoMin,
  936. IN ULONG IoMax,
  937. IN ULONG MemMin,
  938. IN ULONG MemMax,
  939. IN OUT PDEBUG_DEVICE_DESCRIPTOR PciDevice
  940. )
  941. {
  942. UCHAR buffer[PCI_COMMON_HDR_LENGTH];
  943. PPCI_COMMON_CONFIG pciData;
  944. ULONG barLength, bytesRead;
  945. ULONG barContents = 0;
  946. PHYSICAL_ADDRESS physicalAddress;
  947. PCI_SLOT_NUMBER pciSlot;
  948. UCHAR barNo;
  949. UCHAR capPtr;
  950. PCI_PM_CAPABILITY pmCap;
  951. //
  952. // Make sure the device is in D0
  953. //
  954. bytesRead = HalpPhase0GetPciDataByOffset(Bus, Slot, &capPtr,
  955. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type0.CapabilitiesPtr), 1);
  956. if ((bytesRead == 1) && capPtr) {
  957. bytesRead = HalpPhase0GetPciDataByOffset(Bus, Slot, &pmCap, capPtr, sizeof(pmCap));
  958. if ((bytesRead == sizeof(pmCap)) &&
  959. (pmCap.Header.CapabilityID == PCI_CAPABILITY_ID_POWER_MANAGEMENT) &&
  960. (pmCap.PMCSR.ControlStatus.PowerState != 0)) {
  961. pmCap.PMCSR.ControlStatus.PowerState = 0; // set to D0
  962. HalpPhase0SetPciDataByOffset(Bus,
  963. Slot,
  964. &pmCap.PMCSR.ControlStatus,
  965. capPtr + FIELD_OFFSET(PCI_PM_CAPABILITY,PMCSR.ControlStatus),
  966. sizeof(pmCap.PMCSR.ControlStatus));
  967. HalpKdStallExecution(2000);
  968. }
  969. }
  970. pciSlot.u.AsULONG = Slot;
  971. pciData = (PPCI_COMMON_CONFIG)buffer;
  972. bytesRead = HalpPhase0GetPciDataByOffset(Bus,
  973. pciSlot.u.AsULONG,
  974. pciData,
  975. 0,
  976. PCI_COMMON_HDR_LENGTH);
  977. ASSERT(bytesRead != 0);
  978. PciDevice->Bus = Bus;
  979. PciDevice->Slot = pciSlot.u.AsULONG;
  980. PciDevice->VendorID = pciData->VendorID;
  981. PciDevice->DeviceID = pciData->DeviceID;
  982. PciDevice->BaseClass = pciData->BaseClass;
  983. PciDevice->SubClass = pciData->SubClass;
  984. //DbgPrint("Configuring device between %x - %x\n",
  985. // MemMin, MemMax);
  986. //
  987. // Cycle through the BARs, turning them on if necessary,
  988. // and mapping them.
  989. //
  990. for (barNo = 0; barNo < PCI_TYPE0_ADDRESSES; barNo++) {
  991. barContents = 0xffffffff;
  992. PciDevice->BaseAddress[barNo].Valid = FALSE;
  993. HalpPhase0SetPciDataByOffset(Bus,
  994. pciSlot.u.AsULONG,
  995. &barContents,
  996. 0x10 + (4 * barNo),
  997. 4);
  998. HalpPhase0GetPciDataByOffset(Bus,
  999. pciSlot.u.AsULONG,
  1000. &barContents,
  1001. 0x10 + (4 * barNo),
  1002. 4);
  1003. if (pciData->u.type0.BaseAddresses[barNo] &
  1004. PCI_ADDRESS_IO_SPACE) {
  1005. //
  1006. // This is an I/O BAR.
  1007. //
  1008. barLength = (((USHORT)barContents & PCI_ADDRESS_IO_ADDRESS_MASK) - 1) ^
  1009. 0xffff;
  1010. if (!(pciData->u.type0.BaseAddresses[barNo] &
  1011. PCI_ADDRESS_IO_ADDRESS_MASK)) {
  1012. //
  1013. // And it's empty.
  1014. //
  1015. //
  1016. // Try to fit this I/O window half-way between the min and the max.
  1017. //
  1018. if ((ULONG)(IoMax - IoMin) >= (barLength * 3)) {
  1019. //
  1020. // There is plenty of room, make a safe guess. Try
  1021. // to put it half-way between the upper and lower
  1022. // bounds, rounding up to the next natural alignment.
  1023. //
  1024. pciData->u.type0.BaseAddresses[barNo] =
  1025. (((IoMax / 2) + (IoMin / 2)) + barLength) & (barLength -1);
  1026. } else if (barLength >= (IoMax -
  1027. ((IoMin & (barLength -1)) ?
  1028. ((IoMin + barLength) & (barLength -1)) :
  1029. IoMin))) {
  1030. //
  1031. // Space is tight, make a not-so-safe guess. Try
  1032. // to put it at the bottom of the range, rounded
  1033. // up the the next natural alignment.
  1034. //
  1035. pciData->u.type0.BaseAddresses[barNo] =
  1036. ((IoMin & (barLength -1)) ?
  1037. ((IoMin + barLength) & (barLength -1)) :
  1038. IoMin);
  1039. }
  1040. IoMin = (USHORT)pciData->u.type0.BaseAddresses[barNo];
  1041. }
  1042. pciData->Command |= PCI_ENABLE_IO_SPACE;
  1043. PciDevice->BaseAddress[barNo].Type = CmResourceTypePort;
  1044. PciDevice->BaseAddress[barNo].Valid = TRUE;
  1045. PciDevice->BaseAddress[barNo].TranslatedAddress =
  1046. (PUCHAR)(ULONG_PTR)(pciData->u.type0.BaseAddresses[barNo] &
  1047. PCI_ADDRESS_IO_ADDRESS_MASK);
  1048. PciDevice->BaseAddress[barNo].Length = barLength;
  1049. } else {
  1050. //
  1051. // This is a memory BAR.
  1052. //
  1053. barLength = ((barContents & PCI_ADDRESS_MEMORY_ADDRESS_MASK) - 1) ^
  1054. 0xffffffff;
  1055. if (!(pciData->u.type0.BaseAddresses[barNo] &
  1056. PCI_ADDRESS_MEMORY_ADDRESS_MASK)) {
  1057. //
  1058. // And it's empty.
  1059. //
  1060. if (barLength == 0) continue;
  1061. //
  1062. // Try to fit this memory window half-way between the min and the max.
  1063. //
  1064. if ((ULONG)(MemMax - MemMin) >= (barLength * 3)) {
  1065. //
  1066. // There is plenty of room, make a safe guess. Try
  1067. // to put it half-way between the upper and lower
  1068. // bounds, rounding up to the next natural alignment.
  1069. //
  1070. pciData->u.type0.BaseAddresses[barNo] =
  1071. (ULONG)(((MemMax / 2) + (MemMin / 2))
  1072. + barLength) & ~(barLength -1);
  1073. } else if (barLength >= (ULONG)(MemMax -
  1074. ((MemMin & ~(barLength -1)) ?
  1075. ((MemMin + barLength) & ~(barLength -1)) :
  1076. MemMin))) {
  1077. //
  1078. // Space is tight, make a not-so-safe guess. Try
  1079. // to put it at the bottom of the range, rounded
  1080. // up the the next natural alignment.
  1081. //
  1082. pciData->u.type0.BaseAddresses[barNo] =
  1083. (ULONG)((MemMin & ~(barLength -1)) ?
  1084. ((MemMin + barLength) & ~(barLength -1)) :
  1085. MemMin);
  1086. }
  1087. MemMin = pciData->u.type0.BaseAddresses[barNo] &
  1088. PCI_ADDRESS_MEMORY_ADDRESS_MASK;
  1089. }
  1090. pciData->Command |= PCI_ENABLE_MEMORY_SPACE;
  1091. physicalAddress.HighPart = 0;
  1092. physicalAddress.LowPart = pciData->u.type0.BaseAddresses[barNo]
  1093. & PCI_ADDRESS_MEMORY_ADDRESS_MASK;
  1094. PciDevice->BaseAddress[barNo].Type = CmResourceTypeMemory;
  1095. PciDevice->BaseAddress[barNo].Valid = TRUE;
  1096. PciDevice->BaseAddress[barNo].TranslatedAddress =
  1097. HalpMapPhysicalMemory64(physicalAddress,
  1098. ADDRESS_AND_SIZE_TO_SPAN_PAGES(physicalAddress.LowPart, barLength));
  1099. PciDevice->BaseAddress[barNo].Length = barLength;
  1100. }
  1101. }
  1102. pciData->Command |= PCI_ENABLE_BUS_MASTER;
  1103. //
  1104. // Write back any changes we made.
  1105. //
  1106. HalpPhase0SetPciDataByOffset(Bus,
  1107. pciSlot.u.AsULONG,
  1108. pciData,
  1109. 0,
  1110. 0x40);
  1111. return STATUS_SUCCESS;
  1112. }
  1113. NTSTATUS
  1114. HalpSearchForPciDebuggingDevice(
  1115. IN OUT PDEBUG_DEVICE_DESCRIPTOR PciDevice,
  1116. IN ULONG StartBusNumber,
  1117. IN ULONG EndBusNumber,
  1118. IN ULONG MinMem,
  1119. IN ULONG MaxMem,
  1120. IN USHORT MinIo,
  1121. IN USHORT MaxIo,
  1122. IN BOOLEAN ConfigureBridges
  1123. )
  1124. /*++
  1125. Routine Description:
  1126. This routine is a helper function for
  1127. HalpSetupPciDeviceForDebugging.
  1128. Arguments:
  1129. PciDevice - Structure indicating the device
  1130. Return Value:
  1131. STATUS_SUCCESS if the device is configured and usable.
  1132. STATUS_NO_MORE_MATCHES if no device matched the criteria.
  1133. STATUS_UNSUCCESSFUL if the routine fails for other reasons.
  1134. --*/
  1135. #define TARGET_DEVICE_NOT_FOUND 0x10000
  1136. {
  1137. NTSTATUS status;
  1138. UCHAR buffer[PCI_COMMON_HDR_LENGTH];
  1139. PPCI_COMMON_CONFIG pciData;
  1140. UCHAR bus, dev, func, bytesRead;
  1141. PCI_SLOT_NUMBER pciSlot, targetSlot;
  1142. ULONG newMinMem, newMaxMem;
  1143. ULONG newMinIo, newMaxIo;
  1144. ULONG newMinBus, newMaxBus;
  1145. UCHAR barNo;
  1146. BOOLEAN unconfigureBridge;
  1147. pciData = (PPCI_COMMON_CONFIG)buffer;
  1148. pciSlot.u.AsULONG = 0;
  1149. newMinMem = MinMem;
  1150. newMaxMem = MaxMem;
  1151. newMinIo = MinIo;
  1152. newMaxIo = MaxIo;
  1153. newMinBus = StartBusNumber;
  1154. newMaxBus = EndBusNumber;
  1155. bus = (UCHAR)StartBusNumber;
  1156. //DbgPrint("HalpSearchForPciDebuggingDevice:\n"
  1157. // "\tMem: %x-%x\n"
  1158. // "\tI/O: %x-%x\n"
  1159. // "\tBus: %x-%x\n"
  1160. // "\t%s Configuring Bridges\n",
  1161. // MinMem, MaxMem,
  1162. // MinIo, MaxIo,
  1163. // StartBusNumber, EndBusNumber,
  1164. // ConfigureBridges ? "" : "Not");
  1165. //
  1166. // This bit stays set to 1 until we find the device.
  1167. //
  1168. targetSlot.u.bits.Reserved = TARGET_DEVICE_NOT_FOUND;
  1169. while (TRUE) {
  1170. UCHAR nextBus;
  1171. nextBus = bus + 1;
  1172. HalpFindFreeResourceLimits(bus,
  1173. &newMinIo,
  1174. &newMaxIo,
  1175. &newMinMem,
  1176. &newMaxMem,
  1177. &newMinBus,
  1178. &newMaxBus
  1179. );
  1180. for (dev = 0; dev < PCI_MAX_DEVICES; dev++) {
  1181. for (func = 0; func < PCI_MAX_FUNCTION; func++) {
  1182. pciSlot.u.bits.DeviceNumber = dev;
  1183. pciSlot.u.bits.FunctionNumber = func;
  1184. bytesRead = (UCHAR)HalpPhase0GetPciDataByOffset(bus,
  1185. pciSlot.u.AsULONG,
  1186. pciData,
  1187. 0,
  1188. PCI_COMMON_HDR_LENGTH);
  1189. if (bytesRead == 0) continue;
  1190. if (pciData->VendorID != PCI_INVALID_VENDORID) {
  1191. //DbgPrint("%04x:%04x - %x/%x/%x - \tSlot: %x\n",
  1192. // pciData->VendorID,
  1193. // pciData->DeviceID,
  1194. // pciData->BaseClass,
  1195. // pciData->SubClass,
  1196. // pciData->ProgIf,
  1197. // pciSlot.u.AsULONG);
  1198. switch (PCI_CONFIGURATION_TYPE(pciData)) {
  1199. case PCI_DEVICE_TYPE:
  1200. //
  1201. // Match first on Bus/Dev/Func
  1202. //
  1203. if ((PciDevice->Bus == bus) &&
  1204. (PciDevice->Slot == pciSlot.u.AsULONG)) {
  1205. //DbgPrint("\n\nMatched on Bus/Slot\n\n");
  1206. return HalpSetupUnconfiguredDebuggingDevice(
  1207. bus,
  1208. pciSlot.u.AsULONG,
  1209. newMinIo,
  1210. newMaxIo,
  1211. newMinMem,
  1212. newMaxMem,
  1213. PciDevice
  1214. );
  1215. }
  1216. if ((PciDevice->Bus == MAXULONG) &&
  1217. (PciDevice->Slot == MAXULONG)) {
  1218. //
  1219. // Bus and Slot weren't specified. Match
  1220. // on VID/DID.
  1221. //
  1222. if ((pciData->VendorID == PciDevice->VendorID) &&
  1223. (pciData->DeviceID == PciDevice->DeviceID)) {
  1224. //DbgPrint("\n\nMatched on Vend/Dev\n\n");
  1225. return HalpSetupUnconfiguredDebuggingDevice(
  1226. bus,
  1227. pciSlot.u.AsULONG,
  1228. newMinIo,
  1229. newMaxIo,
  1230. newMinMem,
  1231. newMaxMem,
  1232. PciDevice
  1233. );
  1234. }
  1235. if ((PciDevice->VendorID == MAXUSHORT) &&
  1236. (PciDevice->DeviceID == MAXUSHORT)) {
  1237. //
  1238. // VID/DID weren't specified. Match
  1239. // on class codes.
  1240. //
  1241. if ((pciData->BaseClass == PciDevice->BaseClass) &&
  1242. (pciData->SubClass == PciDevice->SubClass)) {
  1243. //DbgPrint("\n\nMatched on Base/Sub\n\n");
  1244. //
  1245. // Further match on Programming Interface,
  1246. // if specified.
  1247. //
  1248. if ((PciDevice->ProgIf != MAXUCHAR) &&
  1249. (PciDevice->ProgIf != pciData->ProgIf)) {
  1250. break;
  1251. }
  1252. //DbgPrint("\n\nMatched on programming interface\n\n");
  1253. return HalpSetupUnconfiguredDebuggingDevice(
  1254. bus,
  1255. pciSlot.u.AsULONG,
  1256. newMinIo,
  1257. newMaxIo,
  1258. newMinMem,
  1259. newMaxMem,
  1260. PciDevice
  1261. );
  1262. }
  1263. }
  1264. }
  1265. break;
  1266. case PCI_CARDBUS_BRIDGE_TYPE:
  1267. {
  1268. ULONG bridgeMemMin = 0, bridgeMemMax = 0;
  1269. USHORT bridgeIoMin, bridgeIoMax;
  1270. ULONG LegacyBaseAddress;
  1271. unconfigureBridge = FALSE;
  1272. //DbgPrint("Found a CardBus bridge\n");
  1273. HalpPhase0GetPciDataByOffset(bus,
  1274. pciSlot.u.AsULONG,
  1275. &LegacyBaseAddress,
  1276. CARDBUS_LEGACY_MODE_BASE_ADDR,
  1277. 4);
  1278. if (!(((LegacyBaseAddress & ~1) == 0) &&
  1279. (pciData->u.type2.SecondaryBus != 0) &&
  1280. (pciData->u.type2.SubordinateBus !=0) &&
  1281. (pciData->u.type2.Range[0].Base != 0) &&
  1282. (pciData->u.type2.SocketRegistersBaseAddress != 0) &&
  1283. (pciData->Command & PCI_ENABLE_MEMORY_SPACE) &&
  1284. (pciData->Command & PCI_ENABLE_IO_SPACE))) {
  1285. //
  1286. // The bridge is unconfigured.
  1287. //
  1288. if (ConfigureBridges){
  1289. //
  1290. // We should configure it now.
  1291. //
  1292. status = HalpConfigureCardBusBridge(
  1293. PciDevice,
  1294. bus,
  1295. pciSlot.u.AsULONG,
  1296. newMinIo,
  1297. newMaxIo,
  1298. newMinMem,
  1299. newMaxMem,
  1300. MAX((UCHAR)newMinBus, (bus + 1)),
  1301. newMaxBus,
  1302. pciData
  1303. );
  1304. if (!NT_SUCCESS(status)) {
  1305. break;
  1306. }
  1307. unconfigureBridge = TRUE;
  1308. } else {
  1309. //
  1310. // We aren't configuring bridges
  1311. // on this pass.
  1312. //
  1313. break;
  1314. }
  1315. }
  1316. bridgeMemMin = pciData->u.type2.Range[0].Base;
  1317. bridgeMemMax = pciData->u.type2.Range[0].Limit | 0xfff;
  1318. bridgeIoMin = (USHORT)pciData->u.type2.Range[2].Base;
  1319. bridgeIoMax = (USHORT)pciData->u.type2.Range[2].Limit | 0x3;
  1320. //DbgPrint("Configured: I/O %x-%x Mem %x-%x\n",
  1321. // bridgeIoMin, bridgeIoMax,
  1322. // bridgeMemMin, bridgeMemMax);
  1323. //
  1324. // Recurse.
  1325. //
  1326. status = HalpSearchForPciDebuggingDevice(
  1327. PciDevice,
  1328. (ULONG)pciData->u.type2.SecondaryBus,
  1329. (ULONG)pciData->u.type2.SubordinateBus,
  1330. bridgeMemMin,
  1331. bridgeMemMax,
  1332. bridgeIoMin,
  1333. bridgeIoMax,
  1334. ConfigureBridges);
  1335. if (NT_SUCCESS(status)) {
  1336. return status;
  1337. }
  1338. if (!unconfigureBridge) {
  1339. //
  1340. // Bump up the bus number so that we don't
  1341. // scan down the busses we just recursed into.
  1342. //
  1343. nextBus = pciData->u.type2.SubordinateBus + 1;
  1344. } else {
  1345. HalpUnconfigureCardBusBridge(bus,
  1346. pciSlot.u.AsULONG);
  1347. }
  1348. }
  1349. break;
  1350. case PCI_BRIDGE_TYPE:
  1351. {
  1352. ULONG bridgeMemMin = 0, bridgeMemMax = 0;
  1353. USHORT bridgeIoMin, bridgeIoMax;
  1354. unconfigureBridge = FALSE;
  1355. //DbgPrint("Found a PCI to PCI bridge\n");
  1356. if (!((pciData->u.type1.SecondaryBus != 0) &&
  1357. (pciData->u.type1.SubordinateBus !=0) &&
  1358. (pciData->Command & PCI_ENABLE_MEMORY_SPACE) &&
  1359. (pciData->Command & PCI_ENABLE_IO_SPACE))) {
  1360. //
  1361. // The bridge is unconfigured.
  1362. //
  1363. if (ConfigureBridges){
  1364. //
  1365. // We should configure it now.
  1366. //
  1367. status = HalpConfigurePciBridge(
  1368. PciDevice,
  1369. bus,
  1370. pciSlot.u.AsULONG,
  1371. newMinIo,
  1372. newMaxIo,
  1373. newMinMem,
  1374. newMaxMem,
  1375. MAX((UCHAR)newMinBus, (bus + 1)),
  1376. newMaxBus,
  1377. pciData
  1378. );
  1379. if (!NT_SUCCESS(status)) {
  1380. break;
  1381. }
  1382. unconfigureBridge = TRUE;
  1383. } else {
  1384. //
  1385. // We aren't configuring bridges
  1386. // on this pass.
  1387. //
  1388. break;
  1389. }
  1390. }
  1391. bridgeMemMin = PciBridgeMemory2Base(pciData->u.type1.MemoryBase);
  1392. bridgeMemMax = PciBridgeMemory2Limit(pciData->u.type1.MemoryLimit);
  1393. bridgeIoMin = (USHORT)PciBridgeIO2Base(pciData->u.type1.IOBase, 0);
  1394. bridgeIoMax = (USHORT)PciBridgeIO2Limit(pciData->u.type1.IOLimit, 0);
  1395. //DbgPrint("Configured: I/O %x-%x Mem %x-%x\n",
  1396. // bridgeIoMin, bridgeIoMax,
  1397. // bridgeMemMin, bridgeMemMax);
  1398. //
  1399. // Recurse.
  1400. //
  1401. status = HalpSearchForPciDebuggingDevice(
  1402. PciDevice,
  1403. (ULONG)pciData->u.type1.SecondaryBus,
  1404. (ULONG)pciData->u.type1.SubordinateBus,
  1405. bridgeMemMin,
  1406. bridgeMemMax,
  1407. bridgeIoMin,
  1408. bridgeIoMax,
  1409. ConfigureBridges);
  1410. if (NT_SUCCESS(status)) {
  1411. return status;
  1412. }
  1413. if (!unconfigureBridge) {
  1414. //
  1415. // Bump up the bus number so that we don't
  1416. // scan down the busses we just recursed into.
  1417. //
  1418. nextBus = pciData->u.type1.SubordinateBus + 1;
  1419. } else {
  1420. HalpUnconfigurePciBridge(bus,
  1421. pciSlot.u.AsULONG);
  1422. }
  1423. }
  1424. break;
  1425. default:
  1426. break;
  1427. }
  1428. }
  1429. if (!PCI_MULTIFUNCTION_DEVICE(pciData) &&
  1430. (func == 0)) {
  1431. break;
  1432. }
  1433. }
  1434. }
  1435. if (nextBus >= EndBusNumber) {
  1436. break;
  1437. }
  1438. bus = nextBus;
  1439. }
  1440. return STATUS_NOT_FOUND;
  1441. }
  1442. NTSTATUS
  1443. HalpReleasePciDeviceForDebugging(
  1444. IN OUT PDEBUG_DEVICE_DESCRIPTOR PciDevice
  1445. )
  1446. /*++
  1447. Routine Description:
  1448. This routine de-allocates any resources acquired in
  1449. HalpSetupPciDeviceForDebugging.
  1450. Arguments:
  1451. PciDevice - Structure indicating the device
  1452. Return Value:
  1453. --*/
  1454. {
  1455. ULONG i;
  1456. for (i = 0; i < PCI_TYPE0_ADDRESSES; i++) {
  1457. if (PciDevice->BaseAddress[i].Valid &&
  1458. PciDevice->BaseAddress[i].Type == CmResourceTypeMemory) {
  1459. PciDevice->BaseAddress[i].Valid = FALSE;
  1460. HalpUnmapVirtualAddress(PciDevice->BaseAddress[i].TranslatedAddress,
  1461. ADDRESS_AND_SIZE_TO_SPAN_PAGES(
  1462. PciDevice->BaseAddress[i].TranslatedAddress,
  1463. PciDevice->BaseAddress[i].Length));
  1464. }
  1465. }
  1466. return STATUS_SUCCESS;
  1467. }
  1468. VOID
  1469. HalpRegisterKdSupportFunctions(
  1470. IN PLOADER_PARAMETER_BLOCK LoaderBlock
  1471. )
  1472. /*++
  1473. Routine Description:
  1474. This routine fills in the HalPrivateDispatchTable
  1475. with the functions needed for debugging through
  1476. PCI devices.
  1477. Arguments:
  1478. LoaderBlock - The Loader Block
  1479. Return Value:
  1480. --*/
  1481. {
  1482. KdSetupPciDeviceForDebugging = HalpSetupPciDeviceForDebugging;
  1483. KdReleasePciDeviceForDebugging = HalpReleasePciDeviceForDebugging;
  1484. #ifdef ACPI_HAL
  1485. KdGetAcpiTablePhase0 = HalpGetAcpiTablePhase0;
  1486. #endif
  1487. KdCheckPowerButton = HalpCheckPowerButton;
  1488. KdMapPhysicalMemory64 = HalpMapPhysicalMemory64;
  1489. KdUnmapVirtualAddress = HalpUnmapVirtualAddress;
  1490. }
  1491. VOID
  1492. HalpRegisterPciDebuggingDeviceInfo(
  1493. VOID
  1494. )
  1495. {
  1496. OBJECT_ATTRIBUTES ObjectAttributes;
  1497. UNICODE_STRING UnicodeString;
  1498. HANDLE BaseHandle = NULL;
  1499. HANDLE Handle = NULL;
  1500. ULONG disposition;
  1501. ULONG bus;
  1502. UCHAR i;
  1503. PCI_SLOT_NUMBER slot;
  1504. NTSTATUS status;
  1505. BOOLEAN debuggerFound = FALSE;
  1506. PAGED_CODE();
  1507. for (i = 0;
  1508. i < MAX_DEBUGGING_DEVICES_SUPPORTED;
  1509. i++) {
  1510. if (HalpPciDebuggingDevice[i].u.bits.Reserved1 == TRUE) {
  1511. //
  1512. // Must be using a PCI device for a debugger.
  1513. //
  1514. debuggerFound = TRUE;
  1515. }
  1516. }
  1517. if (!debuggerFound) {
  1518. return;
  1519. }
  1520. //
  1521. // Open PCI service key.
  1522. //
  1523. RtlInitUnicodeString (&UnicodeString,
  1524. L"\\REGISTRY\\MACHINE\\SYSTEM\\CURRENTCONTROLSET\\SERVICES\\PCI");
  1525. InitializeObjectAttributes(&ObjectAttributes,
  1526. &UnicodeString,
  1527. OBJ_CASE_INSENSITIVE,
  1528. NULL,
  1529. (PSECURITY_DESCRIPTOR) NULL);
  1530. status = ZwOpenKey (&BaseHandle,
  1531. KEY_READ,
  1532. &ObjectAttributes);
  1533. if (!NT_SUCCESS(status)) {
  1534. return;
  1535. }
  1536. // Get the right key
  1537. RtlInitUnicodeString (&UnicodeString,
  1538. L"Debug");
  1539. InitializeObjectAttributes(&ObjectAttributes,
  1540. &UnicodeString,
  1541. OBJ_CASE_INSENSITIVE,
  1542. BaseHandle,
  1543. (PSECURITY_DESCRIPTOR) NULL);
  1544. status = ZwCreateKey (&Handle,
  1545. KEY_READ,
  1546. &ObjectAttributes,
  1547. 0,
  1548. (PUNICODE_STRING) NULL,
  1549. REG_OPTION_VOLATILE,
  1550. &disposition);
  1551. ZwClose(BaseHandle);
  1552. BaseHandle = Handle;
  1553. ASSERT(disposition == REG_CREATED_NEW_KEY);
  1554. if (!NT_SUCCESS(status)) {
  1555. return;
  1556. }
  1557. for (i = 0;
  1558. i < MAX_DEBUGGING_DEVICES_SUPPORTED;
  1559. i++) {
  1560. if (HalpPciDebuggingDevice[i].u.bits.Reserved1 == TRUE) {
  1561. //
  1562. // This entry is populated. Create a key for it.
  1563. //
  1564. RtlInitUnicodeString (&UnicodeString,
  1565. L"0");
  1566. (*(PCHAR)&(UnicodeString.Buffer[0])) += i;
  1567. InitializeObjectAttributes(&ObjectAttributes,
  1568. &UnicodeString,
  1569. OBJ_CASE_INSENSITIVE,
  1570. BaseHandle,
  1571. (PSECURITY_DESCRIPTOR) NULL);
  1572. status = ZwCreateKey (&Handle,
  1573. KEY_READ,
  1574. &ObjectAttributes,
  1575. 0,
  1576. (PUNICODE_STRING) NULL,
  1577. REG_OPTION_VOLATILE,
  1578. &disposition);
  1579. ASSERT(disposition == REG_CREATED_NEW_KEY);
  1580. //
  1581. // Fill in the values below this key.
  1582. //
  1583. bus = HalpPciDebuggingDevice[i].u.bits.BusNumber;
  1584. RtlInitUnicodeString (&UnicodeString,
  1585. L"Bus");
  1586. status = ZwSetValueKey (Handle,
  1587. &UnicodeString,
  1588. 0,
  1589. REG_DWORD,
  1590. &bus,
  1591. sizeof(ULONG));
  1592. //ASSERT(NT_SUCCESS(status));
  1593. slot.u.AsULONG = 0;
  1594. slot.u.bits.FunctionNumber = HalpPciDebuggingDevice[i].u.bits.FunctionNumber;
  1595. slot.u.bits.DeviceNumber = HalpPciDebuggingDevice[i].u.bits.DeviceNumber;
  1596. RtlInitUnicodeString (&UnicodeString,
  1597. L"Slot");
  1598. status = ZwSetValueKey (Handle,
  1599. &UnicodeString,
  1600. 0,
  1601. REG_DWORD,
  1602. &slot.u.AsULONG,
  1603. sizeof(ULONG));
  1604. //ASSERT(NT_SUCCESS(status));
  1605. ZwClose(Handle);
  1606. }
  1607. }
  1608. ZwClose(BaseHandle);
  1609. return;
  1610. }
  1611. NTSTATUS
  1612. HalpConfigurePciBridge(
  1613. IN PDEBUG_DEVICE_DESCRIPTOR PciDevice,
  1614. IN ULONG Bus,
  1615. IN ULONG Slot,
  1616. IN ULONG IoMin,
  1617. IN ULONG IoMax,
  1618. IN ULONG MemMin,
  1619. IN ULONG MemMax,
  1620. IN ULONG BusMin,
  1621. IN ULONG BusMax,
  1622. IN OUT PPCI_COMMON_CONFIG PciData
  1623. )
  1624. {
  1625. USHORT memUnits = 0;
  1626. ULONG memSize;
  1627. PciData->u.type1.PrimaryBus = (UCHAR)Bus;
  1628. PciData->u.type1.SecondaryBus = (UCHAR)BusMin;
  1629. PciData->u.type1.SubordinateBus = (UCHAR)(MIN(BusMax, (BusMin + 2)));
  1630. PciData->Command &= ~PCI_ENABLE_BUS_MASTER;
  1631. //DbgPrint("HalpConfigurePciBridge: P: %x S: %x S: %x\n"
  1632. // "\tI/O %x-%x Mem %x-%x Bus %x-%x\n",
  1633. // PciData->u.type1.PrimaryBus,
  1634. // PciData->u.type1.SecondaryBus,
  1635. // PciData->u.type1.SubordinateBus,
  1636. // IoMin, IoMax,
  1637. // MemMin, MemMax,
  1638. // BusMin, BusMax);
  1639. //
  1640. // Only enable I/O on the bridge if we are looking for
  1641. // something besides a 1394 controller.
  1642. //
  1643. if (!((PciDevice->BaseClass == PCI_CLASS_SERIAL_BUS_CTLR) &&
  1644. (PciDevice->SubClass == PCI_SUBCLASS_SB_IEEE1394))) {
  1645. if (((IoMax & 0xf000) - (IoMin & 0xf000)) >= 0X1000) {
  1646. //
  1647. // There is enough I/O space here to enable
  1648. // an I/O window.
  1649. //
  1650. PciData->u.type1.IOBase =
  1651. (UCHAR)((IoMax & 0xf000) >> 12) - 1;
  1652. PciData->u.type1.IOLimit = PciData->u.type1.IOBase;
  1653. PciData->Command |= PCI_ENABLE_IO_SPACE;
  1654. PciData->Command |= PCI_ENABLE_BUS_MASTER;
  1655. }
  1656. }
  1657. //
  1658. // Enable a memory window if possible.
  1659. //
  1660. memSize = ((MemMax + 1) & 0xfff00000) - (MemMin & 0xfff00000);
  1661. if (memSize >= 0x100000) {
  1662. memUnits = 1;
  1663. }
  1664. if (memSize >= 0x400000) {
  1665. memUnits = 4;
  1666. }
  1667. if (memUnits > 0) {
  1668. //
  1669. // There is enough space.
  1670. //
  1671. PciData->u.type1.MemoryBase =
  1672. (USHORT)((MemMax & 0xfff00000) >> 16) - (memUnits << 4);
  1673. PciData->u.type1.MemoryLimit = PciData->u.type1.MemoryBase + ((memUnits - 1) << 4);
  1674. PciData->Command |= PCI_ENABLE_MEMORY_SPACE;
  1675. PciData->Command |= PCI_ENABLE_BUS_MASTER;
  1676. }
  1677. if (PciData->Command & PCI_ENABLE_BUS_MASTER) {
  1678. HalpPhase0SetPciDataByOffset(Bus,
  1679. Slot,
  1680. PciData,
  1681. 0,
  1682. 0x24);
  1683. return STATUS_SUCCESS;
  1684. } else {
  1685. return STATUS_UNSUCCESSFUL;
  1686. }
  1687. }
  1688. VOID
  1689. HalpUnconfigurePciBridge(
  1690. IN ULONG Bus,
  1691. IN ULONG Slot
  1692. )
  1693. {
  1694. UCHAR buffer[0x20] = {0};
  1695. //
  1696. // Zero the command register.
  1697. //
  1698. HalpPhase0SetPciDataByOffset(Bus,
  1699. Slot,
  1700. buffer,
  1701. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  1702. 2);
  1703. //
  1704. // Zero the address space and bus number registers.
  1705. //
  1706. HalpPhase0SetPciDataByOffset(Bus,
  1707. Slot,
  1708. buffer,
  1709. FIELD_OFFSET (PCI_COMMON_CONFIG, u),
  1710. 0x20);
  1711. }
  1712. BOOLEAN
  1713. HalpValidateCardbusSocketRegs(
  1714. IN PCARDBUS_SOCKET_REGS SocketRegs
  1715. )
  1716. /*++
  1717. Routine Description:
  1718. This routine looks to make sure the cardbus socket registers look
  1719. reasonable, and also that there is a card in the slot.
  1720. Arguments:
  1721. SocketRegs - virtual address of mapped socket registers
  1722. Return Value:
  1723. returns TRUE if everything looks ok, FALSE otherwise
  1724. --*/
  1725. {
  1726. ULONG mask;
  1727. //
  1728. // First verify there are no reserved bits on
  1729. //
  1730. if ((SocketRegs->Event & 0xFFFFFFF0) ||
  1731. (SocketRegs->Mask & 0xFFFFFFF0) ||
  1732. (SocketRegs->PresentState & 0x0FFF0000) ||
  1733. (SocketRegs->ForceEvent) ||
  1734. (SocketRegs->Control & 0xFFFFFF00)) {
  1735. //
  1736. // Some bits we don't expect to be on were on
  1737. //
  1738. return FALSE;
  1739. }
  1740. if ((SocketRegs->PresentState & SKTSTATE_3VSOCKET) == 0) {
  1741. //
  1742. // need to have a 3.3v socket for cardbus
  1743. //
  1744. return FALSE;
  1745. }
  1746. //
  1747. // Stop here if there is no CardBus card in the slot
  1748. //
  1749. if ((SocketRegs->PresentState & SKTSTATE_CARDTYPE_MASK) != SKTSTATE_CBCARD) {
  1750. return FALSE;
  1751. }
  1752. //
  1753. // Also look to make sure this isn't on
  1754. //
  1755. if (SocketRegs->PresentState & SKTSTATE_NOTACARD) {
  1756. return FALSE;
  1757. }
  1758. return TRUE;
  1759. }
  1760. NTSTATUS
  1761. HalpConfigureCardBusBridge(
  1762. IN PDEBUG_DEVICE_DESCRIPTOR PciDevice,
  1763. IN ULONG Bus,
  1764. IN ULONG Slot,
  1765. IN ULONG IoMin,
  1766. IN ULONG IoMax,
  1767. IN ULONG MemMin,
  1768. IN ULONG MemMax,
  1769. IN ULONG BusMin,
  1770. IN ULONG BusMax,
  1771. IN OUT PPCI_COMMON_CONFIG PciData
  1772. )
  1773. {
  1774. NTSTATUS status = STATUS_UNSUCCESSFUL;
  1775. ULONG ulTemp;
  1776. ULONG SocketRegBase, origSocketRegBase = 0;
  1777. USHORT BridgeControl = 0;
  1778. PCARDBUS_SOCKET_REGS SocketRegs;
  1779. PHYSICAL_ADDRESS physicalAddress;
  1780. ULONG bytesRead;
  1781. USHORT command, origCmd;
  1782. UCHAR capPtr = 0;
  1783. PCI_PM_CAPABILITY pmCap;
  1784. BOOLEAN cmdRegChanged = FALSE;
  1785. BOOLEAN socketRegChanged = FALSE;
  1786. BOOLEAN socketRegConfigured = FALSE;
  1787. try{
  1788. //
  1789. // First look for a power management capability on this controller, and
  1790. // if it exists (it should), put the cardbus controller into D0
  1791. //
  1792. bytesRead = HalpPhase0GetPciDataByOffset(Bus, Slot, &capPtr,
  1793. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.CapabilitiesPtr), 1);
  1794. if (bytesRead != 1) {
  1795. // if we can't do this, things must be just too broken
  1796. leave;
  1797. }
  1798. while (capPtr) {
  1799. UCHAR capHeader[2];
  1800. //
  1801. // loop through the capabilities list and look for power management
  1802. //
  1803. if ((capPtr & 3) || (capPtr < 0x40)) {
  1804. //
  1805. // the capabilities pointer isn't valid
  1806. //
  1807. leave;
  1808. }
  1809. bytesRead = HalpPhase0GetPciDataByOffset(Bus, Slot, &capHeader, capPtr, 2);
  1810. if (bytesRead != 2) {
  1811. leave;
  1812. }
  1813. if (capHeader[0] == PCI_CAPABILITY_ID_POWER_MANAGEMENT) {
  1814. //
  1815. // found it
  1816. //
  1817. break;
  1818. }
  1819. if (capPtr == capHeader[1]) {
  1820. //
  1821. // this is bad config data. Avoid an infinite loop
  1822. //
  1823. leave;
  1824. }
  1825. capPtr = capHeader[1];
  1826. }
  1827. if (capPtr) {
  1828. bytesRead = HalpPhase0GetPciDataByOffset(Bus, Slot, &pmCap, capPtr, sizeof(pmCap));
  1829. if ((bytesRead == sizeof(pmCap)) &&
  1830. (pmCap.Header.CapabilityID == PCI_CAPABILITY_ID_POWER_MANAGEMENT) &&
  1831. (pmCap.PMCSR.ControlStatus.PowerState != 0)) {
  1832. pmCap.PMCSR.ControlStatus.PowerState = 0; // set to D0
  1833. HalpPhase0SetPciDataByOffset(Bus,
  1834. Slot,
  1835. &pmCap.PMCSR.ControlStatus,
  1836. capPtr + FIELD_OFFSET(PCI_PM_CAPABILITY,PMCSR.ControlStatus),
  1837. sizeof(pmCap.PMCSR.ControlStatus));
  1838. HalpKdStallExecution(1000);
  1839. }
  1840. }
  1841. //
  1842. // Turn off legacy mode base address... this really should be done
  1843. // via ACPI _INIT. This may not be sufficient to put the host in
  1844. // cardbus mode.
  1845. //
  1846. ulTemp = 0;
  1847. HalpPhase0SetPciDataByOffset(Bus,
  1848. Slot,
  1849. &ulTemp,
  1850. CARDBUS_LEGACY_MODE_BASE_ADDR,
  1851. 4);
  1852. //
  1853. // save away original value of command register, socket reg base
  1854. //
  1855. bytesRead = HalpPhase0GetPciDataByOffset(Bus,
  1856. Slot,
  1857. &origCmd,
  1858. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  1859. 2);
  1860. if (bytesRead != 2) {
  1861. leave;
  1862. }
  1863. bytesRead = HalpPhase0GetPciDataByOffset(Bus,
  1864. Slot,
  1865. &origSocketRegBase,
  1866. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.SocketRegistersBaseAddress),
  1867. 4);
  1868. if (bytesRead != 4) {
  1869. leave;
  1870. }
  1871. //
  1872. // Turn on socket registers in a suitable location
  1873. //
  1874. if (origSocketRegBase) {
  1875. //
  1876. // bios has configured socket reg base... see if it is working
  1877. //
  1878. physicalAddress.HighPart = 0;
  1879. physicalAddress.LowPart = origSocketRegBase;
  1880. SocketRegs = HalpMapPhysicalMemory64(physicalAddress, 1);
  1881. if (!SocketRegs) {
  1882. leave;
  1883. }
  1884. command = origCmd | PCI_ENABLE_MEMORY_SPACE;
  1885. if (command != origCmd) {
  1886. HalpPhase0SetPciDataByOffset(Bus,
  1887. Slot,
  1888. &command,
  1889. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  1890. 2);
  1891. cmdRegChanged = TRUE;
  1892. }
  1893. HalpKdStallExecution(600);
  1894. //
  1895. // make sure it looks like cardbus socket registers
  1896. //
  1897. if (HalpValidateCardbusSocketRegs(SocketRegs)) {
  1898. socketRegConfigured = TRUE;
  1899. SocketRegBase = origSocketRegBase;
  1900. } else {
  1901. //
  1902. // This didn't work... we'll try it again in a moment with a different address
  1903. //
  1904. HalpPhase0SetPciDataByOffset(Bus,
  1905. Slot,
  1906. &origCmd,
  1907. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  1908. 2);
  1909. cmdRegChanged = FALSE;
  1910. }
  1911. }
  1912. //
  1913. // If the socket regs weren't working where they were originally configured by
  1914. // the bios, or if the bios didn't configure them, we will configure them here
  1915. //
  1916. if (!socketRegConfigured) {
  1917. SocketRegBase = ((MemMin + 0xfff) & 0xfffff000);
  1918. if ((SocketRegBase + 0x1000) >= MemMax) {
  1919. leave;
  1920. }
  1921. //
  1922. // Adjust MemMin to allow the socket registers to remain visible
  1923. //
  1924. MemMin = SocketRegBase + 0x1000;
  1925. HalpPhase0SetPciDataByOffset(Bus,
  1926. Slot,
  1927. &SocketRegBase,
  1928. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.SocketRegistersBaseAddress),
  1929. 4);
  1930. socketRegChanged = TRUE;
  1931. physicalAddress.HighPart = 0;
  1932. physicalAddress.LowPart = SocketRegBase;
  1933. SocketRegs = HalpMapPhysicalMemory64(physicalAddress, 1);
  1934. if (!SocketRegs) {
  1935. leave;
  1936. }
  1937. command = origCmd | PCI_ENABLE_MEMORY_SPACE;
  1938. if (command != origCmd) {
  1939. HalpPhase0SetPciDataByOffset(Bus,
  1940. Slot,
  1941. &command,
  1942. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  1943. 2);
  1944. cmdRegChanged = TRUE;
  1945. }
  1946. HalpKdStallExecution(600);
  1947. if (!HalpValidateCardbusSocketRegs(SocketRegs)) {
  1948. //
  1949. // The registers don't look right
  1950. //
  1951. leave;
  1952. }
  1953. }
  1954. PciData->u.type2.SocketRegistersBaseAddress = SocketRegBase;
  1955. //
  1956. // make sure events are disabled
  1957. //
  1958. SocketRegs->Mask = 0;
  1959. //
  1960. // power up the card
  1961. //
  1962. SocketRegs->Control = SKTPOWER_VCC_033V | SKTPOWER_VPP_033V;
  1963. HalpKdStallExecution(600);
  1964. //
  1965. // Make sure CBRST is off
  1966. //
  1967. HalpPhase0GetPciDataByOffset(Bus,
  1968. Slot,
  1969. &BridgeControl,
  1970. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.BridgeControl),
  1971. 2);
  1972. if (BridgeControl & CARDBUS_BRIDGE_CONTROL_RESET) {
  1973. BridgeControl &= ~CARDBUS_BRIDGE_CONTROL_RESET;
  1974. HalpPhase0SetPciDataByOffset(Bus,
  1975. Slot,
  1976. &BridgeControl,
  1977. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.BridgeControl),
  1978. 2);
  1979. }
  1980. PciData->u.type2.PrimaryBus = (UCHAR)Bus;
  1981. PciData->u.type2.SecondaryBus = (UCHAR)BusMin;
  1982. PciData->u.type2.SubordinateBus = (UCHAR)(MIN(BusMax, (BusMin + 2)));
  1983. PciData->Command &= ~PCI_ENABLE_BUS_MASTER;
  1984. //DbgPrint("HalpConfigureCardBusBridge: P: %x S: %x S: %x\n"
  1985. // "\tI/O %x-%x Mem %x-%x Bus %x-%x\n",
  1986. // PciData->u.type2.PrimaryBus,
  1987. // PciData->u.type2.SecondaryBus,
  1988. // PciData->u.type2.SubordinateBus,
  1989. // IoMin, IoMax,
  1990. // MemMin, MemMax,
  1991. // BusMin, BusMax);
  1992. //
  1993. // Only enable I/O on the bridge if we are looking for
  1994. // something besides a 1394 controller.
  1995. //
  1996. if (!((PciDevice->BaseClass == PCI_CLASS_SERIAL_BUS_CTLR) &&
  1997. (PciDevice->SubClass == PCI_SUBCLASS_SB_IEEE1394))) {
  1998. PciData->u.type2.Range[2].Base = IoMin;
  1999. PciData->u.type2.Range[2].Limit = IoMax & 0xffc;
  2000. PciData->Command |= PCI_ENABLE_IO_SPACE;
  2001. PciData->Command |= PCI_ENABLE_BUS_MASTER;
  2002. }
  2003. //
  2004. // Enable a memory window if possible.
  2005. //
  2006. if (MemMax > MemMin) {
  2007. PciData->u.type2.Range[0].Base = MemMin;
  2008. PciData->u.type2.Range[0].Limit = MemMax & 0xfffff000;
  2009. PciData->Command |= PCI_ENABLE_MEMORY_SPACE;
  2010. PciData->Command |= PCI_ENABLE_BUS_MASTER;
  2011. }
  2012. if ((PciData->Command & PCI_ENABLE_BUS_MASTER) == 0) {
  2013. leave;
  2014. }
  2015. HalpPhase0SetPciDataByOffset(Bus,
  2016. Slot,
  2017. PciData,
  2018. 0,
  2019. 0x3c);
  2020. status = STATUS_SUCCESS;
  2021. } finally {
  2022. if (!NT_SUCCESS(status)) {
  2023. if (cmdRegChanged) {
  2024. HalpPhase0SetPciDataByOffset(Bus,
  2025. Slot,
  2026. &origCmd,
  2027. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  2028. 2);
  2029. }
  2030. //
  2031. // restore socket register base
  2032. //
  2033. if (socketRegChanged) {
  2034. HalpPhase0SetPciDataByOffset(Bus,
  2035. Slot,
  2036. &origSocketRegBase,
  2037. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.SocketRegistersBaseAddress),
  2038. 4);
  2039. }
  2040. }
  2041. }
  2042. return status;
  2043. }
  2044. VOID
  2045. HalpUnconfigureCardBusBridge(
  2046. IN ULONG Bus,
  2047. IN ULONG Slot
  2048. )
  2049. {
  2050. UCHAR buffer[0x2c] = {0};
  2051. ULONG SocketRegBase = 0xffffffff;
  2052. PCARDBUS_SOCKET_REGS SocketRegs;
  2053. PHYSICAL_ADDRESS physicalAddress;
  2054. HalpPhase0GetPciDataByOffset(Bus,
  2055. Slot,
  2056. &SocketRegBase,
  2057. FIELD_OFFSET (PCI_COMMON_CONFIG, u.type2.SocketRegistersBaseAddress),
  2058. 4);
  2059. if (SocketRegBase) {
  2060. physicalAddress.HighPart = 0;
  2061. physicalAddress.LowPart = SocketRegBase;
  2062. SocketRegs = HalpMapPhysicalMemory64(physicalAddress, 1);
  2063. //
  2064. // Turn power back off
  2065. //
  2066. if (SocketRegs) {
  2067. SocketRegs->Control = 0;
  2068. }
  2069. }
  2070. //
  2071. // Zero the command register.
  2072. //
  2073. HalpPhase0SetPciDataByOffset(Bus,
  2074. Slot,
  2075. buffer,
  2076. FIELD_OFFSET (PCI_COMMON_CONFIG, Command),
  2077. 2);
  2078. //
  2079. // Zero the address space and bus number registers.
  2080. //
  2081. HalpPhase0SetPciDataByOffset(Bus,
  2082. Slot,
  2083. buffer,
  2084. FIELD_OFFSET (PCI_COMMON_CONFIG, u),
  2085. 0x2c);
  2086. }
  2087. ULONG
  2088. HalpKdStallExecution(
  2089. ULONG LoopCount
  2090. )
  2091. {
  2092. ULONG i,j,b,k,l;
  2093. b = 1;
  2094. for (k=0;k<LoopCount;k++) {
  2095. for (i=1;i<100000;i++) {
  2096. PAUSE_PROCESSOR
  2097. b=b* (i>>k);
  2098. }
  2099. };
  2100. return b;
  2101. }