Module Name:    src
Committed By:   maxv
Date:           Sun Jan 21 08:20:31 UTC 2018

Modified Files:
        src/sys/arch/amd64/amd64: machdep.c

Log Message:
Make it possible for SVS to map in the user page tables a 4K kernel page
contained in a 2MB large page. Will be used soon.


To generate a diff of this commit:
cvs rdiff -u -r1.293 -r1.294 src/sys/arch/amd64/amd64/machdep.c

Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.

Modified files:

Index: src/sys/arch/amd64/amd64/machdep.c
diff -u src/sys/arch/amd64/amd64/machdep.c:1.293 src/sys/arch/amd64/amd64/machdep.c:1.294
--- src/sys/arch/amd64/amd64/machdep.c:1.293	Sat Jan 20 13:42:07 2018
+++ src/sys/arch/amd64/amd64/machdep.c	Sun Jan 21 08:20:30 2018
@@ -1,4 +1,4 @@
-/*	$NetBSD: machdep.c,v 1.293 2018/01/20 13:42:07 maxv Exp $	*/
+/*	$NetBSD: machdep.c,v 1.294 2018/01/21 08:20:30 maxv Exp $	*/
 
 /*
  * Copyright (c) 1996, 1997, 1998, 2000, 2006, 2007, 2008, 2011
@@ -110,7 +110,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: machdep.c,v 1.293 2018/01/20 13:42:07 maxv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: machdep.c,v 1.294 2018/01/21 08:20:30 maxv Exp $");
 
 /* #define XENDEBUG_LOW  */
 
@@ -2278,12 +2278,11 @@ struct svs_utls {
 static pd_entry_t *
 svs_tree_add(struct cpu_info *ci, vaddr_t va)
 {
-	extern pd_entry_t * const normal_pdes[];
 	extern const vaddr_t ptp_masks[];
 	extern const int ptp_shifts[];
 	extern const long nbpd[];
-	pd_entry_t *srcpde, *dstpde;
-	size_t i, idx, pidx, mod;
+	pd_entry_t *dstpde;
+	size_t i, pidx, mod;
 	struct vm_page *pg;
 	paddr_t pa;
 
@@ -2291,12 +2290,6 @@ svs_tree_add(struct cpu_info *ci, vaddr_
 	mod = (size_t)-1;
 
 	for (i = PTP_LEVELS; i > 1; i--) {
-		idx = pl_i(va, i);
-		srcpde = normal_pdes[i - 2];
-
-		if (!pmap_valid_entry(srcpde[idx])) {
-			panic("%s: page not mapped", __func__);
-		}
 		pidx = pl_i(va % mod, i);
 
 		if (!pmap_valid_entry(dstpde[pidx])) {
@@ -2320,19 +2313,44 @@ svs_tree_add(struct cpu_info *ci, vaddr_
 static void
 svs_page_add(struct cpu_info *ci, vaddr_t va)
 {
-	pd_entry_t *srcpde, *dstpde;
+	pd_entry_t *srcpde, *dstpde, pde;
 	size_t idx, pidx;
+	paddr_t pa;
 
 	/* Create levels L4, L3 and L2. */
 	dstpde = svs_tree_add(ci, va);
 
-	/* Enter L1. */
+	pidx = pl1_i(va % NBPD_L2);
+
+	/*
+	 * If 'va' is in a large page, we need to compute its physical
+	 * address manually.
+	 */
+	idx = pl2_i(va);
+	srcpde = L2_BASE;
+	if (!pmap_valid_entry(srcpde[idx])) {
+		panic("%s: L2 page not mapped", __func__);
+	}
+	if (srcpde[idx] & PG_PS) {
+		pa = srcpde[idx] & PG_2MFRAME;
+		pa += (paddr_t)(va % NBPD_L2);
+		pde = (srcpde[idx] & ~(PG_PS|PG_2MFRAME)) | pa;
+
+		if (pmap_valid_entry(dstpde[pidx])) {
+			panic("%s: L1 page already mapped", __func__);
+		}
+		dstpde[pidx] = pde;
+		return;
+	}
+
+	/*
+	 * Normal page, just copy the PDE.
+	 */
 	idx = pl1_i(va);
 	srcpde = L1_BASE;
 	if (!pmap_valid_entry(srcpde[idx])) {
 		panic("%s: L1 page not mapped", __func__);
 	}
-	pidx = pl1_i(va % NBPD_L2);
 	if (pmap_valid_entry(dstpde[pidx])) {
 		panic("%s: L1 page already mapped", __func__);
 	}

Reply via email to