2018-06-04 03:14:51 +02:00
|
|
|
/** @file
|
2018-04-20 10:08:22 +02:00
|
|
|
MADT table parser
|
|
|
|
|
2019-05-31 05:06:51 +02:00
|
|
|
Copyright (c) 2016 - 2019, ARM Limited. All rights reserved.
|
2019-04-04 01:07:06 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
@par Reference(s):
|
2019-05-31 05:06:51 +02:00
|
|
|
- ACPI 6.3 Specification - January 2019
|
|
|
|
- Arm Generic Interrupt Controller Architecture Specification,
|
|
|
|
GIC architecture version 3 and version 4, issue E
|
|
|
|
- Arm Server Base System Architecture 5.0
|
2018-04-20 10:08:22 +02:00
|
|
|
**/
|
|
|
|
|
|
|
|
#include <IndustryStandard/Acpi.h>
|
|
|
|
#include <Library/UefiLib.h>
|
|
|
|
#include "AcpiParser.h"
|
|
|
|
#include "AcpiTableParser.h"
|
2019-05-31 05:06:51 +02:00
|
|
|
#include "MadtParser.h"
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
// Local Variables
|
|
|
|
STATIC CONST UINT8* MadtInterruptControllerType;
|
|
|
|
STATIC CONST UINT8* MadtInterruptControllerLength;
|
|
|
|
STATIC ACPI_DESCRIPTION_HEADER_INFO AcpiHdrInfo;
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function validates the System Vector Base in the GICD.
|
|
|
|
|
|
|
|
@param [in] Ptr Pointer to the start of the field data.
|
|
|
|
@param [in] Context Pointer to context specific information e.g. this
|
|
|
|
could be a pointer to the ACPI table header.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ValidateGICDSystemVectorBase (
|
|
|
|
IN UINT8* Ptr,
|
|
|
|
IN VOID* Context
|
2019-07-19 03:04:59 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
if (*(UINT32*)Ptr != 0) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: System Vector Base must be zero."
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2018-04-20 10:08:22 +02:00
|
|
|
|
2019-05-31 05:06:51 +02:00
|
|
|
/**
|
|
|
|
This function validates the SPE Overflow Interrupt in the GICC.
|
|
|
|
|
|
|
|
@param [in] Ptr Pointer to the start of the field data.
|
|
|
|
@param [in] Context Pointer to context specific information e.g. this
|
|
|
|
could be a pointer to the ACPI table header.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ValidateSpeOverflowInterrupt (
|
|
|
|
IN UINT8* Ptr,
|
|
|
|
IN VOID* Context
|
2019-07-19 03:04:59 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT16 SpeOverflowInterrupt;
|
|
|
|
|
|
|
|
SpeOverflowInterrupt = *(UINT16*)Ptr;
|
|
|
|
|
|
|
|
// SPE not supported by this processor
|
|
|
|
if (SpeOverflowInterrupt == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((SpeOverflowInterrupt < ARM_PPI_ID_MIN) ||
|
|
|
|
((SpeOverflowInterrupt > ARM_PPI_ID_MAX) &&
|
|
|
|
(SpeOverflowInterrupt < ARM_PPI_ID_EXTENDED_MIN)) ||
|
|
|
|
(SpeOverflowInterrupt > ARM_PPI_ID_EXTENDED_MAX)) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: SPE Overflow Interrupt ID of %d is not in the allowed PPI ID "
|
|
|
|
L"ranges of %d-%d or %d-%d (for GICv3.1 or later).",
|
|
|
|
SpeOverflowInterrupt,
|
|
|
|
ARM_PPI_ID_MIN,
|
|
|
|
ARM_PPI_ID_MAX,
|
|
|
|
ARM_PPI_ID_EXTENDED_MIN,
|
|
|
|
ARM_PPI_ID_EXTENDED_MAX
|
|
|
|
);
|
|
|
|
} else if (SpeOverflowInterrupt != ARM_PPI_ID_PMBIRQ) {
|
|
|
|
IncrementWarningCount();
|
|
|
|
Print (
|
|
|
|
L"\nWARNING: SPE Overflow Interrupt ID of %d is not compliant with SBSA "
|
|
|
|
L"Level 3 PPI ID assignment: %d.",
|
|
|
|
SpeOverflowInterrupt,
|
|
|
|
ARM_PPI_ID_PMBIRQ
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2019-05-31 05:06:51 +02:00
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the GICC Interrupt Controller Structure.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER GicCParser[] = {
|
|
|
|
{L"Type", 1, 0, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Length", 1, 1, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"CPU Interface Number", 4, 4, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"ACPI Processor UID", 4, 8, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Flags", 4, 12, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Parking Protocol Version", 4, 16, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"Performance Interrupt GSIV", 4, 20, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Parked Address", 8, 24, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Physical Base Address", 8, 32, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"GICV", 8, 40, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"GICH", 8, 48, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"VGIC Maintenance interrupt", 4, 56, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"GICR Base Address", 8, 60, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"MPIDR", 8, 68, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Processor Power Efficiency Class", 1, 76, L"0x%x", NULL, NULL, NULL,
|
|
|
|
NULL},
|
2019-05-31 05:06:51 +02:00
|
|
|
{L"Reserved", 1, 77, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"SPE overflow Interrupt", 2, 78, L"0x%x", NULL, NULL,
|
|
|
|
ValidateSpeOverflowInterrupt, NULL}
|
2018-04-20 10:08:22 +02:00
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the GICD Interrupt Controller Structure.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER GicDParser[] = {
|
|
|
|
{L"Type", 1, 0, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Length", 1, 1, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"GIC ID", 4, 4, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Physical Base Address", 8, 8, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"System Vector Base", 4, 16, L"0x%x", NULL, NULL,
|
|
|
|
ValidateGICDSystemVectorBase, NULL},
|
|
|
|
{L"GIC Version", 1, 20, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 3, 21, L"%x %x %x", Dump3Chars, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the MSI Frame Interrupt Controller Structure.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER GicMSIFrameParser[] = {
|
|
|
|
{L"Type", 1, 0, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Length", 1, 1, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"MSI Frame ID", 4, 4, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Physical Base Address", 8, 8, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Flags", 4, 16, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"SPI Count", 2, 20, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"SPI Base", 2, 22, L"0x%x", NULL, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the GICR Interrupt Controller Structure.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER GicRParser[] = {
|
|
|
|
{L"Type", 1, 0, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Length", 1, 1, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"Discovery Range Base Address", 8, 4, L"0x%lx", NULL, NULL, NULL,
|
|
|
|
NULL},
|
|
|
|
{L"Discovery Range Length", 4, 12, L"0x%x", NULL, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the GIC ITS Interrupt Controller Structure.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER GicITSParser[] = {
|
|
|
|
{L"Type", 1, 0, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Length", 1, 1, L"%d", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
{L"GIC ITS ID", 4, 4, L"0x%x", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Physical Base Address", 8, 8, L"0x%lx", NULL, NULL, NULL, NULL},
|
|
|
|
{L"Reserved", 4, 16, L"0x%x", NULL, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the ACPI MADT Table.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER MadtParser[] = {
|
|
|
|
PARSE_ACPI_HEADER (&AcpiHdrInfo),
|
|
|
|
{L"Local Interrupt Controller Address", 4, 36, L"0x%x", NULL, NULL, NULL,
|
|
|
|
NULL},
|
|
|
|
{L"Flags", 4, 40, L"0x%x", NULL, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the MADT Interrupt Controller Structure Header Structure.
|
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
STATIC CONST ACPI_PARSER MadtInterruptControllerHeaderParser[] = {
|
|
|
|
{NULL, 1, 0, NULL, NULL, (VOID**)&MadtInterruptControllerType, NULL, NULL},
|
|
|
|
{L"Length", 1, 1, NULL, NULL, (VOID**)&MadtInterruptControllerLength, NULL,
|
|
|
|
NULL},
|
|
|
|
{L"Reserved", 2, 2, NULL, NULL, NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2018-06-04 03:14:51 +02:00
|
|
|
/**
|
|
|
|
This function parses the ACPI MADT table.
|
2018-04-20 10:08:22 +02:00
|
|
|
When trace is enabled this function parses the MADT table and
|
|
|
|
traces the ACPI table fields.
|
|
|
|
|
|
|
|
This function currently parses the following Interrupt Controller
|
|
|
|
Structures:
|
|
|
|
- GICC
|
|
|
|
- GICD
|
|
|
|
- GIC MSI Frame
|
|
|
|
- GICR
|
|
|
|
- GIC ITS
|
|
|
|
|
|
|
|
This function also performs validation of the ACPI table fields.
|
|
|
|
|
|
|
|
@param [in] Trace If TRUE, trace the ACPI fields.
|
|
|
|
@param [in] Ptr Pointer to the start of the buffer.
|
|
|
|
@param [in] AcpiTableLength Length of the ACPI table.
|
|
|
|
@param [in] AcpiTableRevision Revision of the ACPI table.
|
2018-06-04 03:14:51 +02:00
|
|
|
**/
|
2018-04-20 10:08:22 +02:00
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ParseAcpiMadt (
|
|
|
|
IN BOOLEAN Trace,
|
|
|
|
IN UINT8* Ptr,
|
|
|
|
IN UINT32 AcpiTableLength,
|
|
|
|
IN UINT8 AcpiTableRevision
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 Offset;
|
|
|
|
UINT8* InterruptContollerPtr;
|
2018-06-05 03:20:05 +02:00
|
|
|
UINT32 GICDCount;
|
|
|
|
|
|
|
|
GICDCount = 0;
|
2018-04-20 10:08:22 +02:00
|
|
|
|
|
|
|
if (!Trace) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Offset = ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
0,
|
|
|
|
"MADT",
|
|
|
|
Ptr,
|
|
|
|
AcpiTableLength,
|
|
|
|
PARSER_PARAMS (MadtParser)
|
|
|
|
);
|
|
|
|
InterruptContollerPtr = Ptr + Offset;
|
|
|
|
|
|
|
|
while (Offset < AcpiTableLength) {
|
|
|
|
// Parse Interrupt Controller Structure to obtain Length.
|
|
|
|
ParseAcpi (
|
|
|
|
FALSE,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
InterruptContollerPtr,
|
2019-08-02 01:44:05 +02:00
|
|
|
AcpiTableLength - Offset,
|
2018-04-20 10:08:22 +02:00
|
|
|
PARSER_PARAMS (MadtInterruptControllerHeaderParser)
|
|
|
|
);
|
|
|
|
|
2019-07-23 00:50:25 +02:00
|
|
|
// Make sure forward progress is made.
|
|
|
|
if (*MadtInterruptControllerLength < 2) {
|
2018-04-20 10:08:22 +02:00
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
2019-07-23 00:50:25 +02:00
|
|
|
L"ERROR: Structure length is too small: " \
|
|
|
|
L"MadtInterruptControllerLength = %d. " \
|
|
|
|
L"MadtInterruptControllerType = %d. MADT parsing aborted.\n",
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
*MadtInterruptControllerType
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the MADT structure lies inside the table
|
|
|
|
if ((Offset + *MadtInterruptControllerLength) > AcpiTableLength) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Invalid MADT structure length. " \
|
|
|
|
L"MadtInterruptControllerLength = %d. " \
|
|
|
|
L"RemainingTableBufferLength = %d. MADT parsing aborted.\n",
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
AcpiTableLength - Offset
|
|
|
|
);
|
|
|
|
return;
|
2018-04-20 10:08:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (*MadtInterruptControllerType) {
|
2019-05-31 05:06:51 +02:00
|
|
|
case EFI_ACPI_6_3_GIC: {
|
2018-04-20 10:08:22 +02:00
|
|
|
ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"GICC",
|
|
|
|
InterruptContollerPtr,
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
PARSER_PARAMS (GicCParser)
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-31 05:06:51 +02:00
|
|
|
case EFI_ACPI_6_3_GICD: {
|
2018-04-20 10:08:22 +02:00
|
|
|
if (++GICDCount > 1) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Only one GICD must be present,"
|
2018-07-30 03:31:36 +02:00
|
|
|
L" GICDCount = %d\n",
|
2018-04-20 10:08:22 +02:00
|
|
|
GICDCount
|
|
|
|
);
|
|
|
|
}
|
|
|
|
ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"GICD",
|
|
|
|
InterruptContollerPtr,
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
PARSER_PARAMS (GicDParser)
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-31 05:06:51 +02:00
|
|
|
case EFI_ACPI_6_3_GIC_MSI_FRAME: {
|
2018-04-20 10:08:22 +02:00
|
|
|
ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"GIC MSI Frame",
|
|
|
|
InterruptContollerPtr,
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
PARSER_PARAMS (GicMSIFrameParser)
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-31 05:06:51 +02:00
|
|
|
case EFI_ACPI_6_3_GICR: {
|
2018-04-20 10:08:22 +02:00
|
|
|
ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"GICR",
|
|
|
|
InterruptContollerPtr,
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
PARSER_PARAMS (GicRParser)
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-31 05:06:51 +02:00
|
|
|
case EFI_ACPI_6_3_GIC_ITS: {
|
2018-04-20 10:08:22 +02:00
|
|
|
ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"GIC ITS",
|
|
|
|
InterruptContollerPtr,
|
|
|
|
*MadtInterruptControllerLength,
|
|
|
|
PARSER_PARAMS (GicITSParser)
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Unknown Interrupt Controller Structure,"
|
2018-07-30 03:31:36 +02:00
|
|
|
L" Type = %d, Length = %d\n",
|
2018-04-20 10:08:22 +02:00
|
|
|
*MadtInterruptControllerType,
|
|
|
|
*MadtInterruptControllerLength
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} // switch
|
|
|
|
|
|
|
|
InterruptContollerPtr += *MadtInterruptControllerLength;
|
|
|
|
Offset += *MadtInterruptControllerLength;
|
|
|
|
} // while
|
|
|
|
}
|