2 Implement all interfaces for EFI_PCI_IO_PROTOCOL.
4 Copyright (c) 2006 - 2008, Intel Corporation
5 All rights reserved. This program and the accompanying materials
6 are licensed and made available under the terms and conditions of the BSD License
7 which accompanies this distribution. The full text of the license may be found at
8 http://opensource.org/licenses/bsd-license.php
10 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
11 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
23 // Pci Io Protocol Interface
25 EFI_PCI_IO_PROTOCOL PciIoInterface
= {
48 PciIoGetBarAttributes
,
49 PciIoSetBarAttributes
,
55 report a error Status code of PCI bus driver controller.
57 @param PciIoDevice Pci device instance.
58 @param Code status code.
61 ReportErrorStatusCode (
62 IN PCI_IO_DEVICE
*PciIoDevice
,
63 IN EFI_STATUS_CODE_VALUE Code
67 // PCI I/O Support Function Prototypes
72 report a error Status code of PCI bus driver controller.
74 @param PciIoDevice Pci device instance.
75 @param Code status code.
78 ReportErrorStatusCode (
79 IN PCI_IO_DEVICE
*PciIoDevice
,
80 IN EFI_STATUS_CODE_VALUE Code
83 return REPORT_STATUS_CODE_WITH_DEVICE_PATH (
84 EFI_ERROR_CODE
| EFI_ERROR_MINOR
,
86 PciIoDevice
->DevicePath
91 Initializes a PCI I/O Instance.
93 @param PciIoDevice Pci device instance.
97 InitializePciIoInstance (
98 PCI_IO_DEVICE
*PciIoDevice
101 CopyMem (&PciIoDevice
->PciIo
, &PciIoInterface
, sizeof (EFI_PCI_IO_PROTOCOL
));
105 Verifies access to a PCI Base Address Register (BAR)
107 @param PciIoDevice Pci device instance
108 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
109 base address for the memory or I/O operation to perform.
110 @param Type Operation type could be memory or I/O
111 @param Width Signifies the width of the memory or I/O operations.
112 @param Count The number of memory or I/O operations to perform.
113 @param Offset The offset within the PCI configuration space for the PCI controller.
115 @retval EFI_INVALID_PARAMETER Invalid Width/BarIndex or Bar type.
116 @retval EFI_SUCCESS Success Operation.
119 PciIoVerifyBarAccess (
120 PCI_IO_DEVICE
*PciIoDevice
,
123 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
128 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
129 return EFI_INVALID_PARAMETER
;
132 if (BarIndex
== EFI_PCI_IO_PASS_THROUGH_BAR
) {
137 // BarIndex 0-5 is legal
139 if (BarIndex
>= PCI_MAX_BAR
) {
140 return EFI_INVALID_PARAMETER
;
143 if (!CheckBarType (PciIoDevice
, BarIndex
, Type
)) {
144 return EFI_INVALID_PARAMETER
;
148 // If Width is EfiPciIoWidthFifoUintX then convert to EfiPciIoWidthUintX
149 // If Width is EfiPciIoWidthFillUintX then convert to EfiPciIoWidthUintX
151 if (Width
>= EfiPciWidthFifoUint8
&& Width
<= EfiPciWidthFifoUint64
) {
155 Width
= (EFI_PCI_IO_PROTOCOL_WIDTH
) (Width
& 0x03);
157 if ((*Offset
+ Count
* (UINTN
)(1 << Width
)) - 1 >= PciIoDevice
->PciBar
[BarIndex
].Length
) {
158 return EFI_INVALID_PARAMETER
;
161 *Offset
= *Offset
+ PciIoDevice
->PciBar
[BarIndex
].BaseAddress
;
167 Verifies access to a PCI Config Header
169 @param PciIoDevice Pci device instance
170 @param Width Signifies the width of the memory or I/O operations.
171 @param Count The number of memory or I/O operations to perform.
172 @param Offset The offset within the PCI configuration space for the PCI controller.
174 @retval EFI_INVALID_PARAMETER Invalid Width
175 @retval EFI_UNSUPPORTED Offset overflow
176 @retval EFI_SUCCESS Success operation
179 PciIoVerifyConfigAccess (
180 PCI_IO_DEVICE
*PciIoDevice
,
181 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
188 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
189 return EFI_INVALID_PARAMETER
;
193 // If Width is EfiPciIoWidthFillUintX then convert to EfiPciIoWidthUintX
195 Width
= (EFI_PCI_IO_PROTOCOL_WIDTH
) (Width
& 0x03);
197 if (PciIoDevice
->IsPciExp
) {
198 if ((*Offset
+ Count
* (UINTN
)(1 << Width
)) - 1 >= PCI_EXP_MAX_CONFIG_OFFSET
) {
199 return EFI_UNSUPPORTED
;
202 ExtendOffset
= LShiftU64 (*Offset
, 32);
203 *Offset
= EFI_PCI_ADDRESS (PciIoDevice
->BusNumber
, PciIoDevice
->DeviceNumber
, PciIoDevice
->FunctionNumber
, 0);
204 *Offset
= (*Offset
) | ExtendOffset
;
207 if ((*Offset
+ Count
* (UINTN
)(1 << Width
)) - 1 >= PCI_MAX_CONFIG_OFFSET
) {
208 return EFI_UNSUPPORTED
;
211 *Offset
= EFI_PCI_ADDRESS (PciIoDevice
->BusNumber
, PciIoDevice
->DeviceNumber
, PciIoDevice
->FunctionNumber
, *Offset
);
218 Reads from the I/O space of a PCI Root Bridge. Returns when either the polling exit criteria is
219 satisfied or after a defined duration.
221 @param This Pointer to protocol instance of EFI_PCI_IO_PROTOCOL
222 @param Width Signifies the width of the memory or I/O operations.
223 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
224 base address for the memory or I/O operation to perform.
225 @param Offset The offset within the PCI configuration space for the PCI controller.
226 @param Mask Mask used for the polling criteria.
227 @param Value The comparison value used for the polling exit criteria.
228 @param Delay The number of 100 ns units to poll.
229 @param Result Pointer to the last value read from the memory location.
231 @retval EFI_SUCCESS The last data returned from the access matched the poll exit criteria.
232 @retval EFI_TIMEOUT Delay expired before a match occurred.
233 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
234 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
239 IN EFI_PCI_IO_PROTOCOL
*This
,
240 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
250 PCI_IO_DEVICE
*PciIoDevice
;
252 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
254 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
255 return EFI_INVALID_PARAMETER
;
258 Status
= PciIoVerifyBarAccess (PciIoDevice
, BarIndex
, PciBarTypeMem
, Width
, 1, &Offset
);
259 if (EFI_ERROR (Status
)) {
260 return EFI_UNSUPPORTED
;
263 if (Width
> EfiPciIoWidthUint64
) {
264 return EFI_INVALID_PARAMETER
;
267 Status
= PciIoDevice
->PciRootBridgeIo
->PollMem (
268 PciIoDevice
->PciRootBridgeIo
,
269 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
277 if (EFI_ERROR (Status
)) {
278 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
285 Reads from the I/O space of a PCI Root Bridge. Returns when either the polling exit criteria is
286 satisfied or after a defined duration.
288 @param This A pointer to the EFI_PCI_IO_PROTOCOL.
289 @param Width Signifies the width of the memory or I/O operations.
290 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
291 base address for the memory or I/O operation to perform.
292 @param Offset The offset within the selected BAR to start the memory or I/O operation.
293 @param Mask Mask used for the polling criteria.
294 @param Value The comparison value used for the polling exit criteria.
295 @param Delay The number of 100 ns units to poll.
296 @param Result Pointer to the last value read from the memory location.
298 @retval EFI_SUCCESS The last data returned from the access matched the poll exit criteria.
299 @retval EFI_TIMEOUT Delay expired before a match occurred.
300 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
301 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
307 IN EFI_PCI_IO_PROTOCOL
*This
,
308 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
318 PCI_IO_DEVICE
*PciIoDevice
;
320 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
322 if (Width
< 0 || Width
> EfiPciIoWidthUint64
) {
323 return EFI_INVALID_PARAMETER
;
326 Status
= PciIoVerifyBarAccess (PciIoDevice
, BarIndex
, PciBarTypeIo
, Width
, 1, &Offset
);
327 if (EFI_ERROR (Status
)) {
328 return EFI_UNSUPPORTED
;
331 Status
= PciIoDevice
->PciRootBridgeIo
->PollIo (
332 PciIoDevice
->PciRootBridgeIo
,
333 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
341 if (EFI_ERROR (Status
)) {
342 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
349 Enable a PCI driver to access PCI controller registers in the PCI memory or I/O space.
351 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
352 @param Width Signifies the width of the memory or I/O operations.
353 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
354 base address for the memory or I/O operation to perform.
355 @param Offset The offset within the selected BAR to start the memory or I/O operation.
356 @param Count The number of memory or I/O operations to perform.
357 @param Buffer For read operations, the destination buffer to store the results. For write
358 operations, the source buffer to write data from.
360 @retval EFI_SUCCESS The data was read from or written to the PCI controller.
361 @retval EFI_UNSUPPORTED BarIndex not valid for this PCI controller.
362 @retval EFI_UNSUPPORTED The address range specified by Offset, Width, and Count is not
363 valid for the PCI BAR specified by BarIndex.
364 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
365 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
371 IN EFI_PCI_IO_PROTOCOL
*This
,
372 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
380 PCI_IO_DEVICE
*PciIoDevice
;
382 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
384 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
385 return EFI_INVALID_PARAMETER
;
388 if (Buffer
== NULL
) {
389 return EFI_INVALID_PARAMETER
;
392 Status
= PciIoVerifyBarAccess (PciIoDevice
, BarIndex
, PciBarTypeMem
, Width
, Count
, &Offset
);
393 if (EFI_ERROR (Status
)) {
394 return EFI_UNSUPPORTED
;
397 Status
= PciIoDevice
->PciRootBridgeIo
->Mem
.Read (
398 PciIoDevice
->PciRootBridgeIo
,
399 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
405 if (EFI_ERROR (Status
)) {
406 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_READ_ERROR
);
413 Enable a PCI driver to access PCI controller registers in the PCI memory or I/O space.
415 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
416 @param Width Signifies the width of the memory or I/O operations.
417 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
418 base address for the memory or I/O operation to perform.
419 @param Offset The offset within the selected BAR to start the memory or I/O operation.
420 @param Count The number of memory or I/O operations to perform.
421 @param Buffer For read operations, the destination buffer to store the results. For write
422 operations, the source buffer to write data from.
424 @retval EFI_SUCCESS The data was read from or written to the PCI controller.
425 @retval EFI_UNSUPPORTED BarIndex not valid for this PCI controller.
426 @retval EFI_UNSUPPORTED The address range specified by Offset, Width, and Count is not
427 valid for the PCI BAR specified by BarIndex.
428 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
429 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
435 IN EFI_PCI_IO_PROTOCOL
*This
,
436 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
444 PCI_IO_DEVICE
*PciIoDevice
;
446 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
448 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
449 return EFI_INVALID_PARAMETER
;
452 if (Buffer
== NULL
) {
453 return EFI_INVALID_PARAMETER
;
456 Status
= PciIoVerifyBarAccess (PciIoDevice
, BarIndex
, PciBarTypeMem
, Width
, Count
, &Offset
);
457 if (EFI_ERROR (Status
)) {
458 return EFI_UNSUPPORTED
;
461 Status
= PciIoDevice
->PciRootBridgeIo
->Mem
.Write (
462 PciIoDevice
->PciRootBridgeIo
,
463 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
469 if (EFI_ERROR (Status
)) {
470 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_WRITE_ERROR
);
477 Enable a PCI driver to access PCI controller registers in the PCI memory or I/O space.
479 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
480 @param Width Signifies the width of the memory or I/O operations.
481 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
482 base address for the memory or I/O operation to perform.
483 @param Offset The offset within the selected BAR to start the memory or I/O operation.
484 @param Count The number of memory or I/O operations to perform.
485 @param Buffer For read operations, the destination buffer to store the results. For write
486 operations, the source buffer to write data from.
488 @retval EFI_SUCCESS The data was read from or written to the PCI controller.
489 @retval EFI_UNSUPPORTED BarIndex not valid for this PCI controller.
490 @retval EFI_UNSUPPORTED The address range specified by Offset, Width, and Count is not
491 valid for the PCI BAR specified by BarIndex.
492 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
493 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
499 IN EFI_PCI_IO_PROTOCOL
*This
,
500 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
508 PCI_IO_DEVICE
*PciIoDevice
;
510 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
512 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
513 return EFI_INVALID_PARAMETER
;
516 if (Buffer
== NULL
) {
517 return EFI_INVALID_PARAMETER
;
520 Status
= PciIoVerifyBarAccess (PciIoDevice
, BarIndex
, PciBarTypeIo
, Width
, Count
, &Offset
);
521 if (EFI_ERROR (Status
)) {
522 return EFI_UNSUPPORTED
;
525 Status
= PciIoDevice
->PciRootBridgeIo
->Io
.Read (
526 PciIoDevice
->PciRootBridgeIo
,
527 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
533 if (EFI_ERROR (Status
)) {
534 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_READ_ERROR
);
541 Enable a PCI driver to access PCI controller registers in the PCI memory or I/O space.
543 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
544 @param Width Signifies the width of the memory or I/O operations.
545 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
546 base address for the memory or I/O operation to perform.
547 @param Offset The offset within the selected BAR to start the memory or I/O operation.
548 @param Count The number of memory or I/O operations to perform.
549 @param Buffer For read operations, the destination buffer to store the results. For write
550 operations, the source buffer to write data from.
552 @retval EFI_SUCCESS The data was read from or written to the PCI controller.
553 @retval EFI_UNSUPPORTED BarIndex not valid for this PCI controller.
554 @retval EFI_UNSUPPORTED The address range specified by Offset, Width, and Count is not
555 valid for the PCI BAR specified by BarIndex.
556 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
557 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
563 IN EFI_PCI_IO_PROTOCOL
*This
,
564 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
572 PCI_IO_DEVICE
*PciIoDevice
;
574 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
576 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
577 return EFI_INVALID_PARAMETER
;
580 if (Buffer
== NULL
) {
581 return EFI_INVALID_PARAMETER
;
584 Status
= PciIoVerifyBarAccess (PciIoDevice
, BarIndex
, PciBarTypeIo
, Width
, Count
, &Offset
);
585 if (EFI_ERROR (Status
)) {
586 return EFI_UNSUPPORTED
;
589 Status
= PciIoDevice
->PciRootBridgeIo
->Io
.Write (
590 PciIoDevice
->PciRootBridgeIo
,
591 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
597 if (EFI_ERROR (Status
)) {
598 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_WRITE_ERROR
);
605 Enable a PCI driver to access PCI controller registers in PCI configuration space.
607 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
608 @param Width Signifies the width of the memory operations.
609 @param Offset The offset within the PCI configuration space for the PCI controller.
610 @param Count The number of PCI configuration operations to perform.
611 @param Buffer For read operations, the destination buffer to store the results. For write
612 operations, the source buffer to write data from.
615 @retval EFI_SUCCESS The data was read from or written to the PCI controller.
616 @retval EFI_UNSUPPORTED The address range specified by Offset, Width, and Count is not
617 valid for the PCI configuration header of the PCI controller.
618 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
619 @retval EFI_INVALID_PARAMETER Buffer is NULL or Width is invalid.
625 IN EFI_PCI_IO_PROTOCOL
*This
,
626 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
633 PCI_IO_DEVICE
*PciIoDevice
;
636 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
639 Status
= PciIoVerifyConfigAccess (PciIoDevice
, Width
, Count
, &Address
);
640 if (EFI_ERROR (Status
)) {
644 Status
= PciIoDevice
->PciRootBridgeIo
->Pci
.Read (
645 PciIoDevice
->PciRootBridgeIo
,
646 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
652 if (EFI_ERROR (Status
)) {
653 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_READ_ERROR
);
660 Enable a PCI driver to access PCI controller registers in PCI configuration space.
662 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
663 @param Width Signifies the width of the memory operations.
664 @param Offset The offset within the PCI configuration space for the PCI controller.
665 @param Count The number of PCI configuration operations to perform.
666 @param Buffer For read operations, the destination buffer to store the results. For write
667 operations, the source buffer to write data from.
670 @retval EFI_SUCCESS The data was read from or written to the PCI controller.
671 @retval EFI_UNSUPPORTED The address range specified by Offset, Width, and Count is not
672 valid for the PCI configuration header of the PCI controller.
673 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
674 @retval EFI_INVALID_PARAMETER Buffer is NULL or Width is invalid.
680 IN EFI_PCI_IO_PROTOCOL
*This
,
681 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
688 PCI_IO_DEVICE
*PciIoDevice
;
691 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
694 Status
= PciIoVerifyConfigAccess (PciIoDevice
, Width
, Count
, &Address
);
695 if (EFI_ERROR (Status
)) {
699 Status
= PciIoDevice
->PciRootBridgeIo
->Pci
.Write (
700 PciIoDevice
->PciRootBridgeIo
,
701 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
707 if (EFI_ERROR (Status
)) {
708 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_WRITE_ERROR
);
715 Enables a PCI driver to copy one region of PCI memory space to another region of PCI
718 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
719 @param Width Signifies the width of the memory operations.
720 @param DestBarIndex The BAR index in the standard PCI Configuration header to use as the
721 base address for the memory operation to perform.
722 @param DestOffset The destination offset within the BAR specified by DestBarIndex to
723 start the memory writes for the copy operation.
724 @param SrcBarIndex The BAR index in the standard PCI Configuration header to use as the
725 base address for the memory operation to perform.
726 @param SrcOffset The source offset within the BAR specified by SrcBarIndex to start
727 the memory reads for the copy operation.
728 @param Count The number of memory operations to perform. Bytes moved is Width
729 size * Count, starting at DestOffset and SrcOffset.
731 @retval EFI_SUCCESS The data was copied from one memory region to another memory region.
732 @retval EFI_UNSUPPORTED DestBarIndex not valid for this PCI controller.
733 @retval EFI_UNSUPPORTED SrcBarIndex not valid for this PCI controller.
734 @retval EFI_UNSUPPORTED The address range specified by DestOffset, Width, and Count
735 is not valid for the PCI BAR specified by DestBarIndex.
736 @retval EFI_UNSUPPORTED The address range specified by SrcOffset, Width, and Count is
737 not valid for the PCI BAR specified by SrcBarIndex.
738 @retval EFI_INVALID_PARAMETER Width is invalid.
739 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
745 IN EFI_PCI_IO_PROTOCOL
*This
,
746 IN EFI_PCI_IO_PROTOCOL_WIDTH Width
,
747 IN UINT8 DestBarIndex
,
748 IN UINT64 DestOffset
,
749 IN UINT8 SrcBarIndex
,
755 PCI_IO_DEVICE
*PciIoDevice
;
757 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
759 if (Width
< 0 || Width
>= EfiPciIoWidthMaximum
) {
760 return EFI_INVALID_PARAMETER
;
763 if (Width
== EfiPciIoWidthFifoUint8
||
764 Width
== EfiPciIoWidthFifoUint16
||
765 Width
== EfiPciIoWidthFifoUint32
||
766 Width
== EfiPciIoWidthFifoUint64
||
767 Width
== EfiPciIoWidthFillUint8
||
768 Width
== EfiPciIoWidthFillUint16
||
769 Width
== EfiPciIoWidthFillUint32
||
770 Width
== EfiPciIoWidthFillUint64
) {
771 return EFI_INVALID_PARAMETER
;
774 Status
= PciIoVerifyBarAccess (PciIoDevice
, DestBarIndex
, PciBarTypeMem
, Width
, Count
, &DestOffset
);
775 if (EFI_ERROR (Status
)) {
776 return EFI_UNSUPPORTED
;
779 Status
= PciIoVerifyBarAccess (PciIoDevice
, SrcBarIndex
, PciBarTypeMem
, Width
, Count
, &SrcOffset
);
780 if (EFI_ERROR (Status
)) {
781 return EFI_UNSUPPORTED
;
784 Status
= PciIoDevice
->PciRootBridgeIo
->CopyMem (
785 PciIoDevice
->PciRootBridgeIo
,
786 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_WIDTH
) Width
,
792 if (EFI_ERROR (Status
)) {
793 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
800 Provides the PCI controller-Cspecific addresses needed to access system memory.
802 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
803 @param Operation Indicates if the bus master is going to read or write to system memory.
804 @param HostAddress The system memory address to map to the PCI controller.
805 @param NumberOfBytes On input the number of bytes to map. On output the number of bytes
807 @param DeviceAddress The resulting map address for the bus master PCI controller to use to
808 access the hosts HostAddress.
809 @param Mapping A resulting value to pass to Unmap().
811 @retval EFI_SUCCESS The range was mapped for the returned NumberOfBytes.
812 @retval EFI_UNSUPPORTED The HostAddress cannot be mapped as a common buffer.
813 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
814 @retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
815 @retval EFI_DEVICE_ERROR The system hardware could not map the requested address.
821 IN EFI_PCI_IO_PROTOCOL
*This
,
822 IN EFI_PCI_IO_PROTOCOL_OPERATION Operation
,
823 IN VOID
*HostAddress
,
824 IN OUT UINTN
*NumberOfBytes
,
825 OUT EFI_PHYSICAL_ADDRESS
*DeviceAddress
,
830 PCI_IO_DEVICE
*PciIoDevice
;
832 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
834 if (Operation
< 0 || Operation
>= EfiPciIoOperationMaximum
) {
835 return EFI_INVALID_PARAMETER
;
838 if (HostAddress
== NULL
|| NumberOfBytes
== NULL
|| DeviceAddress
== NULL
|| Mapping
== NULL
) {
839 return EFI_INVALID_PARAMETER
;
842 if ((PciIoDevice
->Attributes
& EFI_PCI_IO_ATTRIBUTE_DUAL_ADDRESS_CYCLE
) != 0) {
843 Operation
= (EFI_PCI_IO_PROTOCOL_OPERATION
) (Operation
+ EfiPciOperationBusMasterRead64
);
846 Status
= PciIoDevice
->PciRootBridgeIo
->Map (
847 PciIoDevice
->PciRootBridgeIo
,
848 (EFI_PCI_ROOT_BRIDGE_IO_PROTOCOL_OPERATION
) Operation
,
855 if (EFI_ERROR (Status
)) {
856 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
863 Completes the Map() operation and releases any corresponding resources.
865 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
866 @param Mapping The mapping value returned from Map().
868 @retval EFI_SUCCESS The range was unmapped.
869 @retval EFI_DEVICE_ERROR The data was not committed to the target system memory.
875 IN EFI_PCI_IO_PROTOCOL
*This
,
880 PCI_IO_DEVICE
*PciIoDevice
;
882 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
884 Status
= PciIoDevice
->PciRootBridgeIo
->Unmap (
885 PciIoDevice
->PciRootBridgeIo
,
889 if (EFI_ERROR (Status
)) {
890 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
897 Allocates pages that are suitable for an EfiPciIoOperationBusMasterCommonBuffer
900 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
901 @param Type This parameter is not used and must be ignored.
902 @param MemoryType The type of memory to allocate, EfiBootServicesData or
903 EfiRuntimeServicesData.
904 @param Pages The number of pages to allocate.
905 @param HostAddress A pointer to store the base system memory address of the
907 @param Attributes The requested bit mask of attributes for the allocated range.
909 @retval EFI_SUCCESS The requested memory pages were allocated.
910 @retval EFI_UNSUPPORTED Attributes is unsupported. The only legal attribute bits are
911 MEMORY_WRITE_COMBINE and MEMORY_CACHED.
912 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
913 @retval EFI_OUT_OF_RESOURCES The memory pages could not be allocated.
918 PciIoAllocateBuffer (
919 IN EFI_PCI_IO_PROTOCOL
*This
,
920 IN EFI_ALLOCATE_TYPE Type
,
921 IN EFI_MEMORY_TYPE MemoryType
,
923 OUT VOID
**HostAddress
,
928 PCI_IO_DEVICE
*PciIoDevice
;
931 (~(EFI_PCI_ATTRIBUTE_MEMORY_WRITE_COMBINE
| EFI_PCI_ATTRIBUTE_MEMORY_CACHED
))) != 0){
932 return EFI_UNSUPPORTED
;
935 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
937 if ((PciIoDevice
->Attributes
& EFI_PCI_IO_ATTRIBUTE_DUAL_ADDRESS_CYCLE
) != 0) {
938 Attributes
|= EFI_PCI_ATTRIBUTE_DUAL_ADDRESS_CYCLE
;
941 Status
= PciIoDevice
->PciRootBridgeIo
->AllocateBuffer (
942 PciIoDevice
->PciRootBridgeIo
,
950 if (EFI_ERROR (Status
)) {
951 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
958 Frees memory that was allocated with AllocateBuffer().
960 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
961 @param Pages The number of pages to free.
962 @param HostAddress The base system memory address of the allocated range.
964 @retval EFI_SUCCESS The requested memory pages were freed.
965 @retval EFI_INVALID_PARAMETER The memory range specified by HostAddress and Pages
966 was not allocated with AllocateBuffer().
972 IN EFI_PCI_IO_PROTOCOL
*This
,
978 PCI_IO_DEVICE
*PciIoDevice
;
980 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
982 Status
= PciIoDevice
->PciRootBridgeIo
->FreeBuffer (
983 PciIoDevice
->PciRootBridgeIo
,
988 if (EFI_ERROR (Status
)) {
989 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
996 Flushes all PCI posted write transactions from a PCI host bridge to system memory.
998 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
1000 @retval EFI_SUCCESS The PCI posted write transactions were flushed from the PCI host
1001 bridge to system memory.
1002 @retval EFI_DEVICE_ERROR The PCI posted write transactions were not flushed from the PCI
1003 host bridge due to a hardware error.
1009 IN EFI_PCI_IO_PROTOCOL
*This
1013 PCI_IO_DEVICE
*PciIoDevice
;
1015 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
1017 Status
= PciIoDevice
->PciRootBridgeIo
->Flush (
1018 PciIoDevice
->PciRootBridgeIo
1020 if (EFI_ERROR (Status
)) {
1021 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
1028 Retrieves this PCI controller's current PCI bus number, device number, and function number.
1030 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
1031 @param SegmentNumber The PCI controller's current PCI segment number.
1032 @param BusNumber The PCI controller's current PCI bus number.
1033 @param DeviceNumber The PCI controller's current PCI device number.
1034 @param FunctionNumber The PCI controller's current PCI function number.
1036 @retval EFI_SUCCESS The PCI controller location was returned.
1037 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
1043 IN EFI_PCI_IO_PROTOCOL
*This
,
1050 PCI_IO_DEVICE
*PciIoDevice
;
1052 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
1054 if (Segment
== NULL
|| Bus
== NULL
|| Device
== NULL
|| Function
== NULL
) {
1055 return EFI_INVALID_PARAMETER
;
1058 *Segment
= PciIoDevice
->PciRootBridgeIo
->SegmentNumber
;
1059 *Bus
= PciIoDevice
->BusNumber
;
1060 *Device
= PciIoDevice
->DeviceNumber
;
1061 *Function
= PciIoDevice
->FunctionNumber
;
1067 Check BAR type for PCI resource.
1069 @param PciIoDevice PCI device instance
1070 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
1071 base address for the memory or I/O operation to perform.
1072 @param BarType Memory or I/O
1074 @return whether Pci device's bar type is same with input BarType.
1078 IN PCI_IO_DEVICE
*PciIoDevice
,
1080 PCI_BAR_TYPE BarType
1087 if (PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypeMem32
&&
1088 PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypePMem32
&&
1089 PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypePMem64
&&
1090 PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypeMem64
) {
1097 if (PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypeIo32
&&
1098 PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypeIo16
){
1112 Set/Disable new attributes to a Root Bridge
1114 @param PciIoDevice Pci device instance
1115 @param Attributes New attribute want to be set
1116 @param Operation Set or Disable
1118 @retval EFI_UNSUPPORTED If root bridge does not support change attribute
1119 @retval EFI_SUCCESS Success operation.
1122 ModifyRootBridgeAttributes (
1123 IN PCI_IO_DEVICE
*PciIoDevice
,
1124 IN UINT64 Attributes
,
1125 IN EFI_PCI_IO_PROTOCOL_ATTRIBUTE_OPERATION Operation
1128 UINT64 PciRootBridgeSupports
;
1129 UINT64 PciRootBridgeAttributes
;
1130 UINT64 NewPciRootBridgeAttributes
;
1134 // Get the current attributes of this PCI device's PCI Root Bridge
1136 Status
= PciIoDevice
->PciRootBridgeIo
->GetAttributes (
1137 PciIoDevice
->PciRootBridgeIo
,
1138 &PciRootBridgeSupports
,
1139 &PciRootBridgeAttributes
1141 if (EFI_ERROR (Status
)) {
1142 return EFI_UNSUPPORTED
;
1146 // Record the new attribute of the Root Bridge
1148 if (Operation
== EfiPciIoAttributeOperationEnable
) {
1149 NewPciRootBridgeAttributes
= PciRootBridgeAttributes
| Attributes
;
1151 NewPciRootBridgeAttributes
= PciRootBridgeAttributes
& (~Attributes
);
1155 // Call the PCI Root Bridge to attempt to modify the attributes
1157 if ((NewPciRootBridgeAttributes
^ PciRootBridgeAttributes
) != 0) {
1159 Status
= PciIoDevice
->PciRootBridgeIo
->SetAttributes (
1160 PciIoDevice
->PciRootBridgeIo
,
1161 NewPciRootBridgeAttributes
,
1165 if (EFI_ERROR (Status
)) {
1167 // The PCI Root Bridge could not modify the attributes, so return the error.
1169 return EFI_UNSUPPORTED
;
1174 // Also update the attributes for this Root Bridge structure
1176 PciIoDevice
->Attributes
= NewPciRootBridgeAttributes
;
1182 Check whether this device can be enable/disable to snoop
1184 @param PciIoDevice Pci device instance
1185 @param Operation Enable/Disable
1187 @retval EFI_UNSUPPORTED Pci device is not GFX device or not support snoop
1188 @retval EFI_SUCCESS Snoop can be supported.
1191 SupportPaletteSnoopAttributes (
1192 IN PCI_IO_DEVICE
*PciIoDevice
,
1193 IN EFI_PCI_IO_PROTOCOL_ATTRIBUTE_OPERATION Operation
1196 PCI_IO_DEVICE
*Temp
;
1200 // Snoop attribute can be only modified by GFX
1202 if (!IS_PCI_GFX (&PciIoDevice
->Pci
)) {
1203 return EFI_UNSUPPORTED
;
1207 // Get the boot VGA on the same segement
1209 Temp
= ActiveVGADeviceOnTheSameSegment (PciIoDevice
);
1213 // If there is no VGA device on the segement, set
1214 // this graphics card to decode the palette range
1220 // Check these two agents are on the same path
1222 if (!PciDevicesOnTheSamePath (Temp
, PciIoDevice
)) {
1224 // they are not on the same path, so snoop can be enabled or disabled
1229 // Check if they are on the same bus
1231 if (Temp
->Parent
== PciIoDevice
->Parent
) {
1233 PCI_READ_COMMAND_REGISTER (Temp
, &VGACommand
);
1236 // If they are on the same bus, either one can
1237 // be set to snoop, the other set to decode
1239 if ((VGACommand
& EFI_PCI_COMMAND_VGA_PALETTE_SNOOP
) != 0) {
1241 // VGA has set to snoop, so GFX can be only set to disable snoop
1243 if (Operation
== EfiPciIoAttributeOperationEnable
) {
1244 return EFI_UNSUPPORTED
;
1248 // VGA has disabled to snoop, so GFX can be only enabled
1250 if (Operation
== EfiPciIoAttributeOperationDisable
) {
1251 return EFI_UNSUPPORTED
;
1259 // If they are on the same path but on the different bus
1260 // The first agent is set to snoop, the second one set to
1264 if (Temp
->BusNumber
< PciIoDevice
->BusNumber
) {
1266 // GFX should be set to decode
1268 if (Operation
== EfiPciIoAttributeOperationDisable
) {
1269 PCI_ENABLE_COMMAND_REGISTER (Temp
, EFI_PCI_COMMAND_VGA_PALETTE_SNOOP
);
1270 Temp
->Attributes
|= EFI_PCI_COMMAND_VGA_PALETTE_SNOOP
;
1272 return EFI_UNSUPPORTED
;
1277 // GFX should be set to snoop
1279 if (Operation
== EfiPciIoAttributeOperationEnable
) {
1280 PCI_DISABLE_COMMAND_REGISTER (Temp
, EFI_PCI_COMMAND_VGA_PALETTE_SNOOP
);
1281 Temp
->Attributes
&= (~EFI_PCI_COMMAND_VGA_PALETTE_SNOOP
);
1283 return EFI_UNSUPPORTED
;
1292 Performs an operation on the attributes that this PCI controller supports. The operations include
1293 getting the set of supported attributes, retrieving the current attributes, setting the current
1294 attributes, enabling attributes, and disabling attributes.
1296 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
1297 @param Operation The operation to perform on the attributes for this PCI controller.
1298 @param Attributes The mask of attributes that are used for Set, Enable, and Disable
1300 @param Result A pointer to the result mask of attributes that are returned for the Get
1301 and Supported operations.
1303 @retval EFI_SUCCESS The operation on the PCI controller's attributes was completed.
1304 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
1305 @retval EFI_UNSUPPORTED one or more of the bits set in
1306 Attributes are not supported by this PCI controller or one of
1307 its parent bridges when Operation is Set, Enable or Disable.
1313 IN EFI_PCI_IO_PROTOCOL
* This
,
1314 IN EFI_PCI_IO_PROTOCOL_ATTRIBUTE_OPERATION Operation
,
1315 IN UINT64 Attributes
,
1316 OUT UINT64
*Result OPTIONAL
1321 PCI_IO_DEVICE
*PciIoDevice
;
1322 PCI_IO_DEVICE
*UpStreamBridge
;
1323 PCI_IO_DEVICE
*Temp
;
1326 UINT64 UpStreamAttributes
;
1327 UINT16 BridgeControl
;
1330 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
1332 switch (Operation
) {
1333 case EfiPciIoAttributeOperationGet
:
1334 if (Result
== NULL
) {
1335 return EFI_INVALID_PARAMETER
;
1338 *Result
= PciIoDevice
->Attributes
;
1341 case EfiPciIoAttributeOperationSupported
:
1342 if (Result
== NULL
) {
1343 return EFI_INVALID_PARAMETER
;
1346 *Result
= PciIoDevice
->Supports
;
1349 case EfiPciIoAttributeOperationSet
:
1350 Status
= PciIoDevice
->PciIo
.Attributes (
1351 &(PciIoDevice
->PciIo
),
1352 EfiPciIoAttributeOperationEnable
,
1356 if (EFI_ERROR (Status
)) {
1357 return EFI_UNSUPPORTED
;
1360 Status
= PciIoDevice
->PciIo
.Attributes (
1361 &(PciIoDevice
->PciIo
),
1362 EfiPciIoAttributeOperationDisable
,
1363 (~Attributes
) & (PciIoDevice
->Supports
),
1366 if (EFI_ERROR (Status
)) {
1367 return EFI_UNSUPPORTED
;
1372 case EfiPciIoAttributeOperationEnable
:
1373 case EfiPciIoAttributeOperationDisable
:
1377 return EFI_INVALID_PARAMETER
;
1380 // Just a trick for ENABLE attribute
1381 // EFI_PCI_DEVICE_ENABLE is not defined in UEFI spec, which is the internal usage.
1382 // So, this logic doesn't confrom to UEFI spec, which should be removed.
1383 // But this trick logic is still kept for some binary drivers that depend on it.
1385 if ((Attributes
& EFI_PCI_DEVICE_ENABLE
) == EFI_PCI_DEVICE_ENABLE
) {
1386 Attributes
&= (PciIoDevice
->Supports
);
1389 // Raise the EFI_P_PC_ENABLE Status code
1391 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
1393 EFI_IO_BUS_PCI
| EFI_P_PC_ENABLE
,
1394 PciIoDevice
->DevicePath
1399 // If no attributes can be supported, then return.
1400 // Otherwise, set the attributes that it can support.
1402 Supports
= (PciIoDevice
->Supports
) & Attributes
;
1403 if (Supports
!= Attributes
) {
1404 return EFI_UNSUPPORTED
;
1408 // For Root Bridge, just call RootBridgeIo to set attributes;
1410 if (PciIoDevice
->Parent
== NULL
) {
1411 Status
= ModifyRootBridgeAttributes (PciIoDevice
, Attributes
, Operation
);
1419 // Check VGA and VGA16, they can not be set at the same time
1421 if (((Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_IO
) != 0 &&
1422 (Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_IO_16
) != 0) ||
1423 ((Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_IO
) != 0 &&
1424 (Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO_16
) != 0) ||
1425 ((Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO
) != 0 &&
1426 (Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_IO_16
) != 0) ||
1427 ((Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO
) != 0 &&
1428 (Attributes
& EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO_16
) != 0) ) {
1429 return EFI_UNSUPPORTED
;
1433 // For PPB & P2C, set relevant attribute bits
1435 if (IS_PCI_BRIDGE (&PciIoDevice
->Pci
) || IS_CARDBUS_BRIDGE (&PciIoDevice
->Pci
)) {
1437 if ((Attributes
& (EFI_PCI_IO_ATTRIBUTE_VGA_IO
| EFI_PCI_IO_ATTRIBUTE_VGA_IO_16
)) != 0) {
1438 BridgeControl
|= EFI_PCI_BRIDGE_CONTROL_VGA
;
1441 if ((Attributes
& EFI_PCI_IO_ATTRIBUTE_ISA_IO
) != 0) {
1442 BridgeControl
|= EFI_PCI_BRIDGE_CONTROL_ISA
;
1445 if ((Attributes
& (EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO
| EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO_16
)) != 0) {
1446 Command
|= EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO
;
1449 if ((Attributes
& (EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO_16
| EFI_PCI_IO_ATTRIBUTE_VGA_IO_16
)) != 0) {
1450 BridgeControl
|= EFI_PCI_BRIDGE_CONTROL_VGA_16
;
1455 // Do with the attributes on VGA
1456 // Only for VGA's legacy resource, we just can enable once.
1459 (EFI_PCI_IO_ATTRIBUTE_VGA_IO
|
1460 EFI_PCI_IO_ATTRIBUTE_VGA_IO_16
|
1461 EFI_PCI_IO_ATTRIBUTE_VGA_MEMORY
)) != 0) {
1463 // Check if a VGA has been enabled before enabling a new one
1465 if (Operation
== EfiPciIoAttributeOperationEnable
) {
1467 // Check if there have been an active VGA device on the same segment
1469 Temp
= ActiveVGADeviceOnTheSameSegment (PciIoDevice
);
1470 if (Temp
!= NULL
&& Temp
!= PciIoDevice
) {
1472 // An active VGA has been detected, so can not enable another
1474 return EFI_UNSUPPORTED
;
1480 // Do with the attributes on GFX
1482 if ((Attributes
& (EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO
| EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO_16
)) != 0) {
1484 if (Operation
== EfiPciIoAttributeOperationEnable
) {
1486 // Check if snoop can be enabled in current configuration
1488 Status
= SupportPaletteSnoopAttributes (PciIoDevice
, Operation
);
1490 if (EFI_ERROR (Status
)) {
1493 // Enable operation is forbidden, so mask the bit in attributes
1494 // so as to keep consistent with the actual Status
1496 // Attributes &= (~EFI_PCI_IO_ATTRIBUTE_VGA_PALETTE_IO);
1500 return EFI_UNSUPPORTED
;
1506 // It can be supported, so get ready to set the bit
1508 Command
|= EFI_PCI_COMMAND_VGA_PALETTE_SNOOP
;
1512 if ((Attributes
& EFI_PCI_IO_ATTRIBUTE_IO
) != 0) {
1513 Command
|= EFI_PCI_COMMAND_IO_SPACE
;
1516 if ((Attributes
& EFI_PCI_IO_ATTRIBUTE_MEMORY
) != 0) {
1517 Command
|= EFI_PCI_COMMAND_MEMORY_SPACE
;
1520 if ((Attributes
& EFI_PCI_IO_ATTRIBUTE_BUS_MASTER
) != 0) {
1521 Command
|= EFI_PCI_COMMAND_BUS_MASTER
;
1524 // The upstream bridge should be also set to revelant attribute
1525 // expect for IO, Mem and BusMaster
1527 UpStreamAttributes
= Attributes
&
1528 (~(EFI_PCI_IO_ATTRIBUTE_IO
|
1529 EFI_PCI_IO_ATTRIBUTE_MEMORY
|
1530 EFI_PCI_IO_ATTRIBUTE_BUS_MASTER
1533 UpStreamBridge
= PciIoDevice
->Parent
;
1535 if (Operation
== EfiPciIoAttributeOperationEnable
) {
1537 // Enable relevant attributes to command register and bridge control register
1539 Status
= PCI_ENABLE_COMMAND_REGISTER (PciIoDevice
, Command
);
1540 if (BridgeControl
!= 0) {
1541 Status
= PCI_ENABLE_BRIDGE_CONTROL_REGISTER (PciIoDevice
, BridgeControl
);
1544 PciIoDevice
->Attributes
|= Attributes
;
1547 // Enable attributes of the upstream bridge
1549 Status
= UpStreamBridge
->PciIo
.Attributes (
1550 &(UpStreamBridge
->PciIo
),
1551 EfiPciIoAttributeOperationEnable
,
1558 // Disable relevant attributes to command register and bridge control register
1560 Status
= PCI_DISABLE_COMMAND_REGISTER (PciIoDevice
, Command
);
1561 if (BridgeControl
!= 0) {
1562 Status
= PCI_DISABLE_BRIDGE_CONTROL_REGISTER (PciIoDevice
, BridgeControl
);
1565 PciIoDevice
->Attributes
&= (~Attributes
);
1566 Status
= EFI_SUCCESS
;
1570 if (EFI_ERROR (Status
)) {
1571 ReportErrorStatusCode (PciIoDevice
, EFI_IO_BUS_PCI
| EFI_IOB_EC_CONTROLLER_ERROR
);
1578 Gets the attributes that this PCI controller supports setting on a BAR using
1579 SetBarAttributes(), and retrieves the list of resource descriptors for a BAR.
1581 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
1582 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
1583 base address for resource range. The legal range for this field is 0..5.
1584 @param Supports A pointer to the mask of attributes that this PCI controller supports
1585 setting for this BAR with SetBarAttributes().
1586 @param Resources A pointer to the ACPI 2.0 resource descriptors that describe the current
1587 configuration of this BAR of the PCI controller.
1589 @retval EFI_SUCCESS If Supports is not NULL, then the attributes that the PCI
1590 controller supports are returned in Supports. If Resources
1591 is not NULL, then the ACPI 2.0 resource descriptors that the PCI
1592 controller is currently using are returned in Resources.
1593 @retval EFI_INVALID_PARAMETER Both Supports and Attributes are NULL.
1594 @retval EFI_UNSUPPORTED BarIndex not valid for this PCI controller.
1595 @retval EFI_OUT_OF_RESOURCES There are not enough resources available to allocate
1601 PciIoGetBarAttributes (
1602 IN EFI_PCI_IO_PROTOCOL
* This
,
1604 OUT UINT64
*Supports
, OPTIONAL
1605 OUT VOID
**Resources OPTIONAL
1609 UINT8
*Configuration
;
1611 PCI_IO_DEVICE
*PciIoDevice
;
1612 EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
*Ptr
;
1613 EFI_ACPI_END_TAG_DESCRIPTOR
*PtrEnd
;
1617 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
1619 if (Supports
== NULL
&& Resources
== NULL
) {
1620 return EFI_INVALID_PARAMETER
;
1623 if (BarIndex
>= PCI_MAX_BAR
) {
1624 return EFI_UNSUPPORTED
;
1628 // This driver does not support modifications to the WRITE_COMBINE or
1629 // CACHED attributes for BAR ranges.
1631 if (Supports
!= NULL
) {
1632 *Supports
= PciIoDevice
->Supports
& EFI_PCI_IO_ATTRIBUTE_MEMORY_CACHED
& EFI_PCI_IO_ATTRIBUTE_MEMORY_WRITE_COMBINE
;
1635 if (Resources
!= NULL
) {
1637 if (PciIoDevice
->PciBar
[BarIndex
].BarType
!= PciBarTypeUnknown
) {
1641 Configuration
= AllocatePool (sizeof (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
) * NumConfig
+ sizeof (EFI_ACPI_END_TAG_DESCRIPTOR
));
1642 if (Configuration
== NULL
) {
1643 return EFI_OUT_OF_RESOURCES
;
1648 sizeof (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
) * NumConfig
+ sizeof (EFI_ACPI_END_TAG_DESCRIPTOR
)
1651 Ptr
= (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
*) Configuration
;
1653 if (NumConfig
== 1) {
1654 Ptr
->Desc
= ACPI_ADDRESS_SPACE_DESCRIPTOR
;
1655 Ptr
->Len
= sizeof (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
) - 3;
1657 Ptr
->AddrRangeMin
= PciIoDevice
->PciBar
[BarIndex
].BaseAddress
;
1658 Ptr
->AddrLen
= PciIoDevice
->PciBar
[BarIndex
].Length
;
1659 Ptr
->AddrRangeMax
= PciIoDevice
->PciBar
[BarIndex
].Alignment
;
1661 switch (PciIoDevice
->PciBar
[BarIndex
].BarType
) {
1662 case PciBarTypeIo16
:
1663 case PciBarTypeIo32
:
1667 Ptr
->ResType
= ACPI_ADDRESS_SPACE_TYPE_IO
;
1670 case PciBarTypeMem32
:
1674 Ptr
->ResType
= ACPI_ADDRESS_SPACE_TYPE_MEM
;
1678 Ptr
->AddrSpaceGranularity
= 32;
1681 case PciBarTypePMem32
:
1685 Ptr
->ResType
= ACPI_ADDRESS_SPACE_TYPE_MEM
;
1689 Ptr
->SpecificFlag
= 0x6;
1693 Ptr
->AddrSpaceGranularity
= 32;
1696 case PciBarTypeMem64
:
1700 Ptr
->ResType
= ACPI_ADDRESS_SPACE_TYPE_MEM
;
1704 Ptr
->AddrSpaceGranularity
= 64;
1707 case PciBarTypePMem64
:
1711 Ptr
->ResType
= ACPI_ADDRESS_SPACE_TYPE_MEM
;
1715 Ptr
->SpecificFlag
= 0x6;
1719 Ptr
->AddrSpaceGranularity
= 64;
1726 Ptr
= (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
*) ((UINT8
*) Ptr
+ sizeof (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR
));
1732 PtrEnd
= (EFI_ACPI_END_TAG_DESCRIPTOR
*) ((UINT8
*) Ptr
);
1733 PtrEnd
->Desc
= ACPI_END_TAG_DESCRIPTOR
;
1734 PtrEnd
->Checksum
= 0;
1736 *Resources
= Configuration
;
1743 Sets the attributes for a range of a BAR on a PCI controller.
1745 @param This A pointer to the EFI_PCI_IO_PROTOCOL instance.
1746 @param Attributes The mask of attributes to set for the resource range specified by
1747 BarIndex, Offset, and Length.
1748 @param BarIndex The BAR index of the standard PCI Configuration header to use as the
1749 base address for resource range. The legal range for this field is 0..5.
1750 @param Offset A pointer to the BAR relative base address of the resource range to be
1751 modified by the attributes specified by Attributes.
1752 @param Length A pointer to the length of the resource range to be modified by the
1753 attributes specified by Attributes.
1755 @retval EFI_SUCCESS The set of attributes specified by Attributes for the resource
1756 range specified by BarIndex, Offset, and Length were
1757 set on the PCI controller, and the actual resource range is returned
1758 in Offset and Length.
1759 @retval EFI_INVALID_PARAMETER Offset or Length is NULL.
1760 @retval EFI_UNSUPPORTED BarIndex not valid for this PCI controller.
1761 @retval EFI_OUT_OF_RESOURCES There are not enough resources to set the attributes on the
1762 resource range specified by BarIndex, Offset, and
1768 PciIoSetBarAttributes (
1769 IN EFI_PCI_IO_PROTOCOL
*This
,
1770 IN UINT64 Attributes
,
1772 IN OUT UINT64
*Offset
,
1773 IN OUT UINT64
*Length
1777 PCI_IO_DEVICE
*PciIoDevice
;
1778 UINT64 NonRelativeOffset
;
1781 PciIoDevice
= PCI_IO_DEVICE_FROM_PCI_IO_THIS (This
);
1784 // Make sure Offset and Length are not NULL
1786 if (Offset
== NULL
|| Length
== NULL
) {
1787 return EFI_INVALID_PARAMETER
;
1790 if (PciIoDevice
->PciBar
[BarIndex
].BarType
== PciBarTypeUnknown
) {
1791 return EFI_UNSUPPORTED
;
1794 // This driver does not support setting the WRITE_COMBINE or the CACHED attributes.
1795 // If Attributes is not 0, then return EFI_UNSUPPORTED.
1797 Supports
= PciIoDevice
->Supports
& EFI_PCI_IO_ATTRIBUTE_MEMORY_CACHED
& EFI_PCI_IO_ATTRIBUTE_MEMORY_WRITE_COMBINE
;
1799 if (Attributes
!= (Attributes
& Supports
)) {
1800 return EFI_UNSUPPORTED
;
1803 // Attributes must be supported. Make sure the BAR range describd by BarIndex, Offset, and
1804 // Length are valid for this PCI device.
1806 NonRelativeOffset
= *Offset
;
1807 Status
= PciIoVerifyBarAccess (
1815 if (EFI_ERROR (Status
)) {
1816 return EFI_UNSUPPORTED
;
1823 Program parent bridge's attribute recurrently.
1825 @param PciIoDevice Child Pci device instance
1826 @param Operation The operation to perform on the attributes for this PCI controller.
1827 @param Attributes The mask of attributes that are used for Set, Enable, and Disable
1830 @retval EFI_SUCCESS The operation on the PCI controller's attributes was completed.
1831 @retval EFI_INVALID_PARAMETER One or more parameters are invalid.
1832 @retval EFI_UNSUPPORTED one or more of the bits set in
1833 Attributes are not supported by this PCI controller or one of
1834 its parent bridges when Operation is Set, Enable or Disable.
1838 UpStreamBridgesAttributes (
1839 IN PCI_IO_DEVICE
*PciIoDevice
,
1840 IN EFI_PCI_IO_PROTOCOL_ATTRIBUTE_OPERATION Operation
,
1841 IN UINT64 Attributes
1844 PCI_IO_DEVICE
*Parent
;
1845 EFI_PCI_IO_PROTOCOL
*PciIo
;
1847 Parent
= PciIoDevice
->Parent
;
1849 while (Parent
!= NULL
&& IS_PCI_BRIDGE (&Parent
->Pci
)) {
1852 // Get the PciIo Protocol
1854 PciIo
= &Parent
->PciIo
;
1856 PciIo
->Attributes (PciIo
, Operation
, Attributes
, NULL
);
1858 Parent
= Parent
->Parent
;
1865 Test whether two Pci device has same parent bridge.
1867 @param PciDevice1 the frist pci device for testing
1868 @param PciDevice2 the second pci device for testing
1870 @return whether two Pci device has same parent bridge.
1873 PciDevicesOnTheSamePath (
1874 IN PCI_IO_DEVICE
*PciDevice1
,
1875 IN PCI_IO_DEVICE
*PciDevice2
1881 if (PciDevice1
->Parent
== PciDevice2
->Parent
) {
1885 Existed1
= PciDeviceExisted (PciDevice1
->Parent
, PciDevice2
);
1886 Existed2
= PciDeviceExisted (PciDevice2
->Parent
, PciDevice1
);
1888 return (BOOLEAN
) (Existed1
|| Existed2
);