git: 49c894ddced5 - main - powerpc64: Split out DMAP and non-DMAP implementations of some methods

Justin Hibbits jhibbits at FreeBSD.org
Thu May 6 01:57:54 UTC 2021


The branch main has been updated by jhibbits:

URL: https://cgit.FreeBSD.org/src/commit/?id=49c894ddced5eaad8d536d8603356576ce58c2df

commit 49c894ddced5eaad8d536d8603356576ce58c2df
Author:     Justin Hibbits <jhibbits at FreeBSD.org>
AuthorDate: 2021-05-05 14:21:51 +0000
Commit:     Justin Hibbits <jhibbits at FreeBSD.org>
CommitDate: 2021-05-06 01:57:33 +0000

    powerpc64: Split out DMAP and non-DMAP implementations of some methods
    
    Summary:
    Some methods are split between DMAP and non-DMAP, conditional on
    hw_direct_map variable.  Rather than checking this variable every time,
    use it to install different functions via IFUNCs.
    
    Reviewed By: luporl
    Differential Revision: https://reviews.freebsd.org/D30071
---
 sys/powerpc/aim/mmu_oea64.c     | 127 +++++++++++++++++++++++-----------------
 sys/powerpc/aim/mmu_oea64.h     |   3 +
 sys/powerpc/aim/moea64_native.c |   2 +
 sys/powerpc/powerpc/machdep.c   |   4 ++
 sys/powerpc/ps3/mmu_ps3.c       |   1 +
 sys/powerpc/pseries/mmu_phyp.c  |   2 +
 6 files changed, 86 insertions(+), 53 deletions(-)

diff --git a/sys/powerpc/aim/mmu_oea64.c b/sys/powerpc/aim/mmu_oea64.c
index 0e9b15661758..c2e8e6e49913 100644
--- a/sys/powerpc/aim/mmu_oea64.c
+++ b/sys/powerpc/aim/mmu_oea64.c
@@ -386,8 +386,11 @@ static __inline bool moea64_sp_pvo_in_range(struct pvo_entry *pvo,
  */
 void moea64_clear_modify(vm_page_t);
 void moea64_copy_page(vm_page_t, vm_page_t);
+void moea64_copy_page_dmap(vm_page_t, vm_page_t);
 void moea64_copy_pages(vm_page_t *ma, vm_offset_t a_offset,
     vm_page_t *mb, vm_offset_t b_offset, int xfersize);
+void moea64_copy_pages_dmap(vm_page_t *ma, vm_offset_t a_offset,
+    vm_page_t *mb, vm_offset_t b_offset, int xfersize);
 int moea64_enter(pmap_t, vm_offset_t, vm_page_t, vm_prot_t,
     u_int flags, int8_t psind);
 void moea64_enter_object(pmap_t, vm_offset_t, vm_offset_t, vm_page_t,
@@ -416,6 +419,7 @@ void moea64_remove_all(vm_page_t);
 void moea64_remove_write(vm_page_t);
 void moea64_unwire(pmap_t, vm_offset_t, vm_offset_t);
 void moea64_zero_page(vm_page_t);
+void moea64_zero_page_dmap(vm_page_t);
 void moea64_zero_page_area(vm_page_t, int, int);
 void moea64_activate(struct thread *);
 void moea64_deactivate(struct thread *);
@@ -432,6 +436,7 @@ void moea64_dumpsys_map(vm_paddr_t pa, size_t sz,
     void **va);
 void moea64_scan_init(void);
 vm_offset_t moea64_quick_enter_page(vm_page_t m);
+vm_offset_t moea64_quick_enter_page_dmap(vm_page_t m);
 void moea64_quick_remove_page(vm_offset_t addr);
 boolean_t moea64_page_is_mapped(vm_page_t m);
 static int moea64_map_user_ptr(pmap_t pm,
@@ -935,14 +940,6 @@ moea64_early_bootstrap(vm_offset_t kernelstart, vm_offset_t kernelend)
 		battable[i].batl = 0;
 	}
 #else
-	moea64_probe_large_page();
-
-	/* Use a direct map if we have large page support */
-	if (moea64_large_page_size > 0)
-		hw_direct_map = 1;
-	else
-		hw_direct_map = 0;
-
 	/* Install trap handlers for SLBs */
 	bcopy(&slbtrap, (void *)EXC_DSE,(size_t)&slbtrapend - (size_t)&slbtrap);
 	bcopy(&slbtrap, (void *)EXC_ISE,(size_t)&slbtrapend - (size_t)&slbtrap);
@@ -1462,6 +1459,20 @@ void moea64_set_scratchpage_pa(int which, vm_paddr_t pa)
 
 void
 moea64_copy_page(vm_page_t msrc, vm_page_t mdst)
+{
+	mtx_lock(&moea64_scratchpage_mtx);
+
+	moea64_set_scratchpage_pa(0, VM_PAGE_TO_PHYS(msrc));
+	moea64_set_scratchpage_pa(1, VM_PAGE_TO_PHYS(mdst));
+
+	bcopy((void *)moea64_scratchpage_va[0],
+	    (void *)moea64_scratchpage_va[1], PAGE_SIZE);
+
+	mtx_unlock(&moea64_scratchpage_mtx);
+}
+
+void
+moea64_copy_page_dmap(vm_page_t msrc, vm_page_t mdst)
 {
 	vm_offset_t	dst;
 	vm_offset_t	src;
@@ -1469,23 +1480,11 @@ moea64_copy_page(vm_page_t msrc, vm_page_t mdst)
 	dst = VM_PAGE_TO_PHYS(mdst);
 	src = VM_PAGE_TO_PHYS(msrc);
 
-	if (hw_direct_map) {
-		bcopy((void *)PHYS_TO_DMAP(src), (void *)PHYS_TO_DMAP(dst),
-		    PAGE_SIZE);
-	} else {
-		mtx_lock(&moea64_scratchpage_mtx);
-
-		moea64_set_scratchpage_pa(0, src);
-		moea64_set_scratchpage_pa(1, dst);
-
-		bcopy((void *)moea64_scratchpage_va[0],
-		    (void *)moea64_scratchpage_va[1], PAGE_SIZE);
-
-		mtx_unlock(&moea64_scratchpage_mtx);
-	}
+	bcopy((void *)PHYS_TO_DMAP(src), (void *)PHYS_TO_DMAP(dst),
+	    PAGE_SIZE);
 }
 
-static inline void
+inline void
 moea64_copy_pages_dmap(vm_page_t *ma, vm_offset_t a_offset,
     vm_page_t *mb, vm_offset_t b_offset, int xfersize)
 {
@@ -1511,8 +1510,8 @@ moea64_copy_pages_dmap(vm_page_t *ma, vm_offset_t a_offset,
 	}
 }
 
-static inline void
-moea64_copy_pages_nodmap(vm_page_t *ma, vm_offset_t a_offset,
+void
+moea64_copy_pages(vm_page_t *ma, vm_offset_t a_offset,
     vm_page_t *mb, vm_offset_t b_offset, int xfersize)
 {
 	void *a_cp, *b_cp;
@@ -1539,20 +1538,6 @@ moea64_copy_pages_nodmap(vm_page_t *ma, vm_offset_t a_offset,
 	mtx_unlock(&moea64_scratchpage_mtx);
 }
 
-void
-moea64_copy_pages(vm_page_t *ma, vm_offset_t a_offset,
-    vm_page_t *mb, vm_offset_t b_offset, int xfersize)
-{
-
-	if (hw_direct_map) {
-		moea64_copy_pages_dmap(ma, a_offset, mb, b_offset,
-		    xfersize);
-	} else {
-		moea64_copy_pages_nodmap(ma, a_offset, mb, b_offset,
-		    xfersize);
-	}
-}
-
 void
 moea64_zero_page_area(vm_page_t m, int off, int size)
 {
@@ -1580,20 +1565,26 @@ moea64_zero_page(vm_page_t m)
 	vm_paddr_t pa = VM_PAGE_TO_PHYS(m);
 	vm_offset_t va, off;
 
-	if (!hw_direct_map) {
-		mtx_lock(&moea64_scratchpage_mtx);
+	mtx_lock(&moea64_scratchpage_mtx);
 
-		moea64_set_scratchpage_pa(0, pa);
-		va = moea64_scratchpage_va[0];
-	} else {
-		va = PHYS_TO_DMAP(pa);
-	}
+	moea64_set_scratchpage_pa(0, pa);
+	va = moea64_scratchpage_va[0];
 
 	for (off = 0; off < PAGE_SIZE; off += cacheline_size)
 		__asm __volatile("dcbz 0,%0" :: "r"(va + off));
 
-	if (!hw_direct_map)
-		mtx_unlock(&moea64_scratchpage_mtx);
+	mtx_unlock(&moea64_scratchpage_mtx);
+}
+
+void
+moea64_zero_page_dmap(vm_page_t m)
+{
+	vm_paddr_t pa = VM_PAGE_TO_PHYS(m);
+	vm_offset_t va, off;
+
+	va = PHYS_TO_DMAP(pa);
+	for (off = 0; off < PAGE_SIZE; off += cacheline_size)
+		__asm __volatile("dcbz 0,%0" :: "r"(va + off));
 }
 
 vm_offset_t
@@ -1602,9 +1593,6 @@ moea64_quick_enter_page(vm_page_t m)
 	struct pvo_entry *pvo;
 	vm_paddr_t pa = VM_PAGE_TO_PHYS(m);
 
-	if (hw_direct_map)
-		return (PHYS_TO_DMAP(pa));
-
 	/*
  	 * MOEA64_PTE_REPLACE does some locking, so we can't just grab
 	 * a critical section and access the PCPU data like on i386.
@@ -1625,11 +1613,16 @@ moea64_quick_enter_page(vm_page_t m)
 	return (PCPU_GET(qmap_addr));
 }
 
+vm_offset_t
+moea64_quick_enter_page_dmap(vm_page_t m)
+{
+
+	return (PHYS_TO_DMAP(VM_PAGE_TO_PHYS(m)));
+}
+
 void
 moea64_quick_remove_page(vm_offset_t addr)
 {
-	if (hw_direct_map)
-		return;
 
 	mtx_assert(PCPU_PTR(aim.qmap_lock), MA_OWNED);
 	KASSERT(PCPU_GET(qmap_addr) == addr,
@@ -3532,6 +3525,34 @@ struct moea64_funcs *moea64_ops;
 		return (f != NULL ? f : (moea64_##func##_t)def);\
 	}
 
+void
+moea64_install(void)
+{
+#ifdef __powerpc64__
+	if (hw_direct_map == -1) {
+		moea64_probe_large_page();
+
+		/* Use a direct map if we have large page support */
+		if (moea64_large_page_size > 0)
+			hw_direct_map = 1;
+		else
+			hw_direct_map = 0;
+	}
+#endif
+
+	/*
+	 * Default to non-DMAP, and switch over to DMAP functions once we know
+	 * we have DMAP.
+	 */
+	if (hw_direct_map) {
+		moea64_methods.quick_enter_page = moea64_quick_enter_page_dmap;
+		moea64_methods.quick_remove_page = NULL;
+		moea64_methods.copy_page = moea64_copy_page_dmap;
+		moea64_methods.zero_page = moea64_zero_page_dmap;
+		moea64_methods.copy_pages = moea64_copy_pages_dmap;
+	}
+}
+
 DEFINE_OEA64_IFUNC(int64_t, pte_replace, (struct pvo_entry *, int),
     moea64_pte_replace_default)
 DEFINE_OEA64_IFUNC(int64_t, pte_insert, (struct pvo_entry *), moea64_null_method)
diff --git a/sys/powerpc/aim/mmu_oea64.h b/sys/powerpc/aim/mmu_oea64.h
index e0b47bad8eed..4ee766689380 100644
--- a/sys/powerpc/aim/mmu_oea64.h
+++ b/sys/powerpc/aim/mmu_oea64.h
@@ -77,6 +77,9 @@ void		moea64_mid_bootstrap(vm_offset_t kernelstart,
 void		moea64_late_bootstrap(vm_offset_t kernelstart,
 		    vm_offset_t kernelend);
 
+/* "base" install method for initializing moea64 pmap ifuncs */
+void		moea64_install(void);
+
 int64_t		moea64_pte_replace(struct pvo_entry *, int);
 int64_t		moea64_pte_insert(struct pvo_entry *);
 int64_t		moea64_pte_unset(struct pvo_entry *);
diff --git a/sys/powerpc/aim/moea64_native.c b/sys/powerpc/aim/moea64_native.c
index 29ba51a48587..367b99856502 100644
--- a/sys/powerpc/aim/moea64_native.c
+++ b/sys/powerpc/aim/moea64_native.c
@@ -308,6 +308,8 @@ moea64_install_native()
 
 	/* Install the MOEA64 ops. */
 	moea64_ops = &moea64_native_funcs;
+
+	moea64_install();
 }
 
 static int64_t
diff --git a/sys/powerpc/powerpc/machdep.c b/sys/powerpc/powerpc/machdep.c
index 716f7e43db48..4e8b6089beea 100644
--- a/sys/powerpc/powerpc/machdep.c
+++ b/sys/powerpc/powerpc/machdep.c
@@ -137,7 +137,11 @@ int cacheline_size = 128;
 #else
 int cacheline_size = 32;
 #endif
+#ifdef __powerpc64__
+int hw_direct_map = -1;
+#else
 int hw_direct_map = 1;
+#endif
 
 #ifdef BOOKE
 extern vm_paddr_t kernload;
diff --git a/sys/powerpc/ps3/mmu_ps3.c b/sys/powerpc/ps3/mmu_ps3.c
index 2d9fdd06c5ac..c21faccf631c 100644
--- a/sys/powerpc/ps3/mmu_ps3.c
+++ b/sys/powerpc/ps3/mmu_ps3.c
@@ -94,6 +94,7 @@ static void
 mps3_install()
 {
 	moea64_ops = &mps3_funcs;
+	moea64_install();
 }
 
 static void
diff --git a/sys/powerpc/pseries/mmu_phyp.c b/sys/powerpc/pseries/mmu_phyp.c
index 709a7dffc995..d94fb2aa5ae1 100644
--- a/sys/powerpc/pseries/mmu_phyp.c
+++ b/sys/powerpc/pseries/mmu_phyp.c
@@ -127,6 +127,8 @@ mphyp_install()
 {
 
 	moea64_ops = &mmu_phyp_funcs;
+
+	moea64_install();
 }
 
 static void


More information about the dev-commits-src-all mailing list