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.

2030 lines
57 KiB

  1. /*++
  2. Copyright (c) 1995-2000 Microsoft Corporation
  3. Module Name:
  4. isolate.c
  5. Abstract:
  6. Author:
  7. Shie-Lin Tzong (shielint) July-10-1995
  8. Environment:
  9. Kernel mode only.
  10. Revision History:
  11. --*/
  12. #include "busp.h"
  13. #include "pbios.h"
  14. #include "pnpisa.h"
  15. #if ISOLATE_CARDS
  16. #define RANGE_MASK 0xFF000000
  17. BOOLEAN
  18. PipFindIrqInformation (
  19. IN ULONG IrqLevel,
  20. IN PUCHAR BiosRequirements,
  21. OUT PUCHAR Information
  22. );
  23. BOOLEAN
  24. PipFindMemoryInformation (
  25. IN ULONG Index,
  26. IN ULONG Base,
  27. IN ULONG Limit,
  28. IN PUCHAR BiosRequirements,
  29. OUT PUCHAR NameTag,
  30. OUT PUCHAR Information,
  31. OUT PULONG NewLengh OPTIONAL
  32. );
  33. BOOLEAN
  34. PipFindIoPortInformation (
  35. IN ULONG BaseAddress,
  36. IN PUCHAR BiosRequirements,
  37. OUT PUCHAR Information,
  38. OUT PUCHAR Alignment,
  39. OUT PUCHAR RangeLength
  40. );
  41. BOOLEAN
  42. PipFindDmaInformation (
  43. IN UCHAR ChannelMask,
  44. IN PUCHAR BiosRequirements,
  45. OUT PUCHAR Information
  46. );
  47. NTSTATUS
  48. PipReadCardResourceDataBytes (
  49. IN USHORT BytesToRead,
  50. IN PUCHAR Buffer
  51. );
  52. USHORT
  53. PipIrqLevelRequirementsFromDeviceData(
  54. IN PUCHAR BiosRequirements,
  55. ULONG Length
  56. );
  57. //
  58. // Internal type definitions
  59. //
  60. typedef struct _MEMORY_DESC_{
  61. ULONG Base;
  62. ULONG Length;
  63. BOOLEAN Memory32;
  64. } MEMORY_DESC, *PMEMORY_DESC;
  65. typedef struct _IRQ_DESC_{
  66. UCHAR Level;
  67. ULONG Type;
  68. }IRQ_DESC, *PIRQ_DESC;
  69. #ifdef ALLOC_PRAGMA
  70. //#pragma alloc_text(PAGE, PipFindIrqInformation)
  71. //#pragma alloc_text(PAGE, PipFindMemoryInformation)
  72. //#pragma alloc_text(PAGE, PipFindIoPortInformation)
  73. //#pragma alloc_text(PAGE, PipReadCardResourceData)
  74. #pragma alloc_text(PAGE, PipReadDeviceResources)
  75. //#pragma alloc_text(PAGE, PipWriteDeviceResources)
  76. //#pragma alloc_text(PAGE, PipLFSRInitiation)
  77. //#pragma alloc_text(PAGE, PipIsolateCards)
  78. #pragma alloc_text(PAGE, PipFindNextLogicalDeviceTag)
  79. //#pragma alloc_text(PAGE, PipSelectLogicalDevice)
  80. //#pragma alloc_text(PAGE, PipReadCardResourceDataBytes)
  81. #endif
  82. BOOLEAN
  83. PipFindIrqInformation (
  84. IN ULONG IrqLevel,
  85. IN PUCHAR BiosRequirements,
  86. OUT PUCHAR Information
  87. )
  88. /*++
  89. Routine Description:
  90. This routine searches the Bios resource requirement lists for the corresponding
  91. Irq descriptor information. The search stops when we encounter another logical
  92. device id tag or the END tag. On input, the BiosRequirements points to current
  93. logical id tag.
  94. Arguments:
  95. IrqLevel - Supplies the irq level.
  96. BiosRequirements - Supplies a pointer to the bios resource requirement lists. This
  97. parameter must point to the logical device Id tag.
  98. Information - supplies a pointer to a UCHAR to receive the port information/flags.
  99. Return Value:
  100. TRUE - if memory information found. Else False.
  101. --*/
  102. {
  103. UCHAR tag;
  104. ULONG increment;
  105. USHORT irqMask;
  106. PPNP_IRQ_DESCRIPTOR biosDesc;
  107. //
  108. // Skip current logical id tag
  109. //
  110. tag = *BiosRequirements;
  111. ASSERT((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID);
  112. BiosRequirements += (tag & SMALL_TAG_SIZE_MASK) + 1;
  113. //
  114. // Search the possible resource list to get the information
  115. // for the Irq.
  116. //
  117. irqMask = 1 << IrqLevel;
  118. tag = *BiosRequirements;
  119. while ((tag != TAG_COMPLETE_END) && ((tag & SMALL_TAG_MASK) != TAG_LOGICAL_ID)) {
  120. if ((tag & SMALL_TAG_MASK) == TAG_IRQ) {
  121. biosDesc = (PPNP_IRQ_DESCRIPTOR)BiosRequirements;
  122. if (biosDesc->IrqMask & irqMask) {
  123. if ((tag & SMALL_TAG_SIZE_MASK) == 2) {
  124. //
  125. // if no irq info is available, a value of zero is returned.
  126. // (o is not a valid irq information.)
  127. //
  128. *Information = 0;
  129. } else {
  130. *Information = biosDesc->Information;
  131. }
  132. return TRUE;
  133. }
  134. }
  135. if (tag & LARGE_RESOURCE_TAG) {
  136. increment = *((USHORT UNALIGNED *)(BiosRequirements + 1));
  137. increment += 3; // length of large tag
  138. } else {
  139. increment = tag & SMALL_TAG_SIZE_MASK;
  140. increment += 1; // length of small tag
  141. }
  142. BiosRequirements += increment;
  143. tag = *BiosRequirements;
  144. }
  145. return FALSE;
  146. }
  147. BOOLEAN
  148. PipFindMemoryInformation (
  149. IN ULONG Index,
  150. IN ULONG BaseAddress,
  151. IN ULONG Limit,
  152. IN PUCHAR BiosRequirements,
  153. OUT PUCHAR NameTag,
  154. OUT PUCHAR Information,
  155. OUT PULONG NewLength OPTIONAL
  156. )
  157. /*++
  158. Routine Description:
  159. This routine searches the Bios resource requirement lists for the corresponding
  160. memory descriptor information. The search stops when we encounter another logical
  161. device id tag or the END tag. Note, the memory range specified by Base
  162. and Limit must be within a single Pnp ISA memory descriptor.
  163. Arguments:
  164. Index - Which memory descriptor we're interested in.
  165. BaseAddress - Supplies the base address of the memory range.
  166. Limit - Supplies the upper limit of the memory range.
  167. BiosRequirements - Supplies a pointer to the bios resource requirement lists. This
  168. parameter must point to the logical device Id tag.
  169. NameTag - Supplies a variable to receive the Tag of the memory descriptor which
  170. describes the memory information.
  171. Information - supplies a pointer to a UCHAR to receive the memory information
  172. for the specified memory range.
  173. Return Value:
  174. TRUE - if memory information found. Else False.
  175. --*/
  176. {
  177. UCHAR tag;
  178. BOOLEAN found = FALSE,foundMem24, foundMem;
  179. ULONG minAddr, length, maxAddr, alignment, noMem = 0;
  180. USHORT increment;
  181. //
  182. // Skip current logical id tag.
  183. //
  184. tag = *BiosRequirements;
  185. ASSERT((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID);
  186. BiosRequirements += (tag & SMALL_TAG_SIZE_MASK) + 1;
  187. //
  188. // Search the possible resource list to get the information
  189. // for the memory range described by Base and Limit.
  190. //
  191. if (NewLength) {
  192. *NewLength=0;
  193. }
  194. tag = *BiosRequirements;
  195. while ((tag != TAG_COMPLETE_END) && ((tag & SMALL_TAG_MASK) != TAG_LOGICAL_ID)) {
  196. foundMem = foundMem24 = FALSE;
  197. switch (tag) {
  198. case TAG_MEMORY:
  199. minAddr = ((ULONG)(((PPNP_MEMORY_DESCRIPTOR)BiosRequirements)->MinimumAddress)) << 8;
  200. length = ((ULONG)(((PPNP_MEMORY_DESCRIPTOR)BiosRequirements)->MemorySize)) << 8;
  201. maxAddr = (((ULONG)(((PPNP_MEMORY_DESCRIPTOR)BiosRequirements)->MaximumAddress)) << 8)
  202. + length - 1;
  203. foundMem24 = TRUE;
  204. foundMem = TRUE;
  205. break;
  206. case TAG_MEMORY32:
  207. length = ((PPNP_MEMORY32_DESCRIPTOR)BiosRequirements)->MemorySize;
  208. minAddr = ((PPNP_MEMORY32_DESCRIPTOR)BiosRequirements)->MinimumAddress;
  209. maxAddr = ((PPNP_MEMORY32_DESCRIPTOR)BiosRequirements)->MaximumAddress
  210. + length - 1;
  211. foundMem = TRUE;
  212. break;
  213. case TAG_MEMORY32_FIXED:
  214. length = ((PPNP_FIXED_MEMORY32_DESCRIPTOR)BiosRequirements)->MemorySize;
  215. minAddr = ((PPNP_FIXED_MEMORY32_DESCRIPTOR)BiosRequirements)->BaseAddress;
  216. maxAddr = minAddr + length - 1;
  217. foundMem = TRUE;
  218. break;
  219. }
  220. if (foundMem) {
  221. //
  222. // Work around cards that don't set register 43 correctly.
  223. // if the boot config has a value that equals the rom data, but
  224. // has the range type flipped, allow it, and reset
  225. // the length
  226. //
  227. if ((minAddr <= BaseAddress &&
  228. ((maxAddr >= Limit) || ((foundMem24 && (maxAddr >= (BaseAddress+(~Limit & ~RANGE_MASK))))))) && (noMem == Index)) {
  229. *Information = ((PPNP_MEMORY32_DESCRIPTOR)BiosRequirements)->Information;
  230. *NameTag = tag;
  231. found = TRUE;
  232. //
  233. // did we find a 16-bit tag
  234. //
  235. if (NewLength && foundMem24) {
  236. if (maxAddr >= (BaseAddress+(~Limit & ~RANGE_MASK))) {
  237. *NewLength = length;
  238. }
  239. }
  240. break;
  241. } else {
  242. noMem++;
  243. }
  244. }
  245. //
  246. // Advance to next tag
  247. //
  248. if (tag & LARGE_RESOURCE_TAG) {
  249. increment = *(USHORT UNALIGNED *)(BiosRequirements + 1);
  250. increment += 3; // length of large tag
  251. } else {
  252. increment = tag & SMALL_TAG_SIZE_MASK;
  253. increment += 1; // length of small tag
  254. }
  255. BiosRequirements += increment;
  256. tag = *BiosRequirements;
  257. }
  258. return found;
  259. }
  260. BOOLEAN
  261. PipFindIoPortInformation (
  262. IN ULONG BaseAddress,
  263. IN PUCHAR BiosRequirements,
  264. OUT PUCHAR Information,
  265. OUT PUCHAR Alignment,
  266. OUT PUCHAR RangeLength
  267. )
  268. /*++
  269. Routine Description:
  270. This routine searches the Bios resource requirement lists for the corresponding
  271. Io port descriptor information. The search stops when we encounter another logical
  272. device id tag or the END tag.
  273. Arguments:
  274. BaseAddress - Supplies the base address of the Io port range.
  275. BiosRequirements - Supplies a pointer to the bios resource requirement lists. This
  276. parameter must point to the logical device Id tag.
  277. Information - supplies a pointer to a UCHAR to receive the port information/flags.
  278. Alignment - supplies a pointer to a UCHAR to receive the port alignment
  279. information.
  280. RangeLength - supplies a pointer to a UCHAR to receive the port range length
  281. information.
  282. Return Value:
  283. TRUE - if memory information found. Else False.
  284. --*/
  285. {
  286. UCHAR tag;
  287. BOOLEAN found = FALSE;
  288. ULONG minAddr, length, maxAddr, alignment;
  289. USHORT increment;
  290. PPNP_PORT_DESCRIPTOR portDesc;
  291. PPNP_FIXED_PORT_DESCRIPTOR fixedPortDesc;
  292. tag = *BiosRequirements;
  293. ASSERT((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID);
  294. BiosRequirements += (tag & SMALL_TAG_SIZE_MASK) + 1;
  295. //
  296. // Search the possible resource list to get the information
  297. // for the io port range described by Base.
  298. //
  299. tag = *BiosRequirements;
  300. while ((tag != TAG_COMPLETE_END) && ((tag & SMALL_TAG_MASK) != TAG_LOGICAL_ID)) {
  301. switch (tag & SMALL_TAG_MASK) {
  302. case TAG_IO:
  303. portDesc = (PPNP_PORT_DESCRIPTOR)BiosRequirements;
  304. minAddr = portDesc->MinimumAddress;
  305. maxAddr = portDesc->MaximumAddress;
  306. if (minAddr <= BaseAddress && maxAddr >= BaseAddress) {
  307. *Information = portDesc->Information;
  308. *Alignment = portDesc->Alignment;
  309. *RangeLength = portDesc->Length;
  310. found = TRUE;
  311. }
  312. break;
  313. case TAG_IO_FIXED:
  314. fixedPortDesc = (PPNP_FIXED_PORT_DESCRIPTOR)BiosRequirements;
  315. minAddr = fixedPortDesc->MinimumAddress;
  316. if (BaseAddress == minAddr) {
  317. *Information = 0; // 10 bit decode
  318. *Alignment = 1;
  319. *RangeLength = fixedPortDesc->Length;
  320. found = TRUE;
  321. }
  322. break;
  323. }
  324. if (found) {
  325. break;
  326. }
  327. //
  328. // Advance to next tag
  329. //
  330. if (tag & LARGE_RESOURCE_TAG) {
  331. increment = *(USHORT UNALIGNED *)(BiosRequirements + 1);
  332. increment += 3; // length of large tag
  333. } else {
  334. increment = tag & SMALL_TAG_SIZE_MASK;
  335. increment += 1; // length of small tag
  336. }
  337. BiosRequirements += increment;
  338. tag = *BiosRequirements;
  339. }
  340. return found;
  341. }
  342. BOOLEAN
  343. PipFindDmaInformation (
  344. IN UCHAR ChannelMask,
  345. IN PUCHAR BiosRequirements,
  346. OUT PUCHAR Information
  347. )
  348. /*++
  349. Routine Description:
  350. This routine searches the Bios resource requirement lists for the corresponding
  351. Io port descriptor information. The search stops when we encounter another logical
  352. device id tag or the END tag.
  353. Arguments:
  354. BaseAddress - Supplies the channel mask.
  355. BiosRequirements - Supplies a pointer to the bios resource requirement lists. This
  356. parameter must point to the logical device Id tag.
  357. Information - supplies a pointer to a UCHAR to receive the port information/flags.
  358. Return Value:
  359. TRUE - if memory information found. Else False.
  360. --*/
  361. {
  362. UCHAR tag;
  363. BOOLEAN found = FALSE;
  364. USHORT increment;
  365. PPNP_DMA_DESCRIPTOR dmaDesc;
  366. UCHAR biosMask;
  367. tag = *BiosRequirements;
  368. ASSERT((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID);
  369. BiosRequirements += (tag & SMALL_TAG_SIZE_MASK) + 1;
  370. //
  371. // Search the possible resource list to get the information
  372. // for the io port range described by Base.
  373. //
  374. tag = *BiosRequirements;
  375. while ((tag != TAG_COMPLETE_END) && ((tag & SMALL_TAG_MASK) != TAG_LOGICAL_ID)) {
  376. if ((tag & SMALL_TAG_MASK) == TAG_DMA) {
  377. dmaDesc = (PPNP_DMA_DESCRIPTOR)BiosRequirements;
  378. biosMask = dmaDesc->ChannelMask;
  379. if (ChannelMask & biosMask) {
  380. *Information = dmaDesc->Flags;
  381. found = TRUE;
  382. }
  383. }
  384. if (found) {
  385. break;
  386. }
  387. //
  388. // Advance to next tag
  389. //
  390. if (tag & LARGE_RESOURCE_TAG) {
  391. increment = *(USHORT UNALIGNED *)(BiosRequirements + 1);
  392. increment += 3; // length of large tag
  393. } else {
  394. increment = tag & SMALL_TAG_SIZE_MASK;
  395. increment += 1; // length of small tag
  396. }
  397. BiosRequirements += increment;
  398. tag = *BiosRequirements;
  399. }
  400. return found;
  401. }
  402. NTSTATUS
  403. PipReadCardResourceData (
  404. OUT PULONG NumberLogicalDevices,
  405. IN PUCHAR *ResourceData,
  406. OUT PULONG ResourceDataLength
  407. )
  408. /*++
  409. Routine Description:
  410. This routine reads resources data from a specified PnP ISA card. It is
  411. caller's responsibility to release the memory. Before calling this routine,
  412. the Pnp ISA card should be in sleep state (i.e. Initiation Key was sent.)
  413. After exiting this routine, the card will be left in Config state.
  414. Arguments:
  415. NumberLogicalDevices - supplies a variable to receive the number of logical devices
  416. associated with the Pnp Isa card.
  417. ResourceData - Supplies a variable to receive the pointer to the resource data.
  418. ResourceDataLength - Supplies a variable to receive the length of the ResourceData.
  419. Return Value:
  420. NT STATUS code.
  421. --*/
  422. {
  423. PUCHAR buffer, p;
  424. LONG sizeToRead, limit, i;
  425. USHORT size;
  426. UCHAR tag;
  427. ULONG noDevices;
  428. BOOLEAN failed;
  429. NTSTATUS status;
  430. //
  431. // Allocate memory to store the resource data.
  432. // N.B. The buffer size should cover 99.999% of the machines.
  433. //
  434. sizeToRead = 4096;
  435. tryAgain:
  436. noDevices = 0;
  437. buffer = (PUCHAR)ExAllocatePoolWithTag(NonPagedPool, sizeToRead, 'iPnP');
  438. if (!buffer) {
  439. DebugPrint((DEBUG_ERROR, "PipReadCardResourceData returning STATUS_INSUFFICIENT_RESOURCES\n"));
  440. return STATUS_INSUFFICIENT_RESOURCES;
  441. }
  442. //
  443. // Send card from sleep state to configuration state
  444. // Note, by doing this the resource data includes 9 bytes Id.
  445. //
  446. DebugPrint((DEBUG_STATE, "Read resources\n"));
  447. //
  448. // Read card id bytes
  449. //
  450. p = buffer;
  451. status = PipReadCardResourceDataBytes(NUMBER_CARD_ID_BYTES, p);
  452. if (!NT_SUCCESS(status)) {
  453. ExFreePool(buffer);
  454. DebugPrint((DEBUG_STATE | DEBUG_ERROR,
  455. "Read resources failed\n"));
  456. DebugPrint((DEBUG_ERROR, "PipReadCardResourceDataBytes Failed %x\n",status));
  457. return status;
  458. }
  459. i = NUMBER_CARD_ID_BYTES;
  460. p += NUMBER_CARD_ID_BYTES;
  461. //
  462. // read all the tag descriptors of the card resource data
  463. //
  464. failed = FALSE;
  465. limit = sizeToRead - 4 - NUMBER_CARD_ID_BYTES;;
  466. while (TRUE) {
  467. //
  468. // Read tag byte. Make sure it's a valid tag and determine
  469. // the size of the descriptor.
  470. //
  471. PipReadCardResourceDataBytes(1, p);
  472. tag = *p;
  473. i++;
  474. p++;
  475. if (tag == TAG_COMPLETE_END) {
  476. PipReadCardResourceDataBytes(1, p);
  477. p++;
  478. i++;
  479. break;
  480. } else if (tag == TAG_END) { // With NO checksum
  481. *p = 0;
  482. i++;
  483. p++;
  484. break;
  485. }
  486. if (tag & LARGE_RESOURCE_TAG) {
  487. if (tag & 0x70) {
  488. failed = TRUE;
  489. #if VERBOSE_DEBUG
  490. DbgPrint ("Failing Resource read on large tag: %x\n",tag);
  491. #endif
  492. break;
  493. } else {
  494. PipReadCardResourceDataBytes(2, p);
  495. size = *((USHORT UNALIGNED *)p);
  496. p += 2;
  497. i += 2;
  498. }
  499. } else {
  500. if ((tag & 0x70) == 0x50 || (tag & 0x70) == 0x60 || (((tag & 0x70) == 0) && (tag != 0xa))) {
  501. failed = TRUE;
  502. #if VERBOSE_DEBUG
  503. DbgPrint ("Failing Resource read on small tag: %x\n",tag);
  504. #endif
  505. break;
  506. } else {
  507. if ((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID) {
  508. noDevices++;
  509. }
  510. size = (USHORT)(tag & SMALL_TAG_SIZE_MASK);
  511. }
  512. }
  513. //
  514. // read 'size' number of bytes for the current descriptor
  515. //
  516. i += size;
  517. if (i < limit) {
  518. PipReadCardResourceDataBytes(size, p);
  519. p += size;
  520. } else {
  521. ExFreePool(buffer);
  522. sizeToRead <<= 1; // double the buffer
  523. //
  524. // If we can find the END tag with 32K byte, assume the resource
  525. // requirement list is bad.
  526. //
  527. if (sizeToRead > 0x80000) {
  528. DebugPrint ((DEBUG_STATE | DEBUG_ERROR, "PipReadCardResourceData returning STATUS_INVALID_PARAMETER, Sleep\n"));
  529. return STATUS_INVALID_PARAMETER;
  530. } else {
  531. goto tryAgain;
  532. }
  533. }
  534. }
  535. if (failed) {
  536. ExFreePool(buffer);
  537. #if VERBOSE_DEBUG
  538. DbgPrint ("PipReadCardResourceData returning FAILED\n");
  539. #endif
  540. return STATUS_UNSUCCESSFUL;
  541. }
  542. //
  543. // Determine the real size of the buffer required and
  544. // resize the buffer.
  545. //
  546. size = (USHORT)(p - buffer); // i
  547. p = (PUCHAR)ExAllocatePoolWithTag(NonPagedPool, size, 'iPnP');
  548. if (p) {
  549. RtlMoveMemory(p, buffer, size);
  550. ExFreePool(buffer);
  551. } else {
  552. //
  553. // Fail to resize the buffer. Simply leave it alone.
  554. //
  555. p = buffer;
  556. }
  557. *ResourceData = p;
  558. *NumberLogicalDevices = noDevices;
  559. *ResourceDataLength = size;
  560. return STATUS_SUCCESS;
  561. }
  562. NTSTATUS
  563. PipReadDeviceResources (
  564. IN ULONG BusNumber,
  565. IN PUCHAR BiosRequirements,
  566. IN ULONG CardFlags,
  567. OUT PCM_RESOURCE_LIST *ResourceData,
  568. OUT PULONG Length,
  569. OUT PUSHORT irqFlags
  570. )
  571. /*++
  572. Routine Description:
  573. This routine reads boot resource data from an enabled logical device of a PNP ISA
  574. card. Caller must put the card into configuration state and select the logical
  575. device before calling this function. It is caller's responsibility to release
  576. the memory. ( The boot resource data is the resources that a card assigned during
  577. boot.)
  578. Arguments:
  579. BusNumber - specifies the bus number of the device whose resource data to be read.
  580. BiosRequirements - Supplies a pointer to the resource requirement list for the logical
  581. device. This parameter must point to the logical device Id tag.
  582. CardFlags - Flags that may indicate the need to apply a workaround.
  583. ResourceData - Supplies a variable to receive the pointer to the resource data.
  584. Length - Supplies a variable to recieve the length of the resource data.
  585. Return Value:
  586. NT STATUS code.
  587. --*/
  588. {
  589. UCHAR c, junk1, junk2, info;
  590. PUCHAR base;
  591. ULONG l, resourceCount;
  592. BOOLEAN limit;
  593. LONG i, j, noMemoryDesc = 0, noIoDesc = 0, noDmaDesc =0, noIrqDesc = 0;
  594. MEMORY_DESC memoryDesc[NUMBER_MEMORY_DESCRIPTORS + NUMBER_32_MEMORY_DESCRIPTORS];
  595. IRQ_DESC irqDesc[NUMBER_IRQ_DESCRIPTORS];
  596. UCHAR dmaDesc[NUMBER_DMA_DESCRIPTORS];
  597. USHORT ioDesc[NUMBER_IO_DESCRIPTORS];
  598. PCM_RESOURCE_LIST cmResource;
  599. PCM_PARTIAL_RESOURCE_LIST partialResList;
  600. PCM_PARTIAL_RESOURCE_DESCRIPTOR partialDesc;
  601. ULONG dumpData[2];
  602. //
  603. // First make sure the specified BiosRequirements is valid and at the right tag.
  604. //
  605. if ((*BiosRequirements & SMALL_TAG_MASK) != TAG_LOGICAL_ID) {
  606. return STATUS_INVALID_PARAMETER;
  607. }
  608. //
  609. // If card is not activated, don't read boot resource.
  610. // Because boot resource of non activated NEC98's ISAPNP card is not 0.
  611. //
  612. *irqFlags = CM_RESOURCE_INTERRUPT_LATCHED;
  613. PipWriteAddress(ACTIVATE_PORT);
  614. if (!(PipReadData() & 1)) {
  615. *ResourceData = NULL;
  616. *Length = 0;
  617. return STATUS_UNSUCCESSFUL;
  618. }
  619. //
  620. // Read memory configuration
  621. //
  622. base = (PUCHAR)ADDRESS_MEMORY_BASE;
  623. for (i = 0; i < NUMBER_MEMORY_DESCRIPTORS; i++) {
  624. //
  625. // Read memory base address
  626. //
  627. PipWriteAddress(base + ADDRESS_MEMORY_HI);
  628. c = PipReadData();
  629. l = c;
  630. l <<= 8;
  631. PipWriteAddress(base + ADDRESS_MEMORY_LO);
  632. c = PipReadData();
  633. l |= c;
  634. l <<= 8; // l = memory base address
  635. if (l == 0) {
  636. break;
  637. }
  638. memoryDesc[noMemoryDesc].Base = l;
  639. //
  640. // Read memory control byte
  641. //
  642. PipWriteAddress(base + ADDRESS_MEMORY_CTL);
  643. c= PipReadData();
  644. limit = c & 1;
  645. //
  646. // Read memory upper limit address or range length
  647. //
  648. PipWriteAddress(base + ADDRESS_MEMORY_UPPER_HI);
  649. c = PipReadData();
  650. l = c;
  651. l <<= 8;
  652. PipWriteAddress(base + ADDRESS_MEMORY_UPPER_LO);
  653. c = PipReadData();
  654. l |= c;
  655. l <<= 8;
  656. //
  657. //If bit[0] of memory control is 0, this is the range length.
  658. //If bit[0] of memory control is 1, this is upper limit for memory
  659. //address (equal to memory base address plus the range length allocated).
  660. //
  661. if (limit == ADDRESS_MEMORY_CTL_LIMIT) {
  662. l = l - memoryDesc[noMemoryDesc].Base;
  663. }else {
  664. l = (~l+1) & ~(RANGE_MASK);
  665. }
  666. // IBM0001 Token Ring card has write-only registers 0x4B-0x4C.
  667. // The boot configed length comes back 0 instead of 0x2000
  668. if ((CardFlags & CF_IBM_MEMBOOTCONFIG) && (l == 0) &&
  669. (noMemoryDesc == 1)) {
  670. l = 0x2000;
  671. }
  672. memoryDesc[noMemoryDesc].Length = l;
  673. memoryDesc[noMemoryDesc].Memory32 = FALSE;
  674. noMemoryDesc++;
  675. base += ADDRESS_MEMORY_INCR;
  676. }
  677. //
  678. // Read memory 32 configuration
  679. //
  680. // Spec says you can't mix 24 bit and 32bit memory. Helps on
  681. // cards with flakey 32 bit memory registers until we examine only
  682. // the boot configed resources specified in the requirements.
  683. if (noMemoryDesc == 0) {
  684. for (i = 0; i < NUMBER_32_MEMORY_DESCRIPTORS; i++) {
  685. base = ADDRESS_32_MEMORY_BASE(i);
  686. //
  687. // Read memory base address
  688. //
  689. l = 0;
  690. for (j = ADDRESS_32_MEMORY_B3; j <= ADDRESS_32_MEMORY_B0; j++) {
  691. PipWriteAddress(base + j);
  692. c = PipReadData();
  693. l <<= 8;
  694. l |= c;
  695. }
  696. if (l == 0) {
  697. break;
  698. }
  699. memoryDesc[noMemoryDesc].Base = l;
  700. //
  701. // Read memory control byte
  702. //
  703. PipWriteAddress(base + ADDRESS_32_MEMORY_CTL);
  704. c= PipReadData();
  705. limit = c & 1;
  706. //
  707. // Read memory upper limit address or range length
  708. //
  709. l = 0;
  710. for (j = ADDRESS_32_MEMORY_E3; j <= ADDRESS_32_MEMORY_E0; j++) {
  711. PipWriteAddress(base + j);
  712. c = PipReadData();
  713. l <<= 8;
  714. l |= c;
  715. }
  716. if (limit == ADDRESS_MEMORY_CTL_LIMIT) {
  717. l = l - memoryDesc[noMemoryDesc].Base;
  718. }else {
  719. l = ((~l)+1) & ~(RANGE_MASK);
  720. }
  721. memoryDesc[noMemoryDesc].Length = l;
  722. memoryDesc[noMemoryDesc].Memory32 = TRUE;
  723. noMemoryDesc++;
  724. }
  725. }
  726. //
  727. // Read Io Port Configuration
  728. //
  729. base = (PUCHAR)ADDRESS_IO_BASE;
  730. for (i = 0; i < NUMBER_IO_DESCRIPTORS; i++) {
  731. PipWriteAddress(base + ADDRESS_IO_BASE_HI);
  732. c = PipReadData();
  733. l = c;
  734. PipWriteAddress(base + ADDRESS_IO_BASE_LO);
  735. c = PipReadData();
  736. l <<= 8;
  737. l |= c;
  738. if (l == 0) {
  739. break;
  740. }
  741. ioDesc[noIoDesc++] = (USHORT)l;
  742. base += ADDRESS_IO_INCR;
  743. }
  744. //
  745. // Read Interrupt configuration
  746. //
  747. base = (PUCHAR)ADDRESS_IRQ_BASE;
  748. for (i = 0; i < NUMBER_IRQ_DESCRIPTORS; i++) {
  749. PipWriteAddress(base + ADDRESS_IRQ_VALUE);
  750. c = PipReadData() & 0xf;
  751. if (c == 0) {
  752. break;
  753. }
  754. irqDesc[noIrqDesc].Level = c;
  755. PipWriteAddress(base + ADDRESS_IRQ_TYPE);
  756. c = PipReadData();
  757. irqDesc[noIrqDesc++].Type = c;
  758. base += ADDRESS_IRQ_INCR;
  759. DebugPrint((DEBUG_IRQ, "card boot config byte %x\n", (ULONG) c));
  760. // only if card is configured to low level do we respect level.
  761. // register is probably busted
  762. if ((c & 3) == 1) {
  763. *irqFlags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
  764. }
  765. }
  766. //
  767. // Read DMA configuration
  768. //
  769. base = (PUCHAR)ADDRESS_DMA_BASE;
  770. for (i = 0; i < NUMBER_DMA_DESCRIPTORS; i++) {
  771. PipWriteAddress(base + ADDRESS_DMA_VALUE);
  772. c = PipReadData() & 0x7;
  773. if (c == 4) {
  774. break;
  775. }
  776. if (!PipFindDmaInformation ( (UCHAR)(1 << c), BiosRequirements, &info)) {
  777. break;
  778. }
  779. dmaDesc[noDmaDesc++] = c;
  780. base += ADDRESS_DMA_INCR;
  781. }
  782. //
  783. // Construct CM_RESOURCE_LIST structure based on the resource data
  784. // we collect so far.
  785. //
  786. resourceCount = noMemoryDesc + noIoDesc + noDmaDesc + noIrqDesc;
  787. //
  788. // if empty bios resources, simply return.
  789. //
  790. if (resourceCount == 0) {
  791. *ResourceData = NULL;
  792. *Length = 0;
  793. return STATUS_SUCCESS;
  794. }
  795. l = sizeof(CM_RESOURCE_LIST) + sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR) *
  796. ( resourceCount - 1);
  797. cmResource = ExAllocatePoolWithTag(PagedPool, l, 'iPnP');
  798. if (!cmResource) {
  799. return STATUS_INSUFFICIENT_RESOURCES;
  800. }
  801. RtlZeroMemory(cmResource, l);
  802. *Length = l; // Set returned resource data length
  803. cmResource->Count = 1;
  804. cmResource->List[0].InterfaceType = Isa;
  805. cmResource->List[0].BusNumber = BusNumber;
  806. partialResList = (PCM_PARTIAL_RESOURCE_LIST)&cmResource->List[0].PartialResourceList;
  807. partialResList->Version = 0;
  808. partialResList->Revision = 0x3000;
  809. partialResList->Count = resourceCount;
  810. partialDesc = (PCM_PARTIAL_RESOURCE_DESCRIPTOR)&partialResList->PartialDescriptors[0];
  811. //
  812. // Set up all the CM memory descriptors
  813. //
  814. for (i = 0; i < noMemoryDesc; i++) {
  815. ULONG NewLength;
  816. partialDesc->Type = CmResourceTypeMemory;
  817. partialDesc->ShareDisposition = CmResourceShareDeviceExclusive;
  818. partialDesc->u.Memory.Length = memoryDesc[i].Length;
  819. partialDesc->u.Memory.Start.HighPart = 0;
  820. partialDesc->u.Memory.Start.LowPart = memoryDesc[i].Base;
  821. //
  822. // Need to consult configuration data for the Flags
  823. //
  824. l = memoryDesc[i].Base + memoryDesc[i].Length - 1;
  825. if (PipFindMemoryInformation (i, memoryDesc[i].Base, l, BiosRequirements, &junk1, &c,&NewLength)) {
  826. if (NewLength != 0 ) {
  827. partialDesc->u.Memory.Length = NewLength;
  828. }
  829. // Mark the memory descriptor as read-only if the tags describe as
  830. // expansion ROM or generic non-writable memory
  831. if ((c & PNP_MEMORY_ROM_MASK) ||
  832. !(c & PNP_MEMORY_WRITE_STATUS_MASK)) {
  833. partialDesc->Flags = CM_RESOURCE_MEMORY_READ_ONLY;
  834. }
  835. } else {
  836. DebugPrint((DEBUG_CARDRES|DEBUG_WARN,
  837. "ReadDeviceResources: No matched memory req for %x to %x\n",
  838. memoryDesc[i].Base, l));
  839. ExFreePool(cmResource);
  840. return STATUS_UNSUCCESSFUL;
  841. }
  842. partialDesc->Flags |= CM_RESOURCE_MEMORY_24;
  843. partialDesc++;
  844. }
  845. //
  846. // Set up all the CM io/port descriptors
  847. //
  848. for (i = 0; i < noIoDesc; i++) {
  849. partialDesc->Type = CmResourceTypePort;
  850. partialDesc->ShareDisposition = CmResourceShareDeviceExclusive;
  851. partialDesc->Flags = CM_RESOURCE_PORT_IO;
  852. partialDesc->u.Port.Start.LowPart = ioDesc[i];
  853. //
  854. // Need to consult configuration data for the Port length
  855. //
  856. if (PipFindIoPortInformation (ioDesc[i], BiosRequirements, &info, &junk2, &c)) {
  857. if (info & 1) {
  858. partialDesc->Flags |= CM_RESOURCE_PORT_16_BIT_DECODE;
  859. } else {
  860. partialDesc->Flags |= CM_RESOURCE_PORT_10_BIT_DECODE;
  861. }
  862. partialDesc->u.Port.Length = c;
  863. partialDesc++;
  864. } else {
  865. DebugPrint((DEBUG_CARDRES|DEBUG_WARN,
  866. "ReadDeviceResources: No matched port req for %x\n",
  867. ioDesc[i]));
  868. ExFreePool(cmResource);
  869. return STATUS_UNSUCCESSFUL;
  870. }
  871. }
  872. //
  873. // Set up all the CM DMA descriptors
  874. //
  875. for (i = 0; i < noDmaDesc; i++) {
  876. partialDesc->Type = CmResourceTypeDma;
  877. partialDesc->ShareDisposition = CmResourceShareDeviceExclusive;
  878. partialDesc->Flags = 0; // no flags for DMA descriptor
  879. partialDesc->u.Dma.Channel = (ULONG) dmaDesc[i];
  880. partialDesc->u.Dma.Port = 0;
  881. partialDesc->u.Dma.Reserved1 = 0;
  882. partialDesc++;
  883. }
  884. //
  885. // Set up all the CM interrupt descriptors
  886. //
  887. for (i = 0; i < noIrqDesc; i++) {
  888. partialDesc->Type = CmResourceTypeInterrupt;
  889. partialDesc->ShareDisposition = CmResourceShareDeviceExclusive;
  890. partialDesc->Flags = *irqFlags;
  891. partialDesc->u.Interrupt.Vector =
  892. partialDesc->u.Interrupt.Level = irqDesc[i].Level;
  893. partialDesc->u.Interrupt.Affinity = (ULONG)-1;
  894. partialDesc++;
  895. }
  896. *ResourceData = cmResource;
  897. return STATUS_SUCCESS;
  898. }
  899. NTSTATUS
  900. PipWriteDeviceResources (
  901. IN PUCHAR BiosRequirements,
  902. IN PCM_RESOURCE_LIST CmResources
  903. )
  904. /*++
  905. Routine Description:
  906. This routine writes boot resource data to an enabled logical device of
  907. a Pnp ISA card. Caller must put the card into configuration state and select
  908. the logical device before calling this function.
  909. Arguments:
  910. BiosRequirements - Supplies a pointer to the possible resources for the logical
  911. device. This parameter must point to the logical device Id tag.
  912. ResourceData - Supplies a pointer to the cm resource data.
  913. Return Value:
  914. NT STATUS code.
  915. --*/
  916. {
  917. UCHAR c, information, tag;
  918. ULONG count, i, j, pass, base, limit;
  919. PCM_PARTIAL_RESOURCE_DESCRIPTOR cmDesc;
  920. ULONG noIrq =0, noIo = 0, noDma = 0, noMemory24 = 0, noMemory32 = 0, noMemory;
  921. PUCHAR memoryBase, irqBase, dmaBase, ioBase, tmp;
  922. ULONG memory32Base;
  923. #if 0
  924. PipWriteAddress(ACTIVATE_PORT);
  925. if (!(PipReadData() & 1)) {
  926. DbgPrint("PnpIsa-WriteDeviceBootResourceData:The logical device has not been activated\n");
  927. }
  928. #endif // DBG
  929. //
  930. // First make sure the specified BiosRequirements is valid and at the right tag.
  931. //
  932. if ((*BiosRequirements & SMALL_TAG_MASK) != TAG_LOGICAL_ID) {
  933. return STATUS_INVALID_PARAMETER;
  934. }
  935. count = CmResources->List[0].PartialResourceList.Count;
  936. memoryBase = (PUCHAR)ADDRESS_MEMORY_BASE;
  937. memory32Base = 0;
  938. ioBase = (PUCHAR)ADDRESS_IO_BASE;
  939. irqBase = (PUCHAR)ADDRESS_IRQ_BASE;
  940. dmaBase = (PUCHAR)ADDRESS_DMA_BASE;
  941. for (pass = 1; pass <= 2; pass++) {
  942. //
  943. // First pass we make sure the resources to be set is acceptable.
  944. // Second pass we actually write the resources to the logical device's
  945. // configuration space.
  946. //
  947. noMemory = 0;
  948. cmDesc = CmResources->List[0].PartialResourceList.PartialDescriptors;
  949. for (i = 0; i < count; i++) {
  950. switch (cmDesc->Type) {
  951. case CmResourceTypePort:
  952. if (pass == 1) {
  953. noIo++;
  954. if (noIo > NUMBER_IO_DESCRIPTORS ||
  955. cmDesc->u.Port.Start.HighPart != 0 ||
  956. cmDesc->u.Port.Start.LowPart & 0xffff0000 ||
  957. cmDesc->u.Port.Length & 0xffffff00) {
  958. return STATUS_INVALID_PARAMETER;
  959. }
  960. } else {
  961. //
  962. // Set the Io port base address to logical device configuration space
  963. //
  964. c = (UCHAR)cmDesc->u.Port.Start.LowPart;
  965. PipWriteAddress(ioBase + ADDRESS_IO_BASE_LO);
  966. PipWriteData(c);
  967. c = (UCHAR)(cmDesc->u.Port.Start.LowPart >> 8);
  968. PipWriteAddress(ioBase + ADDRESS_IO_BASE_HI);
  969. PipWriteData(c);
  970. ioBase += ADDRESS_IO_INCR;
  971. }
  972. break;
  973. case CmResourceTypeInterrupt:
  974. if (pass == 1) {
  975. noIrq++;
  976. if (noIrq > NUMBER_IRQ_DESCRIPTORS ||
  977. (cmDesc->u.Interrupt.Level & 0xfffffff0)) {
  978. return STATUS_INVALID_PARAMETER;
  979. }
  980. //
  981. // See if we can get the interrupt information from possible resource
  982. // data. We need it to set the configuration register.
  983. //
  984. if (!PipFindIrqInformation(cmDesc->u.Interrupt.Level, BiosRequirements, &information)) {
  985. return STATUS_INVALID_PARAMETER;
  986. }
  987. } else {
  988. //
  989. // Set the Irq to logical device configuration space
  990. //
  991. c = (UCHAR)cmDesc->u.Interrupt.Level;
  992. PipWriteAddress(irqBase + ADDRESS_IRQ_VALUE);
  993. PipWriteData(c);
  994. // Set IRQ to high edge or low level. Explicitly
  995. // ignore what was in the requirements as it may
  996. // specify low edge or high level which don't
  997. // actually work.
  998. if (cmDesc->Flags & CM_RESOURCE_INTERRUPT_LATCHED) {
  999. c = 2;
  1000. } else {
  1001. c = 1;
  1002. }
  1003. PipWriteAddress(irqBase + ADDRESS_IRQ_TYPE);
  1004. PipWriteData(c);
  1005. DebugPrint((DEBUG_IRQ, "Wrote 0x%x to port %x\n",
  1006. (ULONG) c, irqBase));
  1007. PipWriteAddress(irqBase + ADDRESS_IRQ_TYPE);
  1008. c = PipReadData();
  1009. DebugPrint((DEBUG_IRQ, "Read back 0x%x at port %x\n",
  1010. (ULONG) c, irqBase));
  1011. irqBase += ADDRESS_IRQ_INCR;
  1012. }
  1013. break;
  1014. case CmResourceTypeDma:
  1015. if (pass == 1) {
  1016. noDma++;
  1017. if (noDma > NUMBER_IRQ_DESCRIPTORS ||
  1018. (cmDesc->u.Dma.Channel & 0xfffffff8)) {
  1019. return STATUS_INVALID_PARAMETER;
  1020. }
  1021. } else {
  1022. //
  1023. // Set the Dma channel to logical device configuration space
  1024. //
  1025. c = (UCHAR)cmDesc->u.Dma.Channel;
  1026. PipWriteAddress(dmaBase + ADDRESS_DMA_VALUE);
  1027. PipWriteData(c);
  1028. dmaBase += ADDRESS_DMA_INCR;
  1029. }
  1030. break;
  1031. case CmResourceTypeMemory:
  1032. if (pass == 1) {
  1033. base = cmDesc->u.Memory.Start.LowPart;
  1034. limit = base + cmDesc->u.Memory.Length - 1;
  1035. if (!PipFindMemoryInformation(noMemory, base, limit, BiosRequirements, &tag, &information,NULL)) {
  1036. return STATUS_INVALID_PARAMETER;
  1037. } else {
  1038. if (tag == TAG_MEMORY) {
  1039. noMemory24++;
  1040. //
  1041. // Make sure the lower 8 bits of the base address are zero.
  1042. //
  1043. if (noMemory24 > NUMBER_MEMORY_DESCRIPTORS ||
  1044. base & 0xff) {
  1045. return STATUS_INVALID_PARAMETER;
  1046. }
  1047. } else {
  1048. noMemory32++;
  1049. if (noMemory32 > NUMBER_32_MEMORY_DESCRIPTORS) {
  1050. return STATUS_INVALID_PARAMETER;
  1051. }
  1052. }
  1053. }
  1054. } else {
  1055. //
  1056. // Find information in BiosRequirements to help determine how to write
  1057. // the memory configuration space.
  1058. //
  1059. base = cmDesc->u.Memory.Start.LowPart;
  1060. limit = base + cmDesc->u.Memory.Length - 1;
  1061. PipFindMemoryInformation(noMemory, base, limit, BiosRequirements, &tag, &information,NULL);
  1062. if (tag == TAG_MEMORY) {
  1063. PipWriteAddress(memoryBase + ADDRESS_MEMORY_LO);
  1064. PipWriteData(base >> 0x8);
  1065. PipWriteAddress(memoryBase + ADDRESS_MEMORY_HI);
  1066. PipWriteData(base >> 0x10);
  1067. if ((information & 0x18) == 0) { // 8 bit memory only
  1068. c = 0;
  1069. } else {
  1070. c = 2;
  1071. }
  1072. //
  1073. // Check range or limit
  1074. //
  1075. PipWriteAddress(memoryBase + ADDRESS_MEMORY_CTL);
  1076. if (PipReadData() & ADDRESS_MEMORY_CTL_LIMIT) {
  1077. c += ADDRESS_MEMORY_CTL_LIMIT;
  1078. limit = base + cmDesc->u.Memory.Length;
  1079. } else {
  1080. limit = cmDesc->u.Memory.Length; // Range
  1081. limit = (~limit)+1;
  1082. }
  1083. PipWriteAddress(memoryBase + ADDRESS_MEMORY_CTL);
  1084. PipWriteData(c);
  1085. PipWriteAddress(memoryBase + ADDRESS_MEMORY_UPPER_LO);
  1086. PipWriteData((UCHAR)(limit >> 0x8));
  1087. PipWriteAddress(memoryBase + ADDRESS_MEMORY_UPPER_HI);
  1088. PipWriteData((UCHAR)(limit >> 0x10));
  1089. memoryBase += ADDRESS_MEMORY_INCR;
  1090. } else {
  1091. tmp = ADDRESS_32_MEMORY_BASE(memory32Base);
  1092. PipWriteAddress(tmp + ADDRESS_32_MEMORY_B0);
  1093. PipWriteData(base);
  1094. PipWriteAddress(tmp + ADDRESS_32_MEMORY_B1);
  1095. PipWriteData(base >> 0x8);
  1096. PipWriteAddress(tmp + ADDRESS_32_MEMORY_B2);
  1097. PipWriteData(base >> 0x10);
  1098. PipWriteAddress(tmp + ADDRESS_32_MEMORY_B3);
  1099. PipWriteData(base >> 0x18);
  1100. switch (information & 0x18) {
  1101. case 0: // 8 bit only
  1102. c = 0;
  1103. case 8: // 16 bit only
  1104. case 0x10: // 8 and 16 bit supported
  1105. c = 2;
  1106. break;
  1107. case 0x18: // 32 bit only
  1108. c = 4;
  1109. break;
  1110. }
  1111. PipWriteAddress(ADDRESS_32_MEMORY_CTL);
  1112. if (PipReadData() & ADDRESS_MEMORY_CTL_LIMIT) {
  1113. c += ADDRESS_MEMORY_CTL_LIMIT;
  1114. limit = base + cmDesc->u.Memory.Length;
  1115. } else {
  1116. limit = cmDesc->u.Memory.Length; // Range
  1117. limit = (~limit) + 1;
  1118. }
  1119. PipWriteAddress(ADDRESS_32_MEMORY_CTL);
  1120. PipWriteData(c);
  1121. PipWriteAddress(tmp + ADDRESS_32_MEMORY_E0);
  1122. PipWriteData(limit);
  1123. PipWriteAddress(tmp + ADDRESS_32_MEMORY_E1);
  1124. PipWriteData(limit >> 0x8);
  1125. PipWriteAddress(tmp + ADDRESS_32_MEMORY_E2);
  1126. PipWriteData(limit >> 0x10);
  1127. PipWriteAddress(tmp + ADDRESS_32_MEMORY_E3);
  1128. PipWriteData(limit >> 0x18);
  1129. memory32Base++;
  1130. }
  1131. }
  1132. noMemory++;
  1133. }
  1134. cmDesc++;
  1135. }
  1136. }
  1137. //
  1138. // Finally, mark all the unused descriptors as disabled.
  1139. //
  1140. for (i = noMemory24; i < NUMBER_MEMORY_DESCRIPTORS; i++) {
  1141. for (j = 0; j < 5; j++) {
  1142. PipWriteAddress(memoryBase + j);
  1143. PipWriteData(0);
  1144. }
  1145. memoryBase += ADDRESS_MEMORY_INCR;
  1146. }
  1147. for (i = noMemory32; i < NUMBER_32_MEMORY_DESCRIPTORS; i++) {
  1148. tmp = ADDRESS_32_MEMORY_BASE(memory32Base);
  1149. for (j = 0; j < 9; j++) {
  1150. PipWriteAddress(tmp + j);
  1151. PipWriteData(0);
  1152. }
  1153. memory32Base++;
  1154. }
  1155. for (i = noIo; i < NUMBER_IO_DESCRIPTORS; i++) {
  1156. for (j = 0; j < 2; j++) {
  1157. PipWriteAddress(ioBase + j);
  1158. PipWriteData(0);
  1159. }
  1160. ioBase += ADDRESS_IO_INCR;
  1161. }
  1162. for (i = noIrq; i < NUMBER_IRQ_DESCRIPTORS; i++) {
  1163. for (j = 0; j < 2; j++) {
  1164. PipWriteAddress(irqBase + j);
  1165. PipWriteData(0);
  1166. }
  1167. irqBase += ADDRESS_IRQ_INCR;
  1168. }
  1169. for (i = noDma; i < NUMBER_DMA_DESCRIPTORS; i++) {
  1170. PipWriteAddress(dmaBase);
  1171. PipWriteData(4);
  1172. dmaBase += ADDRESS_DMA_INCR;
  1173. }
  1174. return STATUS_SUCCESS;
  1175. }
  1176. VOID
  1177. PipLFSRInitiation(
  1178. VOID
  1179. )
  1180. /*++
  1181. Routine Description:
  1182. This routine insures the LFSR (linear feedback shift register) is in its
  1183. initial state and then performs 32 writes to the ADDRESS port to initiation
  1184. LFSR function.
  1185. Pnp software sends the initiation key to all the Pnp ISA cards to place them
  1186. into configuration mode. The software then ready to perform isolation
  1187. protocol.
  1188. Arguments:
  1189. None.
  1190. Return Value:
  1191. None.
  1192. --*/
  1193. {
  1194. UCHAR seed, bit7;
  1195. ULONG i;
  1196. ASSERT(PipState == PiSWaitForKey);
  1197. //
  1198. // First perform two writes of value zero to insure the LFSR is in the
  1199. // initial state.
  1200. //
  1201. PipWriteAddress (0);
  1202. PipWriteAddress (0);
  1203. //
  1204. // Perform the initiation key.
  1205. //
  1206. seed = LFSR_SEED; // initial value of 0x6a
  1207. for (i = 0; i < 32; i++) {
  1208. PipWriteAddress (seed);
  1209. bit7=(((seed & 2) >> 1) ^ (seed & 1)) << 7;
  1210. seed =(seed >> 1) | bit7;
  1211. }
  1212. DebugPrint((DEBUG_ISOLATE, "Sent initiation key\n"));
  1213. PipReportStateChange(PiSSleep);
  1214. }
  1215. VOID
  1216. PipIsolateCards (
  1217. OUT PULONG NumberCSNs
  1218. )
  1219. /*++
  1220. Routine Description:
  1221. This routine performs PnP ISA cards isolation sequence.
  1222. Arguments:
  1223. NumberCSNs - supplies the addr of a variable to receive the number of
  1224. Pnp Isa cards isolated.
  1225. ReadDataPort - Supplies the address of a variable to supply ReadData port
  1226. address.
  1227. Return Value:
  1228. None.
  1229. --*/
  1230. {
  1231. USHORT j, i;
  1232. UCHAR cardId[NUMBER_CARD_ID_BYTES];
  1233. UCHAR bit, bit7, checksum, byte1, byte2;
  1234. ULONG csn;
  1235. //
  1236. // First send Initiation Key to all the PNP ISA cards to enable PnP auto-config
  1237. // ports and put all cards into sleep state
  1238. //
  1239. PipLFSRInitiation();
  1240. //
  1241. // Reset all Pnp ISA cards' CSN to 0 and return to wait-for-key state
  1242. //
  1243. PipWriteAddress (CONFIG_CONTROL_PORT);
  1244. PipWriteData (CONTROL_RESET_CSN + CONTROL_WAIT_FOR_KEY);
  1245. DebugPrint((DEBUG_STATE, "Reset CSNs, going to WaitForKey\n"));
  1246. PipReportStateChange(PiSWaitForKey);
  1247. csn=*NumberCSNs = 0;
  1248. //
  1249. // Delay 2 msec for cards to load initial configuration state.
  1250. //
  1251. KeStallExecutionProcessor(2000); // delay 2 msec
  1252. //
  1253. // Put cards into configuration mode to ready isolation process.
  1254. // The hardware on each PnP Isa card expects 72 pairs of I/O read
  1255. // access to the read data port.
  1256. //
  1257. PipLFSRInitiation();
  1258. //
  1259. // Starting Pnp Isa card isolation process.
  1260. //
  1261. //
  1262. // Send WAKE[CSN=0] to force all cards without CSN into isolation
  1263. // state to set READ DATA PORT.
  1264. //
  1265. PipIsolation();
  1266. KeStallExecutionProcessor(1000); // delay 1 msec
  1267. DebugPrint((DEBUG_STATE, "Wake all cards without CSN, Isolation\n"));
  1268. //
  1269. // Set read data port to current testing value.
  1270. //
  1271. PipWriteAddress(SET_READ_DATA_PORT);
  1272. PipWriteData((UCHAR)((ULONG_PTR)PipReadDataPort >> 2));
  1273. DebugPrint((DEBUG_STATE, "Set RDP to %x\n", PipReadDataPort));
  1274. //
  1275. // Isolate one PnP ISA card until fail
  1276. //
  1277. PipIsolation();
  1278. while (TRUE) {
  1279. //
  1280. // Read serial isolation port to cause PnP cards in the isolation
  1281. // state to compare one bit of the boards ID.
  1282. //
  1283. PipWriteAddress(SERIAL_ISOLATION_PORT);
  1284. //
  1285. // We need to delay 1 msec prior to starting the first pair of isolation
  1286. // reads and must wait 250usec between each subsequent pair of isolation
  1287. // reads. This delay gives the ISA cards time to access information from
  1288. // possible very slow storage device.
  1289. //
  1290. KeStallExecutionProcessor(1000); // delay 1 msec
  1291. RtlZeroMemory(cardId, NUMBER_CARD_ID_BYTES);
  1292. checksum = LFSR_SEED;
  1293. for (j = 0; j < NUMBER_CARD_ID_BITS; j++) {
  1294. //
  1295. // Read card id bit by bit
  1296. //
  1297. byte1 = PipReadData();
  1298. byte2 = PipReadData();
  1299. bit = (byte1 == ISOLATION_TEST_BYTE_1) && (byte2 == ISOLATION_TEST_BYTE_2);
  1300. cardId[j / 8] |= bit << (j % 8);
  1301. if (j < CHECKSUMED_BITS) {
  1302. //
  1303. // Calculate checksum and only do it for the first 64 bits
  1304. //
  1305. bit7 = (((checksum & 2) >> 1) ^ (checksum & 1) ^ (bit)) << 7;
  1306. checksum = (checksum >> 1) | bit7;
  1307. }
  1308. KeStallExecutionProcessor(250); // delay 250 usec
  1309. }
  1310. //
  1311. // Verify the card id we read is legitimate
  1312. // First make sure checksum is valid. Note zero checksum is considered valid.
  1313. //
  1314. DebugPrint((DEBUG_ISOLATE, "Card Bytes: %X %X %X %X %X %X %X %X %X\n",cardId[0],cardId[1],cardId[2],cardId[3],cardId[4],cardId[5],cardId[6],cardId[7],cardId[8]));
  1315. if (cardId[8] == 0 || checksum == cardId[8]) {
  1316. //
  1317. // Next make sure cardId is not zero
  1318. //
  1319. byte1 = 0;
  1320. for (j = 0; j < NUMBER_CARD_ID_BYTES; j++) {
  1321. byte1 |= cardId[j];
  1322. }
  1323. if (byte1 != 0) {
  1324. //
  1325. // Make sure the vender EISA ID bytes are nonzero
  1326. //
  1327. if ((cardId[0] & 0x7f) != 0 && cardId[1] != 0) {
  1328. //
  1329. // We found a valid Pnp Isa card, assign it a CSN number
  1330. //
  1331. DebugPrint((DEBUG_ISOLATE, "Assigning csn %d\n",csn+1));
  1332. PipWriteAddress(SET_CSN_PORT);
  1333. PipWriteData(++csn);
  1334. if (PipReadData() != csn) {
  1335. csn--;
  1336. DebugPrint((DEBUG_ISOLATE, "Assigning csn %d FAILED, bailing!\n",csn+1));
  1337. PipIsolation();
  1338. PipSleep();
  1339. *NumberCSNs = csn;
  1340. return;
  1341. }
  1342. //
  1343. // Do Wake[CSN] command to put the newly isolated card to
  1344. // sleep state and other un-isolated cards to isolation
  1345. // state.
  1346. //
  1347. PipIsolation();
  1348. DebugPrint((DEBUG_STATE, "Put card in Sleep, other in Isolation\n"));
  1349. continue; // ... to isolate more cards ...
  1350. }
  1351. }
  1352. }else {
  1353. DebugPrint ((DEBUG_ISOLATE, "invalid read during isolation\n"));
  1354. }
  1355. break; // could not isolate more cards ...
  1356. }
  1357. //
  1358. // Finaly put all cards into sleep state
  1359. //
  1360. PipSleep();
  1361. *NumberCSNs = csn;
  1362. }
  1363. ULONG
  1364. PipFindNextLogicalDeviceTag (
  1365. IN OUT PUCHAR *CardData,
  1366. IN OUT LONG *Limit
  1367. )
  1368. /*++
  1369. Routine Description:
  1370. This function searches the Pnp Isa card data for the Next logical
  1371. device tag encountered. The input *CardData should point to an logical device id tag,
  1372. which is the current logical device tag. If the *CardData does not point to a logical
  1373. device id tag (but, it must point to some kind of tag), it will be moved to next
  1374. logical device id tag.
  1375. Arguments:
  1376. CardData - a variable to supply a pointer to the pnp Isa resource descriptors and to
  1377. receive next logical device tag pointer.
  1378. Limit - a variable to supply the maximum length of the search and to receive the new
  1379. lemit after the search.
  1380. Return Value:
  1381. Length of the data between current and next logical device tags, ie the data length
  1382. of the current logical device.
  1383. In case there is no 'next' logical device tag, the returned *CardData = NULL,
  1384. *Limit = zero and the data length of current logical tag is returned as function
  1385. returned value.
  1386. --*/
  1387. {
  1388. UCHAR tag;
  1389. USHORT size;
  1390. LONG l;
  1391. ULONG retLength;
  1392. PUCHAR p;
  1393. BOOLEAN atIdTag = FALSE;;
  1394. p = *CardData;
  1395. l = *Limit;
  1396. tag = *p;
  1397. retLength = 0;
  1398. while (tag != TAG_COMPLETE_END && l > 0) {
  1399. //
  1400. // Determine the size of the BIOS resource descriptor
  1401. //
  1402. if (!(tag & LARGE_RESOURCE_TAG)) {
  1403. size = (USHORT)(tag & SMALL_TAG_SIZE_MASK);
  1404. size += 1; // length of small tag
  1405. } else {
  1406. size = *((USHORT UNALIGNED *)(p + 1));
  1407. size += 3; // length of large tag
  1408. }
  1409. p += size;
  1410. retLength += size;
  1411. l -= size;
  1412. tag = *p;
  1413. if ((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID) {
  1414. *CardData = p;
  1415. *Limit = l;
  1416. return retLength;
  1417. }
  1418. }
  1419. *CardData = NULL;
  1420. *Limit = 0;
  1421. if (tag == TAG_COMPLETE_END) {
  1422. return (retLength + 2); // add 2 for the length of end tag descriptor
  1423. } else {
  1424. return 0;
  1425. }
  1426. }
  1427. VOID
  1428. PipSelectLogicalDevice (
  1429. IN USHORT Csn,
  1430. IN USHORT LogicalDeviceNumber,
  1431. IN ULONG Control
  1432. )
  1433. /*++
  1434. Routine Description:
  1435. This function selects the logical logical device in the specified device.
  1436. Arguments:
  1437. Csn - Supplies a CardSelectNumber to select the card.
  1438. LogicalDeviceNumber - supplies a logical device number to select the logical device.
  1439. Control - supplies a variable to specify if the logical device should be enabled.
  1440. Return Value:
  1441. None
  1442. --*/
  1443. {
  1444. UCHAR tmp;
  1445. //
  1446. // Put cards into sleep state
  1447. //
  1448. ASSERT(PipState == PiSWaitForKey);
  1449. PipLFSRInitiation();
  1450. //
  1451. // Send WAKE[CSN] to force the card into config state.
  1452. //
  1453. DebugPrint((DEBUG_STATE, "Select Logical Device\n"));
  1454. PipConfig((UCHAR)Csn);
  1455. //
  1456. // Select the logical device.
  1457. //
  1458. PipWriteAddress(LOGICAL_DEVICE_PORT);
  1459. PipWriteData(LogicalDeviceNumber);
  1460. if (Control == SELECT_AND_ACTIVATE) {
  1461. //
  1462. // If we need to activate the logical device, disable its io range check and
  1463. // then enable the device.
  1464. //
  1465. PipActivateDevice();
  1466. } else if (Control == SELECT_AND_DEACTIVATE) {
  1467. PipDeactivateDevice();
  1468. }
  1469. }
  1470. NTSTATUS
  1471. PipReadCardResourceDataBytes (
  1472. IN USHORT BytesToRead,
  1473. IN PUCHAR Buffer
  1474. )
  1475. /*++
  1476. Routine Description:
  1477. This function reads specified number of bytes of card resource data .
  1478. Arguments:
  1479. BytesToRead - supplies number of bytes to read.
  1480. Buffer - supplies a pointer to a buffer to receive the read bytes.
  1481. Return Value:
  1482. None
  1483. --*/
  1484. {
  1485. USHORT i, j;
  1486. PUCHAR p;
  1487. for (i = 0, p = Buffer; i < BytesToRead; i++, p++) {
  1488. PipWriteAddress(CONFIG_DATA_STATUS_PORT);
  1489. //
  1490. // Waiting for data ready status bit
  1491. //
  1492. j = 0;
  1493. while ((PipReadData() & 1) != 1) {
  1494. if (j == 10000) {
  1495. return STATUS_NO_SUCH_DEVICE;
  1496. }
  1497. KeStallExecutionProcessor(1000); // delay 1 msec
  1498. j++;
  1499. }
  1500. //
  1501. // Read the data ...
  1502. //
  1503. PipWriteAddress(CONFIG_DATA_PORT);
  1504. *p = PipReadData();
  1505. }
  1506. return STATUS_SUCCESS;
  1507. }
  1508. USHORT
  1509. PipIrqLevelRequirementsFromDeviceData(
  1510. IN PUCHAR BiosRequirements,
  1511. IN ULONG Length
  1512. )
  1513. /*++
  1514. Routine Description:
  1515. This routine searches the resource data for IRQ tags and extracts
  1516. information on whether edge/level is specified. This is on a per
  1517. logical device basis.
  1518. Arguments:
  1519. BiosRequirements - the per-device tags.
  1520. Length - Length of per-device tag area.
  1521. Return Value:
  1522. edge/level as specified by the device requirements.
  1523. --*/
  1524. {
  1525. UCHAR tag, level;
  1526. ULONG increment;
  1527. USHORT irqFlags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
  1528. PPNP_IRQ_DESCRIPTOR biosDesc;
  1529. BOOLEAN sawIrq = FALSE;
  1530. //
  1531. // Skip current logical id tag
  1532. //
  1533. tag = *BiosRequirements;
  1534. ASSERT((tag & SMALL_TAG_MASK) == TAG_LOGICAL_ID);
  1535. BiosRequirements += (tag & SMALL_TAG_SIZE_MASK) + 1;
  1536. //
  1537. // Search the possible resource list to get the information
  1538. // for the Irq.
  1539. //
  1540. tag = *BiosRequirements;
  1541. while ((tag != TAG_COMPLETE_END) && ((tag & SMALL_TAG_MASK) != TAG_LOGICAL_ID)) {
  1542. if ((tag & SMALL_TAG_MASK) == TAG_IRQ) {
  1543. sawIrq = TRUE;
  1544. biosDesc = (PPNP_IRQ_DESCRIPTOR)BiosRequirements;
  1545. if ((tag & SMALL_TAG_SIZE_MASK) == 2) {
  1546. irqFlags = CM_RESOURCE_INTERRUPT_LATCHED;
  1547. } else {
  1548. level = biosDesc->Information;
  1549. DebugPrint((DEBUG_IRQ, "Irq req info is %x\n", (ULONG) level));
  1550. if (level == 0xF) {
  1551. // register is broken, assume edge
  1552. irqFlags = CM_RESOURCE_INTERRUPT_LATCHED;
  1553. } else if (level & 0x3) {
  1554. irqFlags = CM_RESOURCE_INTERRUPT_LATCHED;
  1555. } else if (level & 0xC) {
  1556. irqFlags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
  1557. }
  1558. }
  1559. }
  1560. if (tag & LARGE_RESOURCE_TAG) {
  1561. increment = *((USHORT UNALIGNED *)(BiosRequirements + 1));
  1562. increment += 3; // length of large tag
  1563. } else {
  1564. increment = tag & SMALL_TAG_SIZE_MASK;
  1565. increment += 1; // length of small tag
  1566. }
  1567. BiosRequirements += increment;
  1568. tag = *BiosRequirements;
  1569. }
  1570. if (!sawIrq) {
  1571. return CM_RESOURCE_INTERRUPT_LATCHED;
  1572. }
  1573. return irqFlags;
  1574. }
  1575. VOID
  1576. PipFixBootConfigIrqs(
  1577. IN PCM_RESOURCE_LIST BootResources,
  1578. IN USHORT irqFlags
  1579. )
  1580. /*++
  1581. Routine Description:
  1582. This routine modifies the boot config resources list to reflect
  1583. whether the devices's irqs should be considered edge or level.
  1584. This is on a per logical device basis.
  1585. Arguments:
  1586. BootResources - Boot config as determined by PipReadDeviceResources()
  1587. irqFlags - level/edge setting to apply to boot config resources
  1588. --*/
  1589. {
  1590. PCM_FULL_RESOURCE_DESCRIPTOR cmFullDesc;
  1591. PCM_PARTIAL_RESOURCE_DESCRIPTOR cmPartDesc;
  1592. ULONG count = 0, size, i, j;
  1593. if (BootResources == NULL) {
  1594. return;
  1595. }
  1596. cmFullDesc = &BootResources->List[0];
  1597. for (i = 0; i < BootResources->Count; i++) {
  1598. cmPartDesc = &cmFullDesc->PartialResourceList.PartialDescriptors[0];
  1599. for (j = 0; j < cmFullDesc->PartialResourceList.Count; j++) {
  1600. size = 0;
  1601. if (cmPartDesc->Type == CmResourceTypeInterrupt) {
  1602. cmPartDesc->Flags = irqFlags;
  1603. if (cmPartDesc->Flags & CM_RESOURCE_INTERRUPT_LATCHED) {
  1604. cmPartDesc->ShareDisposition = CmResourceShareDeviceExclusive;
  1605. }
  1606. } else if (cmPartDesc->Type == CmResourceTypeDeviceSpecific) {
  1607. size = cmPartDesc->u.DeviceSpecificData.DataSize;
  1608. }
  1609. cmPartDesc++;
  1610. cmPartDesc = (PCM_PARTIAL_RESOURCE_DESCRIPTOR) ((PUCHAR)cmPartDesc + size);
  1611. }
  1612. cmFullDesc = (PCM_FULL_RESOURCE_DESCRIPTOR)cmPartDesc;
  1613. }
  1614. }
  1615. #endif