blob: 71f491a2192183b86aabebd0dc8524d691781291 [file] [log] [blame]
Soby Mathew44170c42016-03-22 15:51:08 +00001/*
Antonio Nino Diazb9ae5db2018-05-02 11:23:56 +01002 * Copyright (c) 2014-2018, ARM Limited and Contributors. All rights reserved.
Soby Mathew44170c42016-03-22 15:51:08 +00003 *
dp-armfa3cf0b2017-05-03 09:38:09 +01004 * SPDX-License-Identifier: BSD-3-Clause
Soby Mathew44170c42016-03-22 15:51:08 +00005 */
6
Soby Mathew44170c42016-03-22 15:51:08 +00007#include <assert.h>
Antonio Nino Diaz4b32e622018-08-16 16:52:57 +01008#include <stdint.h>
Antonio Nino Diaze0f90632018-12-14 00:18:21 +00009
10#include <platform_def.h>
11
12#include <arch.h>
13#include <arch_helpers.h>
14#include <common/bl_common.h>
15#include <lib/utils.h>
16#include <lib/xlat_tables/xlat_tables.h>
17#include <lib/xlat_tables/xlat_tables_arch.h>
18#include <plat/common/common_def.h>
19
Soby Mathew44170c42016-03-22 15:51:08 +000020#include "../xlat_tables_private.h"
21
Sandrine Bailleux090c8492017-05-19 09:59:37 +010022#define XLAT_TABLE_LEVEL_BASE \
23 GET_XLAT_TABLE_LEVEL_BASE(PLAT_VIRT_ADDR_SPACE_SIZE)
Antonio Nino Diazd48ae612016-08-02 09:21:41 +010024
Sandrine Bailleux090c8492017-05-19 09:59:37 +010025#define NUM_BASE_LEVEL_ENTRIES \
26 GET_NUM_BASE_LEVEL_ENTRIES(PLAT_VIRT_ADDR_SPACE_SIZE)
Antonio Nino Diazd48ae612016-08-02 09:21:41 +010027
28static uint64_t base_xlation_table[NUM_BASE_LEVEL_ENTRIES]
29 __aligned(NUM_BASE_LEVEL_ENTRIES * sizeof(uint64_t));
Soby Mathew44170c42016-03-22 15:51:08 +000030
31static unsigned long long tcr_ps_bits;
32
33static unsigned long long calc_physical_addr_size_bits(
34 unsigned long long max_addr)
35{
36 /* Physical address can't exceed 48 bits */
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010037 assert((max_addr & ADDR_MASK_48_TO_63) == 0U);
Soby Mathew44170c42016-03-22 15:51:08 +000038
39 /* 48 bits address */
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010040 if ((max_addr & ADDR_MASK_44_TO_47) != 0U)
Soby Mathew44170c42016-03-22 15:51:08 +000041 return TCR_PS_BITS_256TB;
42
43 /* 44 bits address */
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010044 if ((max_addr & ADDR_MASK_42_TO_43) != 0U)
Soby Mathew44170c42016-03-22 15:51:08 +000045 return TCR_PS_BITS_16TB;
46
47 /* 42 bits address */
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010048 if ((max_addr & ADDR_MASK_40_TO_41) != 0U)
Soby Mathew44170c42016-03-22 15:51:08 +000049 return TCR_PS_BITS_4TB;
50
51 /* 40 bits address */
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010052 if ((max_addr & ADDR_MASK_36_TO_39) != 0U)
Soby Mathew44170c42016-03-22 15:51:08 +000053 return TCR_PS_BITS_1TB;
54
55 /* 36 bits address */
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010056 if ((max_addr & ADDR_MASK_32_TO_35) != 0U)
Soby Mathew44170c42016-03-22 15:51:08 +000057 return TCR_PS_BITS_64GB;
58
59 return TCR_PS_BITS_4GB;
60}
61
Antonio Nino Diaz3759e3f2017-03-22 15:48:51 +000062#if ENABLE_ASSERTIONS
Antonio Nino Diazb9ae5db2018-05-02 11:23:56 +010063/*
64 * Physical Address ranges supported in the AArch64 Memory Model. Value 0b110 is
65 * supported in ARMv8.2 onwards.
66 */
Antonio Nino Diazd1beee22016-12-13 15:28:54 +000067static const unsigned int pa_range_bits_arr[] = {
68 PARANGE_0000, PARANGE_0001, PARANGE_0010, PARANGE_0011, PARANGE_0100,
Antonio Nino Diazb9ae5db2018-05-02 11:23:56 +010069 PARANGE_0101, PARANGE_0110
Antonio Nino Diazd1beee22016-12-13 15:28:54 +000070};
71
72static unsigned long long get_max_supported_pa(void)
73{
74 u_register_t pa_range = read_id_aa64mmfr0_el1() &
75 ID_AA64MMFR0_EL1_PARANGE_MASK;
76
77 /* All other values are reserved */
78 assert(pa_range < ARRAY_SIZE(pa_range_bits_arr));
79
80 return (1ULL << pa_range_bits_arr[pa_range]) - 1ULL;
81}
Antonio Nino Diaz3759e3f2017-03-22 15:48:51 +000082#endif /* ENABLE_ASSERTIONS */
Antonio Nino Diazd1beee22016-12-13 15:28:54 +000083
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010084unsigned int xlat_arch_current_el(void)
Antonio Nino Diazefabaa92017-04-27 13:30:22 +010085{
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010086 unsigned int el = (unsigned int)GET_EL(read_CurrentEl());
Antonio Nino Diazefabaa92017-04-27 13:30:22 +010087
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010088 assert(el > 0U);
Antonio Nino Diazefabaa92017-04-27 13:30:22 +010089
90 return el;
91}
92
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010093uint64_t xlat_arch_get_xn_desc(unsigned int el)
Antonio Nino Diazefabaa92017-04-27 13:30:22 +010094{
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010095 if (el == 3U) {
Antonio Nino Diazefabaa92017-04-27 13:30:22 +010096 return UPPER_ATTRS(XN);
97 } else {
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +010098 assert(el == 1U);
Antonio Nino Diazefabaa92017-04-27 13:30:22 +010099 return UPPER_ATTRS(PXN);
100 }
101}
102
Soby Mathew44170c42016-03-22 15:51:08 +0000103void init_xlat_tables(void)
104{
105 unsigned long long max_pa;
106 uintptr_t max_va;
107 print_mmap();
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100108 init_xlation_table(0U, base_xlation_table, XLAT_TABLE_LEVEL_BASE,
Antonio Nino Diazd48ae612016-08-02 09:21:41 +0100109 &max_va, &max_pa);
Antonio Nino Diazd1beee22016-12-13 15:28:54 +0000110
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100111 assert(max_va <= (PLAT_VIRT_ADDR_SPACE_SIZE - 1U));
112 assert(max_pa <= (PLAT_PHY_ADDR_SPACE_SIZE - 1U));
113 assert((PLAT_PHY_ADDR_SPACE_SIZE - 1U) <= get_max_supported_pa());
Antonio Nino Diazd1beee22016-12-13 15:28:54 +0000114
Soby Mathew44170c42016-03-22 15:51:08 +0000115 tcr_ps_bits = calc_physical_addr_size_bits(max_pa);
Soby Mathew44170c42016-03-22 15:51:08 +0000116}
117
118/*******************************************************************************
119 * Macro generating the code for the function enabling the MMU in the given
120 * exception level, assuming that the pagetables have already been created.
121 *
122 * _el: Exception level at which the function will run
123 * _tcr_extra: Extra bits to set in the TCR register. This mask will
124 * be OR'ed with the default TCR value.
125 * _tlbi_fct: Function to invalidate the TLBs at the current
126 * exception level
127 ******************************************************************************/
128#define DEFINE_ENABLE_MMU_EL(_el, _tcr_extra, _tlbi_fct) \
129 void enable_mmu_el##_el(unsigned int flags) \
130 { \
131 uint64_t mair, tcr, ttbr; \
132 uint32_t sctlr; \
133 \
134 assert(IS_IN_EL(_el)); \
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100135 assert((read_sctlr_el##_el() & SCTLR_M_BIT) == 0U); \
Soby Mathew44170c42016-03-22 15:51:08 +0000136 \
137 /* Set attributes in the right indices of the MAIR */ \
138 mair = MAIR_ATTR_SET(ATTR_DEVICE, ATTR_DEVICE_INDEX); \
139 mair |= MAIR_ATTR_SET(ATTR_IWBWA_OWBWA_NTR, \
140 ATTR_IWBWA_OWBWA_NTR_INDEX); \
141 mair |= MAIR_ATTR_SET(ATTR_NON_CACHEABLE, \
142 ATTR_NON_CACHEABLE_INDEX); \
143 write_mair_el##_el(mair); \
144 \
145 /* Invalidate TLBs at the current exception level */ \
146 _tlbi_fct(); \
147 \
148 /* Set TCR bits as well. */ \
Antonio Nino Diazd48ae612016-08-02 09:21:41 +0100149 /* Set T0SZ to (64 - width of virtual address space) */ \
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100150 int t0sz = 64 - __builtin_ctzll(PLAT_VIRT_ADDR_SPACE_SIZE);\
151 \
152 if ((flags & XLAT_TABLE_NC) != 0U) { \
Summer Qindaf5dbb2017-03-16 17:16:34 +0000153 /* Inner & outer non-cacheable non-shareable. */\
154 tcr = TCR_SH_NON_SHAREABLE | \
155 TCR_RGN_OUTER_NC | TCR_RGN_INNER_NC | \
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100156 (uint64_t) t0sz; \
Summer Qindaf5dbb2017-03-16 17:16:34 +0000157 } else { \
158 /* Inner & outer WBWA & shareable. */ \
159 tcr = TCR_SH_INNER_SHAREABLE | \
160 TCR_RGN_OUTER_WBA | TCR_RGN_INNER_WBA | \
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100161 (uint64_t) t0sz; \
Summer Qindaf5dbb2017-03-16 17:16:34 +0000162 } \
Soby Mathew44170c42016-03-22 15:51:08 +0000163 tcr |= _tcr_extra; \
164 write_tcr_el##_el(tcr); \
165 \
166 /* Set TTBR bits as well */ \
Antonio Nino Diazd48ae612016-08-02 09:21:41 +0100167 ttbr = (uint64_t) base_xlation_table; \
Soby Mathew44170c42016-03-22 15:51:08 +0000168 write_ttbr0_el##_el(ttbr); \
169 \
170 /* Ensure all translation table writes have drained */ \
171 /* into memory, the TLB invalidation is complete, */ \
172 /* and translation register writes are committed */ \
173 /* before enabling the MMU */ \
Antonio Nino Diaz3f13c352017-02-24 11:39:22 +0000174 dsbish(); \
Soby Mathew44170c42016-03-22 15:51:08 +0000175 isb(); \
176 \
177 sctlr = read_sctlr_el##_el(); \
178 sctlr |= SCTLR_WXN_BIT | SCTLR_M_BIT; \
179 \
Antonio Nino Diaz50eb3742018-07-24 10:20:53 +0100180 if ((flags & DISABLE_DCACHE) != 0U) \
Soby Mathew44170c42016-03-22 15:51:08 +0000181 sctlr &= ~SCTLR_C_BIT; \
182 else \
183 sctlr |= SCTLR_C_BIT; \
184 \
185 write_sctlr_el##_el(sctlr); \
186 \
187 /* Ensure the MMU enable takes effect immediately */ \
188 isb(); \
Jeenu Viswambharan9f142612018-04-27 15:06:57 +0100189 } \
190 \
191 void enable_mmu_direct_el##_el(unsigned int flags) \
192 { \
193 enable_mmu_el##_el(flags); \
Soby Mathew44170c42016-03-22 15:51:08 +0000194 }
195
196/* Define EL1 and EL3 variants of the function enabling the MMU */
197DEFINE_ENABLE_MMU_EL(1,
Antonio Nino Diazc8274a82017-09-15 10:30:34 +0100198 /*
199 * TCR_EL1.EPD1: Disable translation table walk for addresses
200 * that are translated using TTBR1_EL1.
201 */
202 TCR_EPD1_BIT | (tcr_ps_bits << TCR_EL1_IPS_SHIFT),
Soby Mathew44170c42016-03-22 15:51:08 +0000203 tlbivmalle1)
204DEFINE_ENABLE_MMU_EL(3,
205 TCR_EL3_RES1 | (tcr_ps_bits << TCR_EL3_PS_SHIFT),
206 tlbialle3)