summaryrefslogtreecommitdiffstats
path: root/UefiCpuPkg/Library/CpuPageTableLib/UnitTest/RandomTest.c
diff options
context:
space:
mode:
Diffstat (limited to 'UefiCpuPkg/Library/CpuPageTableLib/UnitTest/RandomTest.c')
-rw-r--r--UefiCpuPkg/Library/CpuPageTableLib/UnitTest/RandomTest.c926
1 files changed, 0 insertions, 926 deletions
diff --git a/UefiCpuPkg/Library/CpuPageTableLib/UnitTest/RandomTest.c b/UefiCpuPkg/Library/CpuPageTableLib/UnitTest/RandomTest.c
deleted file mode 100644
index 97a388ca1c..0000000000
--- a/UefiCpuPkg/Library/CpuPageTableLib/UnitTest/RandomTest.c
+++ /dev/null
@@ -1,926 +0,0 @@
-/** @file
- Random test case for Unit tests of the CpuPageTableLib instance of the CpuPageTableLib class
-
- Copyright (c) 2022, Intel Corporation. All rights reserved.<BR>
- SPDX-License-Identifier: BSD-2-Clause-Patent
-
-**/
-
-#include "CpuPageTableLibUnitTest.h"
-#include "RandomTest.h"
-
-UINTN RandomNumber = 0;
-extern IA32_PAGING_ENTRY mValidMaskNoLeaf[6];
-extern IA32_PAGING_ENTRY mValidMaskLeaf[6];
-extern IA32_PAGING_ENTRY mValidMaskLeafFlag[6];
-UINTN mRandomOption;
-IA32_MAP_ATTRIBUTE mSupportedBit;
-extern UINTN mNumberCount;
-extern UINT8 mNumbers[];
-UINTN mNumberIndex;
-UINT64 AlignedTable[] = {
- ~((UINT64)SIZE_4KB - 1),
- ~((UINT64)SIZE_2MB - 1),
- ~((UINT64)SIZE_1GB - 1)
-};
-
-/**
- Generates a pseudorandom byte stream of the specified size.
-
- Return FALSE to indicate this interface is not supported.
-
- @param[out] Output Pointer to buffer to receive random value.
- @param[in] Size Size of random bytes to generate.
-
- @retval TRUE Always return TRUE
-
-**/
-BOOLEAN
-EFIAPI
-RandomBytesUsingArray (
- OUT UINT8 *Output,
- IN UINTN Size
- )
-{
- UINTN Index;
-
- for (Index = 0; Index < Size; Index++) {
- if (mNumberIndex >= mNumberCount) {
- mNumberIndex = 0;
- }
-
- Output[Index] = mNumbers[mNumberIndex];
- mNumberIndex++;
- }
-
- return TRUE;
-}
-
-/**
- Generates a pseudorandom byte stream of the specified size.
-
- Return FALSE to indicate this interface is not supported.
-
- @param[out] Output Pointer to buffer to receive random value.
- @param[in] Size Size of random bytes to generate.
-
- @retval TRUE Pseudorandom byte stream generated successfully.
- @retval FALSE Pseudorandom number generator fails
-**/
-BOOLEAN
-EFIAPI
-LocalRandomBytes (
- OUT UINT8 *Output,
- IN UINTN Size
- )
-{
- if (mRandomOption & USE_RANDOM_ARRAY) {
- return RandomBytesUsingArray (Output, Size);
- } else {
- return RandomBytes (Output, Size);
- }
-}
-
-/**
- Return a random boolean.
-
- @return boolean
-**/
-BOOLEAN
-RandomBoolean (
- VOID
- )
-{
- BOOLEAN Value;
-
- LocalRandomBytes ((UINT8 *)&Value, sizeof (BOOLEAN));
- return Value%2;
-}
-
-/**
- Return a 32bit random number.
-
- @param Start Start of the random number range.
- @param Limit Limit of the random number range, and return value can be Limit.
- @return 32bit random number
-**/
-UINT32
-Random32 (
- UINT32 Start,
- UINT32 Limit
- )
-{
- UINT64 Value;
-
- LocalRandomBytes ((UINT8 *)&Value, sizeof (UINT64));
- return (UINT32)(Value % (Limit - Start + 1)) + Start;
-}
-
-/**
- Return a 64bit random number.
-
- @param Start Start of the random number range.
- @param Limit Limit of the random number range, and return value can be Limit.
- @return 64bit random number
-**/
-UINT64
-Random64 (
- UINT64 Start,
- UINT64 Limit
- )
-{
- UINT64 Value;
-
- LocalRandomBytes ((UINT8 *)&Value, sizeof (UINT64));
- if (Limit - Start == MAX_UINT64) {
- return (UINT64)(Value);
- }
-
- return (UINT64)(Value % (Limit - Start + 1)) + Start;
-}
-
-/**
- Check if the Page table entry is valid
-
- @param[in] PagingEntry The entry in page table to verify
- @param[in] Level the level of PagingEntry.
- @param[in] MaxLeafLevel Max leaf entry level.
- @param[in] LinearAddress The linear address verified.
-
- @retval Leaf entry.
-**/
-UNIT_TEST_STATUS
-ValidateAndRandomeModifyPageTablePageTableEntry (
- IN IA32_PAGING_ENTRY *PagingEntry,
- IN UINTN Level,
- IN UINTN MaxLeafLevel,
- IN UINT64 Address
- )
-{
- UINT64 Index;
- UINT64 TempPhysicalBase;
- IA32_PAGING_ENTRY *ChildPageEntry;
- UNIT_TEST_STATUS Status;
-
- if (PagingEntry->Pce.Present == 0) {
- return UNIT_TEST_PASSED;
- }
-
- if ((PagingEntry->Uint64 & mValidMaskLeafFlag[Level].Uint64) == mValidMaskLeafFlag[Level].Uint64) {
- //
- // It is a Leaf
- //
- if (Level > MaxLeafLevel) {
- UT_ASSERT_TRUE (Level <= MaxLeafLevel);
- }
-
- if ((PagingEntry->Uint64 & mValidMaskLeaf[Level].Uint64) != PagingEntry->Uint64) {
- UT_ASSERT_EQUAL ((PagingEntry->Uint64 & mValidMaskLeaf[Level].Uint64), PagingEntry->Uint64);
- }
-
- if ((RandomNumber < 100) && RandomBoolean ()) {
- RandomNumber++;
- if (Level == 1) {
- TempPhysicalBase = PagingEntry->Pte4K.Bits.PageTableBaseAddress;
- } else {
- TempPhysicalBase = PagingEntry->PleB.Bits.PageTableBaseAddress;
- }
-
- PagingEntry->Uint64 = (Random64 (0, MAX_UINT64) & mValidMaskLeaf[Level].Uint64) | mValidMaskLeafFlag[Level].Uint64;
- PagingEntry->Pte4K.Bits.Present = 1;
- if (Level == 1) {
- PagingEntry->Pte4K.Bits.PageTableBaseAddress = TempPhysicalBase;
- } else {
- PagingEntry->PleB.Bits.PageTableBaseAddress = TempPhysicalBase;
- }
-
- if ((PagingEntry->Uint64 & mValidMaskLeaf[Level].Uint64) != PagingEntry->Uint64) {
- UT_ASSERT_EQUAL ((PagingEntry->Uint64 & mValidMaskLeaf[Level].Uint64), PagingEntry->Uint64);
- }
- }
-
- return UNIT_TEST_PASSED;
- }
-
- //
- // Not a leaf
- //
- UT_ASSERT_NOT_EQUAL (Level, 1);
- if ((PagingEntry->Uint64 & mValidMaskNoLeaf[Level].Uint64) != PagingEntry->Uint64) {
- DEBUG ((DEBUG_ERROR, "ERROR: Level %d no Leaf entry is 0x%lx, which reserved bit is set \n", Level, PagingEntry->Uint64));
- UT_ASSERT_EQUAL ((PagingEntry->Uint64 & mValidMaskNoLeaf[Level].Uint64), PagingEntry->Uint64);
- }
-
- if ((RandomNumber < 100) && RandomBoolean ()) {
- RandomNumber++;
- TempPhysicalBase = PagingEntry->Pnle.Bits.PageTableBaseAddress;
-
- PagingEntry->Uint64 = Random64 (0, MAX_UINT64) & mValidMaskNoLeaf[Level].Uint64;
- PagingEntry->Pnle.Bits.Present = 1;
- PagingEntry->Pnle.Bits.PageTableBaseAddress = TempPhysicalBase;
- ASSERT ((PagingEntry->Uint64 & mValidMaskLeafFlag[Level].Uint64) != mValidMaskLeafFlag[Level].Uint64);
- }
-
- ChildPageEntry = (IA32_PAGING_ENTRY *)(UINTN)((PagingEntry->Pnle.Bits.PageTableBaseAddress) << 12);
- for (Index = 0; Index < 512; Index++) {
- Status = ValidateAndRandomeModifyPageTablePageTableEntry (&ChildPageEntry[Index], Level-1, MaxLeafLevel, Address + (Index<<(9*(Level-1) + 3)));
- if (Status != UNIT_TEST_PASSED) {
- return Status;
- }
- }
-
- return UNIT_TEST_PASSED;
-}
-
-/**
- Check if the Page table is valid
-
- @param[in] PageTable The pointer to the page table.
- @param[in] PagingMode The paging mode.
-
- @retval UNIT_TEST_PASSED It is a valid Page Table
-**/
-UNIT_TEST_STATUS
-ValidateAndRandomeModifyPageTable (
- IN UINTN PageTable,
- IN PAGING_MODE PagingMode
- )
-{
- UINTN MaxLevel;
- UINTN MaxLeafLevel;
- UINT64 Index;
- UNIT_TEST_STATUS Status;
- IA32_PAGING_ENTRY *PagingEntry;
-
- if ((PagingMode == Paging32bit) || (PagingMode == PagingPae) || (PagingMode >= PagingModeMax)) {
- //
- // 32bit paging is never supported.
- // PAE paging will be supported later.
- //
- return UNIT_TEST_ERROR_TEST_FAILED;
- }
-
- MaxLeafLevel = (UINT8)PagingMode;
- MaxLevel = (UINT8)(PagingMode >> 8);
-
- PagingEntry = (IA32_PAGING_ENTRY *)(UINTN)PageTable;
- for (Index = 0; Index < 512; Index++) {
- Status = ValidateAndRandomeModifyPageTablePageTableEntry (&PagingEntry[Index], MaxLevel, MaxLeafLevel, Index << (9 * MaxLevel + 3));
- if (Status != UNIT_TEST_PASSED) {
- return Status;
- }
- }
-
- return Status;
-}
-
-/**
- Generate single random map entry.
- The map entry can be the input of function PageTableMap
- the LinearAddress and length is aligned to aligned table.
-
- @param MaxAddress Max Address.
- @param MapEntrys Output MapEntrys contains all parameter as input of function PageTableMap
-**/
-VOID
-GenerateSingleRandomMapEntry (
- IN UINT64 MaxAddress,
- IN OUT MAP_ENTRYS *MapEntrys
- )
-{
- UINTN MapsIndex;
- UINT64 FormerLinearAddress;
- UINT64 FormerLinearAddressBottom;
- UINT64 FormerLinearAddressTop;
-
- MapsIndex = MapEntrys->Count;
-
- ASSERT (MapsIndex < MapEntrys->MaxCount);
- //
- // use AlignedTable to avoid that a random number can be very hard to be 1G or 2M aligned
- //
- if ((MapsIndex != 0) && (RandomBoolean ())) {
- FormerLinearAddress = MapEntrys->Maps[Random32 (0, (UINT32)MapsIndex-1)].LinearAddress;
- if (FormerLinearAddress < 2 * (UINT64)SIZE_1GB) {
- FormerLinearAddressBottom = 0;
- } else {
- FormerLinearAddressBottom = FormerLinearAddress - 2 * (UINT64)SIZE_1GB;
- }
-
- if (FormerLinearAddress + 2 * (UINT64)SIZE_1GB > MaxAddress) {
- FormerLinearAddressTop = MaxAddress;
- } else {
- FormerLinearAddressTop = FormerLinearAddress + 2 * (UINT64)SIZE_1GB;
- }
-
- MapEntrys->Maps[MapsIndex].LinearAddress = Random64 (FormerLinearAddressBottom, FormerLinearAddressTop) & AlignedTable[Random32 (0, ARRAY_SIZE (AlignedTable) -1)];
- } else {
- MapEntrys->Maps[MapsIndex].LinearAddress = Random64 (0, MaxAddress) & AlignedTable[Random32 (0, ARRAY_SIZE (AlignedTable) -1)];
- }
-
- //
- // To have better performance, limit the size less than 10G
- //
- MapEntrys->Maps[MapsIndex].Length = Random64 (0, MIN (MaxAddress - MapEntrys->Maps[MapsIndex].LinearAddress, 10 * (UINT64)SIZE_1GB)) & AlignedTable[Random32 (0, ARRAY_SIZE (AlignedTable) -1)];
-
- if ((MapsIndex != 0) && (RandomBoolean ())) {
- MapEntrys->Maps[MapsIndex].Attribute.Uint64 = MapEntrys->Maps[Random32 (0, (UINT32)MapsIndex-1)].Attribute.Uint64;
- MapEntrys->Maps[MapsIndex].Mask.Uint64 = MapEntrys->Maps[Random32 (0, (UINT32)MapsIndex-1)].Mask.Uint64;
- } else {
- MapEntrys->Maps[MapsIndex].Attribute.Uint64 = Random64 (0, MAX_UINT64) & mSupportedBit.Uint64;
- MapEntrys->Maps[MapsIndex].Mask.Uint64 = Random64 (0, MAX_UINT64) & mSupportedBit.Uint64;
- if (MapEntrys->Maps[MapsIndex].Mask.Bits.ProtectionKey != 0) {
- MapEntrys->Maps[MapsIndex].Mask.Bits.ProtectionKey = 0xF;
- }
- }
-
- if (mRandomOption & ONLY_ONE_ONE_MAPPING) {
- MapEntrys->Maps[MapsIndex].Attribute.Bits.PageTableBaseAddress = MapEntrys->Maps[MapsIndex].LinearAddress >> 12;
- MapEntrys->Maps[MapsIndex].Mask.Bits.PageTableBaseAddress = 0xFFFFFFFFFF;
- } else {
- //
- // Todo: If the mask bit for base address is zero, when dump the pagetable, every entry mapping to physical address zeor.
- // This means the map count will be a large number, and impossible to finish in proper time.
- // Need to avoid such case when remove the Random option ONLY_ONE_ONE_MAPPING
- //
- MapEntrys->Maps[MapsIndex].Attribute.Bits.PageTableBaseAddress = (Random64 (0, (((UINT64)1)<<52) - 1) & AlignedTable[Random32 (0, ARRAY_SIZE (AlignedTable) -1)])>> 12;
- if (RandomBoolean ()) {
- MapEntrys->Maps[MapsIndex].Mask.Bits.PageTableBaseAddress = 0;
- }
- }
-
- MapEntrys->Count += 1;
-}
-
-/**
- Compare the attribute for one point.
- MapEntrys records every memory ranges that is used as input
- Map and MapCount are gotten from Page table
- Compare if this point have same attribute.
-
- @param[in] Address Address of one Point.
- @param[in] MapEntrys Record every memory ranges that is used as input
- @param[in] Map Pointer to an array that describes multiple linear address ranges.
- @param[in] MapCount Pointer to a UINTN that hold the number of entries in the Map.
- @param[in] InitMap Pointer to an array that describes init map entries.
- @param[in] InitMapCount Pointer to a UINTN that hold the number of init map entries.
-
- @retval TRUE At least one byte of data is available to be read
- @retval FALSE No data is available to be read
-**/
-BOOLEAN
-CompareEntrysforOnePoint (
- IN UINT64 Address,
- IN MAP_ENTRYS *MapEntrys,
- IN IA32_MAP_ENTRY *Map,
- IN UINTN MapCount,
- IN IA32_MAP_ENTRY *InitMap,
- IN UINTN InitMapCount
- )
-{
- UINTN Index;
- IA32_MAP_ATTRIBUTE AttributeInInitMap;
- IA32_MAP_ATTRIBUTE AttributeInMap;
- IA32_MAP_ATTRIBUTE AttributeInMapEntrys;
- IA32_MAP_ATTRIBUTE MaskInMapEntrys;
-
- AttributeInMap.Uint64 = 0;
- AttributeInMapEntrys.Uint64 = 0;
- AttributeInInitMap.Uint64 = 0;
- MaskInMapEntrys.Uint64 = 0;
- //
- // Assume every entry in maps does not overlap with each other
- //
- for (Index = 0; Index < MapCount; Index++) {
- if ((Address >= Map[Index].LinearAddress) && (Address < (Map[Index].LinearAddress + Map[Index].Length))) {
- AttributeInMap.Uint64 = (Map[Index].Attribute.Uint64 & mSupportedBit.Uint64);
- AttributeInMap.Bits.PageTableBaseAddress = ((Address - Map[Index].LinearAddress) >> 12) + Map[Index].Attribute.Bits.PageTableBaseAddress;
- break;
- }
- }
-
- //
- // Assume every entry in maps does not overlap with each other
- //
- for (Index = 0; Index < InitMapCount; Index++) {
- if ((Address >= InitMap[Index].LinearAddress) && (Address < (InitMap[Index].LinearAddress + InitMap[Index].Length))) {
- AttributeInInitMap.Uint64 = (InitMap[Index].Attribute.Uint64 & mSupportedBit.Uint64);
- AttributeInInitMap.Bits.PageTableBaseAddress = ((Address - InitMap[Index].LinearAddress) >> 12) + InitMap[Index].Attribute.Bits.PageTableBaseAddress;
- break;
- }
- }
-
- AttributeInMapEntrys.Uint64 = AttributeInInitMap.Uint64;
-
- for (Index = MapEntrys->InitCount; Index < MapEntrys->Count; Index++) {
- if ((Address >= MapEntrys->Maps[Index].LinearAddress) && (Address < (MapEntrys->Maps[Index].LinearAddress + MapEntrys->Maps[Index].Length))) {
- if (AttributeInMapEntrys.Bits.Present == 0) {
- AttributeInMapEntrys.Uint64 = 0;
- MaskInMapEntrys.Uint64 = 0;
- }
-
- MaskInMapEntrys.Uint64 |= MapEntrys->Maps[Index].Mask.Uint64;
- AttributeInMapEntrys.Uint64 &= (~MapEntrys->Maps[Index].Mask.Uint64);
- AttributeInMapEntrys.Uint64 |= (MapEntrys->Maps[Index].Attribute.Uint64 & MapEntrys->Maps[Index].Mask.Uint64);
- if (MapEntrys->Maps[Index].Mask.Bits.PageTableBaseAddress != 0) {
- AttributeInMapEntrys.Bits.PageTableBaseAddress = ((Address - MapEntrys->Maps[Index].LinearAddress) >> 12) + MapEntrys->Maps[Index].Attribute.Bits.PageTableBaseAddress;
- }
- }
- }
-
- if (AttributeInMap.Bits.Present == 0) {
- if (AttributeInMapEntrys.Bits.Present == 0) {
- return TRUE;
- }
- }
-
- if ((AttributeInMap.Uint64 & MaskInMapEntrys.Uint64) != (AttributeInMapEntrys.Uint64 & MaskInMapEntrys.Uint64)) {
- DEBUG ((DEBUG_INFO, "======detailed information begin=====\n"));
- DEBUG ((DEBUG_INFO, "\nError: Detect different attribute on a point with linear address: 0x%lx\n", Address));
- DEBUG ((DEBUG_INFO, "By parsing page table, the point has Attribute 0x%lx, and map to physical address 0x%lx\n", IA32_MAP_ATTRIBUTE_ATTRIBUTES (&AttributeInMap) & MaskInMapEntrys.Uint64, AttributeInMap.Bits.PageTableBaseAddress));
- DEBUG ((DEBUG_INFO, "While according to inputs, the point should Attribute 0x%lx, and should map to physical address 0x%lx\n", IA32_MAP_ATTRIBUTE_ATTRIBUTES (&AttributeInMapEntrys) & MaskInMapEntrys.Uint64, AttributeInMapEntrys.Bits.PageTableBaseAddress));
- DEBUG ((DEBUG_INFO, "The total Mask is 0x%lx\n", MaskInMapEntrys.Uint64));
-
- if (MapEntrys->InitCount != 0) {
- DEBUG ((DEBUG_INFO, "Below is the initialization status:\n"));
- for (Index = 0; Index < InitMapCount; Index++) {
- if ((Address >= InitMap[Index].LinearAddress) && (Address < (InitMap[Index].LinearAddress + InitMap[Index].Length))) {
- DEBUG ((DEBUG_INFO, " *"));
- } else {
- DEBUG ((DEBUG_INFO, " "));
- }
-
- DEBUG ((DEBUG_INFO, " %02d: {0x%lx, 0x%lx, 0x%lx}\n", Index, InitMap[Index].LinearAddress, InitMap[Index].LinearAddress + InitMap[Index].Length, InitMap[Index].Attribute.Uint64));
- }
- }
-
- DEBUG ((DEBUG_INFO, "Below is the inputs:\n"));
- DEBUG ((DEBUG_INFO, " Index: {LinearAddress, LinearLimit, Mask, Attribute}\n"));
- for (Index = MapEntrys->InitCount; Index < MapEntrys->Count; Index++) {
- if ((Address >= MapEntrys->Maps[Index].LinearAddress) && (Address < (MapEntrys->Maps[Index].LinearAddress + MapEntrys->Maps[Index].Length))) {
- DEBUG ((DEBUG_INFO, " *"));
- } else {
- DEBUG ((DEBUG_INFO, " "));
- }
-
- DEBUG ((
- DEBUG_INFO,
- " %02d: {0x%lx, 0x%lx, 0x%lx,0x%lx}\n",
- Index,
- MapEntrys->Maps[Index].LinearAddress,
- MapEntrys->Maps[Index].LinearAddress + MapEntrys->Maps[Index].Length,
- MapEntrys->Maps[Index].Mask.Uint64,
- MapEntrys->Maps[Index].Attribute.Uint64
- ));
- }
-
- DEBUG ((DEBUG_INFO, "Below is the dumped from pagetable:\n"));
- for (Index = 0; Index < MapCount; Index++) {
- if ((Address >= Map[Index].LinearAddress) && (Address < (Map[Index].LinearAddress + Map[Index].Length))) {
- DEBUG ((DEBUG_INFO, " *"));
- } else {
- DEBUG ((DEBUG_INFO, " "));
- }
-
- DEBUG ((DEBUG_INFO, "%02d: {0x%lx, 0x%lx, 0x%lx}\n", Index, Map[Index].LinearAddress, Map[Index].LinearAddress + Map[Index].Length, Map[Index].Attribute.Uint64));
- }
-
- DEBUG ((DEBUG_INFO, "======detailed information done=====\n"));
- return FALSE;
- }
-
- return TRUE;
-}
-
-/**
- Append key point of a given address to Buffer
- if buffer is NULL, only count needed count
-
- @param[in, out] Buffer Buffer to contains all key point.
- @param[in, out] Count Count of the key point.
- @param[in] Address given address
-**/
-VOID
-AppendKeyPointToBuffer (
- IN OUT UINT64 *Buffer,
- IN OUT UINTN *Count,
- IN UINT64 Address
- )
-{
- if ( Buffer != NULL) {
- Buffer[*Count] = Address;
- (*Count)++;
- Buffer[*Count] = Address+1;
- (*Count)++;
- Buffer[*Count] = Address-1;
- (*Count)++;
- } else {
- (*Count) = (*Count) +3;
- }
-}
-
-/**
- Get all key points from a buffer
- if buffer is NULL, only count needed count
-
- @param[in] MapEntrys Record every memory ranges that is used as input
- @param[in] Map Pointer to an array that describes multiple linear address ranges.
- @param[in] MapCount Pointer to a UINTN that hold the actual number of entries in the Map.
- @param[in, out] Buffer Buffer to contains all key point.
- @param[in, out] Count Count of the key point.
-**/
-VOID
-GetKeyPointList (
- IN MAP_ENTRYS *MapEntrys,
- IN IA32_MAP_ENTRY *Map,
- IN UINTN MapCount,
- IN OUT UINT64 *Buffer,
- IN OUT UINTN *Count
- )
-{
- UINTN TemCount;
- UINTN Index1;
- UINTN Index2;
-
- TemCount = 0;
-
- for (Index1 = 0; Index1 < MapEntrys->Count; Index1++) {
- AppendKeyPointToBuffer (Buffer, &TemCount, MapEntrys->Maps[Index1].LinearAddress);
- AppendKeyPointToBuffer (Buffer, &TemCount, MapEntrys->Maps[Index1].LinearAddress + MapEntrys->Maps[Index1].Length);
- }
-
- for (Index2 = 0; Index2 < MapCount; Index2++) {
- if (Buffer != NULL) {
- for (Index1 = 0; Index1 < TemCount; Index1++) {
- if (Buffer[Index1] == Map[Index2].LinearAddress) {
- break;
- }
- }
-
- if (Index1 < TemCount) {
- continue;
- }
- }
-
- AppendKeyPointToBuffer (Buffer, &TemCount, Map[Index2].LinearAddress);
- }
-
- for (Index2 = 0; Index2 < MapCount; Index2++) {
- if (Buffer != NULL) {
- for (Index1 = 0; Index1 < TemCount; Index1++) {
- if (Buffer[Index1] == (Map[Index2].LinearAddress + Map[Index2].Length)) {
- break;
- }
- }
-
- if (Index1 < TemCount) {
- continue;
- }
- }
-
- AppendKeyPointToBuffer (Buffer, &TemCount, Map[Index2].LinearAddress + Map[Index2].Length);
- }
-
- *Count = TemCount;
-}
-
-/**
- Generate random one range with randome attribute, and add it into pagetable
- Compare the key point has same attribute
-
- @param[in, out] PageTable The pointer to the page table to update, or pointer to NULL if a new page table is to be created.
- @param[in] PagingMode The paging mode.
- @param[in] MaxAddress Max Address.
- @param[in] MapEntrys Record every memory ranges that is used as input
- @param[in] PagesRecord Used to record memory usage for page table.
- @param[in] InitMap Pointer to an array that describes init map entries.
- @param[in] InitMapCount Pointer to a UINTN that hold the number of init map entries.
-
- @retval UNIT_TEST_PASSED The test is successful.
-**/
-UNIT_TEST_STATUS
-SingleMapEntryTest (
- IN OUT UINTN *PageTable,
- IN PAGING_MODE PagingMode,
- IN UINT64 MaxAddress,
- IN MAP_ENTRYS *MapEntrys,
- IN ALLOCATE_PAGE_RECORDS *PagesRecord,
- IN IA32_MAP_ENTRY *InitMap,
- IN UINTN InitMapCount
- )
-{
- UINTN MapsIndex;
- RETURN_STATUS Status;
- UINTN PageTableBufferSize;
- VOID *Buffer;
- IA32_MAP_ENTRY *Map;
- UINTN MapCount;
- UINTN Index;
- UINTN KeyPointCount;
- UINTN NewKeyPointCount;
- UINT64 *KeyPointBuffer;
- UINTN Level;
- UINT64 Value;
- UNIT_TEST_STATUS TestStatus;
-
- MapsIndex = MapEntrys->Count;
-
- GenerateSingleRandomMapEntry (MaxAddress, MapEntrys);
-
- PageTableBufferSize = 0;
- Status = PageTableMap (
- PageTable,
- PagingMode,
- NULL,
- &PageTableBufferSize,
- MapEntrys->Maps[MapsIndex].LinearAddress,
- MapEntrys->Maps[MapsIndex].Length,
- &MapEntrys->Maps[MapsIndex].Attribute,
- &MapEntrys->Maps[MapsIndex].Mask
- );
- if (PageTableBufferSize != 0) {
- UT_ASSERT_EQUAL (Status, RETURN_BUFFER_TOO_SMALL);
-
- //
- // Allocate memory for Page table
- // Note the memory is used in one complete Random test.
- //
- Buffer = PagesRecord->AllocatePagesForPageTable (PagesRecord, EFI_SIZE_TO_PAGES (PageTableBufferSize));
- UT_ASSERT_NOT_EQUAL (Buffer, NULL);
- Status = PageTableMap (
- PageTable,
- PagingMode,
- Buffer,
- &PageTableBufferSize,
- MapEntrys->Maps[MapsIndex].LinearAddress,
- MapEntrys->Maps[MapsIndex].Length,
- &MapEntrys->Maps[MapsIndex].Attribute,
- &MapEntrys->Maps[MapsIndex].Mask
- );
- }
-
- if (Status != RETURN_SUCCESS ) {
- UT_ASSERT_EQUAL (Status, RETURN_SUCCESS);
- }
-
- UT_ASSERT_EQUAL (Status, RETURN_SUCCESS);
- TestStatus = IsPageTableValid (*PageTable, PagingMode);
- if (TestStatus != UNIT_TEST_PASSED) {
- return TestStatus;
- }
-
- MapCount = 0;
- Status = PageTableParse (*PageTable, PagingMode, NULL, &MapCount);
- if (MapCount != 0) {
- UT_ASSERT_EQUAL (Status, RETURN_BUFFER_TOO_SMALL);
-
- //
- // Allocate memory for Maps
- // Note the memory is only used in this one Single MapEntry Test
- //
- Map = AllocatePages (EFI_SIZE_TO_PAGES (MapCount * sizeof (IA32_MAP_ENTRY)));
- ASSERT (Map != NULL);
- Status = PageTableParse (*PageTable, PagingMode, Map, &MapCount);
- }
-
- UT_ASSERT_EQUAL (Status, RETURN_SUCCESS);
-
- //
- // Allocate memory to record all key point
- // Note the memory is only used in this one Single MapEntry Test
- //
- KeyPointCount = 0;
- GetKeyPointList (MapEntrys, Map, MapCount, NULL, &KeyPointCount);
- KeyPointBuffer = AllocatePages (EFI_SIZE_TO_PAGES (KeyPointCount * sizeof (UINT64)));
- ASSERT (KeyPointBuffer != NULL);
- NewKeyPointCount = 0;
- GetKeyPointList (MapEntrys, Map, MapCount, KeyPointBuffer, &NewKeyPointCount);
-
- //
- // Compare all key point's attribute
- //
- for (Index = 0; Index < NewKeyPointCount; Index++) {
- if (!CompareEntrysforOnePoint (KeyPointBuffer[Index], MapEntrys, Map, MapCount, InitMap, InitMapCount)) {
- DEBUG ((DEBUG_INFO, "Error happens at below key point\n"));
- DEBUG ((DEBUG_INFO, "Index = %d KeyPointBuffer[Index] = 0x%lx\n", Index, KeyPointBuffer[Index]));
- Value = GetEntryFromPageTable (*PageTable, PagingMode, KeyPointBuffer[Index], &Level);
- DEBUG ((DEBUG_INFO, "From Page table, this key point is in level %d entry, with entry value is 0x%lx\n", Level, Value));
- UT_ASSERT_TRUE (FALSE);
- }
- }
-
- FreePages (KeyPointBuffer, EFI_SIZE_TO_PAGES (KeyPointCount * sizeof (UINT64)));
- if (MapCount != 0) {
- FreePages (Map, EFI_SIZE_TO_PAGES (MapCount * sizeof (IA32_MAP_ENTRY)));
- }
-
- return UNIT_TEST_PASSED;
-}
-
-/**
- Allocate page and record the information in PagesRecord
-
- @param[in] PagesRecord Point to a struct to record memory usage
- @param[in] Pages Page count needed to allocate
-
- @return A pointer to the allocated buffer or NULL if allocation fails.
-**/
-VOID *
-EFIAPI
-RecordAllocatePages (
- IN ALLOCATE_PAGE_RECORDS *PagesRecord,
- IN UINTN Pages
- )
-{
- VOID *Buffer;
-
- Buffer = NULL;
- if (PagesRecord->Count < PagesRecord->MaxCount) {
- Buffer = AllocatePages (Pages);
- PagesRecord->Records[PagesRecord->Count].Buffer = Buffer;
- PagesRecord->Records[PagesRecord->Count].Pages = Pages;
- PagesRecord->Count++;
- }
-
- ASSERT (Buffer != NULL);
-
- return Buffer;
-}
-
-/**
- The function is a whole Random test, it will call SingleMapEntryTest for ExpctedEntryNumber times
-
- @param[in] ExpctedEntryNumber The count of random entry
- @param[in] PagingMode The paging mode.
-
- @retval UNIT_TEST_PASSED The test is successful.
-**/
-UNIT_TEST_STATUS
-MultipleMapEntryTest (
- IN UINTN ExpctedEntryNumber,
- IN PAGING_MODE PagingMode
- )
-{
- UINTN PageTable;
- UINT64 MaxAddress;
- MAP_ENTRYS *MapEntrys;
- ALLOCATE_PAGE_RECORDS *PagesRecord;
- UINTN Index;
- UNIT_TEST_STATUS TestStatus;
- RETURN_STATUS Status;
- IA32_MAP_ENTRY *InitMap;
- UINTN InitMapCount;
-
- MaxAddress = GetMaxAddress (PagingMode);
- PageTable = 0;
- MapEntrys = AllocatePages (EFI_SIZE_TO_PAGES (1000*sizeof (MAP_ENTRY) + sizeof (MAP_ENTRYS)));
- ASSERT (MapEntrys != NULL);
- MapEntrys->Count = 0;
- MapEntrys->InitCount = 0;
- MapEntrys->MaxCount = 1000;
- PagesRecord = AllocatePages (EFI_SIZE_TO_PAGES (1000*sizeof (ALLOCATE_PAGE_RECORD) + sizeof (ALLOCATE_PAGE_RECORDS)));
- ASSERT (PagesRecord != NULL);
- PagesRecord->Count = 0;
- PagesRecord->MaxCount = 1000;
- PagesRecord->AllocatePagesForPageTable = RecordAllocatePages;
-
- if (mRandomOption & MANUAL_CHANGE_PAGE_TABLE) {
- ExpctedEntryNumber = ExpctedEntryNumber/2;
- }
-
- for (Index = 0; Index < ExpctedEntryNumber; Index++) {
- TestStatus = SingleMapEntryTest (
- &PageTable,
- PagingMode,
- MaxAddress,
- MapEntrys,
- PagesRecord,
- NULL,
- 0
- );
- if (TestStatus != UNIT_TEST_PASSED) {
- return TestStatus;
- }
- }
-
- if ((mRandomOption & MANUAL_CHANGE_PAGE_TABLE) != 0) {
- MapEntrys->InitCount = ExpctedEntryNumber;
- TestStatus = ValidateAndRandomeModifyPageTable (PageTable, PagingMode);
- RandomNumber = 0;
- if (TestStatus != UNIT_TEST_PASSED) {
- return TestStatus;
- }
-
- InitMapCount = 0;
- Status = PageTableParse (PageTable, PagingMode, NULL, &InitMapCount);
- if (InitMapCount != 0) {
- UT_ASSERT_EQUAL (Status, RETURN_BUFFER_TOO_SMALL);
-
- //
- // Allocate memory for Maps
- // Note the memory is only used in this one Single MapEntry Test
- //
- InitMap = AllocatePages (EFI_SIZE_TO_PAGES (InitMapCount * sizeof (IA32_MAP_ENTRY)));
- ASSERT (InitMap != NULL);
- Status = PageTableParse (PageTable, PagingMode, InitMap, &InitMapCount);
- }
-
- UT_ASSERT_EQUAL (Status, RETURN_SUCCESS);
- for (Index = 0; Index < ExpctedEntryNumber; Index++) {
- TestStatus = SingleMapEntryTest (
- &PageTable,
- PagingMode,
- MaxAddress,
- MapEntrys,
- PagesRecord,
- InitMap,
- InitMapCount
- );
- if (TestStatus != UNIT_TEST_PASSED) {
- return TestStatus;
- }
- }
-
- if (InitMapCount != 0) {
- FreePages (InitMap, EFI_SIZE_TO_PAGES (InitMapCount*sizeof (IA32_MAP_ENTRY)));
- }
- }
-
- FreePages (
- MapEntrys,
- EFI_SIZE_TO_PAGES (1000*sizeof (MAP_ENTRY) + sizeof (MAP_ENTRYS))
- );
-
- for (Index = 0; Index < PagesRecord->Count; Index++) {
- FreePages (PagesRecord->Records[Index].Buffer, PagesRecord->Records[Index].Pages);
- }
-
- FreePages (PagesRecord, EFI_SIZE_TO_PAGES (1000*sizeof (ALLOCATE_PAGE_RECORD) + sizeof (ALLOCATE_PAGE_RECORDS)));
-
- return UNIT_TEST_PASSED;
-}
-
-/**
- Random Test
-
- @param[in] Context [Optional] An optional parameter that enables:
- 1) test-case reuse with varied parameters and
- 2) test-case re-entry for Target tests that need a
- reboot. This parameter is a VOID* and it is the
- responsibility of the test author to ensure that the
- contents are well understood by all test cases that may
- consume it.
-
- @retval UNIT_TEST_PASSED The Unit test has completed and the test
- case was successful.
- @retval UNIT_TEST_ERROR_TEST_FAILED A test case assertion has failed.
-**/
-UNIT_TEST_STATUS
-EFIAPI
-TestCaseforRandomTest (
- IN UNIT_TEST_CONTEXT Context
- )
-{
- UNIT_TEST_STATUS Status;
- UINTN Index;
-
- UT_ASSERT_EQUAL (RandomSeed (NULL, 0), TRUE);
- UT_ASSERT_EQUAL (Random32 (100, 100), 100);
- UT_ASSERT_EQUAL (Random64 (100, 100), 100);
- UT_ASSERT_TRUE ((Random32 (9, 10) >= 9) & (Random32 (9, 10) <= 10));
- UT_ASSERT_TRUE ((Random64 (9, 10) >= 9) & (Random64 (9, 10) <= 10));
-
- mSupportedBit.Bits.Present = 1;
- mSupportedBit.Bits.ReadWrite = 1;
- mSupportedBit.Bits.UserSupervisor = 1;
- mSupportedBit.Bits.WriteThrough = 1;
- mSupportedBit.Bits.CacheDisabled = 1;
- mSupportedBit.Bits.Accessed = 1;
- mSupportedBit.Bits.Dirty = 1;
- mSupportedBit.Bits.Pat = 1;
- mSupportedBit.Bits.Global = 1;
- mSupportedBit.Bits.Reserved1 = 0;
- mSupportedBit.Bits.PageTableBaseAddress = 0;
- mSupportedBit.Bits.Reserved2 = 0;
- mSupportedBit.Bits.ProtectionKey = 0xF;
- mSupportedBit.Bits.Nx = 1;
-
- mRandomOption = ((CPU_PAGE_TABLE_LIB_RANDOM_TEST_CONTEXT *)Context)->RandomOption;
- mNumberIndex = 0;
-
- for (Index = 0; Index < ((CPU_PAGE_TABLE_LIB_RANDOM_TEST_CONTEXT *)Context)->TestCount; Index++) {
- Status = MultipleMapEntryTest (
- ((CPU_PAGE_TABLE_LIB_RANDOM_TEST_CONTEXT *)Context)->TestRangeCount,
- ((CPU_PAGE_TABLE_LIB_RANDOM_TEST_CONTEXT *)Context)->PagingMode
- );
- if (Status != UNIT_TEST_PASSED) {
- return Status;
- }
-
- DEBUG ((DEBUG_INFO, "."));
- }
-
- DEBUG ((DEBUG_INFO, "\n"));
-
- return UNIT_TEST_PASSED;
-}