Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/sh3/sh3 Avoid inline RUN_P2 when accessing TLB via ...



details:   https://anonhg.NetBSD.org/src/rev/967d4fbb6442
branches:  trunk
changeset: 974593:967d4fbb6442
user:      uwe <uwe%NetBSD.org@localhost>
date:      Tue Aug 04 01:55:16 2020 +0000

description:
Avoid inline RUN_P2 when accessing TLB via memory mapped arrays.

Move code that needs to run from P2 into separate functions and call
them via P2 pointer.

diffstat:

 sys/arch/sh3/sh3/mmu_sh4.c |  119 ++++++++++++++++++++++++++------------------
 1 files changed, 69 insertions(+), 50 deletions(-)

diffs (190 lines):

diff -r da4bd3a4d015 -r 967d4fbb6442 sys/arch/sh3/sh3/mmu_sh4.c
--- a/sys/arch/sh3/sh3/mmu_sh4.c        Mon Aug 03 23:01:47 2020 +0000
+++ b/sys/arch/sh3/sh3/mmu_sh4.c        Tue Aug 04 01:55:16 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: mmu_sh4.c,v 1.20 2020/08/03 23:01:47 uwe Exp $ */
+/*     $NetBSD: mmu_sh4.c,v 1.21 2020/08/04 01:55:16 uwe Exp $ */
 
 /*-
  * Copyright (c) 2002 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: mmu_sh4.c,v 1.20 2020/08/03 23:01:47 uwe Exp $");
+__KERNEL_RCSID(0, "$NetBSD: mmu_sh4.c,v 1.21 2020/08/04 01:55:16 uwe Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -39,26 +39,13 @@
 #include <sh3/mmu.h>
 #include <sh3/mmu_sh4.h>
 
-static __noinline void __sh4_tlb_assoc(uint32_t);
-
-/* Must be inlined because we "call" it while running on P2 */
-static inline void __sh4_itlb_invalidate_all(void)
-    __attribute__((always_inline));
+static __noinline void do_tlb_assoc(uint32_t);
+static __noinline void do_invalidate_asid(int);
+static __noinline void do_invalidate_all(uint32_t);
 
 #define        SH4_MMU_HAZARD  __asm volatile("nop;nop;nop;nop;nop;nop;nop;nop;")
 
 
-
-static inline void
-__sh4_itlb_invalidate_all(void)
-{
-
-       _reg_write_4(SH4_ITLB_AA, 0);
-       _reg_write_4(SH4_ITLB_AA | (1 << SH4_ITLB_E_SHIFT), 0);
-       _reg_write_4(SH4_ITLB_AA | (2 << SH4_ITLB_E_SHIFT), 0);
-       _reg_write_4(SH4_ITLB_AA | (3 << SH4_ITLB_E_SHIFT), 0);
-}
-
 void
 sh4_mmu_start(void)
 {
@@ -89,7 +76,7 @@
  * caller is responsible for saving/setting/restoring PTEH.
  */
 static __noinline void
-__sh4_tlb_assoc(uint32_t va)
+do_tlb_assoc(uint32_t va)
 {
 
        _reg_write_4(SH4_UTLB_AA | SH4_UTLB_A, va);
@@ -104,7 +91,7 @@
        uint32_t opteh;
        uint32_t sr;
 
-       tlb_assoc_p2 = SH3_P2SEG_FUNC(__sh4_tlb_assoc);
+       tlb_assoc_p2 = SH3_P2SEG_FUNC(do_tlb_assoc);
 
        va &= SH4_UTLB_AA_VPN_MASK;
 
@@ -119,53 +106,85 @@
 }
 
 
+static __noinline void
+do_invalidate_asid(int asid)
+{
+       int e;
+
+       for (e = 0; e < SH4_UTLB_ENTRY; ++e) {
+               uint32_t addr = SH4_UTLB_AA | (e << SH4_UTLB_E_SHIFT);
+               uint32_t aa = _reg_read_4(addr);
+               if ((aa & SH4_UTLB_AA_ASID_MASK) == asid)
+                       _reg_write_4(addr, 0);
+       }
+
+       for (e = 0; e < SH4_ITLB_ENTRY; ++e) {
+               uint32_t addr = SH4_ITLB_AA | (e << SH4_UTLB_E_SHIFT);
+               uint32_t aa = _reg_read_4(addr);
+               if ((aa & SH4_ITLB_AA_ASID_MASK) == asid)
+                       _reg_write_4(addr, 0);
+       }
+
+       PAD_P1_SWITCH;
+}
+
+
 void
 sh4_tlb_invalidate_asid(int asid)
 {
-       uint32_t a;
-       int e, sr;
+       void (*invalidate_asid_p2)(int);
+       uint32_t sr;
+
+       KDASSERT(asid != 0);
+
+       invalidate_asid_p2 = SH3_P2SEG_FUNC(do_invalidate_asid);
 
        sr = _cpu_exception_suspend();
-       /* Invalidate entry attribute to ASID */
-       RUN_P2;
-       for (e = 0; e < SH4_UTLB_ENTRY; e++) {
-               a = SH4_UTLB_AA | (e << SH4_UTLB_E_SHIFT);
-               if ((_reg_read_4(a) & SH4_UTLB_AA_ASID_MASK) == asid)
-                       _reg_write_4(a, 0);
+       (*invalidate_asid_p2)(asid);
+
+       _cpu_set_sr(sr);
+}
+
+
+static __noinline void
+do_invalidate_all(uint32_t limit)
+{
+       int e;
+
+       for (e = 0; e < limit; ++e) {
+               _reg_write_4(SH4_UTLB_AA  | (e << SH4_UTLB_E_SHIFT), 0);
+               _reg_write_4(SH4_UTLB_DA1 | (e << SH4_UTLB_E_SHIFT), 0);
        }
 
-       __sh4_itlb_invalidate_all();
-       RUN_P1;
-       _cpu_set_sr(sr);
+       for (e = 0; e < SH4_ITLB_ENTRY; ++e) {
+               _reg_write_4(SH4_ITLB_AA  | (e << SH4_ITLB_E_SHIFT), 0);
+               _reg_write_4(SH4_ITLB_DA1 | (e << SH4_ITLB_E_SHIFT), 0);
+       }
+
+       PAD_P1_SWITCH;
 }
 
+
 void
 sh4_tlb_invalidate_all(void)
 {
-       uint32_t a;
-       int e, eend, sr;
+       void (*invalidate_all_p2)(uint32_t);
+       uint32_t limit;
+       uint32_t sr;
+
+       invalidate_all_p2 = SH3_P2SEG_FUNC(do_invalidate_all);
+
+       /* do we resereve TLB entries for wired u-area? */
+       limit = _reg_read_4(SH4_MMUCR) & SH4_MMUCR_URB_MASK;
+       limit = limit ? (limit >> SH4_MMUCR_URB_SHIFT) : SH4_UTLB_ENTRY;
 
        sr = _cpu_exception_suspend();
-       /* If non-wired entry limit is zero, clear all entry. */
-       a = _reg_read_4(SH4_MMUCR) & SH4_MMUCR_URB_MASK;
-       eend = a ? (a >> SH4_MMUCR_URB_SHIFT) : SH4_UTLB_ENTRY;
+       (*invalidate_all_p2)(limit);
 
-       RUN_P2;
-       for (e = 0; e < eend; e++) {
-               a = SH4_UTLB_AA | (e << SH4_UTLB_E_SHIFT);
-               _reg_write_4(a, 0);
-               a = SH4_UTLB_DA1 | (e << SH4_UTLB_E_SHIFT);
-               _reg_write_4(a, 0);
-       }
-       __sh4_itlb_invalidate_all();
-       _reg_write_4(SH4_ITLB_DA1, 0);
-       _reg_write_4(SH4_ITLB_DA1 | (1 << SH4_ITLB_E_SHIFT), 0);
-       _reg_write_4(SH4_ITLB_DA1 | (2 << SH4_ITLB_E_SHIFT), 0);
-       _reg_write_4(SH4_ITLB_DA1 | (3 << SH4_ITLB_E_SHIFT), 0);
-       RUN_P1;
        _cpu_set_sr(sr);
 }
 
+
 void
 sh4_tlb_update(int asid, vaddr_t va, uint32_t pte)
 {
@@ -177,7 +196,7 @@
        KDASSERT(va != 0);
        KDASSERT((pte & ~PGOFSET) != 0);
 
-       tlb_assoc_p2 = SH3_P2SEG_FUNC(__sh4_tlb_assoc);
+       tlb_assoc_p2 = SH3_P2SEG_FUNC(do_tlb_assoc);
 
        sr = _cpu_exception_suspend();
        opteh = _reg_read_4(SH4_PTEH); /* save old ASID */



Home | Main Index | Thread Index | Old Index