mirror of
				https://github.com/acidanthera/audk.git
				synced 2025-10-31 19:23:54 +01:00 
			
		
		
		
	1. Check if the 'Number of System Localities' provided can be represented in the SLIT table. The table 'Length' field is a 32-bit value while the 'Number of System Localities' field is 64-bit long. 2. Check if the SLIT matrix fits in the table buffer. If N is the SLIT locality count, then the matrix used to represent the localities is N*N bytes long. The ACPI table length must be big enough to fit the matrix. 3. Remove (now) redundant 64x64 bit multiplication. Signed-off-by: Krzysztof Koch <krzysztof.koch@arm.com>
		
			
				
	
	
		
			189 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			189 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   SLIT table parser
 | |
| 
 | |
|   Copyright (c) 2016 - 2019, ARM Limited. All rights reserved.
 | |
|   SPDX-License-Identifier: BSD-2-Clause-Patent
 | |
| 
 | |
|   @par Reference(s):
 | |
|     - ACPI 6.2 Specification - Errata A, September 2017
 | |
| **/
 | |
| 
 | |
| #include <IndustryStandard/Acpi.h>
 | |
| #include <Library/PrintLib.h>
 | |
| #include <Library/UefiLib.h>
 | |
| #include "AcpiParser.h"
 | |
| #include "AcpiTableParser.h"
 | |
| 
 | |
| // Local Variables
 | |
| STATIC CONST UINT64* SlitSystemLocalityCount;
 | |
| STATIC ACPI_DESCRIPTION_HEADER_INFO AcpiHdrInfo;
 | |
| 
 | |
| /**
 | |
|   An ACPI_PARSER array describing the ACPI SLIT table.
 | |
| **/
 | |
| STATIC CONST ACPI_PARSER SlitParser[] = {
 | |
|   PARSE_ACPI_HEADER (&AcpiHdrInfo),
 | |
|   {L"Number of System Localities", 8, 36, L"0x%lx", NULL,
 | |
|    (VOID**)&SlitSystemLocalityCount, NULL, NULL}
 | |
| };
 | |
| 
 | |
| /**
 | |
|   Macro to get the value of a System Locality
 | |
| **/
 | |
| #define SLIT_ELEMENT(Ptr, i, j) *(Ptr + (i * LocalityCount) + j)
 | |
| 
 | |
| /**
 | |
|   This function parses the ACPI SLIT table.
 | |
|   When trace is enabled this function parses the SLIT table and
 | |
|   traces the ACPI table fields.
 | |
| 
 | |
|   This function also validates System Localities for the following:
 | |
|     - Diagonal elements have a normalized value of 10
 | |
|     - Relative distance from System Locality at i*N+j is same as
 | |
|       j*N+i
 | |
| 
 | |
|   @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
 | |
| ParseAcpiSlit (
 | |
|   IN BOOLEAN Trace,
 | |
|   IN UINT8*  Ptr,
 | |
|   IN UINT32  AcpiTableLength,
 | |
|   IN UINT8   AcpiTableRevision
 | |
|   )
 | |
| {
 | |
|   UINT32 Offset;
 | |
|   UINT32 Count;
 | |
|   UINT32 Index;
 | |
|   UINT32 LocalityCount;
 | |
|   UINT8* LocalityPtr;
 | |
|   CHAR16 Buffer[80];  // Used for AsciiName param of ParseAcpi
 | |
| 
 | |
|   if (!Trace) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Offset = ParseAcpi (
 | |
|              TRUE,
 | |
|              0,
 | |
|              "SLIT",
 | |
|              Ptr,
 | |
|              AcpiTableLength,
 | |
|              PARSER_PARAMS (SlitParser)
 | |
|              );
 | |
| 
 | |
|   // Check if the values used to control the parsing logic have been
 | |
|   // successfully read.
 | |
|   if (SlitSystemLocalityCount == NULL) {
 | |
|     IncrementErrorCount ();
 | |
|     Print (
 | |
|       L"ERROR: Insufficient table length. AcpiTableLength = %d.\n",
 | |
|       AcpiTableLength
 | |
|       );
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|     Despite the 'Number of System Localities' being a 64-bit field in SLIT,
 | |
|     the maximum number of localities that can be represented in SLIT is limited
 | |
|     by the 'Length' field of the ACPI table.
 | |
| 
 | |
|     Since the ACPI table length field is 32-bit wide. The maximum number of
 | |
|     localities that can be represented in SLIT can be calculated as:
 | |
| 
 | |
|     MaxLocality = sqrt (MAX_UINT32 - sizeof (EFI_ACPI_6_3_SYSTEM_LOCALITY_DISTANCE_INFORMATION_TABLE_HEADER))
 | |
|                 = 65535
 | |
|                 = MAX_UINT16
 | |
|   */
 | |
