Signed-off-by: Ray Ni <ray.ni@intel.com> Cc: Eric Dong <eric.dong@intel.com> Cc: Rahul Kumar <rahul1.kumar@intel.com> Cc: Gerd Hoffmann <kraxel@redhat.com> Reviewed-by: Eric Dong <eric.dong@intel.com> Reviewed-by: Ray Ni <ray.ni@intel.com>
		
			
				
	
	
		
			1102 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1102 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   Unit tests of the MtrrLib instance of the MtrrLib class
 | |
| 
 | |
|   Copyright (c) 2018 - 2023, Intel Corporation. All rights reserved.<BR>
 | |
|   SPDX-License-Identifier: BSD-2-Clause-Patent
 | |
| 
 | |
| **/
 | |
| 
 | |
| #include "MtrrLibUnitTest.h"
 | |
| 
 | |
| MTRR_MEMORY_CACHE_TYPE  mMemoryCacheTypes[] = {
 | |
|   CacheUncacheable, CacheWriteCombining, CacheWriteThrough, CacheWriteProtected, CacheWriteBack
 | |
| };
 | |
| 
 | |
| UINT64                                       mFixedMtrrsValue[MTRR_NUMBER_OF_FIXED_MTRR];
 | |
| MSR_IA32_MTRR_PHYSBASE_REGISTER              mVariableMtrrsPhysBase[MTRR_NUMBER_OF_VARIABLE_MTRR];
 | |
| MSR_IA32_MTRR_PHYSMASK_REGISTER              mVariableMtrrsPhysMask[MTRR_NUMBER_OF_VARIABLE_MTRR];
 | |
| MSR_IA32_MTRR_DEF_TYPE_REGISTER              mDefTypeMsr;
 | |
| MSR_IA32_MTRRCAP_REGISTER                    mMtrrCapMsr;
 | |
| MSR_IA32_TME_ACTIVATE_REGISTER               mTmeActivateMsr;
 | |
| CPUID_VERSION_INFO_EDX                       mCpuidVersionInfoEdx;
 | |
| CPUID_STRUCTURED_EXTENDED_FEATURE_FLAGS_ECX  mCpuidExtendedFeatureFlagsEcx;
 | |
| CPUID_VIR_PHY_ADDRESS_SIZE_EAX               mCpuidVirPhyAddressSizeEax;
 | |
| 
 | |
| BOOLEAN       mRandomInput;
 | |
| UINTN         mNumberIndex = 0;
 | |
| extern UINTN  mNumbers[];
 | |
| extern UINTN  mNumberCount;
 | |
| 
 | |
| /**
 | |
|   Return a random number between 0 and RAND_MAX.
 | |
| 
 | |
|   If mRandomInput is TRUE, the routine directly calls rand().
 | |
|   Otherwise, the routine returns the pre-generated numbers.
 | |
| 
 | |
|   @return a number between 0 and RAND_MAX.
 | |
| **/
 | |
| UINTN
 | |
| Rand (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   if (mRandomInput) {
 | |
|     return rand ();
 | |
|   } else {
 | |
|     return mNumbers[mNumberIndex++ % (mNumberCount - 1)];
 | |
|   }
 | |
| }
 | |
| 
 | |
| CHAR8  mContentTemplate[] = {
 | |
|   "/** @file\n"
 | |
|   "  Pre-generated random number used by MtrrLib test.\n"
 | |
|   "\n"
 | |
|   "  Copyright (c) 2020, Intel Corporation. All rights reserved.<BR>\n"
 | |
|   "  SPDX-License-Identifier: BSD-2-Clause-Patent\n"
 | |
|   "**/\n"
 | |
|   "UINTN mNumberCount = %d;\n"
 | |
|   "UINTN mNumbers[] = {"
 | |
| };
 | |
| 
 | |
| /**
 | |
|   Generate Count random numbers in FilePath.
 | |
| 
 | |
|   @param FilePath  The file path to put the generated random numbers.
 | |
|   @param Count     Count of random numbers.
 | |
| **/
 | |
| VOID
 | |
| GenerateRandomNumbers (
 | |
|   CHAR8  *FilePath,
 | |
|   UINTN  Count
 | |
|   )
 | |
