Changes to Sv39 MMU driver public header

- Define RV_MMU_PT_LEVELS as the arch max
- Add way to find the PPN from a PTE
- Make utility function to create a satp register value, instead of
  combining this to mmu_enable
- Add function to read the current satp value
- Add function to write the satp register, also fix the fence instruction
This commit is contained in:
Ville Juven 2022-01-19 13:05:53 +02:00 committed by Xiang Xiao
parent 926a19217e
commit 33435e76da

View File

@ -71,6 +71,8 @@
*/ */
#ifdef CONFIG_ARCH_MMU_TYPE_SV39 #ifdef CONFIG_ARCH_MMU_TYPE_SV39
#define RV_MMU_PTE_PADDR_SHIFT (10)
#define RV_MMU_PTE_PPN_MASK ((1 << RV_MMU_PTE_PADDR_SHIFT) - 1)
#define RV_MMU_PTE_PPN_SHIFT (2) #define RV_MMU_PTE_PPN_SHIFT (2)
#define RV_MMU_VPN_WIDTH (9) #define RV_MMU_VPN_WIDTH (9)
#define RV_MMU_VPN_MASK ((1 << RV_MMU_VPN_WIDTH) - 1) #define RV_MMU_VPN_MASK ((1 << RV_MMU_VPN_WIDTH) - 1)
@ -83,10 +85,10 @@
#endif /* CONFIG_ARCH_MMU_TYPE_SV39 */ #endif /* CONFIG_ARCH_MMU_TYPE_SV39 */
/**************************************************************************** /****************************************************************************
* Name: mmu_enable * Name: mmu_satp_reg
* *
* Description: * Description:
* Enable MMU and set the base page table address * Utility function to build satp register value for input parameters
* *
* Input Parameters: * Input Parameters:
* pgbase - The physical base address of the translation table base * pgbase - The physical base address of the translation table base
@ -97,26 +99,65 @@
* *
****************************************************************************/ ****************************************************************************/
static inline void mmu_enable(uintptr_t pgbase, uint16_t asid) static inline uintptr_t mmu_satp_reg(uintptr_t pgbase, uint16_t asid)
{ {
uintptr_t reg; uintptr_t reg;
reg = ((RV_MMU_SATP_MODE << SATP_MODE_SHIFT) & SATP_MODE_MASK); reg = ((RV_MMU_SATP_MODE << SATP_MODE_SHIFT) & SATP_MODE_MASK);
reg |= (((uintptr_t)asid << SATP_ASID_SHIFT) & SATP_ASID_MASK); reg |= (((uintptr_t)asid << SATP_ASID_SHIFT) & SATP_ASID_MASK);
reg |= ((SATP_ADDR_TO_PPN(pgbase) << SATP_PPN_SHIFT) & SATP_PPN_MASK); reg |= ((SATP_ADDR_TO_PPN(pgbase) << SATP_PPN_SHIFT) & SATP_PPN_MASK);
return reg;
}
/* Commit to satp and synchronize */ /****************************************************************************
* Name: mmu_write_satp
*
* Description:
* Write satp
*
* Input Parameters:
* reg - satp value
*
****************************************************************************/
static inline void mmu_write_satp(uintptr_t reg)
{
__asm__ __volatile__ __asm__ __volatile__
( (
"csrw satp, %0\n" "csrw satp, %0\n"
"sfence.vma x0, x0\n" "sfence.vma x0, x0\n"
"fence iorw, iorw\n" "fence\n"
: :
: "rK" (reg) : "rK" (reg)
: "memory" : "memory"
); );
} }
/****************************************************************************
* Name: mmu_read_satp
*
* Description:
* Read satp
*
* Returned Value:
* satp register value
*
****************************************************************************/
static inline uintptr_t mmu_read_satp(void)
{
uintptr_t reg;
__asm__ __volatile__
(
"csrr %0, satp\n"
: "=r" (reg)
:
: "memory"
);
return reg;
}
/**************************************************************************** /****************************************************************************
* Name: mmu_invalidate_tlb_by_vaddr * Name: mmu_invalidate_tlb_by_vaddr
* *
@ -158,6 +199,30 @@ static inline void mmu_invalidate_tlbs(void)
); );
} }
/****************************************************************************
* Name: mmu_enable
*
* Description:
* Enable MMU and set the base page table address
*
* Input Parameters:
* pgbase - The physical base address of the translation table base
* asid - Address space identifier. This can be used to identify different
* address spaces. It is not necessary to use this, nor is it necessary
* for the RISC-V implementation to implement such bits. This means in
* practice that the value should not be used in this generic driver.
*
****************************************************************************/
static inline void mmu_enable(uintptr_t pgbase, uint16_t asid)
{
uintptr_t reg = mmu_satp_reg(pgbase, asid);
/* Commit to satp and synchronize */
mmu_write_satp(reg);
}
/**************************************************************************** /****************************************************************************
* Name: mmu_ln_setentry * Name: mmu_ln_setentry
* *