123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195 |
- /*
- * Copyright (c) 2006-2025 RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2021-01-30 lizhirui first version
- * 2023-10-12 Shell Add permission control API
- * 2025-01-22 ZhangJing Add pte default attribute for ur-cp100
- */
- #ifndef __RISCV_MMU_H__
- #define __RISCV_MMU_H__
- #include <rtthread.h>
- #include <rthw.h>
- #include "riscv.h"
- #undef PAGE_SIZE
- #define PAGE_OFFSET_SHIFT 0
- #define PAGE_OFFSET_BIT 12
- #define PAGE_SIZE __SIZE(PAGE_OFFSET_BIT)
- #define PAGE_OFFSET_MASK __MASK(PAGE_OFFSET_BIT)
- #define VPN0_SHIFT (PAGE_OFFSET_SHIFT + PAGE_OFFSET_BIT)
- #define VPN0_BIT 9
- #define VPN1_SHIFT (VPN0_SHIFT + VPN0_BIT)
- #define VPN1_BIT 9
- #define VPN2_SHIFT (VPN1_SHIFT + VPN1_BIT)
- #define VPN2_BIT 9
- #define PPN0_SHIFT (PAGE_OFFSET_SHIFT + PAGE_OFFSET_BIT)
- #define PPN0_BIT 9
- #define PPN1_SHIFT (PPN0_SHIFT + PPN0_BIT)
- #define PPN1_BIT 9
- #define PPN2_SHIFT (PPN1_SHIFT + PPN1_BIT)
- #define PPN2_BIT 26
- #define PPN_BITS (PPN0_BIT + PPN1_BIT + PPN2_BIT)
- #define L1_PAGE_SIZE __SIZE(PAGE_OFFSET_BIT + VPN0_BIT + VPN1_BIT)
- #define L2_PAGE_SIZE __SIZE(PAGE_OFFSET_BIT + VPN0_BIT)
- #define L3_PAGE_SIZE __SIZE(PAGE_OFFSET_BIT)
- #define ARCH_ADDRESS_WIDTH_BITS 64
- #define PHYSICAL_ADDRESS_WIDTH_BITS 56
- #define PAGE_ATTR_BASE (PTE_A | PTE_D)
- #define PAGE_ATTR_NEXT_LEVEL (0)
- #define PAGE_ATTR_RWX (PTE_X | PTE_W | PTE_R)
- #define PAGE_ATTR_READONLY (PTE_R)
- #define PAGE_ATTR_READEXECUTE (PTE_X | PTE_R)
- #define PAGE_ATTR_USER (PTE_U)
- #define PAGE_ATTR_SYSTEM (0)
- #define PAGE_DEFAULT_ATTR_LEAF (PAGE_ATTR_RWX | PAGE_ATTR_USER | PTE_V | PTE_G | PAGE_ATTR_BASE )
- #define PAGE_DEFAULT_ATTR_NEXT (PAGE_ATTR_NEXT_LEVEL | PTE_V | PTE_G)
- #define PAGE_IS_LEAF(pte) __MASKVALUE(pte, PAGE_ATTR_RWX)
- #define PTE_USED(pte) __MASKVALUE(pte, PTE_V)
- /**
- * encoding of SATP (Supervisor Address Translation and Protection register)
- */
- #define SATP_MODE_OFFSET 60
- #define SATP_MODE_BARE 0
- #define SATP_MODE_SV39 8
- #define SATP_MODE_SV48 9
- #define SATP_MODE_SV57 10
- #define SATP_MODE_SV64 11
- #define ARCH_VADDR_WIDTH 39
- #define SATP_MODE SATP_MODE_SV39
- #define MMU_MAP_K_DEVICE (PTE_G | PTE_W | PTE_R | PTE_V | PAGE_ATTR_BASE)
- #define MMU_MAP_K_RWCB (PTE_G | PTE_X | PTE_W | PTE_R | PTE_V | PAGE_ATTR_BASE)
- #define MMU_MAP_K_RW (PTE_G | PTE_X | PTE_W | PTE_R | PTE_V | PAGE_ATTR_BASE)
- #define MMU_MAP_U_RWCB (PTE_U | PTE_X | PTE_W | PTE_R | PTE_V | PAGE_ATTR_BASE)
- #define MMU_MAP_U_RWCB_XN (PTE_U | PTE_W | PTE_R | PTE_V | PAGE_ATTR_BASE)
- #define MMU_MAP_U_RW (PTE_U | PTE_X | PTE_W | PTE_R | PTE_V | PAGE_ATTR_BASE)
- #define MMU_MAP_EARLY (PAGE_ATTR_RWX | PTE_G | PTE_V | PAGE_ATTR_BASE)
- #define PTE_XWR_MASK 0xe
- #define ARCH_PAGE_SIZE PAGE_SIZE
- #define ARCH_PAGE_MASK (ARCH_PAGE_SIZE - 1)
- #define ARCH_PAGE_SHIFT PAGE_OFFSET_BIT
- #define ARCH_INDEX_WIDTH 9
- #define ARCH_INDEX_SIZE (1ul << ARCH_INDEX_WIDTH)
- #define ARCH_INDEX_MASK (ARCH_INDEX_SIZE - 1)
- #define DRAM_SIZE (0x400000000) /* 16GB */
- #define MMIO_SIZE (0x80000000)
- #define MAP_COUNT ((DRAM_SIZE + MMIO_SIZE) >> VPN2_SHIFT)
- #define ARCH_MAP_FAILED ((void *)0x8000000000000000)
- void mmu_set_pagetable(rt_ubase_t addr);
- void mmu_enable_user_page_access(void);
- void mmu_disable_user_page_access(void);
- #define RT_HW_MMU_PROT_READ 1
- #define RT_HW_MMU_PROT_WRITE 2
- #define RT_HW_MMU_PROT_EXECUTE 4
- #define RT_HW_MMU_PROT_KERNEL 8
- #define RT_HW_MMU_PROT_USER 16
- #define RT_HW_MMU_PROT_CACHE 32
- void rt_hw_asid_init(void);
- struct rt_aspace;
- void rt_hw_asid_switch_pgtbl(struct rt_aspace *aspace, rt_ubase_t pgtbl);
- /**
- * @brief Remove permission from attribution
- *
- * @param attr architecture specified mmu attribution
- * @param prot protect that will be removed
- * @return size_t returned attribution
- */
- rt_inline size_t rt_hw_mmu_attr_rm_perm(size_t attr, rt_base_t prot)
- {
- switch (prot)
- {
- /* remove write permission for user */
- case RT_HW_MMU_PROT_WRITE | RT_HW_MMU_PROT_USER:
- attr &= ~PTE_W;
- break;
- /* remove write permission for kernel */
- case RT_HW_MMU_PROT_WRITE | RT_HW_MMU_PROT_KERNEL:
- attr &= ~PTE_W;
- break;
- default:
- RT_ASSERT(0);
- }
- return attr;
- }
- /**
- * @brief Add permission from attribution
- *
- * @param attr architecture specified mmu attribution
- * @param prot protect that will be added
- * @return size_t returned attribution
- */
- rt_inline size_t rt_hw_mmu_attr_add_perm(size_t attr, rt_base_t prot)
- {
- switch (prot)
- {
- /* add write permission for user */
- case RT_HW_MMU_PROT_WRITE | RT_HW_MMU_PROT_USER:
- attr |= (PTE_R | PTE_W | PTE_U);
- break;
- default:
- RT_ASSERT(0);
- }
- return attr;
- }
- /**
- * @brief Test permission from attribution
- *
- * @param attr architecture specified mmu attribution
- * @param prot protect that will be test
- * @return rt_bool_t RT_TRUE if the prot is allowed, otherwise RT_FALSE
- */
- rt_inline rt_bool_t rt_hw_mmu_attr_test_perm(size_t attr, rt_base_t prot)
- {
- rt_bool_t rc = 0;
- switch (prot & ~RT_HW_MMU_PROT_USER)
- {
- /* test write permission for user */
- case RT_HW_MMU_PROT_WRITE:
- rc = ((attr & PTE_W) && (attr & PTE_R));
- break;
- case RT_HW_MMU_PROT_READ:
- rc = !!(attr & PTE_R);
- break;
- case RT_HW_MMU_PROT_EXECUTE:
- rc = !!(attr & PTE_X);
- break;
- default:
- RT_ASSERT(0);
- }
- if (rc && (prot & RT_HW_MMU_PROT_USER))
- {
- rc = !!(attr & PTE_U);
- }
- return rc;
- }
- #endif
|