This commit is contained in:
Berkus Decker 2021-03-22 01:36:01 +02:00
parent cba96c8de3
commit 644426d806
3 changed files with 93 additions and 59 deletions

View File

@ -134,6 +134,10 @@ Individual files contain the following tag instead of the full license text.
This enables machine processing of license information based on the SPDX This enables machine processing of license information based on the SPDX
License Identifiers that are here available: http://spdx.org/licenses/ License Identifiers that are here available: http://spdx.org/licenses/
@todo http://copyfree.org/support/community
-- copyfree
-- no CoC approach of that community
---- ----
For more information please re-read. For more information please re-read.

View File

@ -216,17 +216,17 @@ impl From<PageTableEntry> for u64 {
// So, first we init the dtb, parse mem-regions from there, then init boot_info page and start mmu, // So, first we init the dtb, parse mem-regions from there, then init boot_info page and start mmu,
// this part will be inited in mmu::init(): // this part will be inited in mmu::init():
// @todo do NOT keep these statically, always allocate from available bump memory // // @todo do NOT keep these statically, always allocate from available bump memory
static mut LVL2_TABLE: Table<PageDirectory> = Table::<PageDirectory> { // static mut LVL2_TABLE: Table<PageDirectory> = Table::<PageDirectory> {
entries: [0; NUM_ENTRIES_4KIB as usize], // entries: [0; NUM_ENTRIES_4KIB as usize],
level: PhantomData, // level: PhantomData,
}; // };
//
// @todo do NOT keep these statically, always allocate from available bump memory // // @todo do NOT keep these statically, always allocate from available bump memory
static mut LVL3_TABLE: Table<PageTable> = Table::<PageTable> { // static mut LVL3_TABLE: Table<PageTable> = Table::<PageTable> {
entries: [0; NUM_ENTRIES_4KIB as usize], // entries: [0; NUM_ENTRIES_4KIB as usize],
level: PhantomData, // level: PhantomData,
}; // };
trait BaseAddr { trait BaseAddr {
fn base_addr_u64(&self) -> u64; fn base_addr_u64(&self) -> u64;
@ -254,10 +254,27 @@ pub unsafe fn init() -> Result<(), &'static str> {
// Prepare the memory attribute indirection register. // Prepare the memory attribute indirection register.
mair::set_up(); mair::set_up();
// should receive in args an obtained memory map from DT
let memory_map = Regions {
start: 0x1000,
size: 0x10000,
};
// bump-allocate page tables for entire memory
// also allocate phys memory to kernel space!
//
// separate regions - regular memory, device mmaps,
// initial thread maps ALL the memory??
// instead
// init thread may map only necessary mem
// boot time only map kernel physmem space, and currently loaded kernel data
// PROBABLY only kernel mapping TTBR1 is needed, the rest is not very useful?
// take over protected memory space though anyway.
// Point the first 2 MiB of virtual addresses to the follow-up LVL3 // Point the first 2 MiB of virtual addresses to the follow-up LVL3
// page-table. // page-table.
LVL2_TABLE.entries[0] = // LVL2_TABLE.entries[0] =
PageTableEntry::new_table_descriptor(LVL3_TABLE.entries.base_addr_usize())?.into(); // PageTableEntry::new_table_descriptor(LVL3_TABLE.entries.base_addr_usize())?.into();
// Fill the rest of the LVL2 (2 MiB) entries as block descriptors. // Fill the rest of the LVL2 (2 MiB) entries as block descriptors.
// //
@ -296,51 +313,6 @@ pub unsafe fn init() -> Result<(), &'static str> {
*entry = page_desc.into(); *entry = page_desc.into();
} }
// Point to the LVL2 table base address in TTBR0.
TTBR0_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); // User (lo-)space addresses
// TTBR1_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); // Kernel (hi-)space addresses
// Configure various settings of stage 1 of the EL1 translation regime.
let ips = ID_AA64MMFR0_EL1.read(ID_AA64MMFR0_EL1::PARange);
TCR_EL1.write(
TCR_EL1::TBI0::Ignored // @todo TBI1 also set to Ignored??
+ TCR_EL1::IPS.val(ips) // Intermediate Physical Address Size
// ttbr0 user memory addresses
+ TCR_EL1::TG0::KiB_4 // 4 KiB granule
+ TCR_EL1::SH0::Inner
+ TCR_EL1::ORGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::IRGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::EPD0::EnableTTBR0Walks
+ TCR_EL1::T0SZ.val(34) // ARMv8ARM Table D5-11 minimum TxSZ for starting table level 2
// ttbr1 kernel memory addresses
+ TCR_EL1::TG1::KiB_4 // 4 KiB granule
+ TCR_EL1::SH1::Inner
+ TCR_EL1::ORGN1::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::IRGN1::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::EPD1::EnableTTBR1Walks
+ TCR_EL1::T1SZ.val(34), // ARMv8ARM Table D5-11 minimum TxSZ for starting table level 2
);
// Switch the MMU on.
//
// First, force all previous changes to be seen before the MMU is enabled.
barrier::isb(barrier::SY);
// use cortex_a::regs::RegisterReadWrite;
// Enable the MMU and turn on data and instruction caching.
SCTLR_EL1.modify(SCTLR_EL1::M::Enable + SCTLR_EL1::C::Cacheable + SCTLR_EL1::I::Cacheable);
// Force MMU init to complete before next instruction
/*
* Invalidate the local I-cache so that any instructions fetched
* speculatively from the PoC are discarded, since they may have
* been dynamically patched at the PoU.
*/
barrier::isb(barrier::SY);
Ok(())
} }
// AArch64: // AArch64:

View File

@ -12,10 +12,10 @@
#![allow(dead_code)] #![allow(dead_code)]
use crate::memory::PageSize;
use { use {
crate::memory::{ crate::memory::{
page_size::{Size1GiB, Size2MiB, Size4KiB}, page_size::{Size1GiB, Size2MiB, Size4KiB},
PageSize,
//virt_page::Page, //virt_page::Page,
PhysAddr, PhysAddr,
PhysFrame, PhysFrame,
@ -26,10 +26,68 @@ use {
ops::{Index, IndexMut}, ops::{Index, IndexMut},
ptr::Unique, ptr::Unique,
}, },
cortex_a::barrier,
register::register_bitfields, register::register_bitfields,
snafu::Snafu, snafu::Snafu,
}; };
#[derive(Debug, Snafu)]
enum MmuError {}
pub fn init() -> Result<(), MmuError> {
// Prepare the memory attribute indirection register.
mair::set_up();
// Point to the LVL2 table base address in TTBR0.
TTBR0_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); // User (lo-)space addresses
// TTBR1_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); // Kernel (hi-)space addresses
// Configure various settings of stage 1 of the EL1 translation regime.
let ips = ID_AA64MMFR0_EL1.read(ID_AA64MMFR0_EL1::PARange);
TCR_EL1.write(
TCR_EL1::TBI0::Ignored // @todo TBI1 also set to Ignored??
+ TCR_EL1::IPS.val(ips) // Intermediate Physical Address Size
// ttbr0 user memory addresses
+ TCR_EL1::TG0::KiB_4 // 4 KiB granule
+ TCR_EL1::SH0::Inner
+ TCR_EL1::ORGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::IRGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::EPD0::EnableTTBR0Walks
+ TCR_EL1::T0SZ.val(34) // ARMv8ARM Table D5-11 minimum TxSZ for starting table level 2
// ttbr1 kernel memory addresses
+ TCR_EL1::TG1::KiB_4 // 4 KiB granule
+ TCR_EL1::SH1::Inner
+ TCR_EL1::ORGN1::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::IRGN1::WriteBack_ReadAlloc_WriteAlloc_Cacheable
+ TCR_EL1::EPD1::EnableTTBR1Walks
+ TCR_EL1::T1SZ.val(34), // ARMv8ARM Table D5-11 minimum TxSZ for starting table level 2
);
// Switch the MMU on.
//
// First, force all previous changes to be seen before the MMU is enabled.
unsafe {
barrier::isb(barrier::SY);
}
// use cortex_a::regs::RegisterReadWrite;
// Enable the MMU and turn on data and instruction caching.
SCTLR_EL1.modify(SCTLR_EL1::M::Enable + SCTLR_EL1::C::Cacheable + SCTLR_EL1::I::Cacheable);
// Force MMU init to complete before next instruction
/*
* Invalidate the local I-cache so that any instructions fetched
* speculatively from the PoC are discarded, since they may have
* been dynamically patched at the PoU.
*/
unsafe {
barrier::isb(barrier::SY);
}
Ok(())
}
/* /*
* With 4k page granule, a virtual address is split into 4 lookup parts * With 4k page granule, a virtual address is split into 4 lookup parts
* spanning 9 bits each: * spanning 9 bits each: