Source-Changes-HG archive

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

[src/trunk]: src/sys/arch Add PCID support in SVS. This avoids TLB flushes du...



details:   https://anonhg.NetBSD.org/src/rev/6cdaa3c7bd5f
branches:  trunk
changeset: 456856:6cdaa3c7bd5f
user:      maxv <maxv%NetBSD.org@localhost>
date:      Wed May 29 16:54:41 2019 +0000

description:
Add PCID support in SVS. This avoids TLB flushes during kernel<->user
transitions, which greatly reduces the performance penalty introduced by
SVS.

We use two ASIDs, 0 (kern) and 1 (user), and use invpcid to flush pages
in both ASIDs.

The read-only machdep.svs.pcid={0,1} sysctl is added, and indicates whether
SVS+PCID is in use.

diffstat:

 sys/arch/amd64/amd64/cpufunc.S    |  31 +++++++++++++++++++++++++---
 sys/arch/amd64/amd64/genassym.cf  |  10 ++++++++-
 sys/arch/amd64/include/pmap.h     |   3 +-
 sys/arch/x86/include/cpufunc.h    |  26 +++++++++++++++++++++++-
 sys/arch/x86/include/pmap.h       |   5 +++-
 sys/arch/x86/include/specialreg.h |   9 +++++++-
 sys/arch/x86/x86/cpu.c            |  10 +++++++-
 sys/arch/x86/x86/svs.c            |  42 ++++++++++++++++++++++++++++++++++----
 sys/arch/x86/x86/x86_machdep.c    |  12 ++++++++--
 9 files changed, 129 insertions(+), 19 deletions(-)

diffs (truncated from 365 to 300 lines):

diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/amd64/amd64/cpufunc.S
--- a/sys/arch/amd64/amd64/cpufunc.S    Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/amd64/amd64/cpufunc.S    Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: cpufunc.S,v 1.40 2019/05/19 08:17:02 maxv Exp $        */
+/*     $NetBSD: cpufunc.S,v 1.41 2019/05/29 16:54:41 maxv Exp $        */
 
 /*
  * Copyright (c) 1998, 2007, 2008 The NetBSD Foundation, Inc.
@@ -41,6 +41,7 @@
 #include <machine/segments.h>
 
 #include "opt_xen.h"
+#include "opt_svs.h"
 
 #include "assym.h"
 
@@ -65,6 +66,17 @@
 
 #ifndef XENPV
 ENTRY(invlpg)
+#ifdef SVS
+       movb    _C_LABEL(svs_pcid),%al
+       testb   %al,%al
+       jz      1f
+       pushq   %rdi
+       pushq   $PMAP_PCID_USER
+       movq    $INVPCID_ADDRESS,%rax
+       invpcid (%rsp),%rax
+       addq    $16,%rsp
+1:     /* FALLTHROUGH */
+#endif
        invlpg  (%rdi)
        ret
 END(invlpg)
@@ -112,7 +124,7 @@
 ENTRY(tlbflushg)
        movq    %cr4, %rax
        testq   $CR4_PGE, %rax
-       jz      1f
+       jz      tlbflush
        movq    %rax, %rdx
        andq    $~CR4_PGE, %rdx
        movq    %rdx, %cr4
@@ -121,8 +133,19 @@
 END(tlbflushg)
 
 ENTRY(tlbflush)
-1:
-       movq    %cr3, %rax
+#ifdef SVS
+       movb    _C_LABEL(svs_pcid),%al
+       testb   %al,%al
+       jz      1f
+       xorq    %rax,%rax
+       pushq   %rax
+       pushq   %rax
+       movq    $INVPCID_ALL_NONGLOBAL,%rax
+       invpcid (%rsp),%rax
+       addq    $16,%rsp
+       ret
+#endif
+1:     movq    %cr3, %rax
        movq    %rax, %cr3
        ret
 END(tlbflush)
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/amd64/amd64/genassym.cf
--- a/sys/arch/amd64/amd64/genassym.cf  Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/amd64/amd64/genassym.cf  Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-#      $NetBSD: genassym.cf,v 1.75 2019/05/27 17:32:36 maxv Exp $
+#      $NetBSD: genassym.cf,v 1.76 2019/05/29 16:54:41 maxv Exp $
 
 #
 # Copyright (c) 1998, 2006, 2007, 2008 The NetBSD Foundation, Inc.
@@ -276,6 +276,14 @@
 define ACPI_SUSPEND_CR4        offsetof(struct cpu_info, ci_suspend_cr4)
 define ACPI_SUSPEND_CR8        offsetof(struct cpu_info, ci_suspend_cr8)
 
+define INVPCID_ADDRESS         INVPCID_ADDRESS
+define INVPCID_CONTEXT         INVPCID_CONTEXT
+define INVPCID_ALL             INVPCID_ALL
+define INVPCID_ALL_NONGLOBAL   INVPCID_ALL_NONGLOBAL
+
+define PMAP_PCID_USER          PMAP_PCID_USER
+define PMAP_PCID_KERN          PMAP_PCID_KERN
+
 if NIOAPIC > 0
 define         PIC_IOAPIC              offsetof(struct pic, pic_ioapic)
 define         IOAPIC_SC_REG           offsetof(struct ioapic_softc, sc_reg)
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/amd64/include/pmap.h
--- a/sys/arch/amd64/include/pmap.h     Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/amd64/include/pmap.h     Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: pmap.h,v 1.60 2019/03/09 09:09:56 maxv Exp $   */
+/*     $NetBSD: pmap.h,v 1.61 2019/05/29 16:54:41 maxv Exp $   */
 
 /*
  * Copyright (c) 1997 Charles D. Cranor and Washington University.
@@ -179,6 +179,7 @@
 void svs_pdir_switch(struct pmap *);
 void svs_init(void);
 extern bool svs_enabled;
+extern bool svs_pcid;
 
 #include <x86/pmap.h>
 
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/x86/include/cpufunc.h
--- a/sys/arch/x86/include/cpufunc.h    Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/x86/include/cpufunc.h    Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: cpufunc.h,v 1.30 2019/05/11 13:40:26 christos Exp $    */
+/*     $NetBSD: cpufunc.h,v 1.31 2019/05/29 16:54:41 maxv Exp $        */
 
 /*
  * Copyright (c) 1998, 2007, 2019 The NetBSD Foundation, Inc.
@@ -65,6 +65,30 @@
 void   wbinvd(void);
 void   breakpoint(void);
 
+#define INVPCID_ADDRESS                0
+#define INVPCID_CONTEXT                1
+#define INVPCID_ALL            2
+#define INVPCID_ALL_NONGLOBAL  3
+
+static inline void
+invpcid(register_t op, uint64_t pcid, vaddr_t va)
+{
+       struct {
+               uint64_t pcid;
+               uint64_t addr;
+       } desc = {
+               .pcid = pcid,
+               .addr = va
+       };
+
+       asm volatile (
+               "invpcid %[desc],%[op]"
+               :
+               : [desc] "m" (desc), [op] "r" (op)
+               : "memory"
+       );
+}
+
 static inline uint64_t
 rdtsc(void)
 {
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/x86/include/pmap.h
--- a/sys/arch/x86/include/pmap.h       Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/x86/include/pmap.h       Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: pmap.h,v 1.100 2019/03/10 16:30:01 maxv Exp $  */
+/*     $NetBSD: pmap.h,v 1.101 2019/05/29 16:54:41 maxv Exp $  */
 
 /*
  * Copyright (c) 1997 Charles D. Cranor and Washington University.
@@ -210,6 +210,9 @@
 
 extern struct pcpu_area *pcpuarea;
 
+#define PMAP_PCID_KERN 0
+#define PMAP_PCID_USER 1
+
 /*
  * pmap data structures: see pmap.c for details of locking.
  */
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/x86/include/specialreg.h
--- a/sys/arch/x86/include/specialreg.h Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/x86/include/specialreg.h Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: specialreg.h,v 1.146 2019/05/18 08:17:39 maxv Exp $    */
+/*     $NetBSD: specialreg.h,v 1.147 2019/05/29 16:54:41 maxv Exp $    */
 
 /*
  * Copyright (c) 2014-2019 The NetBSD Foundation, Inc.
@@ -88,6 +88,13 @@
 #define CCR1_RPL       0x01    /* enables RPLSET and RPLVAL# pins */
 
 /*
+ * CR3
+ */
+#define CR3_PCID               __BITS(11,0)
+#define CR3_PA                 __BITS(62,12)
+#define CR3_NO_TLB_FLUSH       __BIT(63)
+
+/*
  * CR4
  */
 #define CR4_VME                0x00000001 /* virtual 8086 mode extension enable */
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/x86/x86/cpu.c
--- a/sys/arch/x86/x86/cpu.c    Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/x86/x86/cpu.c    Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: cpu.c,v 1.170 2019/05/27 18:36:37 maxv Exp $   */
+/*     $NetBSD: cpu.c,v 1.171 2019/05/29 16:54:41 maxv Exp $   */
 
 /*
  * Copyright (c) 2000-2012 NetBSD Foundation, Inc.
@@ -62,7 +62,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: cpu.c,v 1.170 2019/05/27 18:36:37 maxv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: cpu.c,v 1.171 2019/05/29 16:54:41 maxv Exp $");
 
 #include "opt_ddb.h"
 #include "opt_mpbios.h"                /* for MPDEBUG */
@@ -613,6 +613,12 @@
        if (cpu_feature[5] & CPUID_SEF_SMAP)
                cr4 |= CR4_SMAP;
 
+#ifdef SVS
+       /* If PCID is supported, enable it */
+       if (svs_pcid)
+               cr4 |= CR4_PCIDE;
+#endif
+
        if (cr4) {
                cr4 |= rcr4();
                lcr4(cr4);
diff -r 686e9427bf5d -r 6cdaa3c7bd5f sys/arch/x86/x86/svs.c
--- a/sys/arch/x86/x86/svs.c    Wed May 29 14:28:37 2019 +0000
+++ b/sys/arch/x86/x86/svs.c    Wed May 29 16:54:41 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: svs.c,v 1.28 2019/05/27 18:36:37 maxv Exp $    */
+/*     $NetBSD: svs.c,v 1.29 2019/05/29 16:54:41 maxv Exp $    */
 
 /*
  * Copyright (c) 2018-2019 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: svs.c,v 1.28 2019/05/27 18:36:37 maxv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: svs.c,v 1.29 2019/05/29 16:54:41 maxv Exp $");
 
 #include "opt_svs.h"
 
@@ -228,6 +228,10 @@
  */
 
 bool svs_enabled __read_mostly = false;
+bool svs_pcid __read_mostly = false;
+
+static uint64_t svs_pcid_kcr3 __read_mostly;
+static uint64_t svs_pcid_ucr3 __read_mostly;
 
 struct svs_utls {
        paddr_t kpdirpa;
@@ -405,6 +409,19 @@
 }
 
 static void
+svs_pcid_init(struct cpu_info *ci)
+{
+       if (!svs_pcid) {
+               return;
+       }
+
+       svs_pcid_ucr3 = __SHIFTIN(PMAP_PCID_USER, CR3_PCID) | CR3_NO_TLB_FLUSH;
+       svs_pcid_kcr3 = __SHIFTIN(PMAP_PCID_KERN, CR3_PCID) | CR3_NO_TLB_FLUSH;
+
+       ci->ci_svs_updirpa |= svs_pcid_ucr3;
+}
+
+static void
 svs_range_add(struct cpu_info *ci, vaddr_t va, size_t size, bool global)
 {
        size_t i, n;
@@ -454,6 +471,8 @@
 
        svs_rsp0_init(ci);
        svs_utls_init(ci);
+
+       svs_pcid_init(ci);
 }
 
 void
@@ -523,11 +542,14 @@
        utls->scratch = 0;
 
        /*
-        * Enter the user rsp0. We don't need to flush the TLB here, since
-        * the user page tables are not loaded.
+        * Enter the user rsp0. If we're using PCID we must flush the user VA,
+        * if we aren't it will be flushed during the next CR3 reload.
         */
        pte = ci->ci_svs_rsp0_pte;
        *pte = L1_BASE[pl1_i(va)];
+       if (svs_pcid) {
+               invpcid(INVPCID_ADDRESS, PMAP_PCID_USER, ci->ci_svs_rsp0);
+       }
 }
 
 static inline pt_entry_t
@@ -558,7 +580,7 @@
 
        /* Update the info in the UTLS page */
        utls = (struct svs_utls *)ci->ci_svs_utls;
-       utls->kpdirpa = pmap_pdirpa(pmap, 0);



Home | Main Index | Thread Index | Old Index