diff options
author | Lorenzo Pieralisi <lorenzo.pieralisi@arm.com> | 2014-05-08 12:31:40 -0400 |
---|---|---|
committer | Russell King <rmk+kernel@arm.linux.org.uk> | 2014-05-25 18:49:27 -0400 |
commit | 0e0779da2233f2dfc85e9c3a6ea142476d326811 (patch) | |
tree | 8222330d72936e63a8cec7d45956cb669bed6d14 | |
parent | 72e6ae285a1dbff553734985bedadf409d99c02d (diff) |
ARM: 8053/1: kernel: sleep: restore HYP mode configuration in cpu_resume
On CPUs with virtualization extensions the kernel installs HYP mode
configuration on both primary and secondary cpus upon cold boot.
On platforms where CPUs are shutdown in idle paths (ie CPU core gating),
when a CPU resumes from low-power states it currently does not execute
code that reinstalls the HYP configuration, which means that the kernel
cannot run eg KVM properly on such machines.
This patch, mirroring cold-boot behaviour, executes position independent
code that reinstalls HYP configuration and drops to SVC mode safely on
warmboot, so that deep idle states can be enabled in kernel running as
hosts on platforms with power management HW.
Cc: Christoffer Dall <christoffer.dall@linaro.org>
Cc: Dave Martin <dave.martin@arm.com>
Cc: Marc Zyngier <marc.zyngier@arm.com>
Cc: Nicolas Pitre <nico@linaro.org>
Signed-off-by: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com>
Acked-by: Marc Zyngier <marc.zyngier@arm.com>
Reviewed-by: Dave Martin <Dave.Martin@arm.com>
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
-rw-r--r-- | arch/arm/include/asm/assembler.h | 2 | ||||
-rw-r--r-- | arch/arm/kernel/sleep.S | 5 |
2 files changed, 5 insertions, 2 deletions
diff --git a/arch/arm/include/asm/assembler.h b/arch/arm/include/asm/assembler.h index b974184f9941..57f0584e8d97 100644 --- a/arch/arm/include/asm/assembler.h +++ b/arch/arm/include/asm/assembler.h | |||
@@ -312,7 +312,7 @@ | |||
312 | * you cannot return to the original mode. | 312 | * you cannot return to the original mode. |
313 | */ | 313 | */ |
314 | .macro safe_svcmode_maskall reg:req | 314 | .macro safe_svcmode_maskall reg:req |
315 | #if __LINUX_ARM_ARCH__ >= 6 | 315 | #if __LINUX_ARM_ARCH__ >= 6 && !defined(CONFIG_CPU_V7M) |
316 | mrs \reg , cpsr | 316 | mrs \reg , cpsr |
317 | eor \reg, \reg, #HYP_MODE | 317 | eor \reg, \reg, #HYP_MODE |
318 | tst \reg, #MODE_MASK | 318 | tst \reg, #MODE_MASK |
diff --git a/arch/arm/kernel/sleep.S b/arch/arm/kernel/sleep.S index b907d9b790ab..1b880db2a033 100644 --- a/arch/arm/kernel/sleep.S +++ b/arch/arm/kernel/sleep.S | |||
@@ -127,6 +127,10 @@ ENDPROC(cpu_resume_after_mmu) | |||
127 | .align | 127 | .align |
128 | ENTRY(cpu_resume) | 128 | ENTRY(cpu_resume) |
129 | ARM_BE8(setend be) @ ensure we are in BE mode | 129 | ARM_BE8(setend be) @ ensure we are in BE mode |
130 | #ifdef CONFIG_ARM_VIRT_EXT | ||
131 | bl __hyp_stub_install_secondary | ||
132 | #endif | ||
133 | safe_svcmode_maskall r1 | ||
130 | mov r1, #0 | 134 | mov r1, #0 |
131 | ALT_SMP(mrc p15, 0, r0, c0, c0, 5) | 135 | ALT_SMP(mrc p15, 0, r0, c0, c0, 5) |
132 | ALT_UP_B(1f) | 136 | ALT_UP_B(1f) |
@@ -144,7 +148,6 @@ ARM_BE8(setend be) @ ensure we are in BE mode | |||
144 | ldr r0, [r0, #SLEEP_SAVE_SP_PHYS] | 148 | ldr r0, [r0, #SLEEP_SAVE_SP_PHYS] |
145 | ldr r0, [r0, r1, lsl #2] | 149 | ldr r0, [r0, r1, lsl #2] |
146 | 150 | ||
147 | setmode PSR_I_BIT | PSR_F_BIT | SVC_MODE, r1 @ set SVC, irqs off | ||
148 | @ load phys pgd, stack, resume fn | 151 | @ load phys pgd, stack, resume fn |
149 | ARM( ldmia r0!, {r1, sp, pc} ) | 152 | ARM( ldmia r0!, {r1, sp, pc} ) |
150 | THUMB( ldmia r0!, {r1, r2, r3} ) | 153 | THUMB( ldmia r0!, {r1, r2, r3} ) |