547 lines
12 KiB
ArmAsm
547 lines
12 KiB
ArmAsm
/* SPDX-License-Identifier: GPL-2.0 */
|
|
/*
|
|
* Copyright (C) 2020-2022 Loongson Technology Corporation Limited
|
|
*/
|
|
#include <asm/asm.h>
|
|
#include <asm/export.h>
|
|
#include <asm/loongarch.h>
|
|
#include <asm/page.h>
|
|
#include <asm/pgtable.h>
|
|
#include <asm/regdef.h>
|
|
#include <asm/stackframe.h>
|
|
|
|
.macro tlb_do_page_fault, write
|
|
SYM_FUNC_START(tlb_do_page_fault_\write)
|
|
SAVE_ALL
|
|
csrrd a2, LOONGARCH_CSR_BADV
|
|
move a0, sp
|
|
REG_S a2, sp, PT_BVADDR
|
|
li.w a1, \write
|
|
la.abs t0, do_page_fault
|
|
jirl ra, t0, 0
|
|
RESTORE_ALL_AND_RET
|
|
SYM_FUNC_END(tlb_do_page_fault_\write)
|
|
.endm
|
|
|
|
tlb_do_page_fault 0
|
|
tlb_do_page_fault 1
|
|
|
|
SYM_FUNC_START(handle_tlb_protect)
|
|
BACKUP_T0T1
|
|
SAVE_ALL
|
|
move a0, sp
|
|
move a1, zero
|
|
csrrd a2, LOONGARCH_CSR_BADV
|
|
REG_S a2, sp, PT_BVADDR
|
|
la.abs t0, do_page_fault
|
|
jirl ra, t0, 0
|
|
RESTORE_ALL_AND_RET
|
|
SYM_FUNC_END(handle_tlb_protect)
|
|
|
|
SYM_FUNC_START(handle_tlb_load)
|
|
csrwr t0, EXCEPTION_KS0
|
|
csrwr t1, EXCEPTION_KS1
|
|
csrwr ra, EXCEPTION_KS2
|
|
|
|
/*
|
|
* The vmalloc handling is not in the hotpath.
|
|
*/
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
blt t0, $r0, vmalloc_load
|
|
csrrd t1, LOONGARCH_CSR_PGDL
|
|
|
|
vmalloc_done_load:
|
|
/* Get PGD offset in bytes */
|
|
srli.d t0, t0, PGDIR_SHIFT
|
|
andi t0, t0, (PTRS_PER_PGD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#if CONFIG_PGTABLE_LEVELS > 3
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
ld.d t1, t1, 0
|
|
srli.d t0, t0, PUD_SHIFT
|
|
andi t0, t0, (PTRS_PER_PUD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#endif
|
|
#if CONFIG_PGTABLE_LEVELS > 2
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
ld.d t1, t1, 0
|
|
srli.d t0, t0, PMD_SHIFT
|
|
andi t0, t0, (PTRS_PER_PMD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#endif
|
|
ld.d ra, t1, 0
|
|
|
|
/*
|
|
* For huge tlb entries, pmde doesn't contain an address but
|
|
* instead contains the tlb pte. Check the PAGE_HUGE bit and
|
|
* see if we need to jump to huge tlb processing.
|
|
*/
|
|
andi t0, ra, _PAGE_HUGE
|
|
bne t0, $r0, tlb_huge_update_load
|
|
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER)
|
|
andi t0, t0, (PTRS_PER_PTE - 1)
|
|
slli.d t0, t0, _PTE_T_LOG2
|
|
add.d t1, ra, t0
|
|
|
|
#ifdef CONFIG_SMP
|
|
smp_pgtable_change_load:
|
|
#endif
|
|
#ifdef CONFIG_SMP
|
|
ll.d t0, t1, 0
|
|
#else
|
|
ld.d t0, t1, 0
|
|
#endif
|
|
tlbsrch
|
|
|
|
srli.d ra, t0, _PAGE_PRESENT_SHIFT
|
|
andi ra, ra, 1
|
|
beq ra, $r0, nopage_tlb_load
|
|
|
|
ori t0, t0, _PAGE_VALID
|
|
#ifdef CONFIG_SMP
|
|
sc.d t0, t1, 0
|
|
beq t0, $r0, smp_pgtable_change_load
|
|
#else
|
|
st.d t0, t1, 0
|
|
#endif
|
|
ori t1, t1, 8
|
|
xori t1, t1, 8
|
|
ld.d t0, t1, 0
|
|
ld.d t1, t1, 8
|
|
csrwr t0, LOONGARCH_CSR_TLBELO0
|
|
csrwr t1, LOONGARCH_CSR_TLBELO1
|
|
tlbwr
|
|
leave_load:
|
|
csrrd t0, EXCEPTION_KS0
|
|
csrrd t1, EXCEPTION_KS1
|
|
csrrd ra, EXCEPTION_KS2
|
|
ertn
|
|
#ifdef CONFIG_64BIT
|
|
vmalloc_load:
|
|
la.abs t1, swapper_pg_dir
|
|
b vmalloc_done_load
|
|
#endif
|
|
|
|
/*
|
|
* This is the entry point when build_tlbchange_handler_head
|
|
* spots a huge page.
|
|
*/
|
|
tlb_huge_update_load:
|
|
#ifdef CONFIG_SMP
|
|
ll.d t0, t1, 0
|
|
#else
|
|
ld.d t0, t1, 0
|
|
#endif
|
|
srli.d ra, t0, _PAGE_PRESENT_SHIFT
|
|
andi ra, ra, 1
|
|
beq ra, $r0, nopage_tlb_load
|
|
tlbsrch
|
|
|
|
ori t0, t0, _PAGE_VALID
|
|
#ifdef CONFIG_SMP
|
|
sc.d t0, t1, 0
|
|
beq t0, $r0, tlb_huge_update_load
|
|
ld.d t0, t1, 0
|
|
#else
|
|
st.d t0, t1, 0
|
|
#endif
|
|
addu16i.d t1, $r0, -(CSR_TLBIDX_EHINV >> 16)
|
|
addi.d ra, t1, 0
|
|
csrxchg ra, t1, LOONGARCH_CSR_TLBIDX
|
|
tlbwr
|
|
|
|
csrxchg $r0, t1, LOONGARCH_CSR_TLBIDX
|
|
|
|
/*
|
|
* A huge PTE describes an area the size of the
|
|
* configured huge page size. This is twice the
|
|
* of the large TLB entry size we intend to use.
|
|
* A TLB entry half the size of the configured
|
|
* huge page size is configured into entrylo0
|
|
* and entrylo1 to cover the contiguous huge PTE
|
|
* address space.
|
|
*/
|
|
/* Huge page: Move Global bit */
|
|
xori t0, t0, _PAGE_HUGE
|
|
lu12i.w t1, _PAGE_HGLOBAL >> 12
|
|
and t1, t0, t1
|
|
srli.d t1, t1, (_PAGE_HGLOBAL_SHIFT - _PAGE_GLOBAL_SHIFT)
|
|
or t0, t0, t1
|
|
|
|
addi.d ra, t0, 0
|
|
csrwr t0, LOONGARCH_CSR_TLBELO0
|
|
addi.d t0, ra, 0
|
|
|
|
/* Convert to entrylo1 */
|
|
addi.d t1, $r0, 1
|
|
slli.d t1, t1, (HPAGE_SHIFT - 1)
|
|
add.d t0, t0, t1
|
|
csrwr t0, LOONGARCH_CSR_TLBELO1
|
|
|
|
/* Set huge page tlb entry size */
|
|
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16)
|
|
addu16i.d t1, $r0, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
|
|
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
|
|
|
|
tlbfill
|
|
|
|
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16)
|
|
addu16i.d t1, $r0, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
|
|
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
|
|
|
|
nopage_tlb_load:
|
|
dbar 0
|
|
csrrd ra, EXCEPTION_KS2
|
|
la.abs t0, tlb_do_page_fault_0
|
|
jirl $r0, t0, 0
|
|
SYM_FUNC_END(handle_tlb_load)
|
|
|
|
SYM_FUNC_START(handle_tlb_store)
|
|
csrwr t0, EXCEPTION_KS0
|
|
csrwr t1, EXCEPTION_KS1
|
|
csrwr ra, EXCEPTION_KS2
|
|
|
|
/*
|
|
* The vmalloc handling is not in the hotpath.
|
|
*/
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
blt t0, $r0, vmalloc_store
|
|
csrrd t1, LOONGARCH_CSR_PGDL
|
|
|
|
vmalloc_done_store:
|
|
/* Get PGD offset in bytes */
|
|
srli.d t0, t0, PGDIR_SHIFT
|
|
andi t0, t0, (PTRS_PER_PGD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
|
|
#if CONFIG_PGTABLE_LEVELS > 3
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
ld.d t1, t1, 0
|
|
srli.d t0, t0, PUD_SHIFT
|
|
andi t0, t0, (PTRS_PER_PUD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#endif
|
|
#if CONFIG_PGTABLE_LEVELS > 2
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
ld.d t1, t1, 0
|
|
srli.d t0, t0, PMD_SHIFT
|
|
andi t0, t0, (PTRS_PER_PMD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#endif
|
|
ld.d ra, t1, 0
|
|
|
|
/*
|
|
* For huge tlb entries, pmde doesn't contain an address but
|
|
* instead contains the tlb pte. Check the PAGE_HUGE bit and
|
|
* see if we need to jump to huge tlb processing.
|
|
*/
|
|
andi t0, ra, _PAGE_HUGE
|
|
bne t0, $r0, tlb_huge_update_store
|
|
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER)
|
|
andi t0, t0, (PTRS_PER_PTE - 1)
|
|
slli.d t0, t0, _PTE_T_LOG2
|
|
add.d t1, ra, t0
|
|
|
|
#ifdef CONFIG_SMP
|
|
smp_pgtable_change_store:
|
|
#endif
|
|
#ifdef CONFIG_SMP
|
|
ll.d t0, t1, 0
|
|
#else
|
|
ld.d t0, t1, 0
|
|
#endif
|
|
tlbsrch
|
|
|
|
srli.d ra, t0, _PAGE_PRESENT_SHIFT
|
|
andi ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
|
|
xori ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
|
|
bne ra, $r0, nopage_tlb_store
|
|
|
|
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
|
|
#ifdef CONFIG_SMP
|
|
sc.d t0, t1, 0
|
|
beq t0, $r0, smp_pgtable_change_store
|
|
#else
|
|
st.d t0, t1, 0
|
|
#endif
|
|
|
|
ori t1, t1, 8
|
|
xori t1, t1, 8
|
|
ld.d t0, t1, 0
|
|
ld.d t1, t1, 8
|
|
csrwr t0, LOONGARCH_CSR_TLBELO0
|
|
csrwr t1, LOONGARCH_CSR_TLBELO1
|
|
tlbwr
|
|
leave_store:
|
|
csrrd t0, EXCEPTION_KS0
|
|
csrrd t1, EXCEPTION_KS1
|
|
csrrd ra, EXCEPTION_KS2
|
|
ertn
|
|
#ifdef CONFIG_64BIT
|
|
vmalloc_store:
|
|
la.abs t1, swapper_pg_dir
|
|
b vmalloc_done_store
|
|
#endif
|
|
|
|
/*
|
|
* This is the entry point when build_tlbchange_handler_head
|
|
* spots a huge page.
|
|
*/
|
|
tlb_huge_update_store:
|
|
#ifdef CONFIG_SMP
|
|
ll.d t0, t1, 0
|
|
#else
|
|
ld.d t0, t1, 0
|
|
#endif
|
|
srli.d ra, t0, _PAGE_PRESENT_SHIFT
|
|
andi ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
|
|
xori ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
|
|
bne ra, $r0, nopage_tlb_store
|
|
|
|
tlbsrch
|
|
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
|
|
|
|
#ifdef CONFIG_SMP
|
|
sc.d t0, t1, 0
|
|
beq t0, $r0, tlb_huge_update_store
|
|
ld.d t0, t1, 0
|
|
#else
|
|
st.d t0, t1, 0
|
|
#endif
|
|
addu16i.d t1, $r0, -(CSR_TLBIDX_EHINV >> 16)
|
|
addi.d ra, t1, 0
|
|
csrxchg ra, t1, LOONGARCH_CSR_TLBIDX
|
|
tlbwr
|
|
|
|
csrxchg $r0, t1, LOONGARCH_CSR_TLBIDX
|
|
/*
|
|
* A huge PTE describes an area the size of the
|
|
* configured huge page size. This is twice the
|
|
* of the large TLB entry size we intend to use.
|
|
* A TLB entry half the size of the configured
|
|
* huge page size is configured into entrylo0
|
|
* and entrylo1 to cover the contiguous huge PTE
|
|
* address space.
|
|
*/
|
|
/* Huge page: Move Global bit */
|
|
xori t0, t0, _PAGE_HUGE
|
|
lu12i.w t1, _PAGE_HGLOBAL >> 12
|
|
and t1, t0, t1
|
|
srli.d t1, t1, (_PAGE_HGLOBAL_SHIFT - _PAGE_GLOBAL_SHIFT)
|
|
or t0, t0, t1
|
|
|
|
addi.d ra, t0, 0
|
|
csrwr t0, LOONGARCH_CSR_TLBELO0
|
|
addi.d t0, ra, 0
|
|
|
|
/* Convert to entrylo1 */
|
|
addi.d t1, $r0, 1
|
|
slli.d t1, t1, (HPAGE_SHIFT - 1)
|
|
add.d t0, t0, t1
|
|
csrwr t0, LOONGARCH_CSR_TLBELO1
|
|
|
|
/* Set huge page tlb entry size */
|
|
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16)
|
|
addu16i.d t1, $r0, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
|
|
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
|
|
|
|
tlbfill
|
|
|
|
/* Reset default page size */
|
|
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16)
|
|
addu16i.d t1, $r0, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
|
|
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
|
|
|
|
nopage_tlb_store:
|
|
dbar 0
|
|
csrrd ra, EXCEPTION_KS2
|
|
la.abs t0, tlb_do_page_fault_1
|
|
jirl $r0, t0, 0
|
|
SYM_FUNC_END(handle_tlb_store)
|
|
|
|
SYM_FUNC_START(handle_tlb_modify)
|
|
csrwr t0, EXCEPTION_KS0
|
|
csrwr t1, EXCEPTION_KS1
|
|
csrwr ra, EXCEPTION_KS2
|
|
|
|
/*
|
|
* The vmalloc handling is not in the hotpath.
|
|
*/
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
blt t0, $r0, vmalloc_modify
|
|
csrrd t1, LOONGARCH_CSR_PGDL
|
|
|
|
vmalloc_done_modify:
|
|
/* Get PGD offset in bytes */
|
|
srli.d t0, t0, PGDIR_SHIFT
|
|
andi t0, t0, (PTRS_PER_PGD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#if CONFIG_PGTABLE_LEVELS > 3
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
ld.d t1, t1, 0
|
|
srli.d t0, t0, PUD_SHIFT
|
|
andi t0, t0, (PTRS_PER_PUD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#endif
|
|
#if CONFIG_PGTABLE_LEVELS > 2
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
ld.d t1, t1, 0
|
|
srli.d t0, t0, PMD_SHIFT
|
|
andi t0, t0, (PTRS_PER_PMD - 1)
|
|
slli.d t0, t0, 3
|
|
add.d t1, t1, t0
|
|
#endif
|
|
ld.d ra, t1, 0
|
|
|
|
/*
|
|
* For huge tlb entries, pmde doesn't contain an address but
|
|
* instead contains the tlb pte. Check the PAGE_HUGE bit and
|
|
* see if we need to jump to huge tlb processing.
|
|
*/
|
|
andi t0, ra, _PAGE_HUGE
|
|
bne t0, $r0, tlb_huge_update_modify
|
|
|
|
csrrd t0, LOONGARCH_CSR_BADV
|
|
srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER)
|
|
andi t0, t0, (PTRS_PER_PTE - 1)
|
|
slli.d t0, t0, _PTE_T_LOG2
|
|
add.d t1, ra, t0
|
|
|
|
#ifdef CONFIG_SMP
|
|
smp_pgtable_change_modify:
|
|
#endif
|
|
#ifdef CONFIG_SMP
|
|
ll.d t0, t1, 0
|
|
#else
|
|
ld.d t0, t1, 0
|
|
#endif
|
|
tlbsrch
|
|
|
|
srli.d ra, t0, _PAGE_WRITE_SHIFT
|
|
andi ra, ra, 1
|
|
beq ra, $r0, nopage_tlb_modify
|
|
|
|
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
|
|
#ifdef CONFIG_SMP
|
|
sc.d t0, t1, 0
|
|
beq t0, $r0, smp_pgtable_change_modify
|
|
#else
|
|
st.d t0, t1, 0
|
|
#endif
|
|
ori t1, t1, 8
|
|
xori t1, t1, 8
|
|
ld.d t0, t1, 0
|
|
ld.d t1, t1, 8
|
|
csrwr t0, LOONGARCH_CSR_TLBELO0
|
|
csrwr t1, LOONGARCH_CSR_TLBELO1
|
|
tlbwr
|
|
leave_modify:
|
|
csrrd t0, EXCEPTION_KS0
|
|
csrrd t1, EXCEPTION_KS1
|
|
csrrd ra, EXCEPTION_KS2
|
|
ertn
|
|
#ifdef CONFIG_64BIT
|
|
vmalloc_modify:
|
|
la.abs t1, swapper_pg_dir
|
|
b vmalloc_done_modify
|
|
#endif
|
|
|
|
/*
|
|
* This is the entry point when
|
|
* build_tlbchange_handler_head spots a huge page.
|
|
*/
|
|
tlb_huge_update_modify:
|
|
#ifdef CONFIG_SMP
|
|
ll.d t0, t1, 0
|
|
#else
|
|
ld.d t0, t1, 0
|
|
#endif
|
|
|
|
srli.d ra, t0, _PAGE_WRITE_SHIFT
|
|
andi ra, ra, 1
|
|
beq ra, $r0, nopage_tlb_modify
|
|
|
|
tlbsrch
|
|
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
|
|
|
|
#ifdef CONFIG_SMP
|
|
sc.d t0, t1, 0
|
|
beq t0, $r0, tlb_huge_update_modify
|
|
ld.d t0, t1, 0
|
|
#else
|
|
st.d t0, t1, 0
|
|
#endif
|
|
/*
|
|
* A huge PTE describes an area the size of the
|
|
* configured huge page size. This is twice the
|
|
* of the large TLB entry size we intend to use.
|
|
* A TLB entry half the size of the configured
|
|
* huge page size is configured into entrylo0
|
|
* and entrylo1 to cover the contiguous huge PTE
|
|
* address space.
|
|
*/
|
|
/* Huge page: Move Global bit */
|
|
xori t0, t0, _PAGE_HUGE
|
|
lu12i.w t1, _PAGE_HGLOBAL >> 12
|
|
and t1, t0, t1
|
|
srli.d t1, t1, (_PAGE_HGLOBAL_SHIFT - _PAGE_GLOBAL_SHIFT)
|
|
or t0, t0, t1
|
|
|
|
addi.d ra, t0, 0
|
|
csrwr t0, LOONGARCH_CSR_TLBELO0
|
|
addi.d t0, ra, 0
|
|
|
|
/* Convert to entrylo1 */
|
|
addi.d t1, $r0, 1
|
|
slli.d t1, t1, (HPAGE_SHIFT - 1)
|
|
add.d t0, t0, t1
|
|
csrwr t0, LOONGARCH_CSR_TLBELO1
|
|
|
|
/* Set huge page tlb entry size */
|
|
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16)
|
|
addu16i.d t1, $r0, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
|
|
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
|
|
|
|
tlbwr
|
|
|
|
/* Reset default page size */
|
|
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16)
|
|
addu16i.d t1, $r0, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
|
|
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
|
|
|
|
nopage_tlb_modify:
|
|
dbar 0
|
|
csrrd ra, EXCEPTION_KS2
|
|
la.abs t0, tlb_do_page_fault_1
|
|
jirl $r0, t0, 0
|
|
SYM_FUNC_END(handle_tlb_modify)
|
|
|
|
SYM_FUNC_START(handle_tlb_refill)
|
|
csrwr t0, LOONGARCH_CSR_TLBRSAVE
|
|
csrrd t0, LOONGARCH_CSR_PGD
|
|
lddir t0, t0, 3
|
|
#if CONFIG_PGTABLE_LEVELS > 3
|
|
lddir t0, t0, 2
|
|
#endif
|
|
#if CONFIG_PGTABLE_LEVELS > 2
|
|
lddir t0, t0, 1
|
|
#endif
|
|
ldpte t0, 0
|
|
ldpte t0, 1
|
|
tlbfill
|
|
csrrd t0, LOONGARCH_CSR_TLBRSAVE
|
|
ertn
|
|
SYM_FUNC_END(handle_tlb_refill)
|