/** @file Copyright (c) 2017, Intel Corporation. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent **/ #include #include #include #include #include #include #include // // Machine Specific Registers (MSRs) // #define SMM_FEATURES_LIB_IA32_MTRR_CAP 0x0FE #define SMM_FEATURES_LIB_IA32_SMRR_PHYSBASE 0x1F2 #define SMM_FEATURES_LIB_IA32_SMRR_PHYSMASK 0x1F3 #define SMM_FEATURES_LIB_IA32_CORE_SMRR_PHYSBASE 0x0A0 #define SMM_FEATURES_LIB_IA32_CORE_SMRR_PHYSMASK 0x0A1 BOOLEAN mSmrrSupported = FALSE; // // Set default value to assume IA-32 Architectural MSRs are used // UINT32 mSmrrPhysBaseMsr = SMM_FEATURES_LIB_IA32_SMRR_PHYSBASE; UINT32 mSmrrPhysMaskMsr = SMM_FEATURES_LIB_IA32_SMRR_PHYSMASK; EFI_STATUS TestPointCheckSmrr ( VOID ) { UINT64 SmrrBase; UINT64 SmrrMask; UINT64 Length; UINT32 RegEax; UINT32 RegEdx; UINTN FamilyId; UINTN ModelId; BOOLEAN Result; DEBUG ((DEBUG_INFO, "==== TestPointCheckSmrr - Enter\n")); // // Retrieve CPU Family and Model // AsmCpuid (CPUID_VERSION_INFO, &RegEax, NULL, NULL, &RegEdx); FamilyId = (RegEax >> 8) & 0xf; ModelId = (RegEax >> 4) & 0xf; if (FamilyId == 0x06 || FamilyId == 0x0f) { ModelId = ModelId | ((RegEax >> 12) & 0xf0); } // // Check CPUID(CPUID_VERSION_INFO).EDX[12] for MTRR capability // if ((RegEdx & BIT12) != 0) { // // Check MTRR_CAP MSR bit 11 for SMRR support // if ((AsmReadMsr64 (SMM_FEATURES_LIB_IA32_MTRR_CAP) & BIT11) != 0) { mSmrrSupported = TRUE; } } // // Intel(R) 64 and IA-32 Architectures Software Developer's Manual // Volume 3C, Section 35.3 MSRs in the Intel(R) Atom(TM) Processor Family // // If CPU Family/Model is 06_1CH, 06_26H, 06_27H, 06_35H or 06_36H, then // SMRR Physical Base and SMM Physical Mask MSRs are not available. // if (FamilyId == 0x06) { if (ModelId == 0x1C || ModelId == 0x26 || ModelId == 0x27 || ModelId == 0x35 || ModelId == 0x36) { mSmrrSupported = FALSE; } } // // Intel(R) 64 and IA-32 Architectures Software Developer's Manual // Volume 3C, Section 35.2 MSRs in the Intel(R) Core(TM) 2 Processor Family // // If CPU Family/Model is 06_0F or 06_17, then use Intel(R) Core(TM) 2 // Processor Family MSRs // if (FamilyId == 0x06) { if (ModelId == 0x17 || ModelId == 0x0f) { mSmrrPhysBaseMsr = SMM_FEATURES_LIB_IA32_CORE_SMRR_PHYSBASE; mSmrrPhysMaskMsr = SMM_FEATURES_LIB_IA32_CORE_SMRR_PHYSMASK; } } SmrrMask = 0; if (mSmrrSupported) { SmrrBase = AsmReadMsr64 (mSmrrPhysBaseMsr); SmrrMask = AsmReadMsr64 (mSmrrPhysMaskMsr); DEBUG ((DEBUG_INFO, "SMRR : Base=%016lx Make=%016lx\n", SmrrBase, SmrrMask)); } DEBUG ((DEBUG_INFO, "==== TestPointCheckSmrr - Exit\n")); Result = TRUE; if (!mSmrrSupported) { Result = FALSE; DEBUG ((DEBUG_ERROR, "Smrr is not supported\n")); } else { Length = SmrrMask & ~0xFFFull; Length = ~Length + 1; Length = Length & 0xFFFFF000; if (Length != GetPowerOfTwo64 (Length)) { Result = FALSE; DEBUG ((DEBUG_ERROR, "Smrr is not aligned\n")); } } if (!Result) { TestPointLibAppendErrorString ( PLATFORM_TEST_POINT_ROLE_PLATFORM_IBV, NULL, TEST_POINT_BYTE6_SMM_END_OF_DXE_SMRR_FUNCTIONAL_ERROR_CODE \ TEST_POINT_SMM_END_OF_DXE \ TEST_POINT_BYTE6_SMM_END_OF_DXE_SMRR_FUNCTIONAL_ERROR_STRING ); return EFI_INVALID_PARAMETER; } return EFI_SUCCESS; }