mirror of
https://github.com/ARM-software/arm-trusted-firmware.git
synced 2025-04-17 01:54:22 +00:00

At present, the function provided by the translation library to enable MMU constructs appropriate values for translation library, and programs them to the right registers. The construction of initial values, however, is only required once as both the primary and secondaries program the same values. Additionally, the MMU-enabling function is written in C, which means there's an active stack at the time of enabling MMU. On some systems, like Arm DynamIQ, having active stack while enabling MMU during warm boot might lead to coherency problems. This patch addresses both the above problems by: - Splitting the MMU-enabling function into two: one that sets up values to be programmed into the registers, and another one that takes the pre-computed values and writes to the appropriate registers. With this, the primary effectively calls both functions to have the MMU enabled, but secondaries only need to call the latter. - Rewriting the function that enables MMU in assembly so that it doesn't use stack. This patch fixes a bunch of MISRA issues on the way. Change-Id: I0faca97263a970ffe765f0e731a1417e43fbfc45 Signed-off-by: Jeenu Viswambharan <jeenu.viswambharan@arm.com>
189 lines
5.1 KiB
C
189 lines
5.1 KiB
C
/*
|
|
* Copyright (c) 2017-2018, ARM Limited and Contributors. All rights reserved.
|
|
*
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*/
|
|
|
|
#include <arch.h>
|
|
#include <arch_helpers.h>
|
|
#include <assert.h>
|
|
#include <cassert.h>
|
|
#include <platform_def.h>
|
|
#include <utils.h>
|
|
#include <utils_def.h>
|
|
#include <xlat_tables_v2.h>
|
|
#include "../xlat_tables_private.h"
|
|
|
|
#if ARM_ARCH_MAJOR == 7 && !defined(ARMV7_SUPPORTS_LARGE_PAGE_ADDRESSING)
|
|
#error ARMv7 target does not support LPAE MMU descriptors
|
|
#endif
|
|
|
|
uint32_t mmu_cfg_params[MMU_CFG_PARAM_MAX];
|
|
|
|
/*
|
|
* Returns 1 if the provided granule size is supported, 0 otherwise.
|
|
*/
|
|
int xlat_arch_is_granule_size_supported(size_t size)
|
|
{
|
|
/*
|
|
* The Trusted Firmware uses long descriptor translation table format,
|
|
* which supports 4 KiB pages only.
|
|
*/
|
|
return (size == (4U * 1024U));
|
|
}
|
|
|
|
size_t xlat_arch_get_max_supported_granule_size(void)
|
|
{
|
|
return 4U * 1024U;
|
|
}
|
|
|
|
#if ENABLE_ASSERTIONS
|
|
unsigned long long xlat_arch_get_max_supported_pa(void)
|
|
{
|
|
/* Physical address space size for long descriptor format. */
|
|
return (1ULL << 40) - 1ULL;
|
|
}
|
|
#endif /* ENABLE_ASSERTIONS*/
|
|
|
|
int is_mmu_enabled_ctx(const xlat_ctx_t *ctx __unused)
|
|
{
|
|
return (read_sctlr() & SCTLR_M_BIT) != 0;
|
|
}
|
|
|
|
void xlat_arch_tlbi_va(uintptr_t va)
|
|
{
|
|
/*
|
|
* Ensure the translation table write has drained into memory before
|
|
* invalidating the TLB entry.
|
|
*/
|
|
dsbishst();
|
|
|
|
tlbimvaais(TLBI_ADDR(va));
|
|
}
|
|
|
|
void xlat_arch_tlbi_va_regime(uintptr_t va, xlat_regime_t xlat_regime __unused)
|
|
{
|
|
/*
|
|
* Ensure the translation table write has drained into memory before
|
|
* invalidating the TLB entry.
|
|
*/
|
|
dsbishst();
|
|
|
|
tlbimvaais(TLBI_ADDR(va));
|
|
}
|
|
|
|
void xlat_arch_tlbi_va_sync(void)
|
|
{
|
|
/* Invalidate all entries from branch predictors. */
|
|
bpiallis();
|
|
|
|
/*
|
|
* A TLB maintenance instruction can complete at any time after
|
|
* it is issued, but is only guaranteed to be complete after the
|
|
* execution of DSB by the PE that executed the TLB maintenance
|
|
* instruction. After the TLB invalidate instruction is
|
|
* complete, no new memory accesses using the invalidated TLB
|
|
* entries will be observed by any observer of the system
|
|
* domain. See section D4.8.2 of the ARMv8 (issue k), paragraph
|
|
* "Ordering and completion of TLB maintenance instructions".
|
|
*/
|
|
dsbish();
|
|
|
|
/*
|
|
* The effects of a completed TLB maintenance instruction are
|
|
* only guaranteed to be visible on the PE that executed the
|
|
* instruction after the execution of an ISB instruction by the
|
|
* PE that executed the TLB maintenance instruction.
|
|
*/
|
|
isb();
|
|
}
|
|
|
|
int xlat_arch_current_el(void)
|
|
{
|
|
/*
|
|
* If EL3 is in AArch32 mode, all secure PL1 modes (Monitor, System,
|
|
* SVC, Abort, UND, IRQ and FIQ modes) execute at EL3.
|
|
*/
|
|
return 3;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Function for enabling the MMU in Secure PL1, assuming that the page tables
|
|
* have already been created.
|
|
******************************************************************************/
|
|
void setup_mmu_cfg(unsigned int flags,
|
|
const uint64_t *base_table,
|
|
unsigned long long max_pa,
|
|
uintptr_t max_va)
|
|
{
|
|
u_register_t mair0, ttbcr;
|
|
uint64_t ttbr0;
|
|
|
|
assert(IS_IN_SECURE());
|
|
|
|
/* Set attributes in the right indices of the MAIR */
|
|
mair0 = MAIR0_ATTR_SET(ATTR_DEVICE, ATTR_DEVICE_INDEX);
|
|
mair0 |= MAIR0_ATTR_SET(ATTR_IWBWA_OWBWA_NTR,
|
|
ATTR_IWBWA_OWBWA_NTR_INDEX);
|
|
mair0 |= MAIR0_ATTR_SET(ATTR_NON_CACHEABLE,
|
|
ATTR_NON_CACHEABLE_INDEX);
|
|
|
|
/*
|
|
* Configure the control register for stage 1 of the PL1&0 translation
|
|
* regime.
|
|
*/
|
|
|
|
/* Use the Long-descriptor translation table format. */
|
|
ttbcr = TTBCR_EAE_BIT;
|
|
|
|
/*
|
|
* Disable translation table walk for addresses that are translated
|
|
* using TTBR1. Therefore, only TTBR0 is used.
|
|
*/
|
|
ttbcr |= TTBCR_EPD1_BIT;
|
|
|
|
/*
|
|
* Limit the input address ranges and memory region sizes translated
|
|
* using TTBR0 to the given virtual address space size, if smaller than
|
|
* 32 bits.
|
|
*/
|
|
if (max_va != UINT32_MAX) {
|
|
uintptr_t virtual_addr_space_size = max_va + 1;
|
|
assert(CHECK_VIRT_ADDR_SPACE_SIZE(virtual_addr_space_size));
|
|
/*
|
|
* __builtin_ctzll(0) is undefined but here we are guaranteed
|
|
* that virtual_addr_space_size is in the range [1, UINT32_MAX].
|
|
*/
|
|
ttbcr |= 32 - __builtin_ctzll(virtual_addr_space_size);
|
|
}
|
|
|
|
/*
|
|
* Set the cacheability and shareability attributes for memory
|
|
* associated with translation table walks using TTBR0.
|
|
*/
|
|
if (flags & XLAT_TABLE_NC) {
|
|
/* Inner & outer non-cacheable non-shareable. */
|
|
ttbcr |= TTBCR_SH0_NON_SHAREABLE | TTBCR_RGN0_OUTER_NC |
|
|
TTBCR_RGN0_INNER_NC;
|
|
} else {
|
|
/* Inner & outer WBWA & shareable. */
|
|
ttbcr |= TTBCR_SH0_INNER_SHAREABLE | TTBCR_RGN0_OUTER_WBA |
|
|
TTBCR_RGN0_INNER_WBA;
|
|
}
|
|
|
|
/* Set TTBR0 bits as well */
|
|
ttbr0 = (uint64_t)(uintptr_t) base_table;
|
|
#if ARM_ARCH_AT_LEAST(8, 2)
|
|
/*
|
|
* Enable CnP bit so as to share page tables with all PEs.
|
|
* Mandatory for ARMv8.2 implementations.
|
|
*/
|
|
ttbr0 |= TTBR_CNP_BIT;
|
|
#endif
|
|
|
|
/* Now populate MMU configuration */
|
|
mmu_cfg_params[MMU_CFG_MAIR0] = mair0;
|
|
mmu_cfg_params[MMU_CFG_TCR] = ttbcr;
|
|
mmu_cfg_params[MMU_CFG_TTBR0_LO] = (uint32_t) ttbr0;
|
|
mmu_cfg_params[MMU_CFG_TTBR0_HI] = ttbr0 >> 32;
|
|
}
|