|   if (*SlitSystemLocalityCount > MAX_UINT16) {
 | |
|     IncrementErrorCount ();
 | |
|     Print (
 | |
|       L"ERROR: The Number of System Localities provided can't be represented " \
 | |
|         L"in the SLIT table. SlitSystemLocalityCount = %ld. " \
 | |
|         L"MaxLocalityCountAllowed = %d.\n",
 | |
|       *SlitSystemLocalityCount,
 | |
|       MAX_UINT16
 | |
|       );
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   LocalityCount = (UINT32)*SlitSystemLocalityCount;
 | |
| 
 | |
|   // Make sure system localities fit in the table buffer provided
 | |
|   if (Offset + (LocalityCount * LocalityCount) > AcpiTableLength) {
 | |
|     IncrementErrorCount ();
 | |
|     Print (
 | |
|       L"ERROR: Invalid Number of System Localities. " \
 | |
|         L"SlitSystemLocalityCount = %ld. AcpiTableLength = %d.\n",
 | |
|       *SlitSystemLocalityCount,
 | |
|       AcpiTableLength
 | |
|       );
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   LocalityPtr = Ptr + Offset;
 | |
| 
 | |
|   // We only print the Localities if the count is less than 16
 | |
|   // If the locality count is more than 16 then refer to the
 | |
|   // raw data dump.
 | |
|   if (LocalityCount < 16) {
 | |
|     UnicodeSPrint (
 | |
|       Buffer,
 | |
|       sizeof (Buffer),
 | |
|       L"Entry[0x%lx][0x%lx]",
 | |
|       LocalityCount,
 | |
|       LocalityCount
 | |
|       );
 | |
|     PrintFieldName (0, Buffer);
 | |
|     Print (L"\n");
 | |
|     Print (L"       ");
 | |
|     for (Index = 0; Index < LocalityCount; Index++) {
 | |
|       Print (L" (%3d) ", Index);
 | |
|     }
 | |
|     Print (L"\n");
 | |
|     for (Count = 0; Count< LocalityCount; Count++) {
 | |
|       Print (L" (%3d) ", Count);
 | |
|       for (Index = 0; Index < LocalityCount; Index++) {
 | |
|         Print (L"  %3d  ", SLIT_ELEMENT (LocalityPtr, Count, Index));
 | |
|       }
 | |
|       Print (L"\n");
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Validate
 | |
|   for (Count = 0; Count < LocalityCount; Count++) {
 | |
|     for (Index = 0; Index < LocalityCount; Index++) {
 | |
|       // Element[x][x] must be equal to 10
 | |
|       if ((Count == Index) && (SLIT_ELEMENT (LocalityPtr, Count,Index) != 10)) {
 | |
|         IncrementErrorCount ();
 | |
|         Print (
 | |
|           L"ERROR: Diagonal Element[0x%lx][0x%lx] (%3d)."
 | |
|             L" Normalized Value is not 10\n",
 | |
|           Count,
 | |
|           Index,
 | |
|           SLIT_ELEMENT (LocalityPtr, Count, Index)
 | |
|           );
 | |
|       }
 | |
|       // Element[i][j] must be equal to Element[j][i]
 | |
|       if (SLIT_ELEMENT (LocalityPtr, Count, Index) !=
 | |
|           SLIT_ELEMENT (LocalityPtr, Index, Count)) {
 | |
|         IncrementErrorCount ();
 | |
|         Print (
 | |
|           L"ERROR: Relative distances for Element[0x%lx][0x%lx] (%3d) and \n"
 | |
|            L"Element[0x%lx][0x%lx] (%3d) do not match.\n",
 | |
|           Count,
 | |
|           Index,
 | |
|           SLIT_ELEMENT (LocalityPtr, Count, Index),
 | |
|           Index,
 | |
|           Count,
 | |
|           SLIT_ELEMENT (LocalityPtr, Index, Count)
 | |
|           );
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 |