2010-01-12 20:14:09 +01:00
|
|
|
/*++
|
|
|
|
|
2010-04-29 14:15:47 +02:00
|
|
|
Copyright (c) 2009, Hewlett-Packard Company. All rights reserved.<BR>
|
|
|
|
Portions copyright (c) 2010, Apple Inc. All rights reserved.<BR>
|
2020-12-10 17:50:22 +01:00
|
|
|
Portions copyright (c) 2013-2021, Arm Limited. All rights reserved.<BR>
|
2017-02-09 08:20:30 +01:00
|
|
|
Copyright (c) 2017, Intel Corporation. All rights reserved.<BR>
|
2010-01-12 20:14:09 +01:00
|
|
|
|
2019-04-04 01:03:18 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2010-01-12 20:14:09 +01:00
|
|
|
|
|
|
|
|
|
|
|
--*/
|
|
|
|
|
2013-07-26 19:12:12 +02:00
|
|
|
#include <Library/MemoryAllocationLib.h>
|
2010-01-12 20:14:09 +01:00
|
|
|
#include "CpuDxe.h"
|
|
|
|
|
2014-08-19 15:29:52 +02:00
|
|
|
EFI_STATUS
|
2010-01-12 20:14:09 +01:00
|
|
|
SectionToGcdAttributes (
|
|
|
|
IN UINT32 SectionAttributes,
|
|
|
|
OUT UINT64 *GcdAttributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
*GcdAttributes = 0;
|
|
|
|
|
|
|
|
// determine cacheability attributes
|
2011-02-02 23:35:30 +01:00
|
|
|
switch(SectionAttributes & TT_DESCRIPTOR_SECTION_CACHE_POLICY_MASK) {
|
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_STRONGLY_ORDERED:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_UC;
|
|
|
|
break;
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_SHAREABLE_DEVICE:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_UC;
|
|
|
|
break;
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_WRITE_THROUGH_NO_ALLOC:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_WT;
|
|
|
|
break;
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_WRITE_BACK_NO_ALLOC:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_WB;
|
|
|
|
break;
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_NON_CACHEABLE:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_WC;
|
|
|
|
break;
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_WRITE_BACK_ALLOC:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_WB;
|
|
|
|
break;
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_CACHE_POLICY_NON_SHAREABLE_DEVICE:
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_UC;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
2011-02-02 23:35:30 +01:00
|
|
|
|
2010-01-12 20:14:09 +01:00
|
|
|
// determine protection attributes
|
2011-02-02 23:35:30 +01:00
|
|
|
switch(SectionAttributes & TT_DESCRIPTOR_SECTION_AP_MASK) {
|
|
|
|
case TT_DESCRIPTOR_SECTION_AP_NO_NO: // no read, no write
|
2017-02-09 08:20:30 +01:00
|
|
|
//*GcdAttributes |= EFI_MEMORY_RO | EFI_MEMORY_RP;
|
2010-01-12 20:14:09 +01:00
|
|
|
break;
|
|
|
|
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_AP_RW_NO:
|
|
|
|
case TT_DESCRIPTOR_SECTION_AP_RW_RW:
|
2010-01-12 20:14:09 +01:00
|
|
|
// normal read/write access, do not add additional attributes
|
|
|
|
break;
|
|
|
|
|
|
|
|
// read only cases map to write-protect
|
2011-02-02 23:35:30 +01:00
|
|
|
case TT_DESCRIPTOR_SECTION_AP_RO_NO:
|
|
|
|
case TT_DESCRIPTOR_SECTION_AP_RO_RO:
|
2017-02-09 08:20:30 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_RO;
|
2010-01-12 20:14:09 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// now process eXectue Never attribute
|
2011-02-02 23:35:30 +01:00
|
|
|
if ((SectionAttributes & TT_DESCRIPTOR_SECTION_XN_MASK) != 0 ) {
|
2010-01-12 20:14:09 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_XP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2011-03-31 13:23:55 +02:00
|
|
|
EFI_STATUS
|
|
|
|
PageToGcdAttributes (
|
|
|
|
IN UINT32 PageAttributes,
|
|
|
|
OUT UINT64 *GcdAttributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
*GcdAttributes = 0;
|
|
|
|
|
|
|
|
// determine cacheability attributes
|
|
|
|
switch(PageAttributes & TT_DESCRIPTOR_PAGE_CACHE_POLICY_MASK) {
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_STRONGLY_ORDERED:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_UC;
|
|
|
|
break;
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_SHAREABLE_DEVICE:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_UC;
|
|
|
|
break;
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_WRITE_THROUGH_NO_ALLOC:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_WT;
|
|
|
|
break;
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_WRITE_BACK_NO_ALLOC:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_WB;
|
|
|
|
break;
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_NON_CACHEABLE:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_WC;
|
|
|
|
break;
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_WRITE_BACK_ALLOC:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_WB;
|
|
|
|
break;
|
|
|
|
case TT_DESCRIPTOR_PAGE_CACHE_POLICY_NON_SHAREABLE_DEVICE:
|
|
|
|
*GcdAttributes |= EFI_MEMORY_UC;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// determine protection attributes
|
|
|
|
switch(PageAttributes & TT_DESCRIPTOR_PAGE_AP_MASK) {
|
|
|
|
case TT_DESCRIPTOR_PAGE_AP_NO_NO: // no read, no write
|
2017-02-09 08:20:30 +01:00
|
|
|
//*GcdAttributes |= EFI_MEMORY_RO | EFI_MEMORY_RP;
|
2011-03-31 13:23:55 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TT_DESCRIPTOR_PAGE_AP_RW_NO:
|
|
|
|
case TT_DESCRIPTOR_PAGE_AP_RW_RW:
|
|
|
|
// normal read/write access, do not add additional attributes
|
|
|
|
break;
|
|
|
|
|
|
|
|
// read only cases map to write-protect
|
|
|
|
case TT_DESCRIPTOR_PAGE_AP_RO_NO:
|
|
|
|
case TT_DESCRIPTOR_PAGE_AP_RO_RO:
|
2017-02-09 08:20:30 +01:00
|
|
|
*GcdAttributes |= EFI_MEMORY_RO;
|
2011-03-31 13:23:55 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// now process eXectue Never attribute
|
|
|
|
if ((PageAttributes & TT_DESCRIPTOR_PAGE_XN_MASK) != 0 ) {
|
|
|
|
*GcdAttributes |= EFI_MEMORY_XP;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
SyncCacheConfigPage (
|
|
|
|
IN UINT32 SectionIndex,
|
|
|
|
IN UINT32 FirstLevelDescriptor,
|
|
|
|
IN UINTN NumberOfDescriptors,
|
|
|
|
IN EFI_GCD_MEMORY_SPACE_DESCRIPTOR *MemorySpaceMap,
|
|
|
|
IN OUT EFI_PHYSICAL_ADDRESS *NextRegionBase,
|
|
|
|
IN OUT UINT64 *NextRegionLength,
|
|
|
|
IN OUT UINT32 *NextSectionAttributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 i;
|
|
|
|
volatile ARM_PAGE_TABLE_ENTRY *SecondLevelTable;
|
2020-12-10 14:07:43 +01:00
|
|
|
UINT32 NextPageAttributes;
|
|
|
|
UINT32 PageAttributes;
|
2011-03-31 13:23:55 +02:00
|
|
|
UINT32 BaseAddress;
|
|
|
|
UINT64 GcdAttributes;
|
|
|
|
|
|
|
|
// Get the Base Address from FirstLevelDescriptor;
|
|
|
|
BaseAddress = TT_DESCRIPTOR_PAGE_BASE_ADDRESS(SectionIndex << TT_DESCRIPTOR_SECTION_BASE_SHIFT);
|
|
|
|
|
|
|
|
// Convert SectionAttributes into PageAttributes
|
|
|
|
NextPageAttributes =
|
|
|
|
TT_DESCRIPTOR_CONVERT_TO_PAGE_CACHE_POLICY(*NextSectionAttributes,0) |
|
|
|
|
TT_DESCRIPTOR_CONVERT_TO_PAGE_AP(*NextSectionAttributes);
|
|
|
|
|
|
|
|
// obtain page table base
|
|
|
|
SecondLevelTable = (ARM_PAGE_TABLE_ENTRY *)(FirstLevelDescriptor & TT_DESCRIPTOR_SECTION_PAGETABLE_ADDRESS_MASK);
|
|
|
|
|
|
|
|
for (i=0; i < TRANSLATION_TABLE_PAGE_COUNT; i++) {
|
|
|
|
if ((SecondLevelTable[i] & TT_DESCRIPTOR_PAGE_TYPE_MASK) == TT_DESCRIPTOR_PAGE_TYPE_PAGE) {
|
|
|
|
// extract attributes (cacheability and permissions)
|
|
|
|
PageAttributes = SecondLevelTable[i] & (TT_DESCRIPTOR_PAGE_CACHE_POLICY_MASK | TT_DESCRIPTOR_PAGE_AP_MASK);
|
|
|
|
|
|
|
|
if (NextPageAttributes == 0) {
|
|
|
|
// start on a new region
|
|
|
|
*NextRegionLength = 0;
|
|
|
|
*NextRegionBase = BaseAddress | (i << TT_DESCRIPTOR_PAGE_BASE_SHIFT);
|
|
|
|
NextPageAttributes = PageAttributes;
|
|
|
|
} else if (PageAttributes != NextPageAttributes) {
|
|
|
|
// Convert Section Attributes into GCD Attributes
|
|
|
|
Status = PageToGcdAttributes (NextPageAttributes, &GcdAttributes);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
// update GCD with these changes (this will recurse into our own CpuSetMemoryAttributes below which is OK)
|
|
|
|
SetGcdMemorySpaceAttributes (MemorySpaceMap, NumberOfDescriptors, *NextRegionBase, *NextRegionLength, GcdAttributes);
|
|
|
|
|
|
|
|
// start on a new region
|
|
|
|
*NextRegionLength = 0;
|
|
|
|
*NextRegionBase = BaseAddress | (i << TT_DESCRIPTOR_PAGE_BASE_SHIFT);
|
|
|
|
NextPageAttributes = PageAttributes;
|
|
|
|
}
|
|
|
|
} else if (NextPageAttributes != 0) {
|
|
|
|
// Convert Page Attributes into GCD Attributes
|
|
|
|
Status = PageToGcdAttributes (NextPageAttributes, &GcdAttributes);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
// update GCD with these changes (this will recurse into our own CpuSetMemoryAttributes below which is OK)
|
|
|
|
SetGcdMemorySpaceAttributes (MemorySpaceMap, NumberOfDescriptors, *NextRegionBase, *NextRegionLength, GcdAttributes);
|
|
|
|
|
|
|
|
*NextRegionLength = 0;
|
|
|
|
*NextRegionBase = BaseAddress | (i << TT_DESCRIPTOR_PAGE_BASE_SHIFT);
|
|
|
|
NextPageAttributes = 0;
|
|
|
|
}
|
|
|
|
*NextRegionLength += TT_DESCRIPTOR_PAGE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert back PageAttributes into SectionAttributes
|
|
|
|
*NextSectionAttributes =
|
|
|
|
TT_DESCRIPTOR_CONVERT_TO_SECTION_CACHE_POLICY(NextPageAttributes,0) |
|
|
|
|
TT_DESCRIPTOR_CONVERT_TO_SECTION_AP(NextPageAttributes);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
2010-01-12 20:14:09 +01:00
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
SyncCacheConfig (
|
|
|
|
IN EFI_CPU_ARCH_PROTOCOL *CpuProtocol
|
|
|
|
)
|
|
|
|
{
|
2010-01-14 04:25:08 +01:00
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 i;
|
|
|
|
EFI_PHYSICAL_ADDRESS NextRegionBase;
|
|
|
|
UINT64 NextRegionLength;
|
2020-12-10 14:07:43 +01:00
|
|
|
UINT32 NextSectionAttributes;
|
|
|
|
UINT32 SectionAttributes;
|
2010-01-14 04:25:08 +01:00
|
|
|
UINT64 GcdAttributes;
|
2010-01-12 20:14:09 +01:00
|
|
|
volatile ARM_FIRST_LEVEL_DESCRIPTOR *FirstLevelTable;
|
2010-01-14 04:25:08 +01:00
|
|
|
UINTN NumberOfDescriptors;
|
|
|
|
EFI_GCD_MEMORY_SPACE_DESCRIPTOR *MemorySpaceMap;
|
2010-01-12 20:14:09 +01:00
|
|
|
|
|
|
|
|
2019-02-06 16:39:35 +01:00
|
|
|
DEBUG ((DEBUG_PAGE, "SyncCacheConfig()\n"));
|
2010-01-14 04:25:08 +01:00
|
|
|
|
2010-01-12 20:14:09 +01:00
|
|
|
// This code assumes MMU is enabled and filed with section translations
|
|
|
|
ASSERT (ArmMmuEnabled ());
|
|
|
|
|
2010-01-14 04:25:08 +01:00
|
|
|
//
|
|
|
|
// Get the memory space map from GCD
|
|
|
|
//
|
|
|
|
MemorySpaceMap = NULL;
|
|
|
|
Status = gDS->GetMemorySpaceMap (&NumberOfDescriptors, &MemorySpaceMap);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
2010-01-12 20:14:09 +01:00
|
|
|
|
|
|
|
// The GCD implementation maintains its own copy of the state of memory space attributes. GCD needs
|
|
|
|
// to know what the initial memory space attributes are. The CPU Arch. Protocol does not provide a
|
|
|
|
// GetMemoryAttributes function for GCD to get this so we must resort to calling GCD (as if we were
|
|
|
|
// a client) to update its copy of the attributes. This is bad architecture and should be replaced
|
|
|
|
// with a way for GCD to query the CPU Arch. driver of the existing memory space attributes instead.
|
|
|
|
|
|
|
|
// obtain page table base
|
2011-02-02 23:35:30 +01:00
|
|
|
FirstLevelTable = (ARM_FIRST_LEVEL_DESCRIPTOR *)(ArmGetTTBR0BaseAddress ());
|
2010-01-12 20:14:09 +01:00
|
|
|
|
2011-03-31 13:23:55 +02:00
|
|
|
// Get the first region
|
|
|
|
NextSectionAttributes = FirstLevelTable[0] & (TT_DESCRIPTOR_SECTION_CACHE_POLICY_MASK | TT_DESCRIPTOR_SECTION_AP_MASK);
|
2010-01-12 20:14:09 +01:00
|
|
|
|
|
|
|
// iterate through each 1MB descriptor
|
|
|
|
NextRegionBase = NextRegionLength = 0;
|
2011-03-31 13:23:55 +02:00
|
|
|
for (i=0; i < TRANSLATION_TABLE_SECTION_COUNT; i++) {
|
|
|
|
if ((FirstLevelTable[i] & TT_DESCRIPTOR_SECTION_TYPE_MASK) == TT_DESCRIPTOR_SECTION_TYPE_SECTION) {
|
|
|
|
// extract attributes (cacheability and permissions)
|
|
|
|
SectionAttributes = FirstLevelTable[i] & (TT_DESCRIPTOR_SECTION_CACHE_POLICY_MASK | TT_DESCRIPTOR_SECTION_AP_MASK);
|
|
|
|
|
|
|
|
if (NextSectionAttributes == 0) {
|
|
|
|
// start on a new region
|
|
|
|
NextRegionLength = 0;
|
|
|
|
NextRegionBase = TT_DESCRIPTOR_SECTION_BASE_ADDRESS(i << TT_DESCRIPTOR_SECTION_BASE_SHIFT);
|
|
|
|
NextSectionAttributes = SectionAttributes;
|
|
|
|
} else if (SectionAttributes != NextSectionAttributes) {
|
|
|
|
// Convert Section Attributes into GCD Attributes
|
|
|
|
Status = SectionToGcdAttributes (NextSectionAttributes, &GcdAttributes);
|
2010-01-14 04:25:08 +01:00
|
|
|
ASSERT_EFI_ERROR (Status);
|
2010-01-12 20:14:09 +01:00
|
|
|
|
|
|
|
// update GCD with these changes (this will recurse into our own CpuSetMemoryAttributes below which is OK)
|
2010-01-14 04:25:08 +01:00
|
|
|
SetGcdMemorySpaceAttributes (MemorySpaceMap, NumberOfDescriptors, NextRegionBase, NextRegionLength, GcdAttributes);
|
|
|
|
|
2010-01-12 20:14:09 +01:00
|
|
|
// start on a new region
|
|
|
|
NextRegionLength = 0;
|
2011-03-31 13:23:55 +02:00
|
|
|
NextRegionBase = TT_DESCRIPTOR_SECTION_BASE_ADDRESS(i << TT_DESCRIPTOR_SECTION_BASE_SHIFT);
|
|
|
|
NextSectionAttributes = SectionAttributes;
|
2010-01-12 20:14:09 +01:00
|
|
|
}
|
2011-03-31 13:23:55 +02:00
|
|
|
NextRegionLength += TT_DESCRIPTOR_SECTION_SIZE;
|
|
|
|
} else if (TT_DESCRIPTOR_SECTION_TYPE_IS_PAGE_TABLE(FirstLevelTable[i])) {
|
2016-09-27 00:45:02 +02:00
|
|
|
// In this case any bits set in the 'NextSectionAttributes' are garbage and were set from
|
|
|
|
// bits that are actually part of the pagetable address. We clear it out to zero so that
|
|
|
|
// the SyncCacheConfigPage will use the page attributes instead of trying to convert the
|
|
|
|
// section attributes into page attributes
|
|
|
|
NextSectionAttributes = 0;
|
2011-03-31 13:23:55 +02:00
|
|
|
Status = SyncCacheConfigPage (
|
|
|
|
i,FirstLevelTable[i],
|
2011-03-31 17:31:36 +02:00
|
|
|
NumberOfDescriptors, MemorySpaceMap,
|
2011-03-31 13:23:55 +02:00
|
|
|
&NextRegionBase,&NextRegionLength,&NextSectionAttributes);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
} else {
|
|
|
|
// We do not support yet 16MB sections
|
|
|
|
ASSERT ((FirstLevelTable[i] & TT_DESCRIPTOR_SECTION_TYPE_MASK) != TT_DESCRIPTOR_SECTION_TYPE_SUPERSECTION);
|
2010-01-12 20:14:09 +01:00
|
|
|
|
2011-03-31 13:23:55 +02:00
|
|
|
// start on a new region
|
|
|
|
if (NextSectionAttributes != 0) {
|
|
|
|
// Convert Section Attributes into GCD Attributes
|
|
|
|
Status = SectionToGcdAttributes (NextSectionAttributes, &GcdAttributes);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2010-01-12 20:14:09 +01:00
|
|
|
|
2011-03-31 13:23:55 +02:00
|
|
|
// update GCD with these changes (this will recurse into our own CpuSetMemoryAttributes below which is OK)
|
|
|
|
SetGcdMemorySpaceAttributes (MemorySpaceMap, NumberOfDescriptors, NextRegionBase, NextRegionLength, GcdAttributes);
|
2010-01-12 20:14:09 +01:00
|
|
|
|
2011-03-31 13:23:55 +02:00
|
|
|
NextRegionLength = 0;
|
|
|
|
NextRegionBase = TT_DESCRIPTOR_SECTION_BASE_ADDRESS(i << TT_DESCRIPTOR_SECTION_BASE_SHIFT);
|
|
|
|
NextSectionAttributes = 0;
|
|
|
|
}
|
|
|
|
NextRegionLength += TT_DESCRIPTOR_SECTION_SIZE;
|
|
|
|
}
|
2010-01-12 20:14:09 +01:00
|
|
|
} // section entry loop
|
|
|
|
|
2011-03-31 13:23:55 +02:00
|
|
|
if (NextSectionAttributes != 0) {
|
|
|
|
// Convert Section Attributes into GCD Attributes
|
|
|
|
Status = SectionToGcdAttributes (NextSectionAttributes, &GcdAttributes);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
// update GCD with these changes (this will recurse into our own CpuSetMemoryAttributes below which is OK)
|
|
|
|
SetGcdMemorySpaceAttributes (MemorySpaceMap, NumberOfDescriptors, NextRegionBase, NextRegionLength, GcdAttributes);
|
|
|
|
}
|
|
|
|
|
2013-07-26 19:12:12 +02:00
|
|
|
FreePool (MemorySpaceMap);
|
|
|
|
|
2010-01-12 20:14:09 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2013-08-19 19:38:39 +02:00
|
|
|
UINT64
|
|
|
|
EfiAttributeToArmAttribute (
|
|
|
|
IN UINT64 EfiAttributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT64 ArmAttributes;
|
|
|
|
|
|
|
|
switch (EfiAttributes & EFI_MEMORY_CACHETYPE_MASK) {
|
|
|
|
case EFI_MEMORY_UC:
|
|
|
|
// Map to strongly ordered
|
|
|
|
ArmAttributes = TT_DESCRIPTOR_SECTION_CACHE_POLICY_STRONGLY_ORDERED; // TEX[2:0] = 0, C=0, B=0
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EFI_MEMORY_WC:
|
|
|
|
// Map to normal non-cachable
|
|
|
|
ArmAttributes = TT_DESCRIPTOR_SECTION_CACHE_POLICY_NON_CACHEABLE; // TEX [2:0]= 001 = 0x2, B=0, C=0
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EFI_MEMORY_WT:
|
|
|
|
// Write through with no-allocate
|
|
|
|
ArmAttributes = TT_DESCRIPTOR_SECTION_CACHE_POLICY_WRITE_THROUGH_NO_ALLOC; // TEX [2:0] = 0, C=1, B=0
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EFI_MEMORY_WB:
|
|
|
|
// Write back (with allocate)
|
|
|
|
ArmAttributes = TT_DESCRIPTOR_SECTION_CACHE_POLICY_WRITE_BACK_ALLOC; // TEX [2:0] = 001, C=1, B=1
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EFI_MEMORY_UCE:
|
|
|
|
default:
|
|
|
|
ArmAttributes = TT_DESCRIPTOR_SECTION_TYPE_FAULT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Determine protection attributes
|
2020-12-10 17:50:22 +01:00
|
|
|
if ((EfiAttributes & EFI_MEMORY_RO) != 0) {
|
2013-08-19 19:38:39 +02:00
|
|
|
ArmAttributes |= TT_DESCRIPTOR_SECTION_AP_RO_RO;
|
|
|
|
} else {
|
|
|
|
ArmAttributes |= TT_DESCRIPTOR_SECTION_AP_RW_RW;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Determine eXecute Never attribute
|
2020-12-10 17:50:22 +01:00
|
|
|
if ((EfiAttributes & EFI_MEMORY_XP) != 0) {
|
2013-08-19 19:38:39 +02:00
|
|
|
ArmAttributes |= TT_DESCRIPTOR_SECTION_XN_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ArmAttributes;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
GetMemoryRegionPage (
|
|
|
|
IN UINT32 *PageTable,
|
|
|
|
IN OUT UINTN *BaseAddress,
|
|
|
|
OUT UINTN *RegionLength,
|
|
|
|
OUT UINTN *RegionAttributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 PageAttributes;
|
|
|
|
UINT32 TableIndex;
|
|
|
|
UINT32 PageDescriptor;
|
|
|
|
|
|
|
|
// Convert the section attributes into page attributes
|
|
|
|
PageAttributes = ConvertSectionAttributesToPageAttributes (*RegionAttributes, 0);
|
|
|
|
|
|
|
|
// Calculate index into first level translation table for start of modification
|
2013-09-02 15:12:17 +02:00
|
|
|
TableIndex = ((*BaseAddress) & TT_DESCRIPTOR_PAGE_INDEX_MASK) >> TT_DESCRIPTOR_PAGE_BASE_SHIFT;
|
2013-08-19 19:38:39 +02:00
|
|
|
ASSERT (TableIndex < TRANSLATION_TABLE_PAGE_COUNT);
|
|
|
|
|
|
|
|
// Go through the page table to find the end of the section
|
|
|
|
for (; TableIndex < TRANSLATION_TABLE_PAGE_COUNT; TableIndex++) {
|
|
|
|
// Get the section at the given index
|
|
|
|
PageDescriptor = PageTable[TableIndex];
|
|
|
|
|
|
|
|
if ((PageDescriptor & TT_DESCRIPTOR_PAGE_TYPE_MASK) == TT_DESCRIPTOR_PAGE_TYPE_FAULT) {
|
|
|
|
// Case: End of the boundary of the region
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
} else if ((PageDescriptor & TT_DESCRIPTOR_PAGE_TYPE_PAGE) == TT_DESCRIPTOR_PAGE_TYPE_PAGE) {
|
|
|
|
if ((PageDescriptor & TT_DESCRIPTOR_PAGE_ATTRIBUTE_MASK) == PageAttributes) {
|
|
|
|
*RegionLength = *RegionLength + TT_DESCRIPTOR_PAGE_SIZE;
|
|
|
|
} else {
|
|
|
|
// Case: End of the boundary of the region
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// We do not support Large Page yet. We return EFI_SUCCESS that means end of the region.
|
|
|
|
ASSERT(0);
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
GetMemoryRegion (
|
|
|
|
IN OUT UINTN *BaseAddress,
|
|
|
|
OUT UINTN *RegionLength,
|
|
|
|
OUT UINTN *RegionAttributes
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 TableIndex;
|
|
|
|
UINT32 PageAttributes;
|
|
|
|
UINT32 PageTableIndex;
|
|
|
|
UINT32 SectionDescriptor;
|
|
|
|
ARM_FIRST_LEVEL_DESCRIPTOR *FirstLevelTable;
|
|
|
|
UINT32 *PageTable;
|
|
|
|
|
|
|
|
// Initialize the arguments
|
|
|
|
*RegionLength = 0;
|
|
|
|
|
|
|
|
// Obtain page table base
|
|
|
|
FirstLevelTable = (ARM_FIRST_LEVEL_DESCRIPTOR *)ArmGetTTBR0BaseAddress ();
|
|
|
|
|
|
|
|
// Calculate index into first level translation table for start of modification
|
|
|
|
TableIndex = TT_DESCRIPTOR_SECTION_BASE_ADDRESS (*BaseAddress) >> TT_DESCRIPTOR_SECTION_BASE_SHIFT;
|
|
|
|
ASSERT (TableIndex < TRANSLATION_TABLE_SECTION_COUNT);
|
|
|
|
|
|
|
|
// Get the section at the given index
|
|
|
|
SectionDescriptor = FirstLevelTable[TableIndex];
|
2018-11-30 12:28:26 +01:00
|
|
|
if (!SectionDescriptor) {
|
|
|
|
return EFI_NOT_FOUND;
|
|
|
|
}
|
2013-08-19 19:38:39 +02:00
|
|
|
|
|
|
|
// If 'BaseAddress' belongs to the section then round it to the section boundary
|
|
|
|
if (((SectionDescriptor & TT_DESCRIPTOR_SECTION_TYPE_MASK) == TT_DESCRIPTOR_SECTION_TYPE_SECTION) ||
|
|
|
|
((SectionDescriptor & TT_DESCRIPTOR_SECTION_TYPE_MASK) == TT_DESCRIPTOR_SECTION_TYPE_SUPERSECTION))
|
|
|
|
{
|
|
|
|
*BaseAddress = (*BaseAddress) & TT_DESCRIPTOR_SECTION_BASE_ADDRESS_MASK;
|
|
|
|
*RegionAttributes = SectionDescriptor & TT_DESCRIPTOR_SECTION_ATTRIBUTE_MASK;
|
|
|
|
} else {
|
|
|
|
// Otherwise, we round it to the page boundary
|
|
|
|
*BaseAddress = (*BaseAddress) & TT_DESCRIPTOR_PAGE_BASE_ADDRESS_MASK;
|
|
|
|
|
|
|
|
// Get the attribute at the page table level (Level 2)
|
|
|
|
PageTable = (UINT32*)(SectionDescriptor & TT_DESCRIPTOR_SECTION_PAGETABLE_ADDRESS_MASK);
|
|
|
|
|
|
|
|
// Calculate index into first level translation table for start of modification
|
2013-09-02 15:12:17 +02:00
|
|
|
PageTableIndex = ((*BaseAddress) & TT_DESCRIPTOR_PAGE_INDEX_MASK) >> TT_DESCRIPTOR_PAGE_BASE_SHIFT;
|
2013-08-19 19:38:39 +02:00
|
|
|
ASSERT (PageTableIndex < TRANSLATION_TABLE_PAGE_COUNT);
|
|
|
|
|
|
|
|
PageAttributes = PageTable[PageTableIndex] & TT_DESCRIPTOR_PAGE_ATTRIBUTE_MASK;
|
|
|
|
*RegionAttributes = TT_DESCRIPTOR_CONVERT_TO_SECTION_CACHE_POLICY (PageAttributes, 0) |
|
|
|
|
TT_DESCRIPTOR_CONVERT_TO_SECTION_AP (PageAttributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (;TableIndex < TRANSLATION_TABLE_SECTION_COUNT; TableIndex++) {
|
|
|
|
// Get the section at the given index
|
|
|
|
SectionDescriptor = FirstLevelTable[TableIndex];
|
|
|
|
|
|
|
|
// If the entry is a level-2 page table then we scan it to find the end of the region
|
2013-09-23 11:38:53 +02:00
|
|
|
if (TT_DESCRIPTOR_SECTION_TYPE_IS_PAGE_TABLE (SectionDescriptor)) {
|
2013-08-19 19:38:39 +02:00
|
|
|
// Extract the page table location from the descriptor
|
|
|
|
PageTable = (UINT32*)(SectionDescriptor & TT_DESCRIPTOR_SECTION_PAGETABLE_ADDRESS_MASK);
|
|
|
|
|
|
|
|
// Scan the page table to find the end of the region.
|
|
|
|
Status = GetMemoryRegionPage (PageTable, BaseAddress, RegionLength, RegionAttributes);
|
|
|
|
|
|
|
|
// If we have found the end of the region (Status == EFI_SUCCESS) then we exit the for-loop
|
|
|
|
if (Status == EFI_SUCCESS) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (((SectionDescriptor & TT_DESCRIPTOR_SECTION_TYPE_MASK) == TT_DESCRIPTOR_SECTION_TYPE_SECTION) ||
|
|
|
|
((SectionDescriptor & TT_DESCRIPTOR_SECTION_TYPE_MASK) == TT_DESCRIPTOR_SECTION_TYPE_SUPERSECTION)) {
|
|
|
|
if ((SectionDescriptor & TT_DESCRIPTOR_SECTION_ATTRIBUTE_MASK) != *RegionAttributes) {
|
|
|
|
// If the attributes of the section differ from the one targeted then we exit the loop
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
*RegionLength = *RegionLength + TT_DESCRIPTOR_SECTION_SIZE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If we are on an invalid section then it means it is the end of our section.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|