mirror of https://github.com/acidanthera/audk.git
DynamicTablesPkg: SSDT Pci express generator
This generator allows to generate a SSDT table describing a Pci express Bus. It uses the following CmObj: - EArmObjCmRef - EArmObjPciConfigSpaceInfo - EArmObjPciAddressMapInfo - EArmObjPciInterruptMapInfo REF: https://bugzilla.tianocore.org/show_bug.cgi?id=3682 To: Sami Mujawar <sami.mujawar@arm.com> To: Alexei Fedorov <Alexei.Fedorov@arm.com> Signed-off-by: Pierre Gondois <Pierre.Gondois@arm.com> Reviewed-by: Sami Mujawar <sami.mujawar@arm.com>
This commit is contained in:
parent
ce306e48eb
commit
e35a746cf5
|
@ -39,6 +39,7 @@
|
|||
|
||||
# AML Codegen
|
||||
DynamicTablesPkg/Library/Acpi/Arm/AcpiSsdtCpuTopologyLibArm/SsdtCpuTopologyLibArm.inf
|
||||
DynamicTablesPkg/Library/Acpi/Arm/AcpiSsdtPcieLibArm/SsdtPcieLibArm.inf
|
||||
|
||||
#
|
||||
# Dynamic Table Factory Dxe
|
||||
|
@ -62,6 +63,7 @@
|
|||
|
||||
# AML Codegen
|
||||
NULL|DynamicTablesPkg/Library/Acpi/Arm/AcpiSsdtCpuTopologyLibArm/SsdtCpuTopologyLibArm.inf
|
||||
NULL|DynamicTablesPkg/Library/Acpi/Arm/AcpiSsdtPcieLibArm/SsdtPcieLibArm.inf
|
||||
}
|
||||
|
||||
#
|
||||
|
|
|
@ -67,6 +67,10 @@ The Dynamic Tables Framework implements the following ACPI table generators:
|
|||
The SSDT Cpu-Topology generator collates the cpu and LPI
|
||||
information from the Configuration Manager and generates a
|
||||
SSDT table describing the CPU hierarchy.
|
||||
- SSDT Pci-Express:
|
||||
The SSDT Pci Express generator collates the Pci Express
|
||||
information from the Configuration Manager and generates a
|
||||
SSDT table describing a Pci Express bus.
|
||||
*/
|
||||
|
||||
/** The ACPI_TABLE_GENERATOR_ID type describes ACPI table generator ID.
|
||||
|
@ -93,6 +97,7 @@ typedef enum StdAcpiTableId {
|
|||
EStdAcpiTableIdSsdtSerialPort, ///< SSDT Serial-Port Generator
|
||||
EStdAcpiTableIdSsdtCmn600, ///< SSDT Cmn-600 Generator
|
||||
EStdAcpiTableIdSsdtCpuTopology, ///< SSDT Cpu Topology
|
||||
EStdAcpiTableIdSsdtPciExpress, ///< SSDT Pci Express Generator
|
||||
EStdAcpiTableIdMax
|
||||
} ESTD_ACPI_TABLE_ID;
|
||||
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,138 @@
|
|||
/** @file
|
||||
SSDT Pcie Table Generator.
|
||||
|
||||
Copyright (c) 2021, Arm Limited. All rights reserved.<BR>
|
||||
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
|
||||
@par Reference(s):
|
||||
- PCI Firmware Specification - Revision 3.0
|
||||
- ACPI 6.4 specification:
|
||||
- s6.2.13 "_PRT (PCI Routing Table)"
|
||||
- s6.1.1 "_ADR (Address)"
|
||||
- linux kernel code
|
||||
- Arm Base Boot Requirements v1.0
|
||||
**/
|
||||
|
||||
#ifndef SSDT_PCIE_GENERATOR_H_
|
||||
#define SSDT_PCIE_GENERATOR_H_
|
||||
|
||||
/** Pci address attributes.
|
||||
|
||||
This can also be denoted as space code, address space or ss.
|
||||
*/
|
||||
#define PCI_SS_CONFIG 0
|
||||
#define PCI_SS_IO 1
|
||||
#define PCI_SS_M32 2
|
||||
#define PCI_SS_M64 3
|
||||
|
||||
/** Maximum Pci root complexes supported by this generator.
|
||||
|
||||
Note: This is not a hard limitation and can be extended if needed.
|
||||
Corresponding changes would be needed to support the Name and
|
||||
UID fields describing the Pci root complexes.
|
||||
*/
|
||||
#define MAX_PCI_ROOT_COMPLEXES_SUPPORTED 16
|
||||
|
||||
/** Maximum number of Pci legacy interrupts.
|
||||
|
||||
Currently 4 for INTA-INTB-INTC-INTD.
|
||||
*/
|
||||
#define MAX_PCI_LEGACY_INTERRUPT 4
|
||||
|
||||
// _SB scope of the AML namespace.
|
||||
#define SB_SCOPE "\\_SB_"
|
||||
|
||||
/** C array containing the compiled AML template.
|
||||
This symbol is defined in the auto generated C file
|
||||
containing the AML bytecode array.
|
||||
*/
|
||||
extern CHAR8 ssdtpcieosctemplate_aml_code[];
|
||||
|
||||
#pragma pack(1)
|
||||
|
||||
/** Structure used to map integer to an index.
|
||||
*/
|
||||
typedef struct MappingTable {
|
||||
/// Mapping table.
|
||||
/// Contains the Index <-> integer mapping
|
||||
UINT32 *Table;
|
||||
|
||||
/// Last used index of the Table.
|
||||
/// Bound by MaxIndex.
|
||||
UINT32 LastIndex;
|
||||
|
||||
/// Number of entries in the Table.
|
||||
UINT32 MaxIndex;
|
||||
} MAPPING_TABLE;
|
||||
|
||||
/** A structure holding the Pcie generator and additional private data.
|
||||
*/
|
||||
typedef struct AcpiPcieGenerator {
|
||||
/// ACPI Table generator header
|
||||
ACPI_TABLE_GENERATOR Header;
|
||||
|
||||
// Private fields are defined from here.
|
||||
|
||||
/** A structure used to handle the Address and Interrupt Map referencing.
|
||||
|
||||
A CM_ARM_PCI_CONFIG_SPACE_INFO structure references two CM_ARM_OBJ_REF:
|
||||
- one for the address mapping, referencing
|
||||
CM_ARM_PCI_ADDRESS_MAP_INFO structures.
|
||||
- one for the interrupt mapping, referencing
|
||||
CM_ARM_PCI_INTERRUPT_MAP_INFO structures.
|
||||
|
||||
Example:
|
||||
(Pci0)
|
||||
CM_ARM_PCI_CONFIG_SPACE_INFO
|
||||
|
|
||||
+----------------------------------------
|
||||
| |
|
||||
v v
|
||||
CM_ARM_OBJ_REF CM_ARM_OBJ_REF
|
||||
(List of references to (List of references to
|
||||
address mappings) interrupt mappings)
|
||||
| |
|
||||
v v
|
||||
CM_ARM_PCI_ADDRESS_MAP_INFO[0..N] CM_ARM_PCI_INTERRUPT_MAP_INFO[0..M]
|
||||
(A list of address mappings) (A list of interrupt mappings)
|
||||
|
||||
The CM_ARM_PCI_INTERRUPT_MAP_INFO objects cannot be handled individually.
|
||||
Device's Pci legacy interrupts that are mapped to the same CPU interrupt
|
||||
are grouped under a Link device.
|
||||
For instance, the following mapping:
|
||||
- [INTA of device 0] mapped on [GIC irq 168]
|
||||
- [INTB of device 1] mapped on [GIC irq 168]
|
||||
will be represented in an SSDT table as:
|
||||
- [INTA of device 0] mapped on [Link device A]
|
||||
- [INTB of device 1] mapped on [Link device A]
|
||||
- [Link device A] mapped on [GIC irq 168]
|
||||
|
||||
Counting the number of Cpu interrupts used and grouping them in Link
|
||||
devices is done through this IRQ_TABLE.
|
||||
|
||||
ASL code:
|
||||
Scope (_SB) {
|
||||
Device (LNKA) {
|
||||
[...]
|
||||
Name (_PRS, ResourceTemplate () {
|
||||
Interrupt (ResourceProducer, Level, ActiveHigh, Exclusive) { 168 }
|
||||
})
|
||||
}
|
||||
|
||||
Device (PCI0) {
|
||||
Name (_PRT, Package () {
|
||||
Package (0x0FFFF, 0, LNKA, 0) // INTA of device 0 <-> LNKA
|
||||
Package (0x1FFFF, 1, LNKA, 0) // INTB of device 1 <-> LNKA
|
||||
})
|
||||
}
|
||||
}
|
||||
*/
|
||||
MAPPING_TABLE IrqTable;
|
||||
|
||||
/// Table to map: Index <-> Pci device
|
||||
MAPPING_TABLE DeviceTable;
|
||||
} ACPI_PCI_GENERATOR;
|
||||
|
||||
#pragma pack()
|
||||
|
||||
#endif // SSDT_PCIE_GENERATOR_H_
|
|
@ -0,0 +1,32 @@
|
|||
## @file
|
||||
# Ssdt Serial Port Table Generator
|
||||
#
|
||||
# Copyright (c) 2021, Arm Limited. All rights reserved.<BR>
|
||||
#
|
||||
# SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
##
|
||||
|
||||
[Defines]
|
||||
INF_VERSION = 0x0001001B
|
||||
BASE_NAME = SsdtPcieLibArm
|
||||
FILE_GUID = E431D7FD-26BF-4E3D-9064-5B13B0439057
|
||||
VERSION_STRING = 1.0
|
||||
MODULE_TYPE = DXE_DRIVER
|
||||
LIBRARY_CLASS = NULL|DXE_DRIVER
|
||||
CONSTRUCTOR = AcpiSsdtPcieLibConstructor
|
||||
DESTRUCTOR = AcpiSsdtPcieLibDestructor
|
||||
|
||||
[Sources]
|
||||
SsdtPcieGenerator.c
|
||||
SsdtPcieGenerator.h
|
||||
SsdtPcieOscTemplate.asl
|
||||
|
||||
[Packages]
|
||||
DynamicTablesPkg/DynamicTablesPkg.dec
|
||||
EmbeddedPkg/EmbeddedPkg.dec
|
||||
MdePkg/MdePkg.dec
|
||||
|
||||
[LibraryClasses]
|
||||
AcpiHelperLib
|
||||
AmlLib
|
||||
BaseLib
|
|
@ -0,0 +1,80 @@
|
|||
/** @file
|
||||
SSDT Pci Osc (Operating System Capabilities)
|
||||
|
||||
Copyright (c) 2021, Arm Limited. All rights reserved.<BR>
|
||||
|
||||
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||
|
||||
@par Reference(s):
|
||||
- PCI Firmware Specification - Revision 3.3
|
||||
- ACPI 6.4 specification:
|
||||
- s6.2.13 "_PRT (PCI Routing Table)"
|
||||
- s6.1.1 "_ADR (Address)"
|
||||
- linux kernel code
|
||||
**/
|
||||
|
||||
DefinitionBlock ("SsdtPciOsc.aml", "SSDT", 2, "ARMLTD", "PCI-OSC", 1) {
|
||||
|
||||
// This table is just a template and is never installed as a table.
|
||||
// Pci devices are dynamically created at runtime as:
|
||||
// ASL:
|
||||
// Device (PCIx) {
|
||||
// ...
|
||||
// }
|
||||
// and the _OSC method available below is appended to the PCIx device as:
|
||||
// ASL:
|
||||
// Device (PCIx) {
|
||||
// ...
|
||||
// Method (_OSC, 4 {
|
||||
// ...
|
||||
// })
|
||||
// }
|
||||
Method (_OSC, 4) {
|
||||
//
|
||||
// OS Control Handoff
|
||||
//
|
||||
Name (SUPP, Zero) // PCI _OSC Support Field value
|
||||
Name (CTRL, Zero) // PCI _OSC Control Field value
|
||||
|
||||
// Create DWord-addressable fields from the Capabilities Buffer
|
||||
CreateDWordField (Arg3, 0, CDW1)
|
||||
CreateDWordField (Arg3, 4, CDW2)
|
||||
CreateDWordField (Arg3, 8, CDW3)
|
||||
|
||||
// Check for proper UUID
|
||||
If (LEqual (Arg0,ToUUID ("33DB4D5B-1FF7-401C-9657-7441C03DD766"))) {
|
||||
|
||||
// Save Capabilities DWord2 & 3
|
||||
Store (CDW2, SUPP)
|
||||
Store (CDW3, CTRL)
|
||||
|
||||
// Only allow native hot plug control if OS supports:
|
||||
// * ASPM
|
||||
// * Clock PM
|
||||
// * MSI/MSI-X
|
||||
If (LNotEqual (And (SUPP, 0x16), 0x16)) {
|
||||
And (CTRL, 0x1E, CTRL) // Mask bit 0 (and undefined bits)
|
||||
}
|
||||
|
||||
// Always allow native PME, AER (no dependencies)
|
||||
|
||||
// Never allow SHPC (no SHPC controller in this system)
|
||||
And (CTRL, 0x1D, CTRL)
|
||||
|
||||
If (LNotEqual (Arg1, One)) { // Unknown revision
|
||||
Or (CDW1, 0x08, CDW1)
|
||||
}
|
||||
|
||||
If (LNotEqual (CDW3, CTRL)) { // Capabilities bits were masked
|
||||
Or (CDW1, 0x10, CDW1)
|
||||
}
|
||||
|
||||
// Update DWORD3 in the buffer
|
||||
Store (CTRL,CDW3)
|
||||
Return (Arg3)
|
||||
} Else {
|
||||
Or (CDW1, 4, CDW1) // Unrecognized UUID
|
||||
Return (Arg3)
|
||||
} // If
|
||||
} // _OSC
|
||||
}
|
Loading…
Reference in New Issue