mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	MIPS: kdump: Add support
[ralf@linux-mips.org: Original patch by Maxim Uvarov <muvarov@gmail.com> with plenty of further shining, polishing, debugging and testing by me.] Signed-off-by: Maxim Uvarov <muvarov@gmail.com> Cc: linux-mips@linux-mips.org Cc: kexec@lists.infradead.org Cc: horms@verge.net.au Patchwork: https://patchwork.linux-mips.org/patch/1025/ Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
This commit is contained in:
		
							parent
							
								
									98cdee0eae
								
							
						
					
					
						commit
						7aa1c8f47e
					
				
					 11 changed files with 396 additions and 9 deletions
				
			
		| 
						 | 
					@ -2379,6 +2379,29 @@ config KEXEC
 | 
				
			||||||
	  support.  As of this writing the exact hardware interface is
 | 
						  support.  As of this writing the exact hardware interface is
 | 
				
			||||||
	  strongly in flux, so no good recommendation can be made.
 | 
						  strongly in flux, so no good recommendation can be made.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					config CRASH_DUMP
 | 
				
			||||||
 | 
						  bool "Kernel crash dumps"
 | 
				
			||||||
 | 
						  help
 | 
				
			||||||
 | 
						  Generate crash dump after being started by kexec.
 | 
				
			||||||
 | 
						  This should be normally only set in special crash dump kernels
 | 
				
			||||||
 | 
						  which are loaded in the main kernel with kexec-tools into
 | 
				
			||||||
 | 
						  a specially reserved region and then later executed after
 | 
				
			||||||
 | 
						  a crash by kdump/kexec. The crash dump kernel must be compiled
 | 
				
			||||||
 | 
						  to a memory address not used by the main kernel or firmware using
 | 
				
			||||||
 | 
						  PHYSICAL_START.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					config PHYSICAL_START
 | 
				
			||||||
 | 
						  hex "Physical address where the kernel is loaded"
 | 
				
			||||||
 | 
						  default "0xffffffff84000000" if 64BIT
 | 
				
			||||||
 | 
						  default "0x84000000" if 32BIT
 | 
				
			||||||
 | 
						  depends on CRASH_DUMP
 | 
				
			||||||
 | 
						  help
 | 
				
			||||||
 | 
						  This gives the CKSEG0 or KSEG0 address where the kernel is loaded.
 | 
				
			||||||
 | 
						  If you plan to use kernel for capturing the crash dump change
 | 
				
			||||||
 | 
						  this value to start of the reserved region (the "X" value as
 | 
				
			||||||
 | 
						  specified in the "crashkernel=YM@XM" command line boot parameter
 | 
				
			||||||
 | 
						  passed to the panic-ed kernel).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
config SECCOMP
 | 
					config SECCOMP
 | 
				
			||||||
	bool "Enable seccomp to safely compute untrusted bytecode"
 | 
						bool "Enable seccomp to safely compute untrusted bytecode"
 | 
				
			||||||
	depends on PROC_FS
 | 
						depends on PROC_FS
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,22 +9,43 @@
 | 
				
			||||||
#ifndef _MIPS_KEXEC
 | 
					#ifndef _MIPS_KEXEC
 | 
				
			||||||
# define _MIPS_KEXEC
 | 
					# define _MIPS_KEXEC
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <asm/stacktrace.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Maximum physical address we can use pages from */
 | 
					/* Maximum physical address we can use pages from */
 | 
				
			||||||
#define KEXEC_SOURCE_MEMORY_LIMIT (0x20000000)
 | 
					#define KEXEC_SOURCE_MEMORY_LIMIT (0x20000000)
 | 
				
			||||||
/* Maximum address we can reach in physical address mode */
 | 
					/* Maximum address we can reach in physical address mode */
 | 
				
			||||||
#define KEXEC_DESTINATION_MEMORY_LIMIT (0x20000000)
 | 
					#define KEXEC_DESTINATION_MEMORY_LIMIT (0x20000000)
 | 
				
			||||||
 /* Maximum address we can use for the control code buffer */
 | 
					 /* Maximum address we can use for the control code buffer */
 | 
				
			||||||
#define KEXEC_CONTROL_MEMORY_LIMIT (0x20000000)
 | 
					#define KEXEC_CONTROL_MEMORY_LIMIT (0x20000000)
 | 
				
			||||||
 | 
					/* Reserve 3*4096 bytes for board-specific info */
 | 
				
			||||||
#define KEXEC_CONTROL_PAGE_SIZE 4096
 | 
					#define KEXEC_CONTROL_PAGE_SIZE (4096 + 3*4096)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* The native architecture */
 | 
					/* The native architecture */
 | 
				
			||||||
#define KEXEC_ARCH KEXEC_ARCH_MIPS
 | 
					#define KEXEC_ARCH KEXEC_ARCH_MIPS
 | 
				
			||||||
 | 
					#define MAX_NOTE_BYTES 1024
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static inline void crash_setup_regs(struct pt_regs *newregs,
 | 
					static inline void crash_setup_regs(struct pt_regs *newregs,
 | 
				
			||||||
				    struct pt_regs *oldregs)
 | 
									    struct pt_regs *oldregs)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	/* Dummy implementation for now */
 | 
						if (oldregs)
 | 
				
			||||||
 | 
							memcpy(newregs, oldregs, sizeof(*newregs));
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							prepare_frametrace(newregs);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef CONFIG_KEXEC
 | 
				
			||||||
 | 
					struct kimage;
 | 
				
			||||||
 | 
					extern unsigned long kexec_args[4];
 | 
				
			||||||
 | 
					extern int (*_machine_kexec_prepare)(struct kimage *);
 | 
				
			||||||
 | 
					extern void (*_machine_kexec_shutdown)(void);
 | 
				
			||||||
 | 
					extern void (*_machine_crash_shutdown)(struct pt_regs *regs);
 | 
				
			||||||
 | 
					extern void default_machine_crash_shutdown(struct pt_regs *regs);
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
					extern const unsigned char kexec_smp_wait[];
 | 
				
			||||||
 | 
					extern unsigned long secondary_kexec_args[4];
 | 
				
			||||||
 | 
					extern void (*relocated_kexec_smp_wait) (void *);
 | 
				
			||||||
 | 
					extern atomic_t kexec_ready_to_reboot;
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif /* !_MIPS_KEXEC */
 | 
					#endif /* !_MIPS_KEXEC */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -40,6 +40,8 @@ extern int __cpu_logical_map[NR_CPUS];
 | 
				
			||||||
#define SMP_CALL_FUNCTION	0x2
 | 
					#define SMP_CALL_FUNCTION	0x2
 | 
				
			||||||
/* Octeon - Tell another core to flush its icache */
 | 
					/* Octeon - Tell another core to flush its icache */
 | 
				
			||||||
#define SMP_ICACHE_FLUSH	0x4
 | 
					#define SMP_ICACHE_FLUSH	0x4
 | 
				
			||||||
 | 
					/* Used by kexec crashdump to save all cpu's state */
 | 
				
			||||||
 | 
					#define SMP_DUMP		0x8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
extern volatile cpumask_t cpu_callin_map;
 | 
					extern volatile cpumask_t cpu_callin_map;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -91,4 +93,8 @@ static inline void arch_send_call_function_ipi_mask(const struct cpumask *mask)
 | 
				
			||||||
	mp_ops->send_ipi_mask(mask, SMP_CALL_FUNCTION);
 | 
						mp_ops->send_ipi_mask(mask, SMP_CALL_FUNCTION);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#if defined(CONFIG_KEXEC)
 | 
				
			||||||
 | 
					extern void (*dump_ipi_function_ptr)(void *);
 | 
				
			||||||
 | 
					void dump_send_ipi(void (*dump_ipi_callback)(void *));
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
#endif /* __ASM_SMP_H */
 | 
					#endif /* __ASM_SMP_H */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -80,7 +80,8 @@ obj-$(CONFIG_I8253)		+= i8253.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
obj-$(CONFIG_GPIO_TXX9)		+= gpio_txx9.o
 | 
					obj-$(CONFIG_GPIO_TXX9)		+= gpio_txx9.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
obj-$(CONFIG_KEXEC)		+= machine_kexec.o relocate_kernel.o
 | 
					obj-$(CONFIG_KEXEC)		+= machine_kexec.o relocate_kernel.o crash.o
 | 
				
			||||||
 | 
					obj-$(CONFIG_CRASH_DUMP)	+= crash_dump.o
 | 
				
			||||||
obj-$(CONFIG_EARLY_PRINTK)	+= early_printk.o
 | 
					obj-$(CONFIG_EARLY_PRINTK)	+= early_printk.o
 | 
				
			||||||
obj-$(CONFIG_SPINLOCK_TEST)	+= spinlock_test.o
 | 
					obj-$(CONFIG_SPINLOCK_TEST)	+= spinlock_test.o
 | 
				
			||||||
obj-$(CONFIG_MIPS_MACHINE)	+= mips_machine.o
 | 
					obj-$(CONFIG_MIPS_MACHINE)	+= mips_machine.o
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										71
									
								
								arch/mips/kernel/crash.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								arch/mips/kernel/crash.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,71 @@
 | 
				
			||||||
 | 
					#include <linux/kernel.h>
 | 
				
			||||||
 | 
					#include <linux/smp.h>
 | 
				
			||||||
 | 
					#include <linux/reboot.h>
 | 
				
			||||||
 | 
					#include <linux/kexec.h>
 | 
				
			||||||
 | 
					#include <linux/bootmem.h>
 | 
				
			||||||
 | 
					#include <linux/crash_dump.h>
 | 
				
			||||||
 | 
					#include <linux/delay.h>
 | 
				
			||||||
 | 
					#include <linux/init.h>
 | 
				
			||||||
 | 
					#include <linux/irq.h>
 | 
				
			||||||
 | 
					#include <linux/types.h>
 | 
				
			||||||
 | 
					#include <linux/sched.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* This keeps a track of which one is crashing cpu. */
 | 
				
			||||||
 | 
					static int crashing_cpu = -1;
 | 
				
			||||||
 | 
					static cpumask_t cpus_in_crash = CPU_MASK_NONE;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
					static void crash_shutdown_secondary(void *ignore)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct pt_regs *regs;
 | 
				
			||||||
 | 
						int cpu = smp_processor_id();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						regs = task_pt_regs(current);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!cpu_online(cpu))
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						local_irq_disable();
 | 
				
			||||||
 | 
						if (!cpu_isset(cpu, cpus_in_crash))
 | 
				
			||||||
 | 
							crash_save_cpu(regs, cpu);
 | 
				
			||||||
 | 
						cpu_set(cpu, cpus_in_crash);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						while (!atomic_read(&kexec_ready_to_reboot))
 | 
				
			||||||
 | 
							cpu_relax();
 | 
				
			||||||
 | 
						relocated_kexec_smp_wait(NULL);
 | 
				
			||||||
 | 
						/* NOTREACHED */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void crash_kexec_prepare_cpus(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						unsigned int msecs;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						unsigned int ncpus = num_online_cpus() - 1;/* Excluding the panic cpu */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						dump_send_ipi(crash_shutdown_secondary);
 | 
				
			||||||
 | 
						smp_wmb();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/*
 | 
				
			||||||
 | 
						 * The crash CPU sends an IPI and wait for other CPUs to
 | 
				
			||||||
 | 
						 * respond. Delay of at least 10 seconds.
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						pr_emerg("Sending IPI to other cpus...\n");
 | 
				
			||||||
 | 
						msecs = 10000;
 | 
				
			||||||
 | 
						while ((cpus_weight(cpus_in_crash) < ncpus) && (--msecs > 0)) {
 | 
				
			||||||
 | 
							cpu_relax();
 | 
				
			||||||
 | 
							mdelay(1);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#else /* !defined(CONFIG_SMP)  */
 | 
				
			||||||
 | 
					static void crash_kexec_prepare_cpus(void) {}
 | 
				
			||||||
 | 
					#endif /* !defined(CONFIG_SMP)  */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void default_machine_crash_shutdown(struct pt_regs *regs)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						local_irq_disable();
 | 
				
			||||||
 | 
						crashing_cpu = smp_processor_id();
 | 
				
			||||||
 | 
						crash_save_cpu(regs, crashing_cpu);
 | 
				
			||||||
 | 
						crash_kexec_prepare_cpus();
 | 
				
			||||||
 | 
						cpu_set(crashing_cpu, cpus_in_crash);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										77
									
								
								arch/mips/kernel/crash_dump.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								arch/mips/kernel/crash_dump.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,77 @@
 | 
				
			||||||
 | 
					#include <linux/highmem.h>
 | 
				
			||||||
 | 
					#include <linux/bootmem.h>
 | 
				
			||||||
 | 
					#include <linux/crash_dump.h>
 | 
				
			||||||
 | 
					#include <asm/uaccess.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					unsigned long long elfcorehdr_addr = ELFCORE_ADDR_MAX;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int __init parse_savemaxmem(char *p)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						if (p)
 | 
				
			||||||
 | 
							saved_max_pfn = (memparse(p, &p) >> PAGE_SHIFT) - 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 1;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					__setup("savemaxmem=", parse_savemaxmem);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void *kdump_buf_page;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * copy_oldmem_page - copy one page from "oldmem"
 | 
				
			||||||
 | 
					 * @pfn: page frame number to be copied
 | 
				
			||||||
 | 
					 * @buf: target memory address for the copy; this can be in kernel address
 | 
				
			||||||
 | 
					 *	space or user address space (see @userbuf)
 | 
				
			||||||
 | 
					 * @csize: number of bytes to copy
 | 
				
			||||||
 | 
					 * @offset: offset in bytes into the page (based on pfn) to begin the copy
 | 
				
			||||||
 | 
					 * @userbuf: if set, @buf is in user address space, use copy_to_user(),
 | 
				
			||||||
 | 
					 *	otherwise @buf is in kernel address space, use memcpy().
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Copy a page from "oldmem". For this page, there is no pte mapped
 | 
				
			||||||
 | 
					 * in the current kernel.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Calling copy_to_user() in atomic context is not desirable. Hence first
 | 
				
			||||||
 | 
					 * copying the data to a pre-allocated kernel page and then copying to user
 | 
				
			||||||
 | 
					 * space in non-atomic context.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					ssize_t copy_oldmem_page(unsigned long pfn, char *buf,
 | 
				
			||||||
 | 
								 size_t csize, unsigned long offset, int userbuf)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						void  *vaddr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!csize)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						vaddr = kmap_atomic_pfn(pfn);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!userbuf) {
 | 
				
			||||||
 | 
							memcpy(buf, (vaddr + offset), csize);
 | 
				
			||||||
 | 
							kunmap_atomic(vaddr);
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							if (!kdump_buf_page) {
 | 
				
			||||||
 | 
								pr_warning("Kdump: Kdump buffer page not allocated\n");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								return -EFAULT;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							copy_page(kdump_buf_page, vaddr);
 | 
				
			||||||
 | 
							kunmap_atomic(vaddr);
 | 
				
			||||||
 | 
							if (copy_to_user(buf, (kdump_buf_page + offset), csize))
 | 
				
			||||||
 | 
								return -EFAULT;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return csize;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int __init kdump_buf_page_init(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int ret = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						kdump_buf_page = kmalloc(PAGE_SIZE, GFP_KERNEL);
 | 
				
			||||||
 | 
						if (!kdump_buf_page) {
 | 
				
			||||||
 | 
							pr_warning("Kdump: Failed to allocate kdump buffer page\n");
 | 
				
			||||||
 | 
							ret = -ENOMEM;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return ret;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					arch_initcall(kdump_buf_page_init);
 | 
				
			||||||
| 
						 | 
					@ -5,7 +5,7 @@
 | 
				
			||||||
 * This source code is licensed under the GNU General Public License,
 | 
					 * This source code is licensed under the GNU General Public License,
 | 
				
			||||||
 * Version 2.  See the file COPYING for more details.
 | 
					 * Version 2.  See the file COPYING for more details.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					#include <linux/compiler.h>
 | 
				
			||||||
#include <linux/kexec.h>
 | 
					#include <linux/kexec.h>
 | 
				
			||||||
#include <linux/mm.h>
 | 
					#include <linux/mm.h>
 | 
				
			||||||
#include <linux/delay.h>
 | 
					#include <linux/delay.h>
 | 
				
			||||||
| 
						 | 
					@ -19,9 +19,19 @@ extern const size_t relocate_new_kernel_size;
 | 
				
			||||||
extern unsigned long kexec_start_address;
 | 
					extern unsigned long kexec_start_address;
 | 
				
			||||||
extern unsigned long kexec_indirection_page;
 | 
					extern unsigned long kexec_indirection_page;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int (*_machine_kexec_prepare)(struct kimage *) = NULL;
 | 
				
			||||||
 | 
					void (*_machine_kexec_shutdown)(void) = NULL;
 | 
				
			||||||
 | 
					void (*_machine_crash_shutdown)(struct pt_regs *regs) = NULL;
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
					void (*relocated_kexec_smp_wait) (void *);
 | 
				
			||||||
 | 
					atomic_t kexec_ready_to_reboot = ATOMIC_INIT(0);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int
 | 
					int
 | 
				
			||||||
machine_kexec_prepare(struct kimage *kimage)
 | 
					machine_kexec_prepare(struct kimage *kimage)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						if (_machine_kexec_prepare)
 | 
				
			||||||
 | 
							return _machine_kexec_prepare(kimage);
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -33,14 +43,20 @@ machine_kexec_cleanup(struct kimage *kimage)
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
machine_shutdown(void)
 | 
					machine_shutdown(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						if (_machine_kexec_shutdown)
 | 
				
			||||||
 | 
							_machine_kexec_shutdown();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
machine_crash_shutdown(struct pt_regs *regs)
 | 
					machine_crash_shutdown(struct pt_regs *regs)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						if (_machine_crash_shutdown)
 | 
				
			||||||
 | 
							_machine_crash_shutdown(regs);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							default_machine_crash_shutdown(regs);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
typedef void (*noretfun_t)(void) __attribute__((noreturn));
 | 
					typedef void (*noretfun_t)(void) __noreturn;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
machine_kexec(struct kimage *image)
 | 
					machine_kexec(struct kimage *image)
 | 
				
			||||||
| 
						 | 
					@ -52,7 +68,9 @@ machine_kexec(struct kimage *image)
 | 
				
			||||||
	reboot_code_buffer =
 | 
						reboot_code_buffer =
 | 
				
			||||||
	  (unsigned long)page_address(image->control_code_page);
 | 
						  (unsigned long)page_address(image->control_code_page);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	kexec_start_address = image->start;
 | 
						kexec_start_address =
 | 
				
			||||||
 | 
							(unsigned long) phys_to_virt(image->start);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	kexec_indirection_page =
 | 
						kexec_indirection_page =
 | 
				
			||||||
		(unsigned long) phys_to_virt(image->head & PAGE_MASK);
 | 
							(unsigned long) phys_to_virt(image->head & PAGE_MASK);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -63,7 +81,7 @@ machine_kexec(struct kimage *image)
 | 
				
			||||||
	 * The generic kexec code builds a page list with physical
 | 
						 * The generic kexec code builds a page list with physical
 | 
				
			||||||
	 * addresses. they are directly accessible through KSEG0 (or
 | 
						 * addresses. they are directly accessible through KSEG0 (or
 | 
				
			||||||
	 * CKSEG0 or XPHYS if on 64bit system), hence the
 | 
						 * CKSEG0 or XPHYS if on 64bit system), hence the
 | 
				
			||||||
	 * pys_to_virt() call.
 | 
						 * phys_to_virt() call.
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
	for (ptr = &image->head; (entry = *ptr) && !(entry &IND_DONE);
 | 
						for (ptr = &image->head; (entry = *ptr) && !(entry &IND_DONE);
 | 
				
			||||||
	     ptr = (entry & IND_INDIRECTION) ?
 | 
						     ptr = (entry & IND_INDIRECTION) ?
 | 
				
			||||||
| 
						 | 
					@ -81,5 +99,12 @@ machine_kexec(struct kimage *image)
 | 
				
			||||||
	printk("Will call new kernel at %08lx\n", image->start);
 | 
						printk("Will call new kernel at %08lx\n", image->start);
 | 
				
			||||||
	printk("Bye ...\n");
 | 
						printk("Bye ...\n");
 | 
				
			||||||
	__flush_cache_all();
 | 
						__flush_cache_all();
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
						/* All secondary cpus now may jump to kexec_wait cycle */
 | 
				
			||||||
 | 
						relocated_kexec_smp_wait = reboot_code_buffer +
 | 
				
			||||||
 | 
							(void *)(kexec_smp_wait - relocate_new_kernel);
 | 
				
			||||||
 | 
						smp_wmb();
 | 
				
			||||||
 | 
						atomic_set(&kexec_ready_to_reboot, 1);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
	((noretfun_t) reboot_code_buffer)();
 | 
						((noretfun_t) reboot_code_buffer)();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -15,6 +15,11 @@
 | 
				
			||||||
#include <asm/addrspace.h>
 | 
					#include <asm/addrspace.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
LEAF(relocate_new_kernel)
 | 
					LEAF(relocate_new_kernel)
 | 
				
			||||||
 | 
						PTR_L a0,	arg0
 | 
				
			||||||
 | 
						PTR_L a1,	arg1
 | 
				
			||||||
 | 
						PTR_L a2,	arg2
 | 
				
			||||||
 | 
						PTR_L a3,	arg3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	PTR_L		s0, kexec_indirection_page
 | 
						PTR_L		s0, kexec_indirection_page
 | 
				
			||||||
	PTR_L		s1, kexec_start_address
 | 
						PTR_L		s1, kexec_start_address
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -26,7 +31,6 @@ process_entry:
 | 
				
			||||||
	and		s3, s2, 0x1
 | 
						and		s3, s2, 0x1
 | 
				
			||||||
	beq		s3, zero, 1f
 | 
						beq		s3, zero, 1f
 | 
				
			||||||
	and		s4, s2, ~0x1	/* store destination addr in s4 */
 | 
						and		s4, s2, ~0x1	/* store destination addr in s4 */
 | 
				
			||||||
	move		a0, s4
 | 
					 | 
				
			||||||
	b		process_entry
 | 
						b		process_entry
 | 
				
			||||||
 | 
					
 | 
				
			||||||
1:
 | 
					1:
 | 
				
			||||||
| 
						 | 
					@ -60,10 +64,92 @@ copy_word:
 | 
				
			||||||
	b		process_entry
 | 
						b		process_entry
 | 
				
			||||||
 | 
					
 | 
				
			||||||
done:
 | 
					done:
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
						/* kexec_flag reset is signal to other CPUs what kernel
 | 
				
			||||||
 | 
						   was moved to it's location. Note - we need relocated address
 | 
				
			||||||
 | 
						   of kexec_flag.  */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						bal		1f
 | 
				
			||||||
 | 
					 1: 	move		t1,ra;
 | 
				
			||||||
 | 
						PTR_LA		t2,1b
 | 
				
			||||||
 | 
						PTR_LA		t0,kexec_flag
 | 
				
			||||||
 | 
						PTR_SUB		t0,t0,t2;
 | 
				
			||||||
 | 
						PTR_ADD		t0,t1,t0;
 | 
				
			||||||
 | 
						LONG_S		zero,(t0)
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sync
 | 
				
			||||||
	/* jump to kexec_start_address */
 | 
						/* jump to kexec_start_address */
 | 
				
			||||||
	j		s1
 | 
						j		s1
 | 
				
			||||||
	END(relocate_new_kernel)
 | 
						END(relocate_new_kernel)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Other CPUs should wait until code is relocated and
 | 
				
			||||||
 | 
					 * then start at entry (?) point.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					LEAF(kexec_smp_wait)
 | 
				
			||||||
 | 
						PTR_L		a0, s_arg0
 | 
				
			||||||
 | 
						PTR_L		a1, s_arg1
 | 
				
			||||||
 | 
						PTR_L		a2, s_arg2
 | 
				
			||||||
 | 
						PTR_L		a3, s_arg3
 | 
				
			||||||
 | 
						PTR_L		s1, kexec_start_address
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Non-relocated address works for args and kexec_start_address ( old
 | 
				
			||||||
 | 
						 * kernel is not overwritten). But we need relocated address of
 | 
				
			||||||
 | 
						 * kexec_flag.
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						bal		1f
 | 
				
			||||||
 | 
					1:	move		t1,ra;
 | 
				
			||||||
 | 
						PTR_LA		t2,1b
 | 
				
			||||||
 | 
						PTR_LA		t0,kexec_flag
 | 
				
			||||||
 | 
						PTR_SUB		t0,t0,t2;
 | 
				
			||||||
 | 
						PTR_ADD		t0,t1,t0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					1:	LONG_L		s0, (t0)
 | 
				
			||||||
 | 
						bne		s0, zero,1b
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sync
 | 
				
			||||||
 | 
						j		s1
 | 
				
			||||||
 | 
						END(kexec_smp_wait)
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef __mips64
 | 
				
			||||||
 | 
					       /* all PTR's must be aligned to 8 byte in 64-bit mode */
 | 
				
			||||||
 | 
					       .align  3
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* All parameters to new kernel are passed in registers a0-a3.
 | 
				
			||||||
 | 
					 * kexec_args[0..3] are uses to prepare register values.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					kexec_args:
 | 
				
			||||||
 | 
						EXPORT(kexec_args)
 | 
				
			||||||
 | 
					arg0:	PTR		0x0
 | 
				
			||||||
 | 
					arg1:	PTR		0x0
 | 
				
			||||||
 | 
					arg2:	PTR		0x0
 | 
				
			||||||
 | 
					arg3:	PTR		0x0
 | 
				
			||||||
 | 
						.size	kexec_args,PTRSIZE*4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef CONFIG_SMP
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Secondary CPUs may have different kernel parameters in
 | 
				
			||||||
 | 
					 * their registers a0-a3. secondary_kexec_args[0..3] are used
 | 
				
			||||||
 | 
					 * to prepare register values.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					secondary_kexec_args:
 | 
				
			||||||
 | 
						EXPORT(secondary_kexec_args)
 | 
				
			||||||
 | 
					s_arg0:	PTR		0x0
 | 
				
			||||||
 | 
					s_arg1:	PTR		0x0
 | 
				
			||||||
 | 
					s_arg2:	PTR		0x0
 | 
				
			||||||
 | 
					s_arg3:	PTR		0x0
 | 
				
			||||||
 | 
						.size	secondary_kexec_args,PTRSIZE*4
 | 
				
			||||||
 | 
					kexec_flag:
 | 
				
			||||||
 | 
						LONG		0x1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
kexec_start_address:
 | 
					kexec_start_address:
 | 
				
			||||||
	EXPORT(kexec_start_address)
 | 
						EXPORT(kexec_start_address)
 | 
				
			||||||
	PTR		0x0
 | 
						PTR		0x0
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -22,6 +22,7 @@
 | 
				
			||||||
#include <linux/console.h>
 | 
					#include <linux/console.h>
 | 
				
			||||||
#include <linux/pfn.h>
 | 
					#include <linux/pfn.h>
 | 
				
			||||||
#include <linux/debugfs.h>
 | 
					#include <linux/debugfs.h>
 | 
				
			||||||
 | 
					#include <linux/kexec.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <asm/addrspace.h>
 | 
					#include <asm/addrspace.h>
 | 
				
			||||||
#include <asm/bootinfo.h>
 | 
					#include <asm/bootinfo.h>
 | 
				
			||||||
| 
						 | 
					@ -536,12 +537,64 @@ static void __init arch_mem_init(char **cmdline_p)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	bootmem_init();
 | 
						bootmem_init();
 | 
				
			||||||
 | 
					#ifdef CONFIG_KEXEC
 | 
				
			||||||
 | 
						if (crashk_res.start != crashk_res.end)
 | 
				
			||||||
 | 
							reserve_bootmem(crashk_res.start,
 | 
				
			||||||
 | 
									crashk_res.end - crashk_res.start + 1,
 | 
				
			||||||
 | 
									BOOTMEM_DEFAULT);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
	device_tree_init();
 | 
						device_tree_init();
 | 
				
			||||||
	sparse_init();
 | 
						sparse_init();
 | 
				
			||||||
	plat_swiotlb_setup();
 | 
						plat_swiotlb_setup();
 | 
				
			||||||
	paging_init();
 | 
						paging_init();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef CONFIG_KEXEC
 | 
				
			||||||
 | 
					static inline unsigned long long get_total_mem(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						unsigned long long total;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						total = max_pfn - min_low_pfn;
 | 
				
			||||||
 | 
						return total << PAGE_SHIFT;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void __init mips_parse_crashkernel(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						unsigned long long total_mem;
 | 
				
			||||||
 | 
						unsigned long long crash_size, crash_base;
 | 
				
			||||||
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						total_mem = get_total_mem();
 | 
				
			||||||
 | 
						ret = parse_crashkernel(boot_command_line, total_mem,
 | 
				
			||||||
 | 
									&crash_size, &crash_base);
 | 
				
			||||||
 | 
						if (ret != 0 || crash_size <= 0)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						crashk_res.start = crash_base;
 | 
				
			||||||
 | 
						crashk_res.end   = crash_base + crash_size - 1;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void __init request_crashkernel(struct resource *res)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = request_resource(res, &crashk_res);
 | 
				
			||||||
 | 
						if (!ret)
 | 
				
			||||||
 | 
							pr_info("Reserving %ldMB of memory at %ldMB for crashkernel\n",
 | 
				
			||||||
 | 
								(unsigned long)((crashk_res.end -
 | 
				
			||||||
 | 
									crashk_res.start + 1) >> 20),
 | 
				
			||||||
 | 
								(unsigned long)(crashk_res.start  >> 20));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#else /* !defined(CONFIG_KEXEC)  */
 | 
				
			||||||
 | 
					static void __init mips_parse_crashkernel(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void __init request_crashkernel(struct resource *res)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#endif /* !defined(CONFIG_KEXEC)  */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void __init resource_init(void)
 | 
					static void __init resource_init(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int i;
 | 
						int i;
 | 
				
			||||||
| 
						 | 
					@ -557,6 +610,8 @@ static void __init resource_init(void)
 | 
				
			||||||
	/*
 | 
						/*
 | 
				
			||||||
	 * Request address space for all standard RAM.
 | 
						 * Request address space for all standard RAM.
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
 | 
						mips_parse_crashkernel();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (i = 0; i < boot_mem_map.nr_map; i++) {
 | 
						for (i = 0; i < boot_mem_map.nr_map; i++) {
 | 
				
			||||||
		struct resource *res;
 | 
							struct resource *res;
 | 
				
			||||||
		unsigned long start, end;
 | 
							unsigned long start, end;
 | 
				
			||||||
| 
						 | 
					@ -593,6 +648,7 @@ static void __init resource_init(void)
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		request_resource(res, &code_resource);
 | 
							request_resource(res, &code_resource);
 | 
				
			||||||
		request_resource(res, &data_resource);
 | 
							request_resource(res, &data_resource);
 | 
				
			||||||
 | 
							request_crashkernel(res);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -386,3 +386,20 @@ void flush_tlb_one(unsigned long vaddr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
EXPORT_SYMBOL(flush_tlb_page);
 | 
					EXPORT_SYMBOL(flush_tlb_page);
 | 
				
			||||||
EXPORT_SYMBOL(flush_tlb_one);
 | 
					EXPORT_SYMBOL(flush_tlb_one);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#if defined(CONFIG_KEXEC)
 | 
				
			||||||
 | 
					void (*dump_ipi_function_ptr)(void *) = NULL;
 | 
				
			||||||
 | 
					void dump_send_ipi(void (*dump_ipi_callback)(void *))
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int i;
 | 
				
			||||||
 | 
						int cpu = smp_processor_id();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						dump_ipi_function_ptr = dump_ipi_callback;
 | 
				
			||||||
 | 
						smp_mb();
 | 
				
			||||||
 | 
						for_each_online_cpu(i)
 | 
				
			||||||
 | 
							if (i != cpu)
 | 
				
			||||||
 | 
								mp_ops->send_ipi_single(i, SMP_DUMP);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EXPORT_SYMBOL(dump_send_ipi);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -13,6 +13,7 @@
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
#include <linux/bug.h>
 | 
					#include <linux/bug.h>
 | 
				
			||||||
#include <linux/compiler.h>
 | 
					#include <linux/compiler.h>
 | 
				
			||||||
 | 
					#include <linux/kexec.h>
 | 
				
			||||||
#include <linux/init.h>
 | 
					#include <linux/init.h>
 | 
				
			||||||
#include <linux/kernel.h>
 | 
					#include <linux/kernel.h>
 | 
				
			||||||
#include <linux/module.h>
 | 
					#include <linux/module.h>
 | 
				
			||||||
| 
						 | 
					@ -409,6 +410,9 @@ void __noreturn die(const char *str, struct pt_regs *regs)
 | 
				
			||||||
		panic("Fatal exception");
 | 
							panic("Fatal exception");
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (regs && kexec_should_crash(current))
 | 
				
			||||||
 | 
							crash_kexec(regs);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	do_exit(sig);
 | 
						do_exit(sig);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue