Commit 62cac480 authored by Dengcheng Zhu's avatar Dengcheng Zhu Committed by Paul Burton

MIPS: kexec: Make a framework for both jumping and halting on nonboot CPUs

The existing implementation lets machine_kexec() CPU jump to reboot code
buffer, whereas other CPUs to relocated_kexec_smp_wait. The natural way to
bring up an SMP new kernel would be to let CPU0 do it while others being
halted. For those failing to do so, fall back to the jumping method.
Signed-off-by: default avatarDengcheng Zhu <dzhu@wavecomp.com>
[paul.burton@mips.com: Guard kexec_nonboot_cpu_jump with CONFIG_SMP]
Signed-off-by: default avatarPaul Burton <paul.burton@mips.com>
Patchwork: https://patchwork.linux-mips.org/patch/20570/
Cc: pburton@wavecomp.com
Cc: ralf@linux-mips.org
Cc: linux-mips@linux-mips.org
Cc: rachel.mozes@intel.com
parent dc57aaf9
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
#include <linux/sched/task_stack.h> #include <linux/sched/task_stack.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/export.h> #include <linux/export.h>
#include <linux/kexec.h>
#include <asm/mmu_context.h> #include <asm/mmu_context.h>
#include <asm/time.h> #include <asm/time.h>
...@@ -424,6 +425,9 @@ const struct plat_smp_ops octeon_smp_ops = { ...@@ -424,6 +425,9 @@ const struct plat_smp_ops octeon_smp_ops = {
.cpu_disable = octeon_cpu_disable, .cpu_disable = octeon_cpu_disable,
.cpu_die = octeon_cpu_die, .cpu_die = octeon_cpu_die,
#endif #endif
#ifdef CONFIG_KEXEC
.kexec_nonboot_cpu = kexec_nonboot_cpu_jump,
#endif
}; };
static irqreturn_t octeon_78xx_reched_interrupt(int irq, void *dev_id) static irqreturn_t octeon_78xx_reched_interrupt(int irq, void *dev_id)
...@@ -501,6 +505,9 @@ static const struct plat_smp_ops octeon_78xx_smp_ops = { ...@@ -501,6 +505,9 @@ static const struct plat_smp_ops octeon_78xx_smp_ops = {
.cpu_disable = octeon_cpu_disable, .cpu_disable = octeon_cpu_disable,
.cpu_die = octeon_cpu_die, .cpu_die = octeon_cpu_die,
#endif #endif
#ifdef CONFIG_KEXEC
.kexec_nonboot_cpu = kexec_nonboot_cpu_jump,
#endif
}; };
void __init octeon_setup_smp(void) void __init octeon_setup_smp(void)
......
...@@ -39,11 +39,12 @@ extern unsigned long kexec_args[4]; ...@@ -39,11 +39,12 @@ extern unsigned long kexec_args[4];
extern int (*_machine_kexec_prepare)(struct kimage *); extern int (*_machine_kexec_prepare)(struct kimage *);
extern void (*_machine_kexec_shutdown)(void); extern void (*_machine_kexec_shutdown)(void);
extern void (*_machine_crash_shutdown)(struct pt_regs *regs); extern void (*_machine_crash_shutdown)(struct pt_regs *regs);
extern void default_machine_crash_shutdown(struct pt_regs *regs); void default_machine_crash_shutdown(struct pt_regs *regs);
void kexec_nonboot_cpu_jump(void);
void kexec_reboot(void);
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
extern const unsigned char kexec_smp_wait[]; extern const unsigned char kexec_smp_wait[];
extern unsigned long secondary_kexec_args[4]; extern unsigned long secondary_kexec_args[4];
extern void (*relocated_kexec_smp_wait) (void *);
extern atomic_t kexec_ready_to_reboot; extern atomic_t kexec_ready_to_reboot;
extern void (*_crash_smp_send_stop)(void); extern void (*_crash_smp_send_stop)(void);
#endif #endif
......
...@@ -33,6 +33,9 @@ struct plat_smp_ops { ...@@ -33,6 +33,9 @@ struct plat_smp_ops {
int (*cpu_disable)(void); int (*cpu_disable)(void);
void (*cpu_die)(unsigned int cpu); void (*cpu_die)(unsigned int cpu);
#endif #endif
#ifdef CONFIG_KEXEC
void (*kexec_nonboot_cpu)(void);
#endif
}; };
extern void register_smp_ops(const struct plat_smp_ops *ops); extern void register_smp_ops(const struct plat_smp_ops *ops);
......
...@@ -91,6 +91,22 @@ static inline void __cpu_die(unsigned int cpu) ...@@ -91,6 +91,22 @@ static inline void __cpu_die(unsigned int cpu)
extern void play_dead(void); extern void play_dead(void);
#endif #endif
#ifdef CONFIG_KEXEC
static inline void kexec_nonboot_cpu(void)
{
extern const struct plat_smp_ops *mp_ops; /* private */
return mp_ops->kexec_nonboot_cpu();
}
static inline void *kexec_nonboot_cpu_func(void)
{
extern const struct plat_smp_ops *mp_ops; /* private */
return mp_ops->kexec_nonboot_cpu;
}
#endif
/* /*
* This function will set up the necessary IPIs for Linux to communicate * This function will set up the necessary IPIs for Linux to communicate
* with the CPUs in mask. * with the CPUs in mask.
......
...@@ -46,7 +46,9 @@ static void crash_shutdown_secondary(void *passed_regs) ...@@ -46,7 +46,9 @@ static void crash_shutdown_secondary(void *passed_regs)
while (!atomic_read(&kexec_ready_to_reboot)) while (!atomic_read(&kexec_ready_to_reboot))
cpu_relax(); cpu_relax();
relocated_kexec_smp_wait(NULL);
kexec_reboot();
/* NOTREACHED */ /* NOTREACHED */
} }
......
...@@ -19,15 +19,19 @@ extern const size_t relocate_new_kernel_size; ...@@ -19,15 +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; static unsigned long reboot_code_buffer;
void (*_machine_kexec_shutdown)(void) = NULL;
void (*_machine_crash_shutdown)(struct pt_regs *regs) = NULL;
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
void (*relocated_kexec_smp_wait) (void *); static void (*relocated_kexec_smp_wait)(void *);
atomic_t kexec_ready_to_reboot = ATOMIC_INIT(0); atomic_t kexec_ready_to_reboot = ATOMIC_INIT(0);
void (*_crash_smp_send_stop)(void) = NULL; void (*_crash_smp_send_stop)(void) = NULL;
#endif #endif
int (*_machine_kexec_prepare)(struct kimage *) = NULL;
void (*_machine_kexec_shutdown)(void) = NULL;
void (*_machine_crash_shutdown)(struct pt_regs *regs) = NULL;
static void kexec_image_info(const struct kimage *kimage) static void kexec_image_info(const struct kimage *kimage)
{ {
unsigned long i; unsigned long i;
...@@ -51,10 +55,16 @@ static void kexec_image_info(const struct kimage *kimage) ...@@ -51,10 +55,16 @@ static void kexec_image_info(const struct kimage *kimage)
int int
machine_kexec_prepare(struct kimage *kimage) machine_kexec_prepare(struct kimage *kimage)
{ {
#ifdef CONFIG_SMP
if (!kexec_nonboot_cpu_func())
return -EINVAL;
#endif
kexec_image_info(kimage); kexec_image_info(kimage);
if (_machine_kexec_prepare) if (_machine_kexec_prepare)
return _machine_kexec_prepare(kimage); return _machine_kexec_prepare(kimage);
return 0; return 0;
} }
...@@ -63,11 +73,41 @@ machine_kexec_cleanup(struct kimage *kimage) ...@@ -63,11 +73,41 @@ machine_kexec_cleanup(struct kimage *kimage)
{ {
} }
#ifdef CONFIG_SMP
static void kexec_shutdown_secondary(void *param)
{
int cpu = smp_processor_id();
if (!cpu_online(cpu))
return;
/* We won't be sent IPIs any more. */
set_cpu_online(cpu, false);
local_irq_disable();
while (!atomic_read(&kexec_ready_to_reboot))
cpu_relax();
kexec_reboot();
/* NOTREACHED */
}
#endif
void void
machine_shutdown(void) machine_shutdown(void)
{ {
if (_machine_kexec_shutdown) if (_machine_kexec_shutdown)
_machine_kexec_shutdown(); _machine_kexec_shutdown();
#ifdef CONFIG_SMP
smp_call_function(kexec_shutdown_secondary, NULL, 0);
while (num_online_cpus() > 1) {
cpu_relax();
mdelay(1);
}
#endif
} }
void void
...@@ -79,12 +119,47 @@ machine_crash_shutdown(struct pt_regs *regs) ...@@ -79,12 +119,47 @@ machine_crash_shutdown(struct pt_regs *regs)
default_machine_crash_shutdown(regs); default_machine_crash_shutdown(regs);
} }
typedef void (*noretfun_t)(void) __noreturn; #ifdef CONFIG_SMP
void kexec_nonboot_cpu_jump(void)
{
local_flush_icache_range((unsigned long)relocated_kexec_smp_wait,
reboot_code_buffer + relocate_new_kernel_size);
relocated_kexec_smp_wait(NULL);
}
#endif
void kexec_reboot(void)
{
void (*do_kexec)(void) __noreturn;
#ifdef CONFIG_SMP
if (smp_processor_id() > 0) {
/*
* Instead of cpu_relax() or wait, this is needed for kexec
* smp reboot. Kdump usually doesn't require an smp new
* kernel, but kexec may do.
*/
kexec_nonboot_cpu();
/* NOTREACHED */
}
#endif
/*
* Make sure we get correct instructions written by the
* machine_kexec() CPU.
*/
local_flush_icache_range(reboot_code_buffer,
reboot_code_buffer + relocate_new_kernel_size);
do_kexec = (void *)reboot_code_buffer;
do_kexec();
}
void void
machine_kexec(struct kimage *image) machine_kexec(struct kimage *image)
{ {
unsigned long reboot_code_buffer;
unsigned long entry; unsigned long entry;
unsigned long *ptr; unsigned long *ptr;
...@@ -128,6 +203,7 @@ machine_kexec(struct kimage *image) ...@@ -128,6 +203,7 @@ 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");
/* Make reboot code buffer available to the boot CPU. */
__flush_cache_all(); __flush_cache_all();
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
/* All secondary cpus now may jump to kexec_wait cycle */ /* All secondary cpus now may jump to kexec_wait cycle */
...@@ -136,5 +212,5 @@ machine_kexec(struct kimage *image) ...@@ -136,5 +212,5 @@ machine_kexec(struct kimage *image)
smp_wmb(); smp_wmb();
atomic_set(&kexec_ready_to_reboot, 1); atomic_set(&kexec_ready_to_reboot, 1);
#endif #endif
((noretfun_t) reboot_code_buffer)(); kexec_reboot();
} }
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include <linux/linkage.h> #include <linux/linkage.h>
#include <linux/bug.h> #include <linux/bug.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/kexec.h>
#include <asm/time.h> #include <asm/time.h>
#include <asm/pgtable.h> #include <asm/pgtable.h>
...@@ -423,6 +424,9 @@ const struct plat_smp_ops bmips43xx_smp_ops = { ...@@ -423,6 +424,9 @@ const struct plat_smp_ops bmips43xx_smp_ops = {
.cpu_disable = bmips_cpu_disable, .cpu_disable = bmips_cpu_disable,
.cpu_die = bmips_cpu_die, .cpu_die = bmips_cpu_die,
#endif #endif
#ifdef CONFIG_KEXEC
.kexec_nonboot_cpu = kexec_nonboot_cpu_jump,
#endif
}; };
const struct plat_smp_ops bmips5000_smp_ops = { const struct plat_smp_ops bmips5000_smp_ops = {
...@@ -437,6 +441,9 @@ const struct plat_smp_ops bmips5000_smp_ops = { ...@@ -437,6 +441,9 @@ const struct plat_smp_ops bmips5000_smp_ops = {
.cpu_disable = bmips_cpu_disable, .cpu_disable = bmips_cpu_disable,
.cpu_die = bmips_cpu_die, .cpu_die = bmips_cpu_die,
#endif #endif
#ifdef CONFIG_KEXEC
.kexec_nonboot_cpu = kexec_nonboot_cpu_jump,
#endif
}; };
#endif /* CONFIG_SMP */ #endif /* CONFIG_SMP */
......
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#include <linux/sched/task_stack.h> #include <linux/sched/task_stack.h>
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/cpufreq.h> #include <linux/cpufreq.h>
#include <linux/kexec.h>
#include <asm/processor.h> #include <asm/processor.h>
#include <asm/time.h> #include <asm/time.h>
#include <asm/clock.h> #include <asm/clock.h>
...@@ -749,4 +750,7 @@ const struct plat_smp_ops loongson3_smp_ops = { ...@@ -749,4 +750,7 @@ const struct plat_smp_ops loongson3_smp_ops = {
.cpu_disable = loongson3_cpu_disable, .cpu_disable = loongson3_cpu_disable,
.cpu_die = loongson3_cpu_die, .cpu_die = loongson3_cpu_die,
#endif #endif
#ifdef CONFIG_KEXEC
.kexec_nonboot_cpu = kexec_nonboot_cpu_jump,
#endif
}; };
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment