Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch The MSR EFER state is not saved and restored when s...
details: https://anonhg.NetBSD.org/src/rev/6b18c9f2fbb7
branches: trunk
changeset: 346638:6b18c9f2fbb7
user: maxv <maxv%NetBSD.org@localhost>
date: Sun Jul 24 14:09:22 2016 +0000
description:
The MSR EFER state is not saved and restored when sleeping on i386. On PAE,
the CPU crashes right after waking up, since it needs to access NOX-ed
pages, which are to be enabled in an MSR.
Fix this by properly saving and restoring the EFER MSR. It's a little
tricky since the wakeup code uses %edx, but rdmsr overwrites it. We just
save it in %esi.
Now, the CPU sleeps properly on PAE kernels.
diffstat:
sys/arch/i386/acpi/acpi_wakecode.S | 20 +++++++++++++++++++-
sys/arch/i386/acpi/acpi_wakeup_low.S | 17 +++++++++++++++--
sys/arch/x86/acpi/acpi_wakeup.c | 10 ++++------
3 files changed, 38 insertions(+), 9 deletions(-)
diffs (135 lines):
diff -r 64c10341cc0c -r 6b18c9f2fbb7 sys/arch/i386/acpi/acpi_wakecode.S
--- a/sys/arch/i386/acpi/acpi_wakecode.S Sun Jul 24 13:23:25 2016 +0000
+++ b/sys/arch/i386/acpi/acpi_wakecode.S Sun Jul 24 14:09:22 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: acpi_wakecode.S,v 1.17 2016/07/24 13:04:58 maxv Exp $ */
+/* $NetBSD: acpi_wakecode.S,v 1.18 2016/07/24 14:09:22 maxv Exp $ */
/*-
* Copyright (c) 2002 The NetBSD Foundation, Inc.
@@ -159,6 +159,13 @@
movl WAKEUP_r_cr4 + ACPI_WAKEUP_ADDR,%eax
movl %eax,%cr4
+ /*
+ * Load the correct MSR EFER value now to not depend on the
+ * data segment register. After this point, no instruction is
+ * allowed to clobber %ebx until wrmsr.
+ */
+ movl WAKEUP_efer + ACPI_WAKEUP_ADDR,%ebx
+
/* Load temporary page table, we will switch to full page table later */
movl WAKEUP_r_cr3 + ACPI_WAKEUP_ADDR,%eax
movl %eax,%cr3
@@ -175,6 +182,15 @@
nop
+ /*
+ * Load the normal system value of MSR EFER. This includes
+ * enabling NXE (if supported).
+ */
+ movl %ebx,%eax
+ movl $0,%edx
+ movl $MSR_EFER,%ecx
+ wrmsr
+
/* Restore registers */
movl WAKEUP_curcpu + ACPI_WAKEUP_ADDR,%edx
movl WAKEUP_restorecpu + ACPI_WAKEUP_ADDR,%ebx
@@ -209,6 +225,8 @@
WAKEUP_curcpu: .long 0
.global WAKEUP_restorecpu
WAKEUP_restorecpu: .long 0
+ .global WAKEUP_efer
+WAKEUP_efer: .long 0
.global WAKEUP_vbios_reset
WAKEUP_vbios_reset: .byte 0
diff -r 64c10341cc0c -r 6b18c9f2fbb7 sys/arch/i386/acpi/acpi_wakeup_low.S
--- a/sys/arch/i386/acpi/acpi_wakeup_low.S Sun Jul 24 13:23:25 2016 +0000
+++ b/sys/arch/i386/acpi/acpi_wakeup_low.S Sun Jul 24 14:09:22 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: acpi_wakeup_low.S,v 1.6 2016/07/24 13:04:58 maxv Exp $ */
+/* $NetBSD: acpi_wakeup_low.S,v 1.7 2016/07/24 14:09:22 maxv Exp $ */
/*-
* Copyright (c) 2007 Joerg Sonnenberger <joerg%netbsd.org@localhost>
@@ -29,10 +29,11 @@
*/
#include <machine/asm.h>
-__KERNEL_RCSID(0, "$NetBSD: acpi_wakeup_low.S,v 1.6 2016/07/24 13:04:58 maxv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: acpi_wakeup_low.S,v 1.7 2016/07/24 14:09:22 maxv Exp $");
#include "assym.h"
#include <machine/segments.h>
+#include <machine/specialreg.h>
.text
.p2align 2, 0x90
@@ -51,6 +52,13 @@
movw ACPI_SUSPEND_GS(%edx),%ax
movw %ax,%gs
+ movl $MSR_EFER,%ecx
+ movl ACPI_SUSPEND_EFER(%edx),%eax
+ movl %edx,%esi
+ movl $0,%edx
+ wrmsr
+ movl %esi,%edx
+
movl ACPI_SUSPEND_CR2(%edx),%eax
movl %eax,%cr2
movl ACPI_SUSPEND_CR4(%edx),%eax
@@ -96,6 +104,11 @@
movl %ebp,ACPI_SUSPEND_REG+(3*4)(%edx)
movl %esp,ACPI_SUSPEND_REG+(4*4)(%edx)
+ movl $MSR_EFER,%ecx
+ rdmsr
+ movl CPUVAR(SELF),%edx
+ movl %eax,ACPI_SUSPEND_EFER(%edx)
+
movl %cr0,%eax
movl %eax,ACPI_SUSPEND_CR0(%edx)
movl %cr2,%eax
diff -r 64c10341cc0c -r 6b18c9f2fbb7 sys/arch/x86/acpi/acpi_wakeup.c
--- a/sys/arch/x86/acpi/acpi_wakeup.c Sun Jul 24 13:23:25 2016 +0000
+++ b/sys/arch/x86/acpi/acpi_wakeup.c Sun Jul 24 14:09:22 2016 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: acpi_wakeup.c,v 1.39 2015/08/18 10:42:41 christos Exp $ */
+/* $NetBSD: acpi_wakeup.c,v 1.40 2016/07/24 14:09:22 maxv Exp $ */
/*-
* Copyright (c) 2002, 2011 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: acpi_wakeup.c,v 1.39 2015/08/18 10:42:41 christos Exp $");
+__KERNEL_RCSID(0, "$NetBSD: acpi_wakeup.c,v 1.40 2016/07/24 14:09:22 maxv Exp $");
/*-
* Copyright (c) 2001 Takanori Watanabe <takawata%jp.freebsd.org@localhost>
@@ -62,7 +62,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: acpi_wakeup.c,v 1.39 2015/08/18 10:42:41 christos Exp $");
+__KERNEL_RCSID(0, "$NetBSD: acpi_wakeup.c,v 1.40 2016/07/24 14:09:22 maxv Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -165,10 +165,8 @@
#ifdef __i386__
WAKECODE_FIXUP(WAKEUP_r_cr4, uint32_t, ci->ci_suspend_cr4);
-#else
+#endif
WAKECODE_FIXUP(WAKEUP_efer, uint32_t, ci->ci_suspend_efer);
-#endif
-
WAKECODE_FIXUP(WAKEUP_curcpu, void *, ci);
#ifdef __i386__
WAKECODE_FIXUP(WAKEUP_r_cr3, uint32_t, tmp_pdir);
Home |
Main Index |
Thread Index |
Old Index