.text
.balign 16
ENTRY(tme_handler)
.global tlb_entry_i_dat
.global tlb_entry_d_dat
SWITCH_TO_KERNEL_STACK
st r0, @-sp
st r1, @-sp
st r2, @-sp
st r3, @-sp
seth r3,
ld r1, @(MESTS_offset, r3) ; r1: status (MESTS reg.)
ld r0, @(MDEVP_offset, r3) ; r0: PFN + ASID (MDEVP reg.)
st r1, @(MESTS_offset, r3) ; clear status (MESTS reg.)
and3 r1, r1,
bnez r1, 1f ; instruction TLB miss?
;; data TLB miss
;; input
;; r0: PFN + ASID (MDEVP reg.)
;; r1 - r3: free
;; output
;; r0: PFN + ASID
;; r1: TLB entry base address
;; r2: &tlb_entry_{i|d}_dat
;; r3: free
seth r2,
or3 r2, r2,
ldi r1,
seth r2,
or3 r2, r2,
and r1, sp
ld r1, @(16, r1) ; current_thread_info->cpu
slli r1,
add r2, r1
seth r1,
or3 r1, r1,
bra 2f
.balign 16
.fillinsn
1:
;; instrucntion TLB miss
;; input
;; r0: MDEVP reg. (included ASID)
;; r1 - r3: free
;; output
;; r0: PFN + ASID
;; r1: TLB entry base address
;; r2: &tlb_entry_{i|d}_dat
;; r3: free
ldi r3,
and3 r0, r0,
mvfc r1, bpc
and r1, r3
or r0, r1 ; r0: PFN + ASID
seth r2,
or3 r2, r2,
ldi r1,
seth r2,
or3 r2, r2,
and r1, sp
ld r1, @(16, r1) ; current_thread_info->cpu
slli r1,
add r2, r1
seth r1,
or3 r1, r1,
.fillinsn
2:
;; select TLB entry
;; input
;; r0: PFN + ASID
;; r1: TLB entry base address
;; r2: &tlb_entry_{i|d}_dat
;; r3: free
;; output
;; r0: PFN + ASID
;; r1: TLB entry address
;; r2, r3: free
ld r3, @r2 || srli r1,
ld r3, @r2
srli r1,
add r1, r3
; tlb_entry_{d|i}_dat++;
addi r3,
and3 r3, r3,
st r3, @r2 || slli r1,
st r3, @r2
slli r1,
;; load pte
;; input
;; r0: PFN + ASID
;; r1: TLB entry address
;; r2, r3: free
;; output
;; r0: PFN + ASID
;; r1: TLB entry address
;; r2: pte_data
;; r3: free
; pgd = *(unsigned long *)MPTB;
ld24 r2,
srl3 r3, r0,
not r2, r2 || slli r3,
not r2, r2
slli r3,
ld r2, @r2 ; r2: pgd base addr (MPTB reg.)
or r3, r2 ; r3: pmd addr
; pmd = pmd_offset(pgd, address);
ld r3, @r3 ; r3: pmd data
beqz r3, 3f ; pmd_none(*pmd) ?
and3 r2, r3,
add3 r2, r2,
bnez r2, 3f ; pmd_bad(*pmd) ?
ldi r2,
; pte = pte_offset(pmd, address);
and r2, r3 ; r2: pte base addr
srl3 r3, r0,
and3 r3, r3,
or r3, r2
seth r2,
or r3, r2 ; r3: pte addr
; pte_data = (unsigned long)pte_val(*pte);
ld r2, @r3 ; r2: pte data
and3 r3, r2,
beqz r3, 3f
.fillinsn
5:
;; set tlb
;; input
;; r0: PFN + ASID
;; r1: TLB entry address
;; r2: pte_data
;; r3: free
st r0, @r1 ; set_tlb_tag(entry++, address);
st r2, @+r1 ; set_tlb_data(entry, pte_data);
.fillinsn
6:
ld r3, @sp+
ld r2, @sp+
ld r1, @sp+
ld r0, @sp+
rte
.fillinsn
3:
;; error
;; input
;; r0: PFN + ASID
;; r1: TLB entry address
;; r2, r3: free
;; output
;; r0: PFN + ASID
;; r1: TLB entry address
;; r2: pte_data
;; r3: free
bra 5b || ldi r2,
ldi r2,
bra 5b
st sp, @-sp
st r0, @-sp
st r1, @-sp
st r2, @-sp
st r3, @-sp
st r4, @-sp
seth r3,
ld r0, @(MDEVA_offset,r3) ; r0: address (MDEVA reg.)
mvfc r2, bpc ; r2: bpc
ld r1, @(MESTS_offset,r3) ; r1: status (MESTS reg.)
st r1, @(MESTS_offset,r3) ; clear status (MESTS reg.)
and3 r1, r1,
beqz r1, 1f ; data TLB miss?
;; instrucntion TLB miss
mv r0, r2 ; address = bpc;
; entry = (unsigned long *)ITLB_BASE+tlb_entry_i*2;
seth r3,
ld r4, @(low(tlb_entry_i_dat),r3)
sll3 r2, r4,
seth r1,
or3 r1, r1,
add r2, r1 ; r2: entry
addi r4,
and3 r4, r4,
st r4, @(low(tlb_entry_i_dat),r3)
bra 2f
.fillinsn
1:
;; data TLB miss
; entry = (unsigned long *)DTLB_BASE+tlb_entry_d*2;
seth r3,
ld r4, @(low(tlb_entry_d_dat),r3)
sll3 r2, r4,
seth r1,
or3 r1, r1,
add r2, r1 ; r2: entry
addi r4,
and3 r4, r4,
st r4, @(low(tlb_entry_d_dat),r3)
.fillinsn
2:
;; load pte
; r0: address, r2: entry
; r1,r3,r4: (free)
; pgd = *(unsigned long *)MPTB;
ld24 r1,
not r1, r1
ld r1, @r1
srl3 r4, r0,
sll3 r3, r4,
add r3, r1 ; r3: pgd
; pmd = pmd_offset(pgd, address);
ld r1, @r3 ; r1: pmd
beqz r1, 3f ; pmd_none(*pmd) ?
;
and3 r1, r1,
ldi r4,
bne r1, r4, 3f ; pmd_bad(*pmd) ?
.fillinsn
4:
; pte = pte_offset(pmd, address);
ld r4, @r3 ; r4: pte
ldi r3,
and r4, r3
srl3 r3, r0,
and3 r3, r3,
add r4, r3
seth r3,
add r4, r3 ; r4: pte
; pte_data = (unsigned long)pte_val(*pte);
ld r1, @r4 ; r1: pte_data
and3 r3, r1,
beqz r3, 3f
.fillinsn
;; set tlb
; r0: address, r1: pte_data, r2: entry
; r3,r4: (free)
5:
ldi r3,
and r3, r0
seth r4,
ld r4, @(low(MASID),r4) ; r4: MASID
and3 r4, r4,
or r3, r4
st r3, @r2
st r1, @(4,r2) ; set_tlb_data(entry, pte_data);
ld r4, @sp+
ld r3, @sp+
ld r2, @sp+
ld r1, @sp+
ld r0, @sp+
ld sp, @sp+
rte
.fillinsn
3:
ldi r1,
bra 5b
ENTRY(init_tlb)
;; Set MMU Register
seth r0,
or3 r0, r0,
ldi r1,
st r1, @(MPSZ_offset,r0) ; Set MPSZ Reg(Page size 4KB:0 16KB:1 64KB:2)
ldi r1,
st r1, @(MASID_offset,r0) ; Set ASID Zero
;; Set TLB
seth r0,
or3 r0, r0,
seth r1,
or3 r1, r1,
ldi r2,
ldi r3,
addi r0,
addi r1,
clear_tlb:
st r2, @+r0 ; VPA <- 0
st r2, @+r0 ; PPA <- 0
st r2, @+r1 ; VPA <- 0
st r2, @+r1 ; PPA <- 0
addi r3,
bnez r3, clear_tlb
;;
jmp r14
ENTRY(m32r_itlb_entrys)
ENTRY(m32r_otlb_entrys)
.end