@@ -87,25 +87,29 @@ is hardcoded here (`64 KiB` page descriptors).
87
87
The actual page tables are stored in a global instance of the `PageTables ` struct :
88
88
89
89
```rust
90
- // Two newtypes for added type safety, so that you cannot accidentally place a TableDescriptor into
91
- // a PageDescriptor slot in `struct PageTables`, and vice versa.
90
+ /// A table descriptor for 64 KiB aperture.
91
+ ///
92
+ /// The output points to the next table.
92
93
#[derive (Copy , Clone )]
93
94
#[repr (transparent )]
94
- struct RawTableDescriptor (u64 );
95
+ struct TableDescriptor (u64 );
95
96
97
+ /// A page descriptor with 64 KiB aperture.
98
+ ///
99
+ /// The output points to physical memory.
96
100
#[derive(Copy , Clone )]
97
101
#[repr(transparent)]
98
- struct RawPageDescriptor (u64 );
102
+ struct PageDescriptor (u64 );
99
103
100
104
/// Big monolithic struct for storing the page tables. Individual levels must be 64 KiB aligned,
101
105
/// hence the "reverse" order of appearance.
102
106
#[repr(C )]
103
107
#[repr(align(65536))]
104
108
struct PageTables <const N : usize > {
105
109
// Page descriptors, covering 64 KiB windows per entry.
106
- lvl3 : [[RawPageDescriptor ; 8192 ]; N ],
110
+ lvl3 : [[PageDescriptor ; 8192 ]; N ],
107
111
// Table descriptors, covering 512 MiB windows.
108
- lvl2 : [RawTableDescriptor ; N ],
112
+ lvl2 : [TableDescriptor ; N ],
109
113
}
110
114
111
115
/// Usually evaluates to 1 GiB for RPi3 and 4 GiB for RPi 4.
@@ -115,8 +119,8 @@ const ENTRIES_512_MIB: usize = bsp::addr_space_size() >> FIVETWELVE_MIB_SHIFT;
115
119
///
116
120
/// Supposed to land in `.bss`. Therefore, ensure that they boil down to all "0" entries.
117
121
static mut TABLES : PageTables <{ ENTRIES_512_MIB }> = PageTables {
118
- lvl3 : [[RawPageDescriptor (0 ); 8192 ]; ENTRIES_512_MIB ],
119
- lvl2 : [RawTableDescriptor (0 ); ENTRIES_512_MIB ],
122
+ lvl3 : [[PageDescriptor (0 ); 8192 ]; ENTRIES_512_MIB ],
123
+ lvl2 : [TableDescriptor (0 ); ENTRIES_512_MIB ],
120
124
};
121
125
```
122
126
@@ -135,11 +139,11 @@ memory and device memory (which is not cached).
135
139
fn set_up_mair () {
136
140
// Define the memory types being mapped.
137
141
MAIR_EL1 . write (
138
- // Attribute 1 - Cacheable normal DRAM
142
+ // Attribute 1 - Cacheable normal DRAM.
139
143
MAIR_EL1 :: Attr1_HIGH :: Memory_OuterWriteBack_NonTransient_ReadAlloc_WriteAlloc
140
144
+ MAIR_EL1 :: Attr1_LOW_MEMORY :: InnerWriteBack_NonTransient_ReadAlloc_WriteAlloc
141
145
142
- // Attribute 0 - Device
146
+ // Attribute 0 - Device.
143
147
+ MAIR_EL1 :: Attr0_HIGH :: Device
144
148
+ MAIR_EL1 :: Attr0_LOW_DEVICE :: Device_nGnRE ,
145
149
);
@@ -303,7 +307,7 @@ diff -uNr 10_privilege_level/src/arch/aarch64/mmu.rs 11_virtual_memory/src/arch/
303
307
+ // A level 3 page descriptor, as per AArch64 Reference Manual Figure D4-17.
304
308
+ register_bitfields! {u64,
305
309
+ STAGE1_PAGE_DESCRIPTOR [
306
- + /// Privileged execute-never
310
+ + /// Privileged execute-never.
307
311
+ PXN OFFSET(53) NUMBITS(1) [
308
312
+ False = 0,
309
313
+ True = 1
@@ -312,27 +316,27 @@ diff -uNr 10_privilege_level/src/arch/aarch64/mmu.rs 11_virtual_memory/src/arch/
312
316
+ /// Physical address of the next page table (lvl2) or the page descriptor (lvl3).
313
317
+ OUTPUT_ADDR_64KiB OFFSET(16) NUMBITS(32) [], // [47:16]
314
318
+
315
- + /// Access flag
319
+ + /// Access flag.
316
320
+ AF OFFSET(10) NUMBITS(1) [
317
321
+ False = 0,
318
322
+ True = 1
319
323
+ ],
320
324
+
321
- + /// Shareability field
325
+ + /// Shareability field.
322
326
+ SH OFFSET(8) NUMBITS(2) [
323
327
+ OuterShareable = 0b10,
324
328
+ InnerShareable = 0b11
325
329
+ ],
326
330
+
327
- + /// Access Permissions
331
+ + /// Access Permissions.
328
332
+ AP OFFSET(6) NUMBITS(2) [
329
333
+ RW_EL1 = 0b00,
330
334
+ RW_EL1_EL0 = 0b01,
331
335
+ RO_EL1 = 0b10,
332
336
+ RO_EL1_EL0 = 0b11
333
337
+ ],
334
338
+
335
- + /// Memory attributes index into the MAIR_EL1 register
339
+ + /// Memory attributes index into the MAIR_EL1 register.
336
340
+ AttrIndx OFFSET(2) NUMBITS(3) [],
337
341
+
338
342
+ TYPE OFFSET(1) NUMBITS(1) [
@@ -418,7 +422,7 @@ diff -uNr 10_privilege_level/src/arch/aarch64/mmu.rs 11_virtual_memory/src/arch/
418
422
+ for register::FieldValue<u64, STAGE1_PAGE_DESCRIPTOR::Register>
419
423
+ {
420
424
+ fn from(attribute_fields: AttributeFields) -> Self {
421
- + // Memory attributes
425
+ + // Memory attributes.
422
426
+ let mut desc = match attribute_fields.mem_attributes {
423
427
+ MemAttributes::CacheableDRAM => {
424
428
+ STAGE1_PAGE_DESCRIPTOR::SH::InnerShareable
@@ -430,13 +434,13 @@ diff -uNr 10_privilege_level/src/arch/aarch64/mmu.rs 11_virtual_memory/src/arch/
430
434
+ }
431
435
+ };
432
436
+
433
- + // Access Permissions
437
+ + // Access Permissions.
434
438
+ desc += match attribute_fields.acc_perms {
435
439
+ AccessPermissions::ReadOnly => STAGE1_PAGE_DESCRIPTOR::AP::RO_EL1,
436
440
+ AccessPermissions::ReadWrite => STAGE1_PAGE_DESCRIPTOR::AP::RW_EL1,
437
441
+ };
438
442
+
439
- + // Execute Never
443
+ + // Execute Never.
440
444
+ desc += if attribute_fields.execute_never {
441
445
+ STAGE1_PAGE_DESCRIPTOR::PXN::True
442
446
+ } else {
@@ -472,11 +476,11 @@ diff -uNr 10_privilege_level/src/arch/aarch64/mmu.rs 11_virtual_memory/src/arch/
472
476
+ fn set_up_mair() {
473
477
+ // Define the memory types being mapped.
474
478
+ MAIR_EL1.write(
475
- + // Attribute 1 - Cacheable normal DRAM
479
+ + // Attribute 1 - Cacheable normal DRAM.
476
480
+ MAIR_EL1::Attr1_HIGH::Memory_OuterWriteBack_NonTransient_ReadAlloc_WriteAlloc
477
481
+ + MAIR_EL1::Attr1_LOW_MEMORY::InnerWriteBack_NonTransient_ReadAlloc_WriteAlloc
478
482
+
479
- + // Attribute 0 - Device
483
+ + // Attribute 0 - Device.
480
484
+ + MAIR_EL1::Attr0_HIGH::Device
481
485
+ + MAIR_EL1::Attr0_LOW_DEVICE::Device_nGnRE,
482
486
+ );
@@ -560,7 +564,7 @@ diff -uNr 10_privilege_level/src/arch/aarch64/mmu.rs 11_virtual_memory/src/arch/
560
564
+ // Enable the MMU and turn on data and instruction caching.
561
565
+ SCTLR_EL1.modify(SCTLR_EL1::M::Enable + SCTLR_EL1::C::Cacheable + SCTLR_EL1::I::Cacheable);
562
566
+
563
- + // Force MMU init to complete before next instruction
567
+ + // Force MMU init to complete before next instruction.
564
568
+ barrier::isb(barrier::SY);
565
569
+
566
570
+ Ok(())
0 commit comments