1 /*
2 * Copyright (c) 2024 BayLibre, SAS
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/kernel.h>
8 #include <zephyr/ztest.h>
9 #include <kernel_arch_interface.h>
10
11 /*
12 * Virtual and physical addresses used to exercize MMU page table recycling.
13 * Those are completely arbitrary addresses away from any existing addresses
14 * (no worry, the test will fail otherwise). Those addresses don't have
15 * to be valid as we won't attempt any access to the mapped memory.
16 */
17 #define TEST_VIRT_ADDR 0x456560000
18 #define TEST_PHYS_ADDR 0x123230000
19
20 /* special test hooks in arch/arm64/core/mmu.c for test purpose */
21 extern int arm64_mmu_nb_free_tables(void);
22 extern int arm64_mmu_tables_total_usage(void);
23
24 /* initial states to compare against */
25 static int initial_nb_free_tables;
26 static int initial_tables_usage;
27
arm64_mmu_test_init(void)28 static void *arm64_mmu_test_init(void)
29 {
30 /* get initial states */
31 initial_nb_free_tables = arm64_mmu_nb_free_tables();
32 initial_tables_usage = arm64_mmu_tables_total_usage();
33
34 TC_PRINT(" Total page tables: %d\n", CONFIG_MAX_XLAT_TABLES);
35 TC_PRINT(" Initial free tables: %d\n", initial_nb_free_tables);
36 TC_PRINT(" Initial total table usage: %#x\n", initial_tables_usage);
37
38 zassert_true(initial_nb_free_tables > 1,
39 "initial_nb_free_tables = %d", initial_nb_free_tables);
40 zassert_true(initial_tables_usage > 1,
41 "initial_tables_usage = %d", initial_tables_usage);
42
43 return NULL;
44 }
45
mem_map_test(uintptr_t virt_addr,uintptr_t phys_addr,size_t size)46 static int mem_map_test(uintptr_t virt_addr, uintptr_t phys_addr, size_t size)
47 {
48 /*
49 * This is not defined to return any error but the implementation
50 * will call k_panic() if an error occurs.
51 */
52 arch_mem_map((void *)virt_addr, phys_addr, size, K_MEM_ARM_NORMAL_NC);
53
54 int mapped_nb_free_tables = arm64_mmu_nb_free_tables();
55 int mapped_tables_usage = arm64_mmu_tables_total_usage();
56
57 TC_PRINT(" After arch_mem_map:\n");
58 TC_PRINT(" current free tables: %d\n", mapped_nb_free_tables);
59 TC_PRINT(" current total table usage: %#x\n", mapped_tables_usage);
60
61 zassert_true(mapped_nb_free_tables < initial_nb_free_tables,
62 "%d vs %d", mapped_nb_free_tables, initial_nb_free_tables);
63 zassert_true(mapped_tables_usage > initial_tables_usage,
64 "%#x vs %#x", mapped_tables_usage, initial_tables_usage);
65
66 arch_mem_unmap((void *)virt_addr, size);
67
68 int unmapped_nb_free_tables = arm64_mmu_nb_free_tables();
69 int unmapped_tables_usage = arm64_mmu_tables_total_usage();
70
71 TC_PRINT(" After arch_mem_unmap:\n");
72 TC_PRINT(" current free tables: %d\n", unmapped_nb_free_tables);
73 TC_PRINT(" current total table usage: %#x\n", unmapped_tables_usage);
74
75 zassert_true(unmapped_nb_free_tables == initial_nb_free_tables,
76 "%d vs %d", unmapped_nb_free_tables, initial_nb_free_tables);
77 zassert_true(unmapped_tables_usage == initial_tables_usage,
78 "%#x vs %#x", unmapped_tables_usage, initial_tables_usage);
79
80 int tables_used = unmapped_nb_free_tables - mapped_nb_free_tables;
81 return tables_used;
82 }
83
ZTEST(arm64_mmu,test_arm64_mmu_01_single_page)84 ZTEST(arm64_mmu, test_arm64_mmu_01_single_page)
85 {
86 /*
87 * Let's map a single page to start with. This will allocate
88 * multiple tables to reach the deepest level.
89 */
90 uintptr_t virt = TEST_VIRT_ADDR;
91 uintptr_t phys = TEST_PHYS_ADDR;
92 size_t size = CONFIG_MMU_PAGE_SIZE;
93
94 int tables_used = mem_map_test(virt, phys, size);
95
96 zassert_true(tables_used == 2, "used %d tables", tables_used);
97 }
98
ZTEST(arm64_mmu,test_arm64_mmu_02_single_block)99 ZTEST(arm64_mmu, test_arm64_mmu_02_single_block)
100 {
101 /*
102 * Same thing as above, except that we expect a block mapping
103 * this time. Both addresses and the size must be properly aligned.
104 * Table allocation won't go as deep as for a page.
105 */
106 int table_entries = CONFIG_MMU_PAGE_SIZE / sizeof(uint64_t);
107 size_t block_size = table_entries * CONFIG_MMU_PAGE_SIZE;
108 uintptr_t virt = TEST_VIRT_ADDR & ~(block_size - 1);
109 uintptr_t phys = TEST_PHYS_ADDR & ~(block_size - 1);
110
111 int tables_used = mem_map_test(virt, phys, block_size);
112
113 zassert_true(tables_used == 1, "used %d tables", tables_used);
114 }
115
ZTEST(arm64_mmu,test_arm64_mmu_03_block_and_page)116 ZTEST(arm64_mmu, test_arm64_mmu_03_block_and_page)
117 {
118 /*
119 * Same thing as above, except that we expect a block mapping
120 * followed by a page mapping to exercize range splitting.
121 * To achieve that we simply increase the size by one page and keep
122 * starting addresses aligned to a block.
123 */
124 int table_entries = CONFIG_MMU_PAGE_SIZE / sizeof(uint64_t);
125 size_t block_size = table_entries * CONFIG_MMU_PAGE_SIZE;
126 uintptr_t virt = TEST_VIRT_ADDR & ~(block_size - 1);
127 uintptr_t phys = TEST_PHYS_ADDR & ~(block_size - 1);
128 size_t size = block_size + CONFIG_MMU_PAGE_SIZE;
129
130 int tables_used = mem_map_test(virt, phys, size);
131
132 zassert_true(tables_used == 2, "used %d tables", tables_used);
133 }
134
ZTEST(arm64_mmu,test_arm64_mmu_04_page_and_block)135 ZTEST(arm64_mmu, test_arm64_mmu_04_page_and_block)
136 {
137 /*
138 * Same thing as above, except that we expect a page mapping
139 * followed by a block mapping to exercize range splitting.
140 * To achieve that we increase the size by one page and decrease
141 * starting addresses by one page below block alignment.
142 */
143 int table_entries = CONFIG_MMU_PAGE_SIZE / sizeof(uint64_t);
144 size_t block_size = table_entries * CONFIG_MMU_PAGE_SIZE;
145 uintptr_t virt = (TEST_VIRT_ADDR & ~(block_size - 1)) - CONFIG_MMU_PAGE_SIZE;
146 uintptr_t phys = (TEST_PHYS_ADDR & ~(block_size - 1)) - CONFIG_MMU_PAGE_SIZE;
147 size_t size = block_size + CONFIG_MMU_PAGE_SIZE;
148
149 int tables_used = mem_map_test(virt, phys, size);
150
151 zassert_true(tables_used == 2, "used %d tables", tables_used);
152 }
153
ZTEST(arm64_mmu,test_arm64_mmu_05_hole_in_block)154 ZTEST(arm64_mmu, test_arm64_mmu_05_hole_in_block)
155 {
156 /*
157 * Create a block mapping.
158 */
159 int table_entries = CONFIG_MMU_PAGE_SIZE / sizeof(uint64_t);
160 size_t block_size = table_entries * CONFIG_MMU_PAGE_SIZE;
161 uintptr_t virt = TEST_VIRT_ADDR & ~(block_size - 1);
162 uintptr_t phys = TEST_PHYS_ADDR & ~(block_size - 1);
163 size_t size = block_size;
164
165 arch_mem_map((void *)virt, phys, size, K_MEM_ARM_NORMAL_NC);
166
167 int mapped_nb_free_tables = arm64_mmu_nb_free_tables();
168 int mapped_tables_usage = arm64_mmu_tables_total_usage();
169
170 TC_PRINT(" After arch_mem_map:\n");
171 TC_PRINT(" current free tables: %d\n", mapped_nb_free_tables);
172 TC_PRINT(" current total table usage: %#x\n", mapped_tables_usage);
173
174 zassert_true(mapped_nb_free_tables < initial_nb_free_tables,
175 "%d vs %d", mapped_nb_free_tables, initial_nb_free_tables);
176 zassert_true(mapped_tables_usage > initial_tables_usage,
177 "%#x vs %#x", mapped_tables_usage, initial_tables_usage);
178
179 /*
180 * Now poke a hole in the middle of this block mapping, effectively
181 * splitting it in two disjoint sub-mappings.
182 */
183 arch_mem_unmap((void *)(virt + CONFIG_MMU_PAGE_SIZE), CONFIG_MMU_PAGE_SIZE);
184
185 int split_nb_free_tables = arm64_mmu_nb_free_tables();
186 int split_tables_usage = arm64_mmu_tables_total_usage();
187
188 TC_PRINT(" After first arch_mem_unmap:\n");
189 TC_PRINT(" current free tables: %d\n", split_nb_free_tables);
190 TC_PRINT(" current total table usage: %#x\n", split_tables_usage);
191
192 zassert_true(split_nb_free_tables < mapped_nb_free_tables,
193 "%d vs %d", split_nb_free_tables, mapped_nb_free_tables);
194 zassert_true(split_tables_usage > mapped_tables_usage,
195 "%#x vs %#x", split_tables_usage, mapped_tables_usage);
196
197 /*
198 * Make sure the hole is actually freed.
199 */
200 zassert_true(arch_page_phys_get((void *)(virt + CONFIG_MMU_PAGE_SIZE), NULL)
201 == -EFAULT, "");
202
203 /*
204 * Then free it all.
205 */
206 arch_mem_unmap((void *)virt, size);
207
208 int unmapped_nb_free_tables = arm64_mmu_nb_free_tables();
209 int unmapped_tables_usage = arm64_mmu_tables_total_usage();
210
211 TC_PRINT(" After second arch_mem_unmap:\n");
212 TC_PRINT(" current free tables: %d\n", unmapped_nb_free_tables);
213 TC_PRINT(" current total table usage: %#x\n", unmapped_tables_usage);
214
215 zassert_true(unmapped_nb_free_tables == initial_nb_free_tables,
216 "%d vs %d", unmapped_nb_free_tables, initial_nb_free_tables);
217 zassert_true(unmapped_tables_usage == initial_tables_usage,
218 "%#x vs %#x", unmapped_tables_usage, initial_tables_usage);
219 }
220
221 ZTEST_SUITE(arm64_mmu, NULL, arm64_mmu_test_init, NULL, NULL, NULL);
222