genode/repos/base-hw/src/core/spec/arm_v8/cpu.cc

146 lines
3.6 KiB
C++

/*
* \brief ARMv8 cpu context initialization
* \author Stefan Kalkowski
* \date 2017-04-12
*/
/*
* Copyright (C) 2017 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU Affero General Public License version 3.
*/
#include <board.h>
#include <cpu.h>
#include <kernel/thread.h>
#include <cpu/memory_barrier.h>
#include <util/bit_allocator.h>
Genode::Cpu::Context::Context(bool privileged)
{
Spsr::El::set(pstate, privileged ? 1 : 0);
}
void Genode::Cpu::switch_to(Context&, Mmu_context & mmu_context)
{
if (mmu_context.id() == 0) return;
if (mmu_context.id() != Ttbr::Asid::get(Ttbr0_el1::read()))
Ttbr0_el1::write(mmu_context.ttbr);
}
void Genode::Cpu::mmu_fault(Genode::Cpu::Context &,
Kernel::Thread_fault & fault)
{
Esr::access_t esr = Esr_el1::read();
fault.addr = Far_el1::read();
switch (Esr::Iss::Abort::Fsc::get(Esr::Iss::get(esr))) {
case Esr::Iss::Abort::Fsc::TRANSLATION:
fault.type = Kernel::Thread_fault::PAGE_MISSING;
return;
case Esr::Iss::Abort::Fsc::PERMISSION:
fault.type = Esr::Iss::Abort::Write::get(Esr::Iss::get(esr))
? Kernel::Thread_fault::WRITE : Kernel::Thread_fault::EXEC;
return;
default:
Genode::raw("MMU-fault not handled ESR=", Genode::Hex(esr));
fault.type = Kernel::Thread_fault::UNKNOWN;
};
}
using Asid_allocator = Genode::Bit_allocator<65536>;
static Asid_allocator &alloc() {
return *unmanaged_singleton<Asid_allocator>(); }
Genode::Cpu::Mmu_context::Mmu_context(addr_t table)
: ttbr(Ttbr::Baddr::masked(table))
{
Ttbr::Asid::set(ttbr, (Genode::uint16_t)alloc().alloc());
}
Genode::Cpu::Mmu_context::~Mmu_context()
{
alloc().free(id());
}
Genode::size_t Genode::Cpu::cache_line_size()
{
static Genode::size_t cache_line_size = 0;
if (!cache_line_size) {
Genode::size_t i = 1 << Ctr_el0::I_min_line::get(Ctr_el0::read());
Genode::size_t d = 1 << Ctr_el0::D_min_line::get(Ctr_el0::read());
cache_line_size = Genode::min(i,d) * 4; /* word size is fixed in ARM */
}
return cache_line_size;
}
template <typename FUNC>
static inline void cache_maintainance(Genode::addr_t const base,
Genode::size_t const size,
FUNC & func)
{
Genode::addr_t start = (Genode::addr_t) base;
Genode::addr_t const end = base + size;
for (; start < end; start += Genode::Cpu::cache_line_size()) func(start);
}
void Genode::Cpu::cache_coherent_region(addr_t const base,
size_t const size)
{
Genode::memory_barrier();
auto lambda = [] (addr_t const base) {
asm volatile("dc cvau, %0" :: "r" (base));
asm volatile("dsb ish");
asm volatile("ic ivau, %0" :: "r" (base));
asm volatile("dsb ish");
asm volatile("isb");
};
cache_maintainance(base, size, lambda);
}
void Genode::Cpu::clear_memory_region(addr_t const addr,
size_t const size,
bool changed_cache_properties)
{
Genode::memory_barrier();
/* normal memory is cleared by D-cache zeroing */
auto normal = [] (addr_t const base) {
asm volatile("dc zva, %0" :: "r" (base));
asm volatile("ic ivau, %0" :: "r" (base));
};
/* DMA memory gets additionally evicted from the D-cache */
auto dma = [] (addr_t const base) {
asm volatile("dc zva, %0" :: "r" (base));
asm volatile("dc civac, %0" :: "r" (base));
asm volatile("ic ivau, %0" :: "r" (base));
};
if (changed_cache_properties) {
cache_maintainance(addr, size, dma);
} else {
cache_maintainance(addr, size, normal);
}
asm volatile("dsb ish");
asm volatile("isb");
}