2020-11-04 12:17:02 +01:00
|
|
|
/** @file
|
|
|
|
HMAT table parser
|
|
|
|
|
|
|
|
Copyright (c) 2020, Arm Limited.
|
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
|
|
|
|
@par Reference(s):
|
|
|
|
- ACPI 6.3 Specification - January 2019
|
|
|
|
|
|
|
|
@par Glossary:
|
|
|
|
- MPDA - Memory Proximity Domain Attributes
|
|
|
|
- SLLBI - System Locality Latency and Bandwidth Information
|
|
|
|
- MSCI - Memory Side Cache Information
|
|
|
|
- Dom - Domain
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <Library/PrintLib.h>
|
|
|
|
#include <Library/BaseLib.h>
|
|
|
|
#include <Library/UefiLib.h>
|
|
|
|
#include "AcpiParser.h"
|
|
|
|
#include "AcpiView.h"
|
|
|
|
|
|
|
|
// Maximum Memory Domain matrix print size.
|
|
|
|
#define MAX_MEMORY_DOMAIN_TARGET_PRINT_MATRIX 10
|
|
|
|
|
|
|
|
// Local variables
|
|
|
|
STATIC CONST UINT16 *HmatStructureType;
|
|
|
|
STATIC CONST UINT32 *HmatStructureLength;
|
|
|
|
|
|
|
|
STATIC CONST UINT32 *NumberInitiatorProximityDomain;
|
|
|
|
STATIC CONST UINT32 *NumberTargetProximityDomain;
|
|
|
|
STATIC CONST
|
2021-07-14 13:45:12 +02:00
|
|
|
EFI_ACPI_6_4_HMAT_STRUCTURE_SYSTEM_LOCALITY_LATENCY_AND_BANDWIDTH_INFO_FLAGS *
|
2020-11-04 12:17:02 +01:00
|
|
|
SllbiFlags;
|
|
|
|
|
|
|
|
STATIC CONST UINT8 *SllbiDataType;
|
|
|
|
STATIC CONST UINT16 *NumberSMBIOSHandles;
|
|
|
|
|
|
|
|
STATIC ACPI_DESCRIPTION_HEADER_INFO AcpiHdrInfo;
|
|
|
|
|
|
|
|
/**
|
|
|
|
Names of System Locality Latency Bandwidth Information (SLLBI) data types
|
|
|
|
**/
|
|
|
|
STATIC CONST CHAR16 *SllbiNames[] = {
|
|
|
|
L"Access %sLatency%s",
|
|
|
|
L"Read %sLatency%s",
|
|
|
|
L"Write %sLatency%s",
|
|
|
|
L"Access %sBandwidth%s",
|
|
|
|
L"Read %sBandwidth%s",
|
|
|
|
L"Write %sBandwidth%s"
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function validates the Cache Attributes field.
|
|
|
|
|
|
|
|
@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
|
|
|
|
ValidateCacheAttributes (
|
|
|
|
IN UINT8 *Ptr,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
2021-07-14 13:45:12 +02:00
|
|
|
EFI_ACPI_6_4_HMAT_STRUCTURE_MEMORY_SIDE_CACHE_INFO_CACHE_ATTRIBUTES *
|
2020-11-04 12:17:02 +01:00
|
|
|
Attributes;
|
|
|
|
|
|
|
|
Attributes =
|
2021-07-14 13:45:12 +02:00
|
|
|
(EFI_ACPI_6_4_HMAT_STRUCTURE_MEMORY_SIDE_CACHE_INFO_CACHE_ATTRIBUTES *)Ptr;
|
2020-11-04 12:17:02 +01:00
|
|
|
|
|
|
|
if (Attributes->TotalCacheLevels > 0x3) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: Attributes bits [3:0] have invalid value: 0x%x",
|
|
|
|
Attributes->TotalCacheLevels
|
|
|
|
);
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
if (Attributes->CacheLevel > 0x3) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: Attributes bits [7:4] have invalid value: 0x%x",
|
|
|
|
Attributes->CacheLevel
|
|
|
|
);
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
if (Attributes->CacheAssociativity > 0x2) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: Attributes bits [11:8] have invalid value: 0x%x",
|
|
|
|
Attributes->CacheAssociativity
|
|
|
|
);
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
if (Attributes->WritePolicy > 0x2) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"\nERROR: Attributes bits [15:12] have invalid value: 0x%x",
|
|
|
|
Attributes->WritePolicy
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Dumps the cache attributes field
|
|
|
|
|
|
|
|
@param [in] Format Optional format string for tracing the data.
|
|
|
|
@param [in] Ptr Pointer to the start of the buffer.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
DumpCacheAttributes (
|
|
|
|
IN CONST CHAR16 *Format OPTIONAL,
|
|
|
|
IN UINT8 *Ptr
|
|
|
|
)
|
|
|
|
{
|
2021-07-14 13:45:12 +02:00
|
|
|
EFI_ACPI_6_4_HMAT_STRUCTURE_MEMORY_SIDE_CACHE_INFO_CACHE_ATTRIBUTES *
|
2020-11-04 12:17:02 +01:00
|
|
|
Attributes;
|
|
|
|
|
|
|
|
Attributes =
|
2021-07-14 13:45:12 +02:00
|
|
|
(EFI_ACPI_6_4_HMAT_STRUCTURE_MEMORY_SIDE_CACHE_INFO_CACHE_ATTRIBUTES *)Ptr;
|
2020-11-04 12:17:02 +01:00
|
|
|
|
|
|
|
Print (L"\n");
|
|
|
|
PrintFieldName (4, L"Total Cache Levels");
|
|
|
|
Print (L"%d\n", Attributes->TotalCacheLevels);
|
|
|
|
PrintFieldName (4, L"Cache Level");
|
|
|
|
Print (L"%d\n", Attributes->CacheLevel);
|
|
|
|
PrintFieldName (4, L"Cache Associativity");
|
|
|
|
Print (L"%d\n", Attributes->CacheAssociativity);
|
|
|
|
PrintFieldName (4, L"Write Policy");
|
|
|
|
Print (L"%d\n", Attributes->WritePolicy);
|
|
|
|
PrintFieldName (4, L"Cache Line Size");
|
|
|
|
Print (L"%d\n", Attributes->CacheLineSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the ACPI HMAT Table.
|
|
|
|
*/
|
|
|
|
STATIC CONST ACPI_PARSER HmatParser[] = {
|
|
|
|
PARSE_ACPI_HEADER (&AcpiHdrInfo),
|
|
|
|
{ L"Reserved", 4,36, NULL, NULL, NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
An ACPI_PARSER array describing the HMAT structure header.
|
|
|
|
*/
|
|
|
|
STATIC CONST ACPI_PARSER HmatStructureHeaderParser[] = {
|
|
|
|
{ L"Type", 2, 0, NULL, NULL, (VOID **)&HmatStructureType, NULL, NULL },
|
|
|
|
{ L"Reserved", 2, 2, NULL, NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Length", 4, 4, NULL, NULL, (VOID **)&HmatStructureLength, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
An ACPI PARSER array describing the Memory Proximity Domain Attributes
|
|
|
|
Structure - Type 0.
|
|
|
|
*/
|
|
|
|
STATIC CONST ACPI_PARSER MemProximityDomainAttributeParser[] = {
|
|
|
|
{ L"Type", 2, 0, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Length", 4, 4, L"%d", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Flags", 2, 8, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 2, 10, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Proximity Dom for initiator", 4, 12, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Proximity Dom for memory", 4, 16, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 4, 20, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 8, 24, L"0x%lx", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 8, 32, L"0x%lx", NULL, NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
An ACPI PARSER array describing the System Locality Latency and Bandwidth
|
|
|
|
Information Structure - Type 1.
|
|
|
|
*/
|
|
|
|
STATIC CONST ACPI_PARSER SllbiParser[] = {
|
|
|
|
{ L"Type", 2, 0, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Length", 4, 4, L"%d", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Flags", 1, 8, L"0x%x", NULL, (VOID **)&SllbiFlags, NULL, NULL },
|
|
|
|
{ L"Data type", 1, 9, L"0x%x", NULL, (VOID **)&SllbiDataType, NULL, NULL },
|
2021-07-14 13:45:12 +02:00
|
|
|
{ L"Min Transfer Size", 1, 10, L"%d", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 1, 11, L"0x%x", NULL, NULL, NULL, NULL },
|
2020-11-04 12:17:02 +01:00
|
|
|
{ L"Initiator Proximity Dom Count", 4, 12, L"%d", NULL,
|
|
|
|
(VOID **)&NumberInitiatorProximityDomain, NULL, NULL },
|
|
|
|
{ L"Target Proximity Dom Count", 4, 16, L"%d", NULL,
|
|
|
|
(VOID **)&NumberTargetProximityDomain, NULL, NULL },
|
|
|
|
{ L"Reserved", 4, 20, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Entry Base Unit", 8, 24, L"0x%lx", NULL, NULL, NULL, NULL }
|
|
|
|
// initiator Proximity Domain list ...
|
|
|
|
// target Proximity Domain list ...
|
|
|
|
// Latency/Bandwidth matrix ...
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
An ACPI PARSER array describing the Memory Side Cache Information
|
|
|
|
Structure - Type 2.
|
|
|
|
*/
|
|
|
|
STATIC CONST ACPI_PARSER MemSideCacheInfoParser[] = {
|
|
|
|
{ L"Type", 2, 0, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 2, 2, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Length", 4, 4, L"%d", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Proximity Dom for memory", 4, 8, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Reserved", 4, 12, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Memory Side Cache Size", 8, 16, L"0x%lx", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"Cache Attributes", 4, 24, NULL, DumpCacheAttributes, NULL,
|
|
|
|
ValidateCacheAttributes, NULL },
|
|
|
|
{ L"Reserved", 2, 28, L"0x%x", NULL, NULL, NULL, NULL },
|
|
|
|
{ L"SMBIOS Handle Count", 2, 30, L"%d", NULL,
|
|
|
|
(VOID **)&NumberSMBIOSHandles, NULL, NULL }
|
|
|
|
// SMBIOS handles List ...
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function parses the Memory Proximity Domain Attributes
|
|
|
|
Structure (Type 0).
|
|
|
|
|
|
|
|
@param [in] Ptr Pointer to the start of the Memory Proximity Domain
|
|
|
|
Attributes Structure data.
|
|
|
|
@param [in] Length Length of the Memory Proximity Domain Attributes
|
|
|
|
Structure.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
DumpMpda (
|
|
|
|
IN UINT8 *Ptr,
|
|
|
|
IN UINT32 Length
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"Memory Proximity Domain Attributes Structure",
|
|
|
|
Ptr,
|
|
|
|
Length,
|
|
|
|
PARSER_PARAMS (MemProximityDomainAttributeParser)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function parses the System Locality Latency and Bandwidth Information
|
|
|
|
Structure (Type 1).
|
|
|
|
|
|
|
|
@param [in] Ptr Pointer to the start of the System Locality Latency and
|
|
|
|
Bandwidth Information Structure data.
|
|
|
|
@param [in] Length Length of the System Locality Latency and Bandwidth
|
|
|
|
Information Structure.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
DumpSllbi (
|
|
|
|
IN UINT8 *Ptr,
|
|
|
|
IN UINT32 Length
|
|
|
|
)
|
|
|
|
{
|
|
|
|
CONST UINT32 *InitiatorProximityDomainList;
|
|
|
|
CONST UINT32 *TargetProximityDomainList;
|
|
|
|
CONST UINT16 *LatencyBandwidthMatrix;
|
|
|
|
UINT32 Offset;
|
|
|
|
CHAR16 Buffer[OUTPUT_FIELD_COLUMN_WIDTH];
|
|
|
|
CHAR16 SecondBuffer[OUTPUT_FIELD_COLUMN_WIDTH];
|
|
|
|
UINT32 RequiredTableSize;
|
|
|
|
UINT32 Index;
|
|
|
|
UINT32 IndexInitiator;
|
|
|
|
UINT32 IndexTarget;
|
|
|
|
UINT32 TargetStartOffset;
|
|
|
|
|
|
|
|
Offset = ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"System Locality Latency and Bandwidth Information Structure",
|
|
|
|
Ptr,
|
|
|
|
Length,
|
|
|
|
PARSER_PARAMS (SllbiParser)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check if the values used to control the parsing logic have been
|
|
|
|
// successfully read.
|
|
|
|
if ((SllbiFlags == NULL) ||
|
|
|
|
(SllbiDataType == NULL) ||
|
|
|
|
(NumberInitiatorProximityDomain == NULL) ||
|
|
|
|
(NumberTargetProximityDomain == NULL))
|
|
|
|
{
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Insufficient remaining table buffer length to read the " \
|
|
|
|
L"SLLBI structure header. Length = %d.\n",
|
|
|
|
Length
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RequiredTableSize = (*NumberInitiatorProximityDomain * sizeof (UINT32)) +
|
|
|
|
(*NumberTargetProximityDomain * sizeof (UINT32)) +
|
|
|
|
(*NumberInitiatorProximityDomain *
|
|
|
|
*NumberTargetProximityDomain * sizeof (UINT16)) +
|
|
|
|
Offset;
|
|
|
|
|
|
|
|
if (RequiredTableSize > Length) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Insufficient System Locality Latency and Bandwidth" \
|
|
|
|
L"Information Structure length. TableLength = %d. " \
|
|
|
|
L"RequiredTableLength = %d.\n",
|
|
|
|
Length,
|
|
|
|
RequiredTableSize
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitiatorProximityDomainList = (UINT32 *)(Ptr + Offset);
|
|
|
|
TargetProximityDomainList = InitiatorProximityDomainList +
|
|
|
|
*NumberInitiatorProximityDomain;
|
|
|
|
LatencyBandwidthMatrix = (UINT16 *)(TargetProximityDomainList +
|
|
|
|
*NumberTargetProximityDomain);
|
|
|
|
|
|
|
|
// Display each element of the Initiator Proximity Domain list
|
|
|
|
for (Index = 0; Index < *NumberInitiatorProximityDomain; Index++) {
|
|
|
|
UnicodeSPrint (
|
|
|
|
Buffer,
|
|
|
|
sizeof (Buffer),
|
|
|
|
L"Initiator Proximity Dom [%d]",
|
|
|
|
Index
|
|
|
|
);
|
|
|
|
|
|
|
|
PrintFieldName (4, Buffer);
|
|
|
|
Print (
|
|
|
|
L"0x%x\n",
|
|
|
|
InitiatorProximityDomainList[Index]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Display each element of the Target Proximity Domain list
|
|
|
|
for (Index = 0; Index < *NumberTargetProximityDomain; Index++) {
|
|
|
|
UnicodeSPrint (
|
|
|
|
Buffer,
|
|
|
|
sizeof (Buffer),
|
|
|
|
L"Target Proximity Dom [%d]",
|
|
|
|
Index
|
|
|
|
);
|
|
|
|
|
|
|
|
PrintFieldName (4, Buffer);
|
|
|
|
Print (
|
|
|
|
L"0x%x\n",
|
|
|
|
TargetProximityDomainList[Index]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create base name depending on Data Type in this Structure
|
|
|
|
if (*SllbiDataType >= ARRAY_SIZE (SllbiNames)) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (L"Error: Unkown Data Type. DataType = 0x%x.\n", *SllbiDataType);
|
|
|
|
return;
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
StrCpyS (Buffer, sizeof (Buffer), SllbiNames[*SllbiDataType]);
|
|
|
|
|
|
|
|
// Adjust base name depending on Memory Hierarchy in this Structure
|
|
|
|
switch (SllbiFlags->MemoryHierarchy) {
|
|
|
|
case 0:
|
|
|
|
UnicodeSPrint (
|
|
|
|
SecondBuffer,
|
|
|
|
sizeof (SecondBuffer),
|
|
|
|
Buffer,
|
|
|
|
L"",
|
|
|
|
L"%s"
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
UnicodeSPrint (
|
|
|
|
SecondBuffer,
|
|
|
|
sizeof (SecondBuffer),
|
|
|
|
Buffer,
|
|
|
|
L"Hit ",
|
|
|
|
L"%s"
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"Error: Invalid Memory Hierarchy. MemoryHierarchy = %d.\n",
|
|
|
|
SllbiFlags->MemoryHierarchy
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
} // switch
|
|
|
|
|
|
|
|
if (*NumberTargetProximityDomain <= MAX_MEMORY_DOMAIN_TARGET_PRINT_MATRIX) {
|
|
|
|
// Display the latency/bandwidth matrix as a matrix
|
|
|
|
UnicodeSPrint (
|
|
|
|
Buffer,
|
|
|
|
sizeof (Buffer),
|
|
|
|
SecondBuffer,
|
|
|
|
L""
|
|
|
|
);
|
|
|
|
PrintFieldName (4, Buffer);
|
|
|
|
|
|
|
|
Print (L"\n Target : X-axis (Horizontal)");
|
|
|
|
Print (L"\n Initiator : Y-axis (Vertical)");
|
|
|
|
Print (L"\n |");
|
|
|
|
|
|
|
|
for (IndexTarget = 0;
|
|
|
|
IndexTarget < *NumberTargetProximityDomain;
|
|
|
|
IndexTarget++)
|
|
|
|
{
|
|
|
|
Print (L" %2d", IndexTarget);
|
|
|
|
}
|
|
|
|
|
|
|
|
Print (L"\n ---+");
|
|
|
|
for (IndexTarget = 0;
|
|
|
|
IndexTarget < *NumberTargetProximityDomain;
|
|
|
|
IndexTarget++)
|
|
|
|
{
|
|
|
|
Print (L"------");
|
|
|
|
}
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
Print (L"\n");
|
|
|
|
|
|
|
|
TargetStartOffset = 0;
|
|
|
|
for (IndexInitiator = 0;
|
|
|
|
IndexInitiator < *NumberInitiatorProximityDomain;
|
|
|
|
IndexInitiator++)
|
|
|
|
{
|
|
|
|
Print (L" %2d |", IndexInitiator);
|
|
|
|
for (IndexTarget = 0;
|
|
|
|
IndexTarget < *NumberTargetProximityDomain;
|
|
|
|
IndexTarget++)
|
|
|
|
{
|
|
|
|
Print (
|
|
|
|
L" %5d",
|
|
|
|
LatencyBandwidthMatrix[TargetStartOffset + IndexTarget]
|
|
|
|
);
|
|
|
|
} // for Target
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
Print (L"\n");
|
|
|
|
TargetStartOffset += (*NumberTargetProximityDomain);
|
|
|
|
} // for Initiator
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
Print (L"\n");
|
|
|
|
} else {
|
|
|
|
// Display the latency/bandwidth matrix as a list
|
|
|
|
UnicodeSPrint (
|
|
|
|
Buffer,
|
|
|
|
sizeof (Buffer),
|
|
|
|
SecondBuffer,
|
|
|
|
L" [%d][%d]"
|
|
|
|
);
|
|
|
|
|
|
|
|
TargetStartOffset = 0;
|
|
|
|
for (IndexInitiator = 0;
|
|
|
|
IndexInitiator < *NumberInitiatorProximityDomain;
|
|
|
|
IndexInitiator++)
|
|
|
|
{
|
|
|
|
for (IndexTarget = 0;
|
|
|
|
IndexTarget < *NumberTargetProximityDomain;
|
|
|
|
IndexTarget++)
|
|
|
|
{
|
|
|
|
UnicodeSPrint (
|
|
|
|
SecondBuffer,
|
|
|
|
sizeof (SecondBuffer),
|
|
|
|
Buffer,
|
|
|
|
IndexInitiator,
|
|
|
|
IndexTarget
|
|
|
|
);
|
|
|
|
|
|
|
|
PrintFieldName (4, SecondBuffer);
|
|
|
|
Print (
|
|
|
|
L"%d\n",
|
|
|
|
LatencyBandwidthMatrix[TargetStartOffset + IndexTarget]
|
|
|
|
);
|
|
|
|
} // for Target
|
2021-12-05 23:54:13 +01:00
|
|
|
|
2020-11-04 12:17:02 +01:00
|
|
|
TargetStartOffset += (*NumberTargetProximityDomain);
|
|
|
|
} // for Initiator
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function parses the Memory Side Cache Information Structure (Type 2).
|
|
|
|
|
|
|
|
@param [in] Ptr Pointer to the start of the Memory Side Cache Information
|
|
|
|
Structure data.
|
|
|
|
@param [in] Length Length of the Memory Side Cache Information Structure.
|
|
|
|
**/
|
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
DumpMsci (
|
|
|
|
IN UINT8 *Ptr,
|
|
|
|
IN UINT32 Length
|
|
|
|
)
|
|
|
|
{
|
|
|
|
CONST UINT16 *SMBIOSHandlesList;
|
|
|
|
CHAR16 Buffer[OUTPUT_FIELD_COLUMN_WIDTH];
|
|
|
|
UINT32 Offset;
|
|
|
|
UINT16 Index;
|
|
|
|
|
|
|
|
Offset = ParseAcpi (
|
|
|
|
TRUE,
|
|
|
|
2,
|
|
|
|
"Memory Side Cache Information Structure",
|
|
|
|
Ptr,
|
|
|
|
Length,
|
|
|
|
PARSER_PARAMS (MemSideCacheInfoParser)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check if the values used to control the parsing logic have been
|
|
|
|
// successfully read.
|
|
|
|
if (NumberSMBIOSHandles == NULL) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Insufficient remaining table buffer length to read the " \
|
|
|
|
L"MSCI structure header. Length = %d.\n",
|
|
|
|
Length
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((*NumberSMBIOSHandles * sizeof (UINT16)) > (Length - Offset)) {
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Invalid Number of SMBIOS Handles. SMBIOSHandlesCount = %d." \
|
|
|
|
L"RemainingBufferLength = %d.\n",
|
|
|
|
*NumberSMBIOSHandles,
|
|
|
|
Length - Offset
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SMBIOSHandlesList = (UINT16 *)(Ptr + Offset);
|
|
|
|
|
|
|
|
for (Index = 0; Index < *NumberSMBIOSHandles; Index++) {
|
|
|
|
UnicodeSPrint (
|
|
|
|
Buffer,
|
|
|
|
sizeof (Buffer),
|
|
|
|
L"SMBIOS Handles [%d]",
|
|
|
|
Index
|
|
|
|
);
|
|
|
|
|
|
|
|
PrintFieldName (4, Buffer);
|
|
|
|
Print (
|
|
|
|
L"0x%x\n",
|
|
|
|
SMBIOSHandlesList[Index]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function parses the ACPI HMAT table.
|
|
|
|
When trace is enabled this function parses the HMAT table and
|
|
|
|
traces the ACPI table fields.
|
|
|
|
|
|
|
|
This function parses the following HMAT structures:
|
|
|
|
- Memory Proximity Domain Attributes Structure (Type 0)
|
|
|
|
- System Locality Latency and Bandwidth Info Structure (Type 1)
|
|
|
|
- Memory Side Cache Info structure (Type 2)
|
|
|
|
|
|
|
|
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.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
ParseAcpiHmat (
|
|
|
|
IN BOOLEAN Trace,
|
|
|
|
IN UINT8 *Ptr,
|
|
|
|
IN UINT32 AcpiTableLength,
|
|
|
|
IN UINT8 AcpiTableRevision
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 Offset;
|
|
|
|
UINT8 *HmatStructurePtr;
|
|
|
|
|
|
|
|
if (!Trace) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Offset = ParseAcpi (
|
|
|
|
Trace,
|
|
|
|
0,
|
|
|
|
"HMAT",
|
|
|
|
Ptr,
|
|
|
|
AcpiTableLength,
|
|
|
|
PARSER_PARAMS (HmatParser)
|
|
|
|
);
|
|
|
|
|
|
|
|
HmatStructurePtr = Ptr + Offset;
|
|
|
|
|
|
|
|
while (Offset < AcpiTableLength) {
|
|
|
|
// Parse HMAT Structure Header to obtain Type and Length.
|
|
|
|
ParseAcpi (
|
|
|
|
FALSE,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
HmatStructurePtr,
|
|
|
|
AcpiTableLength - Offset,
|
|
|
|
PARSER_PARAMS (HmatStructureHeaderParser)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check if the values used to control the parsing logic have been
|
|
|
|
// successfully read.
|
|
|
|
if ((HmatStructureType == NULL) ||
|
|
|
|
(HmatStructureLength == NULL))
|
|
|
|
{
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Insufficient remaining table buffer length to read the " \
|
|
|
|
L"HMAT structure header. Length = %d.\n",
|
|
|
|
AcpiTableLength - Offset
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validate HMAT Structure length.
|
|
|
|
if ((*HmatStructureLength == 0) ||
|
|
|
|
((Offset + (*HmatStructureLength)) > AcpiTableLength))
|
|
|
|
{
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Invalid HMAT Structure length. " \
|
|
|
|
L"Length = %d. Offset = %d. AcpiTableLength = %d.\n",
|
|
|
|
*HmatStructureLength,
|
|
|
|
Offset,
|
|
|
|
AcpiTableLength
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (*HmatStructureType) {
|
2021-07-14 13:45:12 +02:00
|
|
|
case EFI_ACPI_6_4_HMAT_TYPE_MEMORY_PROXIMITY_DOMAIN_ATTRIBUTES:
|
2020-11-04 12:17:02 +01:00
|
|
|
DumpMpda (
|
|
|
|
HmatStructurePtr,
|
|
|
|
*HmatStructureLength
|
|
|
|
);
|
|
|
|
break;
|
2021-07-14 13:45:12 +02:00
|
|
|
case EFI_ACPI_6_4_HMAT_TYPE_SYSTEM_LOCALITY_LATENCY_AND_BANDWIDTH_INFO:
|
2020-11-04 12:17:02 +01:00
|
|
|
DumpSllbi (
|
|
|
|
HmatStructurePtr,
|
|
|
|
*HmatStructureLength
|
|
|
|
);
|
|
|
|
break;
|
2021-07-14 13:45:12 +02:00
|
|
|
case EFI_ACPI_6_4_HMAT_TYPE_MEMORY_SIDE_CACHE_INFO:
|
2020-11-04 12:17:02 +01:00
|
|
|
DumpMsci (
|
|
|
|
HmatStructurePtr,
|
|
|
|
*HmatStructureLength
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
IncrementErrorCount ();
|
|
|
|
Print (
|
|
|
|
L"ERROR: Unknown HMAT structure:"
|
|
|
|
L" Type = %d, Length = %d\n",
|
|
|
|
*HmatStructureType,
|
|
|
|
*HmatStructureLength
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
} // switch
|
|
|
|
|
|
|
|
HmatStructurePtr += *HmatStructureLength;
|
|
|
|
Offset += *HmatStructureLength;
|
|
|
|
} // while
|
|
|
|
}
|