[PATCH] Fix kexec reboot on ARM
Peter Chubb
peter.chubb at nicta.com.au
Tue Dec 13 23:02:10 EST 2011
When kexec() runs, it eventually sets up a 1-to-1 memory map, then
invokes cpu_reset(). When it invokes cpu_reset (which turns
off the MMU), it does so at its virtual address. Across the code that
disables the MMU, virtual and physical addresses have to be the same,
otherwise after disabling the MMU, the PC is invalid.
The simplest fix is to invoke cpu_reset() at its one-to-one mapped
address.
I've tested on KZM (arm v6) and Beagleboard (omap)
---
arch/arm/kernel/machine_kexec.c | 10 +++++++++-
1 file changed, 9 insertions(+), 1 deletion(-)
Index: linux-2.6/arch/arm/kernel/machine_kexec.c
===================================================================
--- linux-2.6.orig/arch/arm/kernel/machine_kexec.c 2011-12-04 21:11:53.280725573 +1100
+++ linux-2.6/arch/arm/kernel/machine_kexec.c 2011-12-14 14:37:08.227654151 +1100
@@ -120,5 +120,13 @@ void machine_kexec(struct kimage *image)
cpu_proc_fin();
outer_inv_all();
flush_cache_all();
- cpu_reset(reboot_code_buffer_phys);
+ /*
+ * cpu_reset disables the MMU, so branch to its (1-to-1 mapped)
+ * physical address not its virtual one.
+ */
+ {
+ void (*cpu_reset_phys)(unsigned long dest) =
+ virt_to_phys(cpu_reset);
+ cpu_reset_phys(reboot_code_buffer_phys);
+ }
}
--
Dr Peter Chubb http://www.gelato.unsw.edu.au peterc AT gelato.unsw.edu.au
http://www.ertos.nicta.com.au ERTOS within National ICT Australia
More information about the kexec
mailing list