| {
 | |
|   FILE   *File;
 | |
|   UINTN  Index;
 | |
| 
 | |
|   File = fopen (FilePath, "w");
 | |
|   fprintf (File, mContentTemplate, Count);
 | |
|   for (Index = 0; Index < Count; Index++) {
 | |
|     if (Index % 10 == 0) {
 | |
|       fprintf (File, "\n ");
 | |
|     }
 | |
| 
 | |
|     fprintf (File, " %d,", rand ());
 | |
|   }
 | |
| 
 | |
|   fprintf (File, "\n};\n");
 | |
|   fclose (File);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Retrieves CPUID information using an extended leaf identifier.
 | |
| 
 | |
|   Executes the CPUID instruction with EAX set to the value specified by Index
 | |
|   and ECX set to the value specified by SubIndex. This function always returns
 | |
|   Index. This function is only available on IA-32 and x64.
 | |
| 
 | |
|   If Eax is not NULL, then the value of EAX after CPUID is returned in Eax.
 | |
|   If Ebx is not NULL, then the value of EBX after CPUID is returned in Ebx.
 | |
|   If Ecx is not NULL, then the value of ECX after CPUID is returned in Ecx.
 | |
|   If Edx is not NULL, then the value of EDX after CPUID is returned in Edx.
 | |
| 
 | |
|   @param  Index     The 32-bit value to load into EAX prior to invoking the
 | |
|                     CPUID instruction.
 | |
|   @param  SubIndex  The 32-bit value to load into ECX prior to invoking the
 | |
|                     CPUID instruction.
 | |
|   @param  Eax       The pointer to the 32-bit EAX value returned by the CPUID
 | |
|                     instruction. This is an optional parameter that may be
 | |
|                     NULL.
 | |
|   @param  Ebx       The pointer to the 32-bit EBX value returned by the CPUID
 | |
|                     instruction. This is an optional parameter that may be
 | |
|                     NULL.
 | |
|   @param  Ecx       The pointer to the 32-bit ECX value returned by the CPUID
 | |
|                     instruction. This is an optional parameter that may be
 | |
|                     NULL.
 | |
|   @param  Edx       The pointer to the 32-bit EDX value returned by the CPUID
 | |
|                     instruction. This is an optional parameter that may be
 | |
|                     NULL.
 | |
| 
 | |
|   @return Index.
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| EFIAPI
 | |
| UnitTestMtrrLibAsmCpuidEx (
 | |
|   IN      UINT32  Index,
 | |
|   IN      UINT32  SubIndex,
 | |
|   OUT     UINT32  *Eax   OPTIONAL,
 | |
|   OUT     UINT32  *Ebx   OPTIONAL,
 | |
|   OUT     UINT32  *Ecx   OPTIONAL,
 | |
|   OUT     UINT32  *Edx   OPTIONAL
 | |
|   )
 | |
| {
 | |
|   switch (Index) {
 | |
|     case CPUID_SIGNATURE:
 | |
|       if (Eax != NULL) {
 | |
|         *Eax = CPUID_STRUCTURED_EXTENDED_FEATURE_FLAGS;
 | |
|       }
 | |
| 
 | |
|       return Index;
 | |
|       break;
 | |
|     case CPUID_VERSION_INFO:
 | |
|       if (Edx != NULL) {
 | |
|         *Edx = mCpuidVersionInfoEdx.Uint32;
 | |
|       }
 | |
| 
 | |
|       return Index;
 | |
|       break;
 | |
|     case CPUID_STRUCTURED_EXTENDED_FEATURE_FLAGS:
 | |
|       if (Ecx != NULL) {
 | |
|         *Ecx = mCpuidExtendedFeatureFlagsEcx.Uint32;
 | |
|       }
 | |
| 
 | |
|       return Index;
 | |
|       break;
 | |
|     case CPUID_EXTENDED_FUNCTION:
 | |
|       if (Eax != NULL) {
 | |
|         *Eax = CPUID_VIR_PHY_ADDRESS_SIZE;
 | |
|       }
 | |
| 
 | |
|       return Index;
 | |
|       break;
 | |
|     case CPUID_VIR_PHY_ADDRESS_SIZE:
 | |
|       if (Eax != NULL) {
 | |
|         *Eax = mCpuidVirPhyAddressSizeEax.Uint32;
 | |
|       }
 | |
| 
 | |
|       return Index;
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Should never fall through to here
 | |
|   //
 | |
|   ASSERT (FALSE);
 | |
|   return Index;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Retrieves CPUID information.
 | |
| 
 | |
|   Executes the CPUID instruction with EAX set to the value specified by Index.
 | |
|   This function always returns Index.
 | |
|   If Eax is not NULL, then the value of EAX after CPUID is returned in Eax.
 | |
|   If Ebx is not NULL, then the value of EBX after CPUID is returned in Ebx.
 | |
|   If Ecx is not NULL, then the value of ECX after CPUID is returned in Ecx.
 | |
|   If Edx is not NULL, then the value of EDX after CPUID is returned in Edx.
 | |
|   This function is only available on IA-32 and x64.
 | |
| 
 | |
|   @param  Index The 32-bit value to load into EAX prior to invoking the CPUID
 | |
|                 instruction.
 | |
|   @param  Eax   The pointer to the 32-bit EAX value returned by the CPUID
 | |
|                 instruction. This is an optional parameter that may be NULL.
 | |
|   @param  Ebx   The pointer to the 32-bit EBX value returned by the CPUID
 | |
|                 instruction. This is an optional parameter that may be NULL.
 | |
|   @param  Ecx   The pointer to the 32-bit ECX value returned by the CPUID
 | |
|                 instruction. This is an optional parameter that may be NULL.
 | |
|   @param  Edx   The pointer to the 32-bit EDX value returned by the CPUID
 | |
|                 instruction. This is an optional parameter that may be NULL.
 | |
| 
 | |
|   @return Index.
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| EFIAPI
 | |
| UnitTestMtrrLibAsmCpuid (
 | |
|   IN      UINT32  Index,
 | |
|   OUT     UINT32  *Eax   OPTIONAL,
 | |
|   OUT     UINT32  *Ebx   OPTIONAL,
 | |
|   OUT     UINT32  *Ecx   OPTIONAL,
 | |
|   OUT     UINT32  *Edx   OPTIONAL
 | |
|   )
 | |
| {
 | |
|   return UnitTestMtrrLibAsmCpuidEx (Index, 0, Eax, Ebx, Ecx, Edx);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Returns a 64-bit Machine Specific Register(MSR).
 | |
| 
 | |
|   Reads and returns the 64-bit MSR specified by Index. No parameter checking is
 | |
|   performed on Index, and some Index values may cause CPU exceptions. The
 | |
|   caller must either guarantee that Index is valid, or the caller must set up
 | |
|   exception handlers to catch the exceptions. This function is only available
 | |
|   on IA-32 and x64.
 | |
| 
 | |
|   @param  MsrIndex The 32-bit MSR index to read.
 | |
| 
 | |
|   @return The value of the MSR identified by MsrIndex.
 | |
| 
 | |
| **/
 | |
| UINT64
 | |
| EFIAPI
 | |
| UnitTestMtrrLibAsmReadMsr64 (
 | |
|   IN UINT32  MsrIndex
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
| 
 | |
|   UT_ASSERT_EQUAL (mCpuidVersionInfoEdx.Bits.MTRR, 1);
 | |
| 
 | |
|   for (Index = 0; Index < ARRAY_SIZE (mFixedMtrrsValue); Index++) {
 | |
|     if (MsrIndex == mFixedMtrrsIndex[Index]) {
 | |
|       UT_ASSERT_EQUAL (mMtrrCapMsr.Bits.FIX, 1);
 | |
|       return mFixedMtrrsValue[Index];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if ((MsrIndex >= MSR_IA32_MTRR_PHYSBASE0) &&
 | |
|       (MsrIndex <= MSR_IA32_MTRR_PHYSMASK0 + (MTRR_NUMBER_OF_VARIABLE_MTRR << 1)))
 | |
|   {
 | |
|     UT_ASSERT_TRUE (((MsrIndex - MSR_IA32_MTRR_PHYSBASE0) >> 1) < mMtrrCapMsr.Bits.VCNT);
 | |
|     if (MsrIndex % 2 == 0) {
 | |
|       Index = (MsrIndex - MSR_IA32_MTRR_PHYSBASE0) >> 1;
 | |
|       return mVariableMtrrsPhysBase[Index].Uint64;
 | |
|     } else {
 | |
|       Index = (MsrIndex - MSR_IA32_MTRR_PHYSMASK0) >> 1;
 | |
|       return mVariableMtrrsPhysMask[Index].Uint64;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (MsrIndex == MSR_IA32_MTRR_DEF_TYPE) {
 | |
|     return mDefTypeMsr.Uint64;
 | |
|   }
 | |
| 
 | |
|   if (MsrIndex == MSR_IA32_MTRRCAP) {
 | |
|     return mMtrrCapMsr.Uint64;
 | |
|   }
 | |
| 
 | |
|   if (MsrIndex == MSR_IA32_TME_ACTIVATE) {
 | |
|     return mTmeActivateMsr.Uint64;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Should never fall through to here
 | |
|   //
 | |
|   ASSERT (FALSE);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Writes a 64-bit value to a Machine Specific Register(MSR), and returns the
 | |
|   value.
 | |
| 
 | |
|   Writes the 64-bit value specified by Value to the MSR specified by Index. The
 | |
|   64-bit value written to the MSR is returned. No parameter checking is
 | |
|   performed on Index or Value, and some of these may cause CPU exceptions. The
 | |
|   caller must either guarantee that Index and Value are valid, or the caller
 | |
|   must establish proper exception handlers. This function is only available on
 | |
|   IA-32 and x64.
 | |
| 
 | |
|   @param  MsrIndex The 32-bit MSR index to write.
 | |
|   @param  Value The 64-bit value to write to the MSR.
 | |
| 
 | |
|   @return Value
 | |
| 
 | |
| **/
 | |
| UINT64
 | |
| EFIAPI
 | |
| UnitTestMtrrLibAsmWriteMsr64 (
 | |
|   IN      UINT32  MsrIndex,
 | |
|   IN      UINT64  Value
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
| 
 | |
|   UT_ASSERT_EQUAL (mCpuidVersionInfoEdx.Bits.MTRR, 1);
 | |
| 
 | |
|   for (Index = 0; Index < ARRAY_SIZE (mFixedMtrrsValue); Index++) {
 | |
|     if (MsrIndex == mFixedMtrrsIndex[Index]) {
 | |
|       UT_ASSERT_EQUAL (mMtrrCapMsr.Bits.FIX, 1);
 | |
|       mFixedMtrrsValue[Index] = Value;
 | |
|       return Value;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if ((MsrIndex >= MSR_IA32_MTRR_PHYSBASE0) &&
 | |
|       (MsrIndex <= MSR_IA32_MTRR_PHYSMASK0 + (MTRR_NUMBER_OF_VARIABLE_MTRR << 1)))
 | |
|   {
 | |
|     UT_ASSERT_TRUE (((MsrIndex - MSR_IA32_MTRR_PHYSBASE0) >> 1) < mMtrrCapMsr.Bits.VCNT);
 | |
|     if (MsrIndex % 2 == 0) {
 | |
|       Index                                = (MsrIndex - MSR_IA32_MTRR_PHYSBASE0) >> 1;
 | |
|       mVariableMtrrsPhysBase[Index].Uint64 = Value;
 | |
|       return Value;
 | |
|     } else {
 | |
|       Index                                = (MsrIndex - MSR_IA32_MTRR_PHYSMASK0) >> 1;
 | |
|       mVariableMtrrsPhysMask[Index].Uint64 = Value;
 | |
|       return Value;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (MsrIndex == MSR_IA32_MTRR_DEF_TYPE) {
 | |
|     if (((MSR_IA32_MTRR_DEF_TYPE_REGISTER *)&Value)->Bits.FE == 1) {
 | |
|       UT_ASSERT_EQUAL (mMtrrCapMsr.Bits.FIX, 1);
 | |
|     }
 | |
| 
 | |
|     mDefTypeMsr.Uint64 = Value;
 | |
|     return Value;
 | |
|   }
 | |
| 
 | |
|   if (MsrIndex == MSR_IA32_MTRRCAP) {
 | |
|     mMtrrCapMsr.Uint64 = Value;
 | |
|     return Value;
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Should never fall through to here
 | |
|   //
 | |
|   ASSERT (FALSE);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Initialize the MTRR registers.
 | |
| 
 | |
|   @param SystemParameter System parameter that controls the MTRR registers initialization.
 | |
| **/
 | |
| UNIT_TEST_STATUS
 | |
| EFIAPI
 | |
| InitializeMtrrRegs (
 | |
|   IN MTRR_LIB_SYSTEM_PARAMETER  *SystemParameter
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
| 
 | |
|   SetMem (mFixedMtrrsValue, sizeof (mFixedMtrrsValue), SystemParameter->DefaultCacheType);
 | |
| 
 | |
|   for (Index = 0; Index < ARRAY_SIZE (mVariableMtrrsPhysBase); Index++) {
 | |
|     mVariableMtrrsPhysBase[Index].Uint64 = 0;
 | |
|     mVariableMtrrsPhysMask[Index].Uint64 = 0;
 | |
|   }
 | |
| 
 | |
|   mDefTypeMsr.Bits.E         = 1;
 | |
|   mDefTypeMsr.Bits.FE        = 0;
 | |
|   mDefTypeMsr.Bits.Type      = SystemParameter->DefaultCacheType;
 | |
|   mDefTypeMsr.Bits.Reserved1 = 0;
 | |
|   mDefTypeMsr.Bits.Reserved2 = 0;
 | |
|   mDefTypeMsr.Bits.Reserved3 = 0;
 | |
| 
 | |
|   mMtrrCapMsr.Bits.SMRR      = 0;
 | |
|   mMtrrCapMsr.Bits.WC        = 0;
 | |
|   mMtrrCapMsr.Bits.VCNT      = SystemParameter->VariableMtrrCount;
 | |
|   mMtrrCapMsr.Bits.FIX       = SystemParameter->FixedMtrrSupported;
 | |
|   mMtrrCapMsr.Bits.Reserved1 = 0;
 | |
|   mMtrrCapMsr.Bits.Reserved2 = 0;
 | |
|   mMtrrCapMsr.Bits.Reserved3 = 0;
 | |
| 
 | |
|   mCpuidVersionInfoEdx.Bits.MTRR                      = SystemParameter->MtrrSupported;
 | |
|   mCpuidVirPhyAddressSizeEax.Bits.PhysicalAddressBits = SystemParameter->PhysicalAddressBits;
 | |
| 
 | |
|   //
 | |
|   // Hook BaseLib functions used by MtrrLib that require some emulation.
 | |
|   //
 | |
|   gUnitTestHostBaseLib.X86->AsmCpuid   = UnitTestMtrrLibAsmCpuid;
 | |
|   gUnitTestHostBaseLib.X86->AsmCpuidEx = UnitTestMtrrLibAsmCpuidEx;
 | |
| 
 | |
|   gUnitTestHostBaseLib.X86->AsmReadMsr64  = UnitTestMtrrLibAsmReadMsr64;
 | |
|   gUnitTestHostBaseLib.X86->AsmWriteMsr64 = UnitTestMtrrLibAsmWriteMsr64;
 | |
| 
 | |
|   if (SystemParameter->MkTmeKeyidBits != 0) {
 | |
|     mCpuidExtendedFeatureFlagsEcx.Bits.TME_EN = 1;
 | |
|     mTmeActivateMsr.Bits.TmeEnable            = 1;
 | |
|     mTmeActivateMsr.Bits.MkTmeKeyidBits       = SystemParameter->MkTmeKeyidBits;
 | |
|   } else {
 | |
|     mCpuidExtendedFeatureFlagsEcx.Bits.TME_EN = 0;
 | |
|     mTmeActivateMsr.Bits.TmeEnable            = 0;
 | |
|     mTmeActivateMsr.Bits.MkTmeKeyidBits       = 0;
 | |
|   }
 | |
| 
 | |
|   return UNIT_TEST_PASSED;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Initialize the MTRR registers.
 | |
| 
 | |
|   @param Context System parameter that controls the MTRR registers initialization.
 | |
| **/
 | |
| UNIT_TEST_STATUS
 | |
| EFIAPI
 | |
| InitializeSystem (
 | |
|   IN UNIT_TEST_CONTEXT  Context
 | |
|   )
 | |
| {
 | |
|   return InitializeMtrrRegs ((MTRR_LIB_SYSTEM_PARAMETER *)Context);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Collect the test result.
 | |
| 
 | |
|   @param DefaultType          Default memory type.
 | |
|   @param PhysicalAddressBits  Physical address bits.
 | |
|   @param VariableMtrrCount    Count of variable MTRRs.
 | |
|   @param Mtrrs                MTRR settings to collect from.
 | |
|   @param Ranges               Return the memory ranges.
 | |
|   @param RangeCount           Return the count of memory ranges.
 | |
|   @param MtrrCount            Return the count of variable MTRRs being used.
 | |
| **/
 | |
| VOID
 | |
| CollectTestResult (
 | |
|   IN     MTRR_MEMORY_CACHE_TYPE  DefaultType,
 | |
|   IN     UINT32                  PhysicalAddressBits,
 | |
|   IN     UINT32                  VariableMtrrCount,
 | |
|   IN     MTRR_SETTINGS           *Mtrrs,
 | |
|   OUT    MTRR_MEMORY_RANGE       *Ranges,
 | |
|   IN OUT UINTN                   *RangeCount,
 | |
|   OUT    UINT32                  *MtrrCount
 | |
|   )
 | |
| {
 | |
|   UINTN              Index;
 | |
|   UINT64             MtrrValidBitsMask;
 | |
|   UINT64             MtrrValidAddressMask;
 | |
|   MTRR_MEMORY_RANGE  RawMemoryRanges[ARRAY_SIZE (Mtrrs->Variables.Mtrr)];
 | |
| 
 | |
|   ASSERT (Mtrrs != NULL);
 | |
|   ASSERT (VariableMtrrCount <= ARRAY_SIZE (Mtrrs->Variables.Mtrr));
 | |
| 
 | |
|   MtrrValidBitsMask    = (1ull << PhysicalAddressBits) - 1;
 | |
|   MtrrValidAddressMask = MtrrValidBitsMask & ~0xFFFull;
 | |
| 
 | |
|   *MtrrCount = 0;
 | |
|   for (Index = 0; Index < VariableMtrrCount; Index++) {
 | |
|     if (((MSR_IA32_MTRR_PHYSMASK_REGISTER *)&Mtrrs->Variables.Mtrr[Index].Mask)->Bits.V == 1) {
 | |
|       RawMemoryRanges[*MtrrCount].BaseAddress = Mtrrs->Variables.Mtrr[Index].Base & MtrrValidAddressMask;
 | |
|       RawMemoryRanges[*MtrrCount].Type        =
 | |
|         ((MSR_IA32_MTRR_PHYSBASE_REGISTER *)&Mtrrs->Variables.Mtrr[Index].Base)->Bits.Type;
 | |
|       RawMemoryRanges[*MtrrCount].Length =
 | |
|         ((~(Mtrrs->Variables.Mtrr[Index].Mask & MtrrValidAddressMask)) & MtrrValidBitsMask) + 1;
 | |
|       (*MtrrCount)++;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   GetEffectiveMemoryRanges (DefaultType, PhysicalAddressBits, RawMemoryRanges, *MtrrCount, Ranges, RangeCount);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return a 32bit random number.
 | |
| 
 | |
|   @param Start  Start of the random number range.
 | |
|   @param Limit  Limit of the random number range.
 | |
|   @return 32bit random number
 | |
| **/
 | |
| UINT32
 | |
| Random32 (
 | |
|   UINT32  Start,
 | |
|   UINT32  Limit
 | |
|   )
 | |
| {
 | |
|   return (UINT32)(((double)Rand () / RAND_MAX) * (Limit - Start)) + Start;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return a 64bit random number.
 | |
| 
 | |
|   @param Start  Start of the random number range.
 | |
|   @param Limit  Limit of the random number range.
 | |
|   @return 64bit random number
 | |
| **/
 | |
| UINT64
 | |
| Random64 (
 | |
|   UINT64  Start,
 | |
|   UINT64  Limit
 | |
|   )
 | |
| {
 | |
|   return (UINT64)(((double)Rand () / RAND_MAX) * (Limit - Start)) + Start;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Generate random MTRR BASE/MASK for a specified type.
 | |
| 
 | |
|   @param PhysicalAddressBits Physical address bits.
 | |
|   @param CacheType           Cache type.
 | |
|   @param MtrrPair            Return the random MTRR.
 | |
|   @param MtrrMemoryRange     Return the random memory range.
 | |
| **/
 | |
| VOID
 | |
| GenerateRandomMtrrPair (
 | |
|   IN  UINT32                  PhysicalAddressBits,
 | |
|   IN  MTRR_MEMORY_CACHE_TYPE  CacheType,
 | |
|   OUT MTRR_VARIABLE_SETTING   *MtrrPair        OPTIONAL,
 | |
|   OUT MTRR_MEMORY_RANGE       *MtrrMemoryRange OPTIONAL
 | |
|   )
 | |
| {
 | |
|   MSR_IA32_MTRR_PHYSBASE_REGISTER  PhysBase;
 | |
|   MSR_IA32_MTRR_PHYSMASK_REGISTER  PhysMask;
 | |
|   UINT32                           SizeShift;
 | |
|   UINT32                           BaseShift;
 | |
|   UINT64                           RandomBoundary;
 | |
|   UINT64                           MaxPhysicalAddress;
 | |
|   UINT64                           RangeSize;
 | |
|   UINT64                           RangeBase;
 | |
|   UINT64                           PhysBasePhyMaskValidBitsMask;
 | |
| 
 | |
|   MaxPhysicalAddress = 1ull << PhysicalAddressBits;
 | |
|   do {
 | |
|     SizeShift = Random32 (12, PhysicalAddressBits - 1);
 | |
|     RangeSize = 1ull << SizeShift;
 | |
| 
 | |
|     BaseShift      = Random32 (SizeShift, PhysicalAddressBits - 1);
 | |
|     RandomBoundary = Random64 (0, 1ull << (PhysicalAddressBits - BaseShift));
 | |
|     RangeBase      = RandomBoundary << BaseShift;
 | |
|   } while (RangeBase < SIZE_1MB || RangeBase > MaxPhysicalAddress - 1);
 | |
| 
 | |
|   PhysBasePhyMaskValidBitsMask = (MaxPhysicalAddress - 1) & 0xfffffffffffff000ULL;
 | |
| 
 | |
|   PhysBase.Uint64    = 0;
 | |
|   PhysBase.Bits.Type = CacheType;
 | |
|   PhysBase.Uint64   |= RangeBase & PhysBasePhyMaskValidBitsMask;
 | |
|   PhysMask.Uint64    = 0;
 | |
|   PhysMask.Bits.V    = 1;
 | |
|   PhysMask.Uint64   |= ((~RangeSize) + 1) & PhysBasePhyMaskValidBitsMask;
 | |
| 
 | |
|   if (MtrrPair != NULL) {
 | |
|     MtrrPair->Base = PhysBase.Uint64;
 | |
|     MtrrPair->Mask = PhysMask.Uint64;
 | |
|   }
 | |
| 
 | |
|   if (MtrrMemoryRange != NULL) {
 | |
|     MtrrMemoryRange->BaseAddress = RangeBase;
 | |
|     MtrrMemoryRange->Length      = RangeSize;
 | |
|     MtrrMemoryRange->Type        = CacheType;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Check whether the Range overlaps with any one in Ranges.
 | |
| 
 | |
|   @param Range  The memory range to check.
 | |
|   @param Ranges The memory ranges.
 | |
|   @param Count  Count of memory ranges.
 | |
| 
 | |
|   @return TRUE when overlap exists.
 | |
| **/
 | |
| BOOLEAN
 | |
| RangesOverlap (
 | |
|   IN MTRR_MEMORY_RANGE  *Range,
 | |
|   IN MTRR_MEMORY_RANGE  *Ranges,
 | |
|   IN UINTN              Count
 | |
|   )
 | |
| {
 | |
|   while (Count-- != 0) {
 | |
|     //
 | |
|     // Two ranges overlap when:
 | |
|     // 1. range#2.base is in the middle of range#1
 | |
|     // 2. range#1.base is in the middle of range#2
 | |
|     //
 | |
|     if (  ((Range->BaseAddress <= Ranges[Count].BaseAddress) && (Ranges[Count].BaseAddress < Range->BaseAddress + Range->Length))
 | |
|        || ((Ranges[Count].BaseAddress <= Range->BaseAddress) && (Range->BaseAddress < Ranges[Count].BaseAddress + Ranges[Count].Length)))
 | |
|     {
 | |
|       return TRUE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Generate random MTRRs.
 | |
| 
 | |
|   @param PhysicalAddressBits  Physical address bits.
 | |
|   @param RawMemoryRanges      Return the randomly generated MTRRs.
 | |
|   @param UcCount              Count of Uncacheable MTRRs.
 | |
|   @param WtCount              Count of Write Through MTRRs.
 | |
|   @param WbCount              Count of Write Back MTRRs.
 | |
|   @param WpCount              Count of Write Protected MTRRs.
 | |
|   @param WcCount              Count of Write Combine MTRRs.
 | |
| **/
 | |
| VOID
 | |
| GenerateValidAndConfigurableMtrrPairs (
 | |
|   IN     UINT32             PhysicalAddressBits,
 | |
|   IN OUT MTRR_MEMORY_RANGE  *RawMemoryRanges,
 | |
|   IN     UINT32             UcCount,
 | |
|   IN     UINT32             WtCount,
 | |
|   IN     UINT32             WbCount,
 | |
|   IN     UINT32             WpCount,
 | |
|   IN     UINT32             WcCount
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
| 
 | |
|   //
 | |
|   // 1. Generate UC, WT, WB in order.
 | |
|   //
 | |
|   for (Index = 0; Index < UcCount; Index++) {
 | |
|     GenerateRandomMtrrPair (PhysicalAddressBits, CacheUncacheable, NULL, &RawMemoryRanges[Index]);
 | |
|   }
 | |
| 
 | |
|   for (Index = UcCount; Index < UcCount + WtCount; Index++) {
 | |
|     GenerateRandomMtrrPair (PhysicalAddressBits, CacheWriteThrough, NULL, &RawMemoryRanges[Index]);
 | |
|   }
 | |
| 
 | |
|   for (Index = UcCount + WtCount; Index < UcCount + WtCount + WbCount; Index++) {
 | |
|     GenerateRandomMtrrPair (PhysicalAddressBits, CacheWriteBack, NULL, &RawMemoryRanges[Index]);
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // 2. Generate WP MTRR and DO NOT overlap with WT, WB.
 | |
|   //
 | |
|   for (Index = UcCount + WtCount + WbCount; Index < UcCount + WtCount + WbCount + WpCount; Index++) {
 | |
|     GenerateRandomMtrrPair (PhysicalAddressBits, CacheWriteProtected, NULL, &RawMemoryRanges[Index]);
 | |
|     while (RangesOverlap (&RawMemoryRanges[Index], &RawMemoryRanges[UcCount], WtCount + WbCount)) {
 | |
|       GenerateRandomMtrrPair (PhysicalAddressBits, CacheWriteProtected, NULL, &RawMemoryRanges[Index]);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // 3. Generate WC MTRR and DO NOT overlap with WT, WB, WP.
 | |
|   //
 | |
|   for (Index = UcCount + WtCount + WbCount + WpCount; Index < UcCount + WtCount + WbCount + WpCount + WcCount; Index++) {
 | |
|     GenerateRandomMtrrPair (PhysicalAddressBits, CacheWriteCombining, NULL, &RawMemoryRanges[Index]);
 | |
|     while (RangesOverlap (&RawMemoryRanges[Index], &RawMemoryRanges[UcCount], WtCount + WbCount + WpCount)) {
 | |
|       GenerateRandomMtrrPair (PhysicalAddressBits, CacheWriteCombining, NULL, &RawMemoryRanges[Index]);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return a random memory cache type.
 | |
| **/
 | |
| MTRR_MEMORY_CACHE_TYPE
 | |
| GenerateRandomCacheType (
 | |
|   VOID
 | |
|   )
 | |
| {
 | |
|   return mMemoryCacheTypes[Random32 (0, ARRAY_SIZE (mMemoryCacheTypes) - 1)];
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Compare function used by qsort().
 | |
| **/
 | |
| 
 | |
| /**
 | |
|   Compare function used by qsort().
 | |
| 
 | |
|   @param Left   Left operand to compare.
 | |
|   @param Right  Right operand to compare.
 | |
| 
 | |
|   @retval 0  Left == Right
 | |
|   @retval -1 Left < Right
 | |
|   @retval 1  Left > Right
 | |
| **/
 | |
| INT32
 | |
| CompareFuncUint64 (
 | |
|   CONST VOID  *Left,
 | |
|   CONST VOID  *Right
 | |
|   )
 | |
| {
 | |
|   INT64  Delta;
 | |
| 
 | |
|   Delta = (*(UINT64 *)Left - *(UINT64 *)Right);
 | |
|   if (Delta > 0) {
 | |
|     return 1;
 | |
|   } else if (Delta == 0) {
 | |
|     return 0;
 | |
|   } else {
 | |
|     return -1;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Determin the memory cache type for the Range.
 | |
| 
 | |
|   @param DefaultType Default cache type.
 | |
|   @param Range       The memory range to determin the cache type.
 | |
|   @param Ranges      The entire memory ranges.
 | |
|   @param RangeCount  Count of the entire memory ranges.
 | |
| **/
 | |
| VOID
 | |
| DetermineMemoryCacheType (
 | |
|   IN     MTRR_MEMORY_CACHE_TYPE  DefaultType,
 | |
|   IN OUT MTRR_MEMORY_RANGE       *Range,
 | |
|   IN     MTRR_MEMORY_RANGE       *Ranges,
 | |
|   IN     UINT32                  RangeCount
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
| 
 | |
|   Range->Type = CacheInvalid;
 | |
|   for (Index = 0; Index < RangeCount; Index++) {
 | |
|     if (RangesOverlap (Range, &Ranges[Index], 1)) {
 | |
|       if (Ranges[Index].Type < Range->Type) {
 | |
|         Range->Type = Ranges[Index].Type;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (Range->Type == CacheInvalid) {
 | |
|     Range->Type = DefaultType;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get the index of the element that does NOT equals to Array[Index].
 | |
| 
 | |
|   @param Index   Current element.
 | |
|   @param Array   Array to scan.
 | |
|   @param Count   Count of the array.
 | |
| 
 | |
|   @return Next element that doesn't equal to current one.
 | |
| **/
 | |
| UINT32
 | |
| GetNextDifferentElementInSortedArray (
 | |
|   IN UINT32  Index,
 | |
|   IN UINT64  *Array,
 | |
|   IN UINT32  Count
 | |
|   )
 | |
| {
 | |
|   UINT64  CurrentElement;
 | |
| 
 | |
|   CurrentElement = Array[Index];
 | |
|   while (CurrentElement == Array[Index] && Index < Count) {
 | |
|     Index++;
 | |
|   }
 | |
| 
 | |
|   return Index;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Remove the duplicates from the array.
 | |
| 
 | |
|   @param Array  The array to operate on.
 | |
|   @param Count  Count of the array.
 | |
| **/
 | |
| VOID
 | |
| RemoveDuplicatesInSortedArray (
 | |
|   IN OUT UINT64  *Array,
 | |
|   IN OUT UINT32  *Count
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
|   UINT32  NewCount;
 | |
| 
 | |
|   Index    = 0;
 | |
|   NewCount = 0;
 | |
|   while (Index < *Count) {
 | |
|     Array[NewCount] = Array[Index];
 | |
|     NewCount++;
 | |
|     Index = GetNextDifferentElementInSortedArray (Index, Array, *Count);
 | |
|   }
 | |
| 
 | |
|   *Count = NewCount;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return TRUE when Address is in the Range.
 | |
| 
 | |
|   @param Address The address to check.
 | |
|   @param Range   The range to check.
 | |
|   @return TRUE when Address is in the Range.
 | |
| **/
 | |
| BOOLEAN
 | |
| AddressInRange (
 | |
|   IN UINT64             Address,
 | |
|   IN MTRR_MEMORY_RANGE  Range
 | |
|   )
 | |
| {
 | |
|   return (Address >= Range.BaseAddress) && (Address <= Range.BaseAddress + Range.Length - 1);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Get the overlap bit flag.
 | |
| 
 | |
|   @param RawMemoryRanges     Raw memory ranges.
 | |
|   @param RawMemoryRangeCount Count of raw memory ranges.
 | |
|   @param Address             The address to check.
 | |
| **/
 | |
| UINT64
 | |
| GetOverlapBitFlag (
 | |
|   IN MTRR_MEMORY_RANGE  *RawMemoryRanges,
 | |
|   IN UINT32             RawMemoryRangeCount,
 | |
|   IN UINT64             Address
 | |
|   )
 | |
| {
 | |
|   UINT64  OverlapBitFlag;
 | |
|   UINT32  Index;
 | |
| 
 | |
|   OverlapBitFlag = 0;
 | |
|   for (Index = 0; Index < RawMemoryRangeCount; Index++) {
 | |
|     if (AddressInRange (Address, RawMemoryRanges[Index])) {
 | |
|       OverlapBitFlag |= (1ull << Index);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return OverlapBitFlag;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return the relationship between flags.
 | |
| 
 | |
|   @param Flag1 Flag 1
 | |
|   @param Flag2 Flag 2
 | |
| 
 | |
|   @retval 0   Flag1 == Flag2
 | |
|   @retval 1   Flag1 is a subset of Flag2
 | |
|   @retval 2   Flag2 is a subset of Flag1
 | |
|   @retval 3   No subset relations between Flag1 and Flag2.
 | |
| **/
 | |
| UINT32
 | |
| CheckOverlapBitFlagsRelation (
 | |
|   IN UINT64  Flag1,
 | |
|   IN UINT64  Flag2
 | |
|   )
 | |
| {
 | |
|   if (Flag1 == Flag2) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if ((Flag1 | Flag2) == Flag2) {
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   if ((Flag1 | Flag2) == Flag1) {
 | |
|     return 2;
 | |
|   }
 | |
| 
 | |
|   return 3;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Return TRUE when the Endpoint is in any of the Ranges.
 | |
| 
 | |
|   @param Endpoint    The endpoint to check.
 | |
|   @param Ranges      The memory ranges.
 | |
|   @param RangeCount  Count of memory ranges.
 | |
| 
 | |
|   @retval TRUE  Endpoint is in one of the range.
 | |
|   @retval FALSE Endpoint is not in any of the ranges.
 | |
| **/
 | |
| BOOLEAN
 | |
| IsEndpointInRanges (
 | |
|   IN UINT64             Endpoint,
 | |
|   IN MTRR_MEMORY_RANGE  *Ranges,
 | |
|   IN UINTN              RangeCount
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
| 
 | |
|   for (Index = 0; Index < RangeCount; Index++) {
 | |
|     if (AddressInRange (Endpoint, Ranges[Index])) {
 | |
|       return TRUE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Compact adjacent ranges of the same type.
 | |
| 
 | |
|   @param DefaultType                    Default memory type.
 | |
|   @param PhysicalAddressBits            Physical address bits.
 | |
|   @param EffectiveMtrrMemoryRanges      Memory ranges to compact.
 | |
|   @param EffectiveMtrrMemoryRangesCount Return the new count of memory ranges.
 | |
| **/
 | |
| VOID
 | |
| CompactAndExtendEffectiveMtrrMemoryRanges (
 | |
|   IN     MTRR_MEMORY_CACHE_TYPE  DefaultType,
 | |
|   IN     UINT32                  PhysicalAddressBits,
 | |
|   IN OUT MTRR_MEMORY_RANGE       **EffectiveMtrrMemoryRanges,
 | |
|   IN OUT UINTN                   *EffectiveMtrrMemoryRangesCount
 | |
|   )
 | |
| {
 | |
|   UINT64                  MaxAddress;
 | |
|   UINTN                   NewRangesCountAtMost;
 | |
|   MTRR_MEMORY_RANGE       *NewRanges;
 | |
|   UINTN                   NewRangesCountActual;
 | |
|   MTRR_MEMORY_RANGE       *CurrentRangeInNewRanges;
 | |
|   MTRR_MEMORY_CACHE_TYPE  CurrentRangeTypeInOldRanges;
 | |
| 
 | |
|   MTRR_MEMORY_RANGE  *OldRanges;
 | |
|   MTRR_MEMORY_RANGE  OldLastRange;
 | |
|   UINTN              OldRangesIndex;
 | |
| 
 | |
|   NewRangesCountActual = 0;
 | |
|   NewRangesCountAtMost = *EffectiveMtrrMemoryRangesCount + 2;   // At most with 2 more range entries.
 | |
|   NewRanges            = (MTRR_MEMORY_RANGE *)calloc (NewRangesCountAtMost, sizeof (MTRR_MEMORY_RANGE));
 | |
|   OldRanges            = *EffectiveMtrrMemoryRanges;
 | |
|   if (OldRanges[0].BaseAddress > 0) {
 | |
|     NewRanges[NewRangesCountActual].BaseAddress = 0;
 | |
|     NewRanges[NewRangesCountActual].Length      = OldRanges[0].BaseAddress;
 | |
|     NewRanges[NewRangesCountActual].Type        = DefaultType;
 | |
|     NewRangesCountActual++;
 | |
|   }
 | |
| 
 | |
|   OldRangesIndex = 0;
 | |
|   while (OldRangesIndex < *EffectiveMtrrMemoryRangesCount) {
 | |
|     CurrentRangeTypeInOldRanges = OldRanges[OldRangesIndex].Type;
 | |
|     CurrentRangeInNewRanges     = NULL;
 | |
|     if (NewRangesCountActual > 0) {
 | |
|       // We need to check CurrentNewRange first before generate a new NewRange.
 | |
|       CurrentRangeInNewRanges = &NewRanges[NewRangesCountActual - 1];
 | |
|     }
 | |
| 
 | |
|     if ((CurrentRangeInNewRanges != NULL) && (CurrentRangeInNewRanges->Type == CurrentRangeTypeInOldRanges)) {
 | |
|       CurrentRangeInNewRanges->Length += OldRanges[OldRangesIndex].Length;
 | |
|     } else {
 | |
|       NewRanges[NewRangesCountActual].BaseAddress = OldRanges[OldRangesIndex].BaseAddress;
 | |
|       NewRanges[NewRangesCountActual].Length     += OldRanges[OldRangesIndex].Length;
 | |
|       NewRanges[NewRangesCountActual].Type        = CurrentRangeTypeInOldRanges;
 | |
|       while (OldRangesIndex + 1 < *EffectiveMtrrMemoryRangesCount && OldRanges[OldRangesIndex + 1].Type == CurrentRangeTypeInOldRanges) {
 | |
|         OldRangesIndex++;
 | |
|         NewRanges[NewRangesCountActual].Length += OldRanges[OldRangesIndex].Length;
 | |
|       }
 | |
| 
 | |
|       NewRangesCountActual++;
 | |
|     }
 | |
| 
 | |
|     OldRangesIndex++;
 | |
|   }
 | |
| 
 | |
|   MaxAddress              = (1ull << PhysicalAddressBits) - 1;
 | |
|   OldLastRange            = OldRanges[(*EffectiveMtrrMemoryRangesCount) - 1];
 | |
|   CurrentRangeInNewRanges = &NewRanges[NewRangesCountActual - 1];
 | |
|   if (OldLastRange.BaseAddress + OldLastRange.Length - 1 < MaxAddress) {
 | |
|     if (CurrentRangeInNewRanges->Type == DefaultType) {
 | |
|       CurrentRangeInNewRanges->Length = MaxAddress - CurrentRangeInNewRanges->BaseAddress + 1;
 | |
|     } else {
 | |
|       NewRanges[NewRangesCountActual].BaseAddress = OldLastRange.BaseAddress + OldLastRange.Length;
 | |
|       NewRanges[NewRangesCountActual].Length      = MaxAddress - NewRanges[NewRangesCountActual].BaseAddress + 1;
 | |
|       NewRanges[NewRangesCountActual].Type        = DefaultType;
 | |
|       NewRangesCountActual++;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   free (*EffectiveMtrrMemoryRanges);
 | |
|   *EffectiveMtrrMemoryRanges      = NewRanges;
 | |
|   *EffectiveMtrrMemoryRangesCount = NewRangesCountActual;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Collect all the endpoints in the raw memory ranges.
 | |
| 
 | |
|   @param Endpoints           Return the collected endpoints.
 | |
|   @param EndPointCount       Return the count of endpoints.
 | |
|   @param RawMemoryRanges     Raw memory ranges.
 | |
|   @param RawMemoryRangeCount Count of raw memory ranges.
 | |
| **/
 | |
| VOID
 | |
| CollectEndpoints (
 | |
|   IN OUT UINT64         *Endpoints,
 | |
|   IN OUT UINT32         *EndPointCount,
 | |
|   IN MTRR_MEMORY_RANGE  *RawMemoryRanges,
 | |
|   IN UINT32             RawMemoryRangeCount
 | |
|   )
 | |
| {
 | |
|   UINT32  Index;
 | |
|   UINT32  RawRangeIndex;
 | |
| 
 | |
|   ASSERT ((RawMemoryRangeCount << 1) == *EndPointCount);
 | |
| 
 | |
|   for (Index = 0; Index < *EndPointCount; Index += 2) {
 | |
|     RawRangeIndex        = Index >> 1;
 | |
|     Endpoints[Index]     = RawMemoryRanges[RawRangeIndex].BaseAddress;
 | |
|     Endpoints[Index + 1] = RawMemoryRanges[RawRangeIndex].BaseAddress + RawMemoryRanges[RawRangeIndex].Length - 1;
 | |
|   }
 | |
| 
 | |
|   qsort (Endpoints, *EndPointCount, sizeof (UINT64), CompareFuncUint64);
 | |
|   RemoveDuplicatesInSortedArray (Endpoints, EndPointCount);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Convert the MTRR BASE/MASK array to memory ranges.
 | |
| 
 | |
|   @param DefaultType          Default memory type.
 | |
|   @param PhysicalAddressBits  Physical address bits.
 | |
|   @param RawMemoryRanges      Raw memory ranges.
 | |
|   @param RawMemoryRangeCount  Count of raw memory ranges.
 | |
|   @param MemoryRanges         Memory ranges.
 | |
|   @param MemoryRangeCount     Count of memory ranges.
 | |
| **/
 | |
| VOID
 | |
| GetEffectiveMemoryRanges (
 | |
|   IN MTRR_MEMORY_CACHE_TYPE  DefaultType,
 | |
|   IN UINT32                  PhysicalAddressBits,
 | |
|   IN MTRR_MEMORY_RANGE       *RawMemoryRanges,
 | |
|   IN UINT32                  RawMemoryRangeCount,
 | |
|   OUT MTRR_MEMORY_RANGE      *MemoryRanges,
 | |
|   OUT UINTN                  *MemoryRangeCount
 | |
|   )
 | |
| {
 | |
|   UINTN              Index;
 | |
|   UINT32             AllEndPointsCount;
 | |
|   UINT64             *AllEndPointsInclusive;
 | |
|   UINT32             AllRangePiecesCountMax;
 | |
|   MTRR_MEMORY_RANGE  *AllRangePieces;
 | |
|   UINTN              AllRangePiecesCountActual;
 | |
|   UINT64             OverlapBitFlag1;
 | |
|   UINT64             OverlapBitFlag2;
 | |
|   INT32              OverlapFlagRelation;
 | |
| 
 | |
|   if (RawMemoryRangeCount == 0) {
 | |
|     MemoryRanges[0].BaseAddress = 0;
 | |
|     MemoryRanges[0].Length      = (1ull << PhysicalAddressBits);
 | |
|     MemoryRanges[0].Type        = DefaultType;
 | |
|     *MemoryRangeCount           = 1;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   AllEndPointsCount      = RawMemoryRangeCount << 1;
 | |
|   AllEndPointsInclusive  = calloc (AllEndPointsCount, sizeof (UINT64));
 | |
|   AllRangePiecesCountMax = RawMemoryRangeCount * 3 + 1;
 | |
|   AllRangePieces         = calloc (AllRangePiecesCountMax, sizeof (MTRR_MEMORY_RANGE));
 | |
|   CollectEndpoints (AllEndPointsInclusive, &AllEndPointsCount, RawMemoryRanges, RawMemoryRangeCount);
 | |
| 
 | |
|   for (Index = 0, AllRangePiecesCountActual = 0; Index < AllEndPointsCount - 1; Index++) {
 | |
|     OverlapBitFlag1     = GetOverlapBitFlag (RawMemoryRanges, RawMemoryRangeCount, AllEndPointsInclusive[Index]);
 | |
|     OverlapBitFlag2     = GetOverlapBitFlag (RawMemoryRanges, RawMemoryRangeCount, AllEndPointsInclusive[Index + 1]);
 | |
|     OverlapFlagRelation = CheckOverlapBitFlagsRelation (OverlapBitFlag1, OverlapBitFlag2);
 | |
|     switch (OverlapFlagRelation) {
 | |
|       case 0:   // [1, 2]
 | |
|         AllRangePieces[AllRangePiecesCountActual].BaseAddress = AllEndPointsInclusive[Index];
 | |
|         AllRangePieces[AllRangePiecesCountActual].Length      = AllEndPointsInclusive[Index + 1] - AllEndPointsInclusive[Index] + 1;
 | |
|         AllRangePiecesCountActual++;
 | |
|         break;
 | |
|       case 1:   // [1, 2)
 | |
|         AllRangePieces[AllRangePiecesCountActual].BaseAddress = AllEndPointsInclusive[Index];
 | |
|         AllRangePieces[AllRangePiecesCountActual].Length      = (AllEndPointsInclusive[Index + 1] - 1) - AllEndPointsInclusive[Index] + 1;
 | |
|         AllRangePiecesCountActual++;
 | |
|         break;
 | |
|       case 2:   // (1, 2]
 | |
|         AllRangePieces[AllRangePiecesCountActual].BaseAddress = AllEndPointsInclusive[Index] + 1;
 | |
|         AllRangePieces[AllRangePiecesCountActual].Length      = AllEndPointsInclusive[Index + 1] - (AllEndPointsInclusive[Index] + 1) + 1;
 | |
|         AllRangePiecesCountActual++;
 | |
| 
 | |
|         if (!IsEndpointInRanges (AllEndPointsInclusive[Index], AllRangePieces, AllRangePiecesCountActual)) {
 | |
|           AllRangePieces[AllRangePiecesCountActual].BaseAddress = AllEndPointsInclusive[Index];
 | |
|           AllRangePieces[AllRangePiecesCountActual].Length      = 1;
 | |
|           AllRangePiecesCountActual++;
 | |
|         }
 | |
| 
 | |
|         break;
 | |
|       case 3:   // (1, 2)
 | |
|         AllRangePieces[AllRangePiecesCountActual].BaseAddress = AllEndPointsInclusive[Index] + 1;
 | |
|         AllRangePieces[AllRangePiecesCountActual].Length      = (AllEndPointsInclusive[Index + 1] - 1) - (AllEndPointsInclusive[Index] + 1) + 1;
 | |
|         if (AllRangePieces[AllRangePiecesCountActual].Length == 0) {
 | |
|           // Only in case 3 can exists Length=0, we should skip such "segment".
 | |
|           break;
 | |
|         }
 | |
| 
 | |
|         AllRangePiecesCountActual++;
 | |
|         if (!IsEndpointInRanges (AllEndPointsInclusive[Index], AllRangePieces, AllRangePiecesCountActual)) {
 | |
|           AllRangePieces[AllRangePiecesCountActual].BaseAddress = AllEndPointsInclusive[Index];
 | |
|           AllRangePieces[AllRangePiecesCountActual].Length      = 1;
 | |
|           AllRangePiecesCountActual++;
 | |
|         }
 | |
| 
 | |
|         break;
 | |
|       default:
 | |
|         ASSERT (FALSE);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (Index = 0; Index < AllRangePiecesCountActual; Index++) {
 | |
|     DetermineMemoryCacheType (DefaultType, &AllRangePieces[Index], RawMemoryRanges, RawMemoryRangeCount);
 | |
|   }
 | |
| 
 | |
|   CompactAndExtendEffectiveMtrrMemoryRanges (DefaultType, PhysicalAddressBits, &AllRangePieces, &AllRangePiecesCountActual);
 | |
|   ASSERT (*MemoryRangeCount >= AllRangePiecesCountActual);
 | |
|   memcpy (MemoryRanges, AllRangePieces, AllRangePiecesCountActual * sizeof (MTRR_MEMORY_RANGE));
 | |
|   *MemoryRangeCount = AllRangePiecesCountActual;
 | |
| 
 | |
|   free (AllEndPointsInclusive);
 | |
|   free (AllRangePieces);
 | |
| }
 |