]> git.proxmox.com Git - mirror_edk2.git/blob - ShellPkg/Library/UefiShellAcpiViewCommandLib/Parsers/Slit/SlitParser.c
ShellPkg: acpiview: SLIT: Validate System Locality count
[mirror_edk2.git] / ShellPkg / Library / UefiShellAcpiViewCommandLib / Parsers / Slit / SlitParser.c
1 /** @file
2 SLIT table parser
3
4 Copyright (c) 2016 - 2019, ARM Limited. All rights reserved.
5 SPDX-License-Identifier: BSD-2-Clause-Patent
6
7 @par Reference(s):
8 - ACPI 6.2 Specification - Errata A, September 2017
9 **/
10
11 #include <IndustryStandard/Acpi.h>
12 #include <Library/PrintLib.h>
13 #include <Library/UefiLib.h>
14 #include "AcpiParser.h"
15 #include "AcpiTableParser.h"
16
17 // Local Variables
18 STATIC CONST UINT64* SlitSystemLocalityCount;
19 STATIC ACPI_DESCRIPTION_HEADER_INFO AcpiHdrInfo;
20
21 /**
22 An ACPI_PARSER array describing the ACPI SLIT table.
23 **/
24 STATIC CONST ACPI_PARSER SlitParser[] = {
25 PARSE_ACPI_HEADER (&AcpiHdrInfo),
26 {L"Number of System Localities", 8, 36, L"0x%lx", NULL,
27 (VOID**)&SlitSystemLocalityCount, NULL, NULL}
28 };
29
30 /**
31 Macro to get the value of a System Locality
32 **/
33 #define SLIT_ELEMENT(Ptr, i, j) *(Ptr + (i * LocalityCount) + j)
34
35 /**
36 This function parses the ACPI SLIT table.
37 When trace is enabled this function parses the SLIT table and
38 traces the ACPI table fields.
39
40 This function also validates System Localities for the following:
41 - Diagonal elements have a normalized value of 10
42 - Relative distance from System Locality at i*N+j is same as
43 j*N+i
44
45 @param [in] Trace If TRUE, trace the ACPI fields.
46 @param [in] Ptr Pointer to the start of the buffer.
47 @param [in] AcpiTableLength Length of the ACPI table.
48 @param [in] AcpiTableRevision Revision of the ACPI table.
49 **/
50 VOID
51 EFIAPI
52 ParseAcpiSlit (
53 IN BOOLEAN Trace,
54 IN UINT8* Ptr,
55 IN UINT32 AcpiTableLength,
56 IN UINT8 AcpiTableRevision
57 )
58 {
59 UINT32 Offset;
60 UINT32 Count;
61 UINT32 Index;
62 UINT32 LocalityCount;
63 UINT8* LocalityPtr;
64 CHAR16 Buffer[80]; // Used for AsciiName param of ParseAcpi
65
66 if (!Trace) {
67 return;
68 }
69
70 Offset = ParseAcpi (
71 TRUE,
72 0,
73 "SLIT",
74 Ptr,
75 AcpiTableLength,
76 PARSER_PARAMS (SlitParser)
77 );
78
79 // Check if the values used to control the parsing logic have been
80 // successfully read.
81 if (SlitSystemLocalityCount == NULL) {
82 IncrementErrorCount ();
83 Print (
84 L"ERROR: Insufficient table length. AcpiTableLength = %d.\n",
85 AcpiTableLength
86 );
87 return;
88 }
89
90 /*
91 Despite the 'Number of System Localities' being a 64-bit field in SLIT,
92 the maximum number of localities that can be represented in SLIT is limited
93 by the 'Length' field of the ACPI table.
94
95 Since the ACPI table length field is 32-bit wide. The maximum number of
96 localities that can be represented in SLIT can be calculated as:
97
98 MaxLocality = sqrt (MAX_UINT32 - sizeof (EFI_ACPI_6_3_SYSTEM_LOCALITY_DISTANCE_INFORMATION_TABLE_HEADER))
99 = 65535
100 = MAX_UINT16
101 */
102 if (*SlitSystemLocalityCount > MAX_UINT16) {
103 IncrementErrorCount ();
104 Print (
105 L"ERROR: The Number of System Localities provided can't be represented " \
106 L"in the SLIT table. SlitSystemLocalityCount = %ld. " \
107 L"MaxLocalityCountAllowed = %d.\n",
108 *SlitSystemLocalityCount,
109 MAX_UINT16
110 );
111 return;
112 }
113
114 LocalityCount = (UINT32)*SlitSystemLocalityCount;
115
116 // Make sure system localities fit in the table buffer provided
117 if (Offset + (LocalityCount * LocalityCount) > AcpiTableLength) {
118 IncrementErrorCount ();
119 Print (
120 L"ERROR: Invalid Number of System Localities. " \
121 L"SlitSystemLocalityCount = %ld. AcpiTableLength = %d.\n",
122 *SlitSystemLocalityCount,
123 AcpiTableLength
124 );
125 return;
126 }
127
128 LocalityPtr = Ptr + Offset;
129
130 // We only print the Localities if the count is less than 16
131 // If the locality count is more than 16 then refer to the
132 // raw data dump.
133 if (LocalityCount < 16) {
134 UnicodeSPrint (
135 Buffer,
136 sizeof (Buffer),
137 L"Entry[0x%lx][0x%lx]",
138 LocalityCount,
139 LocalityCount
140 );
141 PrintFieldName (0, Buffer);
142 Print (L"\n");
143 Print (L" ");
144 for (Index = 0; Index < LocalityCount; Index++) {
145 Print (L" (%3d) ", Index);
146 }
147 Print (L"\n");
148 for (Count = 0; Count< LocalityCount; Count++) {
149 Print (L" (%3d) ", Count);
150 for (Index = 0; Index < LocalityCount; Index++) {
151 Print (L" %3d ", SLIT_ELEMENT (LocalityPtr, Count, Index));
152 }
153 Print (L"\n");
154 }
155 }
156
157 // Validate
158 for (Count = 0; Count < LocalityCount; Count++) {
159 for (Index = 0; Index < LocalityCount; Index++) {
160 // Element[x][x] must be equal to 10
161 if ((Count == Index) && (SLIT_ELEMENT (LocalityPtr, Count,Index) != 10)) {
162 IncrementErrorCount ();
163 Print (
164 L"ERROR: Diagonal Element[0x%lx][0x%lx] (%3d)."
165 L" Normalized Value is not 10\n",
166 Count,
167 Index,
168 SLIT_ELEMENT (LocalityPtr, Count, Index)
169 );
170 }
171 // Element[i][j] must be equal to Element[j][i]
172 if (SLIT_ELEMENT (LocalityPtr, Count, Index) !=
173 SLIT_ELEMENT (LocalityPtr, Index, Count)) {
174 IncrementErrorCount ();
175 Print (
176 L"ERROR: Relative distances for Element[0x%lx][0x%lx] (%3d) and \n"
177 L"Element[0x%lx][0x%lx] (%3d) do not match.\n",
178 Count,
179 Index,
180 SLIT_ELEMENT (LocalityPtr, Count, Index),
181 Index,
182 Count,
183 SLIT_ELEMENT (LocalityPtr, Index, Count)
184 );
185 }
186 }
187 }
188 }