blob: e4a9f84d334ac3c99ec7a0f6cb2be8d3e945d820 [file] [log] [blame]
/*
* Copyright (c) 2020 Intel Corporation.
*
* SPDX-License-Identifier: Apache-2.0
*/
#ifndef KERNEL_INCLUDE_MMU_H
#define KERNEL_INCLUDE_MMU_H
#ifdef CONFIG_MMU
#include <stdint.h>
#include <zephyr/sys/sflist.h>
#include <zephyr/sys/__assert.h>
#include <zephyr/sys/util.h>
#include <zephyr/kernel/mm.h>
#include <zephyr/linker/linker-defs.h>
/** Start address of physical memory. */
#define K_MEM_PHYS_RAM_START ((uintptr_t)CONFIG_SRAM_BASE_ADDRESS)
/** Size of physical memory. */
#define K_MEM_PHYS_RAM_SIZE (KB(CONFIG_SRAM_SIZE))
/** End address (exclusive) of physical memory. */
#define K_MEM_PHYS_RAM_END (K_MEM_PHYS_RAM_START + K_MEM_PHYS_RAM_SIZE)
/** Start address of virtual memory. */
#define K_MEM_VIRT_RAM_START ((uint8_t *)CONFIG_KERNEL_VM_BASE)
/** Size of virtual memory. */
#define K_MEM_VIRT_RAM_SIZE ((size_t)CONFIG_KERNEL_VM_SIZE)
/** End address (exclusive) of virtual memory. */
#define K_MEM_VIRT_RAM_END (K_MEM_VIRT_RAM_START + K_MEM_VIRT_RAM_SIZE)
/** Boot-time virtual start address of the kernel image. */
#define K_MEM_KERNEL_VIRT_START ((uint8_t *)&z_mapped_start[0])
/** Boot-time virtual end address of the kernel image. */
#define K_MEM_KERNEL_VIRT_END ((uint8_t *)&z_mapped_end[0])
/** Boot-time virtual address space size of the kernel image. */
#define K_MEM_KERNEL_VIRT_SIZE (K_MEM_KERNEL_VIRT_END - K_MEM_KERNEL_VIRT_START)
/**
* @brief Offset for translating between static physical and virtual addresses.
*
* @note Do not use directly unless you know exactly what you are going.
*/
#define K_MEM_VM_OFFSET \
((CONFIG_KERNEL_VM_BASE + CONFIG_KERNEL_VM_OFFSET) - \
(CONFIG_SRAM_BASE_ADDRESS + CONFIG_SRAM_OFFSET))
/**
* @brief Get physical address from virtual address for boot RAM mappings.
*
* @note Only applies to boot RAM mappings within the Zephyr image that have never
* been remapped or paged out. Never use this unless you know exactly what you
* are doing.
*
* @param virt Virtual address.
*
* @return Physical address.
*/
#define K_MEM_BOOT_VIRT_TO_PHYS(virt) ((uintptr_t)(((uint8_t *)(virt)) - K_MEM_VM_OFFSET))
/**
* @brief Get virtual address from physical address for boot RAM mappings.
*
* @note Only applies to boot RAM mappings within the Zephyr image that have never
* been remapped or paged out. Never use this unless you know exactly what you
* are doing.
*
* @param phys Physical address.
*
* @return Virtual address.
*/
#define K_MEM_BOOT_PHYS_TO_VIRT(phys) ((uint8_t *)(((uintptr_t)(phys)) + K_MEM_VM_OFFSET))
/**
* @def K_MEM_VM_FREE_START
* @brief Start address of unused, available virtual addresses.
*
* This is the start address of the virtual memory region where
* addresses can be allocated for memory mapping. This depends on whether
* CONFIG_ARCH_MAPS_ALL_RAM is enabled:
*
* - If it is enabled, which means all physical memory are mapped in virtual
* memory address space, and it is the same as
* (CONFIG_SRAM_BASE_ADDRESS + CONFIG_SRAM_SIZE).
*
* - If it is disabled, K_MEM_VM_FREE_START is the same K_MEM_KERNEL_VIRT_END which
* is the end of the kernel image.
*
*/
#ifdef CONFIG_ARCH_MAPS_ALL_RAM
#define K_MEM_VM_FREE_START K_MEM_BOOT_PHYS_TO_VIRT(K_MEM_PHYS_RAM_END)
#else
#define K_MEM_VM_FREE_START K_MEM_KERNEL_VIRT_END
#endif /* CONFIG_ARCH_MAPS_ALL_RAM */
/**
* @defgroup kernel_mm_page_frame_apis Kernel Memory Page Frame Management APIs
* @ingroup kernel_mm_internal_apis
* @{
*
* Macros and data structures for physical page frame accounting,
* APIs for use by eviction and backing store algorithms. This code
* is otherwise not application-facing.
*/
/**
* @brief Number of page frames.
*
* At present, page frame management is only done for main system RAM,
* and we generate paging structures based on CONFIG_SRAM_BASE_ADDRESS
* and CONFIG_SRAM_SIZE.
*
* If we have other RAM regions (DCCM, etc) these typically have special
* properties and shouldn't be used generically for demand paging or
* anonymous mappings. We don't currently maintain an ontology of these in the
* core kernel.
*/
#define K_MEM_NUM_PAGE_FRAMES (K_MEM_PHYS_RAM_SIZE / (size_t)CONFIG_MMU_PAGE_SIZE)
/*
* k_mem_page_frame flags bits
*
* Requirements:
* - K_MEM_PAGE_FRAME_FREE must be one of the possible sfnode flag bits
* - All bit values must be lower than CONFIG_MMU_PAGE_SIZE
*/
/** This physical page is free and part of the free list */
#define K_MEM_PAGE_FRAME_FREE BIT(0)
/** This physical page is reserved by hardware; we will never use it */
#define K_MEM_PAGE_FRAME_RESERVED BIT(1)
/** This page contains critical kernel data and will never be swapped */
#define K_MEM_PAGE_FRAME_PINNED BIT(2)
/**
* This physical page is mapped to some virtual memory address
*
* Currently, we just support one mapping per page frame. If a page frame
* is mapped to multiple virtual pages then it must be pinned.
*/
#define K_MEM_PAGE_FRAME_MAPPED BIT(3)
/**
* This page frame is currently involved in a page-in/out operation
*/
#define K_MEM_PAGE_FRAME_BUSY BIT(4)
/**
* This page frame has a clean copy in the backing store
*/
#define K_MEM_PAGE_FRAME_BACKED BIT(5)
/**
* Data structure for physical page frames
*
* An array of these is instantiated, one element per physical RAM page.
* Hence it's necessary to constrain its size as much as possible.
*/
struct k_mem_page_frame {
union {
/*
* If mapped, K_MEM_PAGE_FRAME_* flags and virtual address
* this page is mapped to.
*/
uintptr_t va_and_flags;
/*
* If unmapped and available, free pages list membership
* with the K_MEM_PAGE_FRAME_FREE flag.
*/
sys_sfnode_t node;
};
/* Backing store and eviction algorithms may both need to
* require additional per-frame custom data for accounting purposes.
* They should declare their own array with indices matching
* k_mem_page_frames[] ones whenever possible.
* They may also want additional flags bits that could be stored here
* and they shouldn't clobber each other. At all costs the total
* size of struct k_mem_page_frame must be minimized.
*/
};
/* Note: this must be false for the other flag bits to be valid */
static inline bool k_mem_page_frame_is_free(struct k_mem_page_frame *pf)
{
return (pf->va_and_flags & K_MEM_PAGE_FRAME_FREE) != 0U;
}
static inline bool k_mem_page_frame_is_pinned(struct k_mem_page_frame *pf)
{
return (pf->va_and_flags & K_MEM_PAGE_FRAME_PINNED) != 0U;
}
static inline bool k_mem_page_frame_is_reserved(struct k_mem_page_frame *pf)
{
return (pf->va_and_flags & K_MEM_PAGE_FRAME_RESERVED) != 0U;
}
static inline bool k_mem_page_frame_is_mapped(struct k_mem_page_frame *pf)
{
return (pf->va_and_flags & K_MEM_PAGE_FRAME_MAPPED) != 0U;
}
static inline bool k_mem_page_frame_is_busy(struct k_mem_page_frame *pf)
{
return (pf->va_and_flags & K_MEM_PAGE_FRAME_BUSY) != 0U;
}
static inline bool k_mem_page_frame_is_backed(struct k_mem_page_frame *pf)
{
return (pf->va_and_flags & K_MEM_PAGE_FRAME_BACKED) != 0U;
}
static inline bool k_mem_page_frame_is_evictable(struct k_mem_page_frame *pf)
{
return (!k_mem_page_frame_is_free(pf) &&
!k_mem_page_frame_is_reserved(pf) &&
k_mem_page_frame_is_mapped(pf) &&
!k_mem_page_frame_is_pinned(pf) &&
!k_mem_page_frame_is_busy(pf));
}
/* If true, page is not being used for anything, is not reserved, is not
* a member of some free pages list, isn't busy, and is ready to be mapped
* in memory
*/
static inline bool k_mem_page_frame_is_available(struct k_mem_page_frame *page)
{
return page->va_and_flags == 0U;
}
static inline void k_mem_page_frame_set(struct k_mem_page_frame *pf, uint8_t flags)
{
pf->va_and_flags |= flags;
}
static inline void k_mem_page_frame_clear(struct k_mem_page_frame *pf, uint8_t flags)
{
/* ensure bit inversion to follow is done on the proper type width */
uintptr_t wide_flags = flags;
pf->va_and_flags &= ~wide_flags;
}
static inline void k_mem_assert_phys_aligned(uintptr_t phys)
{
__ASSERT(phys % CONFIG_MMU_PAGE_SIZE == 0U,
"physical address 0x%lx is not page-aligned", phys);
(void)phys;
}
extern struct k_mem_page_frame k_mem_page_frames[K_MEM_NUM_PAGE_FRAMES];
static inline uintptr_t k_mem_page_frame_to_phys(struct k_mem_page_frame *pf)
{
return (uintptr_t)((pf - k_mem_page_frames) * CONFIG_MMU_PAGE_SIZE) +
K_MEM_PHYS_RAM_START;
}
/* Presumes there is but one mapping in the virtual address space */
static inline void *k_mem_page_frame_to_virt(struct k_mem_page_frame *pf)
{
uintptr_t flags_mask = CONFIG_MMU_PAGE_SIZE - 1;
return (void *)(pf->va_and_flags & ~flags_mask);
}
static inline bool k_mem_is_page_frame(uintptr_t phys)
{
k_mem_assert_phys_aligned(phys);
return IN_RANGE(phys, (uintptr_t)K_MEM_PHYS_RAM_START,
(uintptr_t)(K_MEM_PHYS_RAM_END - 1));
}
static inline struct k_mem_page_frame *k_mem_phys_to_page_frame(uintptr_t phys)
{
__ASSERT(k_mem_is_page_frame(phys),
"0x%lx not an SRAM physical address", phys);
return &k_mem_page_frames[(phys - K_MEM_PHYS_RAM_START) /
CONFIG_MMU_PAGE_SIZE];
}
static inline void k_mem_assert_virtual_region(uint8_t *addr, size_t size)
{
__ASSERT((uintptr_t)addr % CONFIG_MMU_PAGE_SIZE == 0U,
"unaligned addr %p", addr);
__ASSERT(size % CONFIG_MMU_PAGE_SIZE == 0U,
"unaligned size %zu", size);
__ASSERT(!Z_DETECT_POINTER_OVERFLOW(addr, size),
"region %p size %zu zero or wraps around", addr, size);
__ASSERT(IN_RANGE((uintptr_t)addr,
(uintptr_t)K_MEM_VIRT_RAM_START,
((uintptr_t)K_MEM_VIRT_RAM_END - 1)) &&
IN_RANGE(((uintptr_t)addr + size - 1),
(uintptr_t)K_MEM_VIRT_RAM_START,
((uintptr_t)K_MEM_VIRT_RAM_END - 1)),
"invalid virtual address region %p (%zu)", addr, size);
}
/**
* @brief Pretty-print page frame information for all page frames.
*
* Debug function, pretty-print page frame information for all frames
* concisely to printk.
*/
void k_mem_page_frames_dump(void);
/* Convenience macro for iterating over all page frames */
#define K_MEM_PAGE_FRAME_FOREACH(_phys, _pageframe) \
for ((_phys) = K_MEM_PHYS_RAM_START, (_pageframe) = k_mem_page_frames; \
(_phys) < K_MEM_PHYS_RAM_END; \
(_phys) += CONFIG_MMU_PAGE_SIZE, (_pageframe)++)
/** @} */
/**
* @def K_MEM_VM_RESERVED
* @brief Reserve space at the end of virtual memory.
*/
#ifdef CONFIG_DEMAND_PAGING
/* We reserve a virtual page as a scratch area for page-ins/outs at the end
* of the address space
*/
#define K_MEM_VM_RESERVED CONFIG_MMU_PAGE_SIZE
/**
* @brief Location of the scratch page used for demand paging.
*/
#define K_MEM_SCRATCH_PAGE ((void *)((uintptr_t)CONFIG_KERNEL_VM_BASE + \
(uintptr_t)CONFIG_KERNEL_VM_SIZE - \
CONFIG_MMU_PAGE_SIZE))
#else
#define K_MEM_VM_RESERVED 0
#endif /* CONFIG_DEMAND_PAGING */
#ifdef CONFIG_DEMAND_PAGING
/*
* Core kernel demand paging APIs
*/
/**
* Number of page faults since system startup
*
* Counts only those page faults that were handled successfully by the demand
* paging mechanism and were not errors.
*
* @return Number of successful page faults
*/
unsigned long k_mem_num_pagefaults_get(void);
/**
* Free a page frame physical address by evicting its contents
*
* The indicated page frame, if it contains a data page, will have that
* data page evicted to the backing store. The page frame will then be
* marked as available for mappings or page-ins.
*
* This is useful for freeing up entire memory banks so that they may be
* deactivated to save power.
*
* If CONFIG_DEMAND_PAGING_ALLOW_IRQ is enabled, this function may not be
* called by ISRs as the backing store may be in-use.
*
* @param phys Page frame physical address
* @retval 0 Success
* @retval -ENOMEM Insufficient backing store space
*/
int k_mem_page_frame_evict(uintptr_t phys);
/**
* Handle a page fault for a virtual data page
*
* This is invoked from the architecture page fault handler.
*
* If a valid page fault, the core kernel will obtain a page frame,
* populate it with the data page that was evicted to the backing store,
* update page tables, and return so that the faulting instruction may be
* re-tried.
*
* The architecture must not call this function if the page was mapped and
* not paged out at the time the exception was triggered (i.e. a protection
* violation for a mapped page).
*
* If the faulting context had interrupts disabled when the page fault was
* triggered, the entire page fault handling path must have interrupts
* disabled, including the invocation of this function.
*
* Otherwise, interrupts may be enabled and the page fault handler may be
* preemptible. Races to page-in will be appropriately handled by the kernel.
*
* @param addr Faulting virtual address
* @retval true Page fault successfully handled, or nothing needed to be done.
* The arch layer should retry the faulting instruction.
* @retval false This page fault was from an un-mapped page, should
* be treated as an error, and not re-tried.
*/
bool k_mem_page_fault(void *addr);
#endif /* CONFIG_DEMAND_PAGING */
#endif /* CONFIG_MMU */
#endif /* KERNEL_INCLUDE_MMU_H */