Newer
Older
AMI-Aptio-BIOS-Reversed / StaticSkuDataDxeNeonCityEPECB / StaticSkuDataDxeNeonCityEPECB.c
@Ajax Dong Ajax Dong 2 days ago 20 KB Init
/** @file
  StaticSkuDataDxeNeonCityEPECB - Static SKU Data for Neon City EPECB (Purley Platform)

  This UEFI DXE driver provides board-specific static SKU data tables for the
  Neon City EPECB UBA variant. It registers ACPI name-space paths for core count
  tables (CCT), core frequency headers (CFH), NVDIMM driver entries (NVDR), and
  PCIe slot FIX methods (FIX0-X, FIXV, FIXY, FIXW) via UBA protocol GUIDs.

  Driver entry point (ModuleEntryPoint):
    1. Initializes global UEFI table pointers (ImageHandle, SystemTable, etc.)
    2. Locates the HOB list via DxeHobLib
    3. Locates the UMPT, PIRQ, and FIX protocol interfaces
    4. Registers the ACPI path format table for downstream UBA dispatch code

  Build:
    VS2015, X64, DEBUG
    Source: PurleyRpPkg\Uba\UbaMain\Dxe\TypeNeonCityEPECB\StaticSkuDataDxe\

  Copyright (c) 2017-2018, Intel Corporation. All rights reserved.
  SPDX-License-Identifier: BSD-2-Clause-Patent
**/

#include "StaticSkuDataDxeNeonCityEPECB.h"

//
// ---------------------------------------------------------------------------
// Global UEFI Table Pointers
// ---------------------------------------------------------------------------
//

///
/// The image handle of this driver (set by ModuleEntryPoint).
///
EFI_HANDLE           gImageHandle = NULL;

///
/// Pointer to the UEFI System Table (set by ModuleEntryPoint).
///
EFI_SYSTEM_TABLE    *gSystemTable = NULL;

///
/// Pointer to the UEFI Boot Services Table (set by ModuleEntryPoint).
///
EFI_BOOT_SERVICES   *gBootServices = NULL;

///
/// Pointer to the UEFI Runtime Services Table (set by ModuleEntryPoint).
///
EFI_RUNTIME_SERVICES *gRuntimeServices = NULL;

///
/// Pointer to the HOB (Hand-Off Block) list, initialized via DxeHobLib.GetBootModeHobList().
///
VOID *mHobList = NULL;

//
// ---------------------------------------------------------------------------
// Protocol GUID definitions (stored in .data section at 0x4060-0x40E0)
// ---------------------------------------------------------------------------
//

///
/// OEM-defined GUID for UMPT protocol (untyped).
/// Bytes at 0x4060-0x407F (32 bytes, first 16 = GUID, next 8 = name, next 8 = reserved/ptr)
///
/// GUID: 36292936-0E76-31C8-A13A-3AF2FC1C3932
/// This is a UEFI configuration table GUID used to locate platform-specific data.
///
GUID mUbaOemGuid1 = { 0x36292936, 0x0E76, 0x31C8, { 0xA1, 0x3A, 0x3A, 0xF2, 0xFC, 0x1C, 0x39, 0x32 } };

///
/// OEM-defined GUID for the second protocol table.
/// Identifies the UMPT protocol handler GUID.
///
GUID mUbaOemGuid2 = { 0x460D3EE0, 0x6352, 0x4548, { 0xB0, 0xA4, 0x58, 0xD5, 0x7B, 0x31, 0x77, 0xE2 } };

///
/// UMPT (UEFI Memory Protocol Table) GUID.
/// { 0x7739F24C, 0x93D7, 0x11D4, { 0x9A, 0x3A, 0x00, 0x90, 0x27, 0x3F, 0xC1, 0x4D } }
///
/// This GUID identifies the UMPT protocol. The protocol provides memory topology
/// and configuration data used by the board-specific SKU dispatch code.
///
EFI_GUID gUmpProtocolGuid = UMPT_PROTOCOL_GUID;

///
/// Name string "UMPT" stored alongside the GUID in the protocol registration entry.
///
CHAR8 mUmpProtocolName[4] = "UMPT";

///
/// PIRQ (PCI Interrupt Routing) protocol GUID.
/// { 0xCFA1F80F, 0xABA0, 0xC04A, { 0xBF, 0xC9, 0x34, 0xA7, 0x8F, 0x68, 0xDD, 0x8A } }
///
/// This GUID identifies the PIRQ protocol, which provides PCI interrupt routing
/// configuration for the board.
///
EFI_GUID gPirqProtocolGuid = PIRQ_PROTOCOL_GUID;

///
/// Name string "PIRQ" stored alongside the GUID in the protocol registration entry.
///
CHAR8 mPirqProtocolName[4] = "PIRQ";

///
/// FIX protocol GUID.
/// { 0xA5481F4C, 0x76C9, 0x904D, { 0x9F, 0x03, 0x8E, 0x9B, 0x1C, 0x32, 0x7F, 0xCF } }
///
/// This GUID identifies the FIX protocol, which provides board-specific fixup
/// configuration methods (FIX0-X, FIXV, FIXY, FIXW).
///
EFI_GUID gFixProtocolGuid = FIX_PROTOCOL_GUID;

///
/// ACPF (ACPI Path Format) protocol GUID.
/// { 0xF89E1281, 0x1D39, 0x4F63, { 0xAE, 0x99, 0x58, 0x51, 0x7E, 0xC0, 0x77, 0xE3 } }
///
/// This GUID identifies the ACPF protocol, which exposes the ACPI path format
/// table to UBA dispatch code.
///
EFI_GUID gAcpfProtocolGuid = ACPF_PROTOCOL_GUID;

//
// ---------------------------------------------------------------------------
// NVDIMM ACPI Path Name Strings (stored in .rdata section)
// ---------------------------------------------------------------------------
//
// The .rdata section contains ACPI path strings organized as:
//   _SB_.CCT0..CCT8     - Core Count Tables (9 entries)
//   _SB_.CFH0..CFH8     - Core Frequency Headers (9 entries)
//   _SB_.NVDR.DRVT      - NVDIMM Driver root
//   _SB_.NVDR.Nnnn.FXxx - NVDIMM function methods (FXCD, FXST, FXIN, FXOU, FXBS, FXFH, CENA, CFIS)
//   _SB_.PCnn.FIXx       - PCIe Root Port FIX methods
//   _SB_.PCnn.MCPx.MCTL  - PCIe MCTL method
//   _SB_.PCnn.SRxx.MCTL  - PCIe SR MCTL method
//   _SB_.PCnn.RRxx.MCTL  - PCIe RR MCTL method
//

//
// CCT (Core Count Table) path strings: _SB_.CCT0 through _SB_.CCT8
// These map CPU core count configurations per socket.
//
// CFH (Core Frequency Header) path strings: _SB_.CFH0 through _SB_.CFH8
// These map CPU frequency configuration headers per socket.
//

//
// NVDIMM namespace: _SB_.NVDR
// NVDIMM sub-namespace devices (N000 through N151, organized by socket/channel)
//
// Each NVDIMM entry has ACPI methods:
//   FXCD - Function Code
//   FXST - Function Status
//   FXIN - Function Input
//   FXOU - Function Output
//   FXBS - Function Buffer Size
//   FXFH - Function File Handle
//   CENA - Command Enable
//   CFIS - Command Finish Status
//

//
// PCIe Root Port FIX methods (per port):
//   FIX1-FIX4 - Fixed method entries
//   FIX5-FIX7 - Variable method entries
//   MCTL      - Management Control
//
// FIX types:
//   FIX0  - Base FIX method (single entry)
//   FIXX  - FIX Extension (0..8 per device) with type=0x5B80
//   FIXV  - FIX Value (0..8 per device) with type=0x5B80
//   FIXY  - NVDIMM FIX Y (per sub-instance) with type=0x0008
//   FIXW  - NVDIMM FIX W (per sub-instance) with type=0x0008
//

//
// ---------------------------------------------------------------------------
// ACPI Path Entry Table (stored in .data section at 0x40E0+)
// ---------------------------------------------------------------------------
//
// This table is organized as arrays of 32-byte ACPI_PATH_ENTRY structures.
// The table is anchored by the ACPI_PATH_FORMAT_TABLE header at 0x9000.
//
// Entry sections:
//
//   1. PSYS-rooted entries (type 0x5B80):
//      - FIX0 base entry (index 0)
//      - FIXX entries (indices 0..8) for PCIe slot configuration
//      - FIXV entries (indices 0..8) for PCIe slot value configuration
//
//   2. NVDR-rooted entries (type 0x0008):
//      - NVDIMM sub-instance entries organized in groups of 8 per NVDR instance
//      - Each group contains: 6 data entries + 1 FIXY + 1 FIXW
//      - Groups correspond to: N000, N001, N010, N011, N020, N021, ..., N151
//      - NVDIMM instances span sockets (0-1), channels (0-5 per socket), slots (0-1)
//
// Total entries: ~200+ entries covering all board-configurable ACPI paths
//

//
// ACPI Path Format Table header at 0x9000
//
// Signature: "ACPF" (0x46504341)
// Version: 1
// EntryTable pointer: 0x40E0 (gAcpiPathEntries)
//
ACPI_PATH_FORMAT_TABLE mAcpiPathFormatTable = {
  .Signature    = 0x46504341,   ///< "ACPF"
  .Version      = 1,
  .EntryTable   = (ACPI_PATH_ENTRY *) 0x40E0,
};

//
// Protocol registration handles (initialized by entry point)
//
UINT64 gUmpProtocolHandle  = 0;
UINT64 gPirqProtocolHandle = 0;
UINT64 gFixProtocolHandle  = 0;

//
// ---------------------------------------------------------------------------
// Forward Function Declarations
// ---------------------------------------------------------------------------
//

/**
  Allocate and initialize the protocol handle cache.
  Uses BootServices->AllocatePool to allocate memory and then maps the
  UMPT protocol from the configuration table.
**/
UINT64
InitializeUmpProtocol (
  VOID
  );

/**
  Locate the HOB list from SystemTable and validate it.
  Uses HOB list pointer from DXE HOB Library.
**/
VOID *
GetBootModeHobList (
  VOID
  );

/**
  Compare two 8-byte values for equality (unaligned read).

  @param[in] Buffer  Pointer to the data to read.

  @return The 64-bit value read from the buffer.
**/
UINT64
ReadUnaligned64 (
  IN CONST VOID *Buffer
  );

/**
  Check if a HOB entry matches the expected GUID.

  @param[in]  Entry       Pointer to the HOB entry.
  @param[in]  Expected    Pointer to the expected GUID structure.

  @retval TRUE   The HOB entry matches.
  @retval FALSE  The HOB entry does not match.
**/
BOOLEAN
IsHobMatchingGuid (
  IN VOID   *Entry,
  IN VOID   *Expected
  );

/**
  Output ASSERT message via protocol.

  @param[in]  FileName    Source file name string.
  @param[in]  LineNumber  Line number of the assertion.
  @param[in]  Description Assertion description string.
**/
VOID
AssertPrint (
  IN CONST CHAR8 *FileName,
  IN UINTN       LineNumber,
  IN CONST CHAR8 *Description
  );

/**
  Check an EFI status and assert on error.

  @param[in]  Status  The EFI_STATUS value to check.
**/
VOID
CheckEfiStatus (
  IN EFI_STATUS Status
  );

/**
  Read CMOS index register 0x70 and return a board-type dependent value.

  @param[in]  InputValue  Input parameter (unused in current implementation).

  @return Board-type specific value derived from CMOS index 0x4B.
**/
UINT8
GetBoardTypeFromCmos (
  IN UINT64 InputValue
  );

//
// ---------------------------------------------------------------------------
// Function Implementations
// ---------------------------------------------------------------------------
//

/**
  Allocate and initialize the UMP protocol handle cache.

  Allocates a pool buffer, checks system memory size (must be <= 16),
  then locates the UMPT configuration table entry and caches the protocol handle.

  @return The protocol handle pointer, or NULL on failure.
**/
UINT64
InitializeUmpProtocol (
  VOID
  )
{
  UINT64                Handle;
  UINT64                PoolSize;
  UINT64                Result;
  UINT64                CachedHandle;

  Handle = gUmpProtocolHandle;
  if (Handle != 0) {
    return Handle;
  }

  //
  // Allocate a pool buffer using BootServices->AllocatePool (index 24 = 3 entries).
  //
  PoolSize = gBootServices->AllocatePool (31);
  gBootServices->FreePool (PoolSize);

  if (PoolSize <= 0x10) {
    //
    // Locate the UMPT protocol configuration table entry.
    //
    Result = gBootServices->LocateProtocol (&gUmpOemGuid2, NULL, &CachedHandle);
    if (Result < 0) {
      CachedHandle = 0;
    }
    gUmpProtocolHandle = CachedHandle;
    return CachedHandle;
  }

  return 0;
}

/**
  Locate the HOB list from the UEFI System Table via the DXE HOB Library.

  Searches the HOB list for an entry matching the expected GUID and caches it.
  If no matching HOB is found, triggers an ASSERT.

  @param[in]  A1  Parameter (currently unused, retained for compatibility).

  @return The cached HOB list pointer.
**/
VOID *
GetBootModeHobList (
  VOID
  )
{
  VOID   *HobList;
  UINT64  EntryCount;
  UINT64  EntryBase;
  UINT64  CurrentEntry;

  HobList = mHobList;
  if (HobList != NULL) {
    return HobList;
  }

  //
  // Initialize search state.
  //
  EntryCount = 0;
  mHobList   = NULL;

  //
  // Check if the SystemTable has a HOB list pointer at offset 104.
  // (Offset 104 in EFI_SYSTEM_TABLE is the HOB list pointer in PI-spec DXE phase.)
  //
  if (*(UINT64 *)((UINT8 *)gSystemTable + 104) != 0) {
    EntryBase = *(UINT64 *)((UINT8 *)gSystemTable + 112);

    //
    // Linear search through the HOB list.
    //
    for (CurrentEntry = EntryBase; ; CurrentEntry += 24) {
      if (!IsHobMatchingGuid ((VOID *)CurrentEntry, (VOID *)CurrentEntry)) {
        break;
      }
      EntryCount++;
      if (EntryCount >= *(UINT64 *)((UINT8 *)gSystemTable + 104)) {
        goto FoundEntries;
      }
    }

    //
    // Found matching entry: extract the HOB pointer at offset +16.
    //
    HobList = *(VOID **)(CurrentEntry + 16);
    mHobList = HobList;
    return HobList;

FoundEntries:
    //
    // No matching HOB entry found; trigger ASSERT.
    //
    CheckEfiStatus ((EFI_STATUS)(UINTN)0x800000000000000EULL);
    return mHobList;
  }

  //
  // SystemTable has no HOB list; trigger ASSERT.
  //
  CheckEfiStatus ((EFI_STATUS)(UINTN)0x800000000000000EULL);
  return mHobList;
}

/**
  Read an unaligned 64-bit value from the specified buffer.

  @param[in]  Buffer  Pointer to the data to read (need not be aligned).

  @return The 64-bit value at the buffer address.
**/
UINT64
ReadUnaligned64 (
  IN CONST VOID *Buffer
  )
{
  if (Buffer == NULL) {
    AssertPrint (
      "e:\\hs\\MdePkg\\Library\\BaseLib\\Unaligned.c",
      192,
      "Buffer != ((void *) 0)"
      );
  }
  return *(const UINT64 *)Buffer;
}

/**
  Compare two GUID entries in the HOB list by matching their first and second
  64-bit fields.

  @param[in]  Entry     Pointer to the HOB entry GUID.
  @param[in]  Expected  Pointer to the expected GUID structure.

  @retval TRUE   The two GUIDs match.
  @retval FALSE  The two GUIDs do not match (or NULL parameters).
**/
BOOLEAN
IsHobMatchingGuid (
  IN VOID   *Entry,
  IN VOID   *Expected
  )
{
  UINT64 GuidFirst;
  UINT64 GuidSecond;
  UINT64 ExpectedFirst;
  UINT64 ExpectedSecond;

  //
  // Read both GUIDs via unaligned 64-bit reads.
  //
  GuidFirst     = ReadUnaligned64 (Entry);
  GuidSecond    = ReadUnaligned64 ((UINT8 *)Entry + 8);
  ExpectedFirst = ReadUnaligned64 (Expected);
  ExpectedSecond= ReadUnaligned64 ((UINT8 *)Expected + 8);

  return (BOOLEAN)(GuidFirst == ExpectedFirst && GuidSecond == ExpectedSecond);
}

/**
  Output an assertion message using the registered debug/report protocol.

  If the assertion protocol is available, it formats and dispatches the
  assertion string. Otherwise, falls back to a simple ASSERT output.

  @param[in]  FileName     Source file name string.
  @param[in]  LineNumber   Line number of the assertion.
  @param[in]  Description  Assertion description string.
**/
VOID
AssertPrint (
  IN CONST CHAR8 *FileName,
  IN UINTN       LineNumber,
  IN CONST CHAR8 *Description
  )
{
  UINT64 ProtocolHandle;

  ProtocolHandle = InitializeUmpProtocol ();
  if (ProtocolHandle != 0) {
    //
    // Dispatch assertion via the protocol's report function (at offset +8 in vtable).
    //
    ((VOID (*)(CONST CHAR8 *, UINTN, CONST CHAR8 *)) \
      (*(UINT64 **)ProtocolHandle)[1]) (FileName, LineNumber, Description);
  }
}

/**
  Check an EFI status value and assert on error.

  If the status has the high error bit set, triggers a debug assertion
  with the status code formatted as %r.

  @param[in]  Status  The EFI_STATUS value to check.
**/
VOID
CheckEfiStatus (
  IN EFI_STATUS Status
  )
{
  //
  // Check if this is a HOB list pointer validation by examining
  // the SystemTable HOB count entry.
  //
  GetBootModeHobList ((UINT64)Status);
}

/**
  Read the CMOS configuration register and return a board-type dependent value
  used for SKU data dispatch.

  Reads CMOS index 0x4B (preserving bit 7 of index 0x70), interprets the value,
  and maps it to a protocol call type.

  @param[in]  InputValue  Input parameter (used as a bitmask for protocol dispatch).

  @return Board-type dependent value, or 4 if the CMOS value is out of expected range.
**/
UINT8
GetBoardTypeFromCmos (
  IN UINT64 InputValue
  )
{
  UINT64                ProtocolHandle;
  UINT64                DispatchType;
  UINT64                (*ProtocolFunc)(UINT64, UINT64, UINT64 *);
  UINT8                 CmosValue;
  UINT8                 RawCmosResult;
  UINT8                 Result;

  //
  // Get the UMP protocol handle for dispatch.
  //
  ProtocolHandle = InitializeUmpProtocol ();
  DispatchType   = 0;
  ProtocolFunc   = (VOID *)ProtocolHandle;

  if (ProtocolHandle != 0) {
    //
    // Read CMOS index register 0x70, set index to 0x4B (preserving NMI enable bit).
    //
    CmosValue = IoRead8 (0x70);
    IoWrite8 (0x70, CmosValue & 0x80 | 0x4B);

    //
    // Read the CMOS data register 0x71.
    //
    RawCmosResult = IoRead8 (0x71);
    Result = RawCmosResult;

    //
    // If the value is > 3, it may indicate a special configuration.
    // However, the inner check for value == 0 is unreachable in this path
    // (0 <= 3, so it never enters this block). This is compiler-intrinsic
    // behavior preserved from the original binary.
    //
    if (RawCmosResult > 3) {
      if (RawCmosResult == 0) {
        Result = (*(volatile UINT8 *)0xFDAF0490 & 2) | 1;
      }
    }

    //
    // Map the CMOS result to a dispatch type.
    //   Raw CMOS value >= 1 and <= 254: valid range (n3_1 - 1 <= 0xFD).
    //     Value 1 -> DispatchType = 0x8000000C (NVDIMM path data dispatch)
    //     Value 2..254 -> DispatchType = 0x80000006 (FIX/PCIe path data dispatch)
    //   Raw CMOS value 0 or 255: out of range, no dispatch performed,
    //     returns default value of (RawCmosResult - 1).
    //
    if ((UINT8)(RawCmosResult - 1) <= 0xFD) {
      Result = 4;
      DispatchType = 0x80000006LL;
      if (RawCmosResult == 1) {
        DispatchType = 0x8000000CLL;
      }
    } else {
      Result = RawCmosResult - 1;
    }

    //
    // If the dispatch type matches the input mask, call the protocol function.
    //
    if ((DispatchType & (UINT64)InputValue) != 0) {
      Result = ProtocolFunc ((UINT64)InputValue, (UINT64)Result, (UINT64 *)&Result);
    }
  }

  return Result;
}

/**
  Driver entry point.

  Initializes UEFI global table pointers, locates the HOB list,
  then registers the ACPI path format table, UMPT, PIRQ, and FIX
  protocols via the UEFI boot services configuration table mechanism.

  @param[in]  ImageHandle  The firmware allocated handle for the EFI image.
  @param[in]  SystemTable  A pointer to the EFI System Table.

  @retval EFI_SUCCESS           The entry point executed successfully.
  @retval EFI_NOT_FOUND         A required protocol or configuration table was not found.
  @retval EFI_OUT_OF_RESOURCES  Memory allocation failed.
**/
EFI_STATUS
EFIAPI
ModuleEntryPoint (
  IN EFI_HANDLE        ImageHandle,
  IN EFI_SYSTEM_TABLE  *SystemTable
  )
{
  EFI_STATUS  Status;
  UINT64      ProtocolHandle;

  //
  // Save the global UEFI table pointers.
  //
  gImageHandle = ImageHandle;
  if (ImageHandle == NULL) {
    AssertPrint (
      "e:\\hs\\MdePkg\\Library\\UefiBootServicesTableLib\\UefiBootServicesTableLib.c",
      51,
      "gImageHandle != ((void *) 0)"
      );
  }

  gSystemTable = SystemTable;
  if (SystemTable == NULL) {
    AssertPrint (
      "e:\\hs\\MdePkg\\Library\\UefiBootServicesTableLib\\UefiBootServicesTableLib.c",
      57,
      "gST != ((void *) 0)"
      );
  }

  gBootServices = SystemTable->BootServices;
  if (gBootServices == NULL) {
    AssertPrint (
      "e:\\hs\\MdePkg\\Library\\UefiBootServicesTableLib\\UefiBootServicesTableLib.c",
      63,
      "gBS != ((void *) 0)"
      );
  }

  gRuntimeServices = SystemTable->RuntimeServices;
  if (gRuntimeServices == NULL) {
    AssertPrint (
      "e:\\hs\\MdePkg\\Library\\UefiRuntimeServicesTableLib\\UefiRuntimeServicesTableLib.c",
      47,
      "gRT != ((void *) 0)"
      );
  }

  //
  // Initialize the HOB list for later use (DxeHobLib compatibility).
  //
  GetBootModeHobList ();

  //
  // Locate the OEM configuration table entry (UMPT protocol GUID)
  // to obtain the protocol handle for subsequent registrations.
  //
  ProtocolHandle = 0;
  Status = gBootServices->LocateProtocol (
                            &gUbaOemGuid2,
                            NULL,
                            &ProtocolHandle
                            );
  if (!EFI_ERROR (Status)) {
    //
    // Register the ACPI path format table.
    // This installs the mAcpiPathFormatTable (containing "ACPF" signature
    // and pointer to the ACPI path entries) as the protocol interface.
    //
    Status = ((EFI_STATUS (*)(UINT64, VOID *, VOID *, UINTN)) \
              (*((UINT64 **)ProtocolHandle))[2]) (
               ProtocolHandle,
               &gAcpfProtocolGuid,
               &mAcpiPathFormatTable,
               sizeof (mAcpiPathFormatTable)
               );
    if (!EFI_ERROR (Status)) {
      //
      // Register the PIRQ protocol interface.
      //
      Status = ((EFI_STATUS (*)(UINT64, VOID *, VOID *, UINTN)) \
                (*((UINT64 **)ProtocolHandle))[2]) (
                 ProtocolHandle,
                 &gPirqProtocolGuid,
                 &gPirqProtocolHandle,
                 sizeof (gPirqProtocolHandle)
                 );
      if (!EFI_ERROR (Status)) {
        //
        // Register the FIX protocol interface.
        //
        return ((EFI_STATUS (*)(UINT64, VOID *, VOID *, UINTN)) \
                (*((UINT64 **)ProtocolHandle))[2]) (
                 ProtocolHandle,
                 &gFixProtocolGuid,
                 &gFixProtocolHandle,
                 sizeof (gFixProtocolHandle)
                 );
      }
    }
  }

  return Status;
}