2019-06-04 16:11:33 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* linux/arch/arm/kernel/smp.c
|
|
|
|
*
|
|
|
|
* Copyright (C) 2002 ARM Limited, All Rights Reserved.
|
|
|
|
*/
|
2006-10-25 20:59:16 +08:00
|
|
|
#include <linux/module.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/spinlock.h>
|
2017-02-02 02:08:20 +08:00
|
|
|
#include <linux/sched/mm.h>
|
2017-02-09 01:51:36 +08:00
|
|
|
#include <linux/sched/hotplug.h>
|
2017-02-09 01:51:37 +08:00
|
|
|
#include <linux/sched/task_stack.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/cache.h>
|
|
|
|
#include <linux/profile.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/mm.h>
|
2007-07-30 06:36:13 +08:00
|
|
|
#include <linux/err.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <linux/cpu.h>
|
|
|
|
#include <linux/seq_file.h>
|
2006-10-25 20:59:16 +08:00
|
|
|
#include <linux/irq.h>
|
2014-09-04 06:57:13 +08:00
|
|
|
#include <linux/nmi.h>
|
2009-05-18 01:58:34 +08:00
|
|
|
#include <linux/percpu.h>
|
|
|
|
#include <linux/clockchips.h>
|
2010-11-30 19:07:35 +08:00
|
|
|
#include <linux/completion.h>
|
2012-09-05 07:08:59 +08:00
|
|
|
#include <linux/cpufreq.h>
|
2013-10-30 03:32:56 +08:00
|
|
|
#include <linux/irq_work.h>
|
2020-06-24 04:15:00 +08:00
|
|
|
#include <linux/kernel_stat.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-07-27 07:09:06 +08:00
|
|
|
#include <linux/atomic.h>
|
2018-05-10 20:00:43 +08:00
|
|
|
#include <asm/bugs.h>
|
2011-09-08 16:06:10 +08:00
|
|
|
#include <asm/smp.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <asm/cacheflush.h>
|
|
|
|
#include <asm/cpu.h>
|
2009-06-11 22:35:00 +08:00
|
|
|
#include <asm/cputype.h>
|
2011-10-08 18:20:42 +08:00
|
|
|
#include <asm/exception.h>
|
2011-09-30 18:43:29 +08:00
|
|
|
#include <asm/idmap.h>
|
2011-08-08 20:21:59 +08:00
|
|
|
#include <asm/topology.h>
|
2005-06-18 16:33:31 +08:00
|
|
|
#include <asm/mmu_context.h>
|
2018-07-19 19:21:31 +08:00
|
|
|
#include <asm/procinfo.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <asm/processor.h>
|
2010-10-01 22:38:24 +08:00
|
|
|
#include <asm/sections.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <asm/tlbflush.h>
|
|
|
|
#include <asm/ptrace.h>
|
2011-08-24 05:19:29 +08:00
|
|
|
#include <asm/smp_plat.h>
|
2012-02-18 00:54:28 +08:00
|
|
|
#include <asm/virt.h>
|
2011-09-08 16:06:10 +08:00
|
|
|
#include <asm/mach/arch.h>
|
2013-02-23 02:51:30 +08:00
|
|
|
#include <asm/mpu.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2014-07-26 04:05:31 +08:00
|
|
|
#include <trace/events/ipi.h>
|
|
|
|
|
2005-06-18 16:33:31 +08:00
|
|
|
/*
|
|
|
|
* as from 2.5, kernels no longer have an init_tasks structure
|
|
|
|
* so we need some other way of telling a new secondary core
|
|
|
|
* where to place its SVC stack
|
|
|
|
*/
|
|
|
|
struct secondary_data secondary_data;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
enum ipi_msg_type {
|
2012-09-19 15:16:07 +08:00
|
|
|
IPI_WAKEUP,
|
|
|
|
IPI_TIMER,
|
2005-04-17 06:20:36 +08:00
|
|
|
IPI_RESCHEDULE,
|
|
|
|
IPI_CALL_FUNC,
|
|
|
|
IPI_CPU_STOP,
|
2013-10-30 03:32:56 +08:00
|
|
|
IPI_IRQ_WORK,
|
2012-11-28 10:54:41 +08:00
|
|
|
IPI_COMPLETION,
|
2020-06-24 04:15:00 +08:00
|
|
|
NR_IPI,
|
2019-05-15 06:41:48 +08:00
|
|
|
/*
|
|
|
|
* CPU_BACKTRACE is special and not included in NR_IPI
|
|
|
|
* or tracable with trace_ipi_*
|
|
|
|
*/
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
IPI_CPU_BACKTRACE = NR_IPI,
|
2015-12-18 18:06:47 +08:00
|
|
|
/*
|
|
|
|
* SGI8-15 can be reserved by secure firmware, and thus may
|
|
|
|
* not be usable by the kernel. Please keep the above limited
|
|
|
|
* to at most 8 entries.
|
|
|
|
*/
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
MAX_IPI
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
static int ipi_irq_base __read_mostly;
|
|
|
|
static int nr_ipi __read_mostly = NR_IPI;
|
|
|
|
static struct irq_desc *ipi_desc[MAX_IPI] __read_mostly;
|
|
|
|
|
|
|
|
static void ipi_setup(int cpu);
|
|
|
|
|
2012-01-18 23:59:45 +08:00
|
|
|
static DECLARE_COMPLETION(cpu_running);
|
|
|
|
|
2016-08-11 05:46:49 +08:00
|
|
|
static struct smp_operations smp_ops __ro_after_init;
|
2011-09-08 16:06:10 +08:00
|
|
|
|
2015-08-26 14:49:11 +08:00
|
|
|
void __init smp_set_ops(const struct smp_operations *ops)
|
2011-09-08 16:06:10 +08:00
|
|
|
{
|
|
|
|
if (ops)
|
|
|
|
smp_ops = *ops;
|
|
|
|
};
|
|
|
|
|
2012-07-22 03:55:04 +08:00
|
|
|
static unsigned long get_arch_pgd(pgd_t *pgd)
|
|
|
|
{
|
2015-04-05 03:09:46 +08:00
|
|
|
#ifdef CONFIG_ARM_LPAE
|
|
|
|
return __phys_to_pfn(virt_to_phys(pgd));
|
|
|
|
#else
|
|
|
|
return virt_to_phys(pgd);
|
|
|
|
#endif
|
2012-07-22 03:55:04 +08:00
|
|
|
}
|
|
|
|
|
2018-07-19 19:21:31 +08:00
|
|
|
#if defined(CONFIG_BIG_LITTLE) && defined(CONFIG_HARDEN_BRANCH_PREDICTOR)
|
|
|
|
static int secondary_biglittle_prepare(unsigned int cpu)
|
|
|
|
{
|
|
|
|
if (!cpu_vtable[cpu])
|
|
|
|
cpu_vtable[cpu] = kzalloc(sizeof(*cpu_vtable[cpu]), GFP_KERNEL);
|
|
|
|
|
|
|
|
return cpu_vtable[cpu] ? 0 : -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void secondary_biglittle_init(void)
|
|
|
|
{
|
|
|
|
init_proc_vtable(lookup_processor(read_cpuid_id())->proc);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static int secondary_biglittle_prepare(unsigned int cpu)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void secondary_biglittle_init(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-06-18 03:43:14 +08:00
|
|
|
int __cpu_up(unsigned int cpu, struct task_struct *idle)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2014-08-21 03:49:54 +08:00
|
|
|
if (!smp_ops.smp_boot_secondary)
|
|
|
|
return -ENOSYS;
|
|
|
|
|
2018-07-19 19:21:31 +08:00
|
|
|
ret = secondary_biglittle_prepare(cpu);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2005-06-18 16:33:31 +08:00
|
|
|
/*
|
|
|
|
* We need to tell the secondary core where to find
|
|
|
|
* its stack and the page tables.
|
|
|
|
*/
|
2006-01-12 17:05:58 +08:00
|
|
|
secondary_data.stack = task_stack_page(idle) + THREAD_START_SP;
|
2013-02-23 02:51:30 +08:00
|
|
|
#ifdef CONFIG_ARM_MPU
|
2017-10-16 19:54:05 +08:00
|
|
|
secondary_data.mpu_rgn_info = &mpu_rgn_info;
|
2013-02-23 02:51:30 +08:00
|
|
|
#endif
|
|
|
|
|
2012-02-28 21:02:59 +08:00
|
|
|
#ifdef CONFIG_MMU
|
2015-04-05 03:09:46 +08:00
|
|
|
secondary_data.pgdir = virt_to_phys(idmap_pgd);
|
2012-07-22 03:55:04 +08:00
|
|
|
secondary_data.swapper_pg_dir = get_arch_pgd(swapper_pg_dir);
|
2012-02-28 21:02:59 +08:00
|
|
|
#endif
|
2021-09-18 16:44:35 +08:00
|
|
|
secondary_data.task = idle;
|
2013-12-09 23:10:18 +08:00
|
|
|
sync_cache_w(&secondary_data);
|
2005-06-18 16:33:31 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Now bring the CPU into our world.
|
|
|
|
*/
|
2014-08-21 03:49:54 +08:00
|
|
|
ret = smp_ops.smp_boot_secondary(cpu, idle);
|
2005-06-18 16:33:31 +08:00
|
|
|
if (ret == 0) {
|
|
|
|
/*
|
|
|
|
* CPU was successfully started, wait for it
|
|
|
|
* to come online or time out.
|
|
|
|
*/
|
2012-01-18 23:59:45 +08:00
|
|
|
wait_for_completion_timeout(&cpu_running,
|
|
|
|
msecs_to_jiffies(1000));
|
2005-06-18 16:33:31 +08:00
|
|
|
|
2010-12-18 20:34:39 +08:00
|
|
|
if (!cpu_online(cpu)) {
|
|
|
|
pr_crit("CPU%u: failed to come online\n", cpu);
|
2005-06-18 16:33:31 +08:00
|
|
|
ret = -EIO;
|
2010-12-18 20:34:39 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pr_err("CPU%u: failed to boot: %d\n", cpu, ret);
|
2005-06-18 16:33:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-23 02:51:30 +08:00
|
|
|
memset(&secondary_data, 0, sizeof(secondary_data));
|
2005-04-17 06:20:36 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-09-08 16:06:10 +08:00
|
|
|
/* platform specific SMP operations */
|
2011-09-27 21:48:23 +08:00
|
|
|
void __init smp_init_cpus(void)
|
2011-09-08 16:06:10 +08:00
|
|
|
{
|
|
|
|
if (smp_ops.smp_init_cpus)
|
|
|
|
smp_ops.smp_init_cpus();
|
|
|
|
}
|
|
|
|
|
2015-04-01 20:36:57 +08:00
|
|
|
int platform_can_secondary_boot(void)
|
|
|
|
{
|
|
|
|
return !!smp_ops.smp_boot_secondary;
|
|
|
|
}
|
|
|
|
|
2013-08-03 03:52:49 +08:00
|
|
|
int platform_can_cpu_hotplug(void)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_HOTPLUG_CPU
|
|
|
|
if (smp_ops.cpu_kill)
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-03 06:24:33 +08:00
|
|
|
#ifdef CONFIG_HOTPLUG_CPU
|
2011-09-27 21:48:23 +08:00
|
|
|
static int platform_cpu_kill(unsigned int cpu)
|
2011-09-08 16:06:10 +08:00
|
|
|
{
|
|
|
|
if (smp_ops.cpu_kill)
|
|
|
|
return smp_ops.cpu_kill(cpu);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-09-27 21:48:23 +08:00
|
|
|
static int platform_cpu_disable(unsigned int cpu)
|
2011-09-08 16:06:10 +08:00
|
|
|
{
|
|
|
|
if (smp_ops.cpu_disable)
|
|
|
|
return smp_ops.cpu_disable(cpu);
|
|
|
|
|
2015-07-29 07:34:48 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int platform_can_hotplug_cpu(unsigned int cpu)
|
|
|
|
{
|
|
|
|
/* cpu_die must be specified to support hotplug */
|
|
|
|
if (!smp_ops.cpu_die)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (smp_ops.cpu_can_disable)
|
|
|
|
return smp_ops.cpu_can_disable(cpu);
|
|
|
|
|
2011-09-08 16:06:10 +08:00
|
|
|
/*
|
|
|
|
* By default, allow disabling all CPUs except the first one,
|
|
|
|
* since this is special on a lot of platforms, e.g. because
|
|
|
|
* of clock tick interrupts.
|
|
|
|
*/
|
2015-07-29 07:34:48 +08:00
|
|
|
return cpu != 0;
|
2011-09-08 16:06:10 +08:00
|
|
|
}
|
2015-07-29 07:34:48 +08:00
|
|
|
|
2020-09-19 00:19:46 +08:00
|
|
|
static void ipi_teardown(int cpu)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (WARN_ON_ONCE(!ipi_irq_base))
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < nr_ipi; i++)
|
|
|
|
disable_percpu_irq(ipi_irq_base + i);
|
|
|
|
}
|
|
|
|
|
2005-11-03 06:24:33 +08:00
|
|
|
/*
|
|
|
|
* __cpu_disable runs on the processor to be shutdown.
|
|
|
|
*/
|
2013-06-18 03:43:14 +08:00
|
|
|
int __cpu_disable(void)
|
2005-11-03 06:24:33 +08:00
|
|
|
{
|
|
|
|
unsigned int cpu = smp_processor_id();
|
|
|
|
int ret;
|
|
|
|
|
2010-05-15 17:18:05 +08:00
|
|
|
ret = platform_cpu_disable(cpu);
|
2005-11-03 06:24:33 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2019-11-29 23:23:02 +08:00
|
|
|
#ifdef CONFIG_GENERIC_ARCH_TOPOLOGY
|
|
|
|
remove_cpu_topology(cpu);
|
|
|
|
#endif
|
|
|
|
|
2005-11-03 06:24:33 +08:00
|
|
|
/*
|
|
|
|
* Take this CPU offline. Once we clear this, we can't return,
|
|
|
|
* and we must not schedule until we're ready to give up the cpu.
|
|
|
|
*/
|
2009-05-28 21:16:52 +08:00
|
|
|
set_cpu_online(cpu, false);
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
ipi_teardown(cpu);
|
2005-11-03 06:24:33 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* OK - migrate IRQs away from this CPU
|
|
|
|
*/
|
2019-01-21 21:42:42 +08:00
|
|
|
irq_migrate_all_off_this_cpu();
|
2005-11-03 06:24:33 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Flush user cache and TLB mappings, and then remove this CPU
|
|
|
|
* from the vm mask set of all processes.
|
2012-09-07 13:39:15 +08:00
|
|
|
*
|
|
|
|
* Caches are flushed to the Level of Unification Inner Shareable
|
|
|
|
* to write-back dirty lines to unified caches shared by all CPUs.
|
2005-11-03 06:24:33 +08:00
|
|
|
*/
|
2012-09-07 13:39:15 +08:00
|
|
|
flush_cache_louis();
|
2005-11-03 06:24:33 +08:00
|
|
|
local_flush_tlb_all();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2023-05-13 05:07:32 +08:00
|
|
|
* called on the thread which is asking for a CPU to be shutdown after the
|
|
|
|
* shutdown completed.
|
2005-11-03 06:24:33 +08:00
|
|
|
*/
|
2023-05-13 05:07:32 +08:00
|
|
|
void arch_cpuhp_cleanup_dead_cpu(unsigned int cpu)
|
2005-11-03 06:24:33 +08:00
|
|
|
{
|
2017-01-19 08:26:28 +08:00
|
|
|
pr_debug("CPU%u: shutdown\n", cpu);
|
2010-11-30 19:07:35 +08:00
|
|
|
|
2018-05-08 22:19:40 +08:00
|
|
|
clear_tasks_mm_cpumask(cpu);
|
2013-04-19 01:05:29 +08:00
|
|
|
/*
|
|
|
|
* platform_cpu_kill() is generally expected to do the powering off
|
|
|
|
* and/or cutting of clocks to the dying CPU. Optionally, this may
|
|
|
|
* be done by the CPU which is dying in preference to supporting
|
|
|
|
* this call, but that means there is _no_ synchronisation between
|
|
|
|
* the requesting CPU and the dying CPU actually losing power.
|
|
|
|
*/
|
2005-11-03 06:24:33 +08:00
|
|
|
if (!platform_cpu_kill(cpu))
|
2014-10-28 19:26:42 +08:00
|
|
|
pr_err("CPU%u: unable to kill\n", cpu);
|
2005-11-03 06:24:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Called from the idle thread for the CPU which has been shutdown.
|
|
|
|
*
|
|
|
|
* Note that we disable IRQs here, but do not re-enable them
|
|
|
|
* before returning to the caller. This is also the behaviour
|
|
|
|
* of the other hotplug-cpu capable cores, so presumably coming
|
|
|
|
* out of idle fixes this.
|
|
|
|
*/
|
2023-02-14 15:05:58 +08:00
|
|
|
void __noreturn arch_cpu_idle_dead(void)
|
2005-11-03 06:24:33 +08:00
|
|
|
{
|
|
|
|
unsigned int cpu = smp_processor_id();
|
|
|
|
|
|
|
|
idle_task_exit();
|
|
|
|
|
2010-11-30 20:21:30 +08:00
|
|
|
local_irq_disable();
|
|
|
|
|
2013-04-19 01:05:29 +08:00
|
|
|
/*
|
|
|
|
* Flush the data out of the L1 cache for this CPU. This must be
|
|
|
|
* before the completion to ensure that data is safely written out
|
|
|
|
* before platform_cpu_kill() gets called - which may disable
|
|
|
|
* *this* CPU and power down its cache.
|
|
|
|
*/
|
|
|
|
flush_cache_louis();
|
|
|
|
|
|
|
|
/*
|
2023-05-13 05:07:32 +08:00
|
|
|
* Tell cpuhp_bp_sync_dead() that this CPU is now safe to dispose
|
|
|
|
* of. Once this returns, power and/or clocks can be removed at
|
|
|
|
* any point from this CPU and its cache by platform_cpu_kill().
|
2013-04-19 01:05:29 +08:00
|
|
|
*/
|
2023-05-13 05:07:32 +08:00
|
|
|
cpuhp_ap_report_dead();
|
2010-11-30 19:07:35 +08:00
|
|
|
|
2005-11-03 06:24:33 +08:00
|
|
|
/*
|
2013-04-19 01:05:29 +08:00
|
|
|
* Ensure that the cache lines associated with that completion are
|
|
|
|
* written out. This covers the case where _this_ CPU is doing the
|
|
|
|
* powering down, to ensure that the completion is visible to the
|
|
|
|
* CPU waiting for this one.
|
|
|
|
*/
|
|
|
|
flush_cache_louis();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The actual CPU shutdown procedure is at least platform (if not
|
|
|
|
* CPU) specific. This may remove power, or it may simply spin.
|
|
|
|
*
|
|
|
|
* Platforms are generally expected *NOT* to return from this call,
|
|
|
|
* although there are some which do because they have no way to
|
|
|
|
* power down the CPU. These platforms are the _only_ reason we
|
|
|
|
* have a return path which uses the fragment of assembly below.
|
|
|
|
*
|
|
|
|
* The return path should not be used for platforms which can
|
|
|
|
* power off the CPU.
|
2005-11-03 06:24:33 +08:00
|
|
|
*/
|
2013-01-14 23:54:28 +08:00
|
|
|
if (smp_ops.cpu_die)
|
|
|
|
smp_ops.cpu_die(cpu);
|
2005-11-03 06:24:33 +08:00
|
|
|
|
2014-01-11 19:25:37 +08:00
|
|
|
pr_warn("CPU%u: smp_ops.cpu_die() returned, trying to resuscitate\n",
|
|
|
|
cpu);
|
|
|
|
|
2005-11-03 06:24:33 +08:00
|
|
|
/*
|
|
|
|
* Do not return to the idle loop - jump back to the secondary
|
|
|
|
* cpu initialisation. There's some initialisation which needs
|
|
|
|
* to be repeated to undo the effects of taking the CPU offline.
|
|
|
|
*/
|
|
|
|
__asm__("mov sp, %0\n"
|
2010-12-21 00:58:19 +08:00
|
|
|
" mov fp, #0\n"
|
2021-09-18 16:44:35 +08:00
|
|
|
" mov r0, %1\n"
|
2005-11-03 06:24:33 +08:00
|
|
|
" b secondary_start_kernel"
|
|
|
|
:
|
2021-09-18 16:44:35 +08:00
|
|
|
: "r" (task_stack_page(current) + THREAD_SIZE - 8),
|
|
|
|
"r" (current)
|
|
|
|
: "r0");
|
2023-02-17 02:38:55 +08:00
|
|
|
|
|
|
|
unreachable();
|
2005-11-03 06:24:33 +08:00
|
|
|
}
|
|
|
|
#endif /* CONFIG_HOTPLUG_CPU */
|
|
|
|
|
2010-12-03 19:09:48 +08:00
|
|
|
/*
|
|
|
|
* Called by both boot and secondaries to move global data into
|
|
|
|
* per-processor storage.
|
|
|
|
*/
|
2013-06-18 03:43:14 +08:00
|
|
|
static void smp_store_cpu_info(unsigned int cpuid)
|
2010-12-03 19:09:48 +08:00
|
|
|
{
|
|
|
|
struct cpuinfo_arm *cpu_info = &per_cpu(cpu_data, cpuid);
|
|
|
|
|
|
|
|
cpu_info->loops_per_jiffy = loops_per_jiffy;
|
2012-11-06 19:57:43 +08:00
|
|
|
cpu_info->cpuid = read_cpuid_id();
|
2011-08-08 20:21:59 +08:00
|
|
|
|
|
|
|
store_cpu_topology(cpuid);
|
2019-05-28 16:38:14 +08:00
|
|
|
check_cpu_icache_size(cpuid);
|
2010-12-03 19:09:48 +08:00
|
|
|
}
|
|
|
|
|
ARM: implement THREAD_INFO_IN_TASK for uniprocessor systems
On UP systems, only a single task can be 'current' at the same time,
which means we can use a global variable to track it. This means we can
also enable THREAD_INFO_IN_TASK for those systems, as in that case,
thread_info is accessed via current rather than the other way around,
removing the need to store thread_info at the base of the task stack.
This, in turn, permits us to enable IRQ stacks and vmap'ed stacks on UP
systems as well.
To partially mitigate the performance overhead of this arrangement, use
a ADD/ADD/LDR sequence with the appropriate PC-relative group
relocations to load the value of current when needed. This means that
accessing current will still only require a single load as before,
avoiding the need for a literal to carry the address of the global
variable in each function. However, accessing thread_info will now
require this load as well.
Acked-by: Linus Walleij <linus.walleij@linaro.org>
Acked-by: Nicolas Pitre <nico@fluxnic.net>
Signed-off-by: Ard Biesheuvel <ardb@kernel.org>
Tested-by: Marc Zyngier <maz@kernel.org>
Tested-by: Vladimir Murzin <vladimir.murzin@arm.com> # ARMv7M
2021-11-24 21:08:11 +08:00
|
|
|
static void set_current(struct task_struct *cur)
|
|
|
|
{
|
|
|
|
/* Set TPIDRURO */
|
|
|
|
asm("mcr p15, 0, %0, c13, c0, 3" :: "r"(cur) : "memory");
|
|
|
|
}
|
|
|
|
|
2005-06-18 16:33:31 +08:00
|
|
|
/*
|
|
|
|
* This is the secondary CPU boot entry. We're using this CPUs
|
|
|
|
* idle thread stack, but a set of temporary page tables.
|
|
|
|
*/
|
2021-09-18 16:44:35 +08:00
|
|
|
asmlinkage void secondary_start_kernel(struct task_struct *task)
|
2005-06-18 16:33:31 +08:00
|
|
|
{
|
|
|
|
struct mm_struct *mm = &init_mm;
|
2012-10-20 00:53:01 +08:00
|
|
|
unsigned int cpu;
|
|
|
|
|
ARM: smp: Store current pointer in TPIDRURO register if available
Now that the user space TLS register is assigned on every return to user
space, we can use it to keep the 'current' pointer while running in the
kernel. This removes the need to access it via thread_info, which is
located at the base of the stack, but will be moved out of there in a
subsequent patch.
Use the __builtin_thread_pointer() helper when available - this will
help GCC understand that reloading the value within the same function is
not necessary, even when using the per-task stack protector (which also
generates accesses via the TLS register). For example, the generated
code below loads TPIDRURO only once, and uses it to access both the
stack canary and the preempt_count fields.
<do_one_initcall>:
e92d 41f0 stmdb sp!, {r4, r5, r6, r7, r8, lr}
ee1d 4f70 mrc 15, 0, r4, cr13, cr0, {3}
4606 mov r6, r0
b094 sub sp, #80 ; 0x50
f8d4 34e8 ldr.w r3, [r4, #1256] ; 0x4e8 <- stack canary
9313 str r3, [sp, #76] ; 0x4c
f8d4 8004 ldr.w r8, [r4, #4] <- preempt count
Co-developed-by: Keith Packard <keithpac@amazon.com>
Signed-off-by: Keith Packard <keithpac@amazon.com>
Signed-off-by: Ard Biesheuvel <ardb@kernel.org>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
Tested-by: Amit Daniel Kachhap <amit.kachhap@arm.com>
2021-09-18 16:44:37 +08:00
|
|
|
set_current(task);
|
|
|
|
|
2018-07-19 19:21:31 +08:00
|
|
|
secondary_biglittle_init();
|
|
|
|
|
2012-10-20 00:53:01 +08:00
|
|
|
/*
|
|
|
|
* The identity mapping is uncached (strongly ordered), so
|
|
|
|
* switch away from it before attempting any exclusive accesses.
|
|
|
|
*/
|
|
|
|
cpu_switch_mm(mm->pgd, mm);
|
2013-03-01 00:48:40 +08:00
|
|
|
local_flush_bp_all();
|
2012-10-20 00:53:01 +08:00
|
|
|
enter_lazy_tlb(mm, current);
|
|
|
|
local_flush_tlb_all();
|
2005-06-18 16:33:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* All kernel threads share the same mm context; grab a
|
|
|
|
* reference and switch to it.
|
|
|
|
*/
|
2012-10-20 00:53:01 +08:00
|
|
|
cpu = smp_processor_id();
|
2017-02-28 06:30:07 +08:00
|
|
|
mmgrab(mm);
|
2005-06-18 16:33:31 +08:00
|
|
|
current->active_mm = mm;
|
2009-09-24 23:34:49 +08:00
|
|
|
cpumask_set_cpu(cpu, mm_cpumask(mm));
|
2005-06-18 16:33:31 +08:00
|
|
|
|
2012-11-30 03:39:54 +08:00
|
|
|
cpu_init();
|
|
|
|
|
2017-12-27 17:38:55 +08:00
|
|
|
#ifndef CONFIG_MMU
|
|
|
|
setup_vectors_base();
|
|
|
|
#endif
|
2014-10-28 20:08:34 +08:00
|
|
|
pr_debug("CPU%u: Booted secondary processor\n", cpu);
|
2012-05-06 03:58:13 +08:00
|
|
|
|
2010-12-03 23:00:49 +08:00
|
|
|
trace_hardirqs_off();
|
2005-06-18 16:33:31 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Give the platform a chance to do its own initialisation.
|
|
|
|
*/
|
2013-01-14 23:54:28 +08:00
|
|
|
if (smp_ops.smp_secondary_init)
|
|
|
|
smp_ops.smp_secondary_init(cpu);
|
2005-06-18 16:33:31 +08:00
|
|
|
|
2008-09-07 22:57:22 +08:00
|
|
|
notify_cpu_starting(cpu);
|
2008-02-05 00:30:57 +08:00
|
|
|
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
ipi_setup(cpu);
|
|
|
|
|
2005-06-18 16:33:31 +08:00
|
|
|
calibrate_delay();
|
|
|
|
|
|
|
|
smp_store_cpu_info(cpu);
|
|
|
|
|
|
|
|
/*
|
2011-06-20 23:46:01 +08:00
|
|
|
* OK, now it's safe to let the boot CPU continue. Wait for
|
|
|
|
* the CPU migration code to notice that the CPU is online
|
2012-01-18 23:59:45 +08:00
|
|
|
* before we continue - which happens after __cpu_up returns.
|
2005-06-18 16:33:31 +08:00
|
|
|
*/
|
2009-05-28 21:16:52 +08:00
|
|
|
set_cpu_online(cpu, true);
|
2018-05-10 20:00:43 +08:00
|
|
|
|
|
|
|
check_other_bugs();
|
|
|
|
|
2012-01-18 23:59:45 +08:00
|
|
|
complete(&cpu_running);
|
2011-10-14 19:44:41 +08:00
|
|
|
|
|
|
|
local_irq_enable();
|
|
|
|
local_fiq_enable();
|
2015-08-25 20:52:09 +08:00
|
|
|
local_abt_enable();
|
2011-10-14 19:44:41 +08:00
|
|
|
|
2005-06-18 16:33:31 +08:00
|
|
|
/*
|
|
|
|
* OK, it's off to the idle thread for us
|
|
|
|
*/
|
2016-02-27 02:43:40 +08:00
|
|
|
cpu_startup_entry(CPUHP_AP_ONLINE_IDLE);
|
2005-06-18 16:33:31 +08:00
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
void __init smp_cpus_done(unsigned int max_cpus)
|
|
|
|
{
|
2015-01-05 03:01:23 +08:00
|
|
|
int cpu;
|
|
|
|
unsigned long bogosum = 0;
|
|
|
|
|
|
|
|
for_each_online_cpu(cpu)
|
|
|
|
bogosum += per_cpu(cpu_data, cpu).loops_per_jiffy;
|
|
|
|
|
|
|
|
printk(KERN_INFO "SMP: Total of %d processors activated "
|
|
|
|
"(%lu.%02lu BogoMIPS).\n",
|
|
|
|
num_online_cpus(),
|
|
|
|
bogosum / (500000/HZ),
|
|
|
|
(bogosum / (5000/HZ)) % 100);
|
|
|
|
|
2012-02-18 00:54:28 +08:00
|
|
|
hyp_mode_check();
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void __init smp_prepare_boot_cpu(void)
|
|
|
|
{
|
2012-11-30 03:39:54 +08:00
|
|
|
set_my_cpu_offset(per_cpu_offset(smp_processor_id()));
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2010-12-03 19:09:48 +08:00
|
|
|
void __init smp_prepare_cpus(unsigned int max_cpus)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2010-12-03 19:09:48 +08:00
|
|
|
unsigned int ncores = num_possible_cpus();
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2011-08-08 20:21:59 +08:00
|
|
|
init_cpu_topology();
|
|
|
|
|
2010-12-03 19:09:48 +08:00
|
|
|
smp_store_cpu_info(smp_processor_id());
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*
|
2010-12-03 19:09:48 +08:00
|
|
|
* are we trying to boot more cores than exist?
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2010-12-03 19:09:48 +08:00
|
|
|
if (max_cpus > ncores)
|
|
|
|
max_cpus = ncores;
|
2011-07-07 08:56:51 +08:00
|
|
|
if (ncores > 1 && max_cpus) {
|
|
|
|
/*
|
|
|
|
* Initialise the present map, which describes the set of CPUs
|
|
|
|
* actually populated at the present time. A platform should
|
2013-01-14 23:54:28 +08:00
|
|
|
* re-initialize the map in the platforms smp_prepare_cpus()
|
|
|
|
* if present != possible (e.g. physical hotplug).
|
2011-07-07 08:56:51 +08:00
|
|
|
*/
|
2012-03-29 13:08:30 +08:00
|
|
|
init_cpu_present(cpu_possible_mask);
|
2011-07-07 08:56:51 +08:00
|
|
|
|
2010-12-03 19:09:48 +08:00
|
|
|
/*
|
|
|
|
* Initialise the SCU if there are more than one CPU
|
|
|
|
* and let them know where to start.
|
|
|
|
*/
|
2013-01-14 23:54:28 +08:00
|
|
|
if (smp_ops.smp_prepare_cpus)
|
|
|
|
smp_ops.smp_prepare_cpus(max_cpus);
|
2010-12-03 19:09:48 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2014-07-26 04:05:31 +08:00
|
|
|
static const char *ipi_types[NR_IPI] __tracepoint_string = {
|
2020-11-30 18:24:09 +08:00
|
|
|
[IPI_WAKEUP] = "CPU wakeup interrupts",
|
|
|
|
[IPI_TIMER] = "Timer broadcast interrupts",
|
|
|
|
[IPI_RESCHEDULE] = "Rescheduling interrupts",
|
|
|
|
[IPI_CALL_FUNC] = "Function call interrupts",
|
|
|
|
[IPI_CPU_STOP] = "CPU stop interrupts",
|
|
|
|
[IPI_IRQ_WORK] = "IRQ work interrupts",
|
|
|
|
[IPI_COMPLETION] = "completion interrupts",
|
2010-11-15 22:40:29 +08:00
|
|
|
};
|
|
|
|
|
2020-06-23 05:15:54 +08:00
|
|
|
static void smp_cross_call(const struct cpumask *target, unsigned int ipinr);
|
2014-07-26 04:05:31 +08:00
|
|
|
|
2010-11-15 22:33:51 +08:00
|
|
|
void show_ipi_list(struct seq_file *p, int prec)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2010-11-15 22:40:29 +08:00
|
|
|
unsigned int cpu, i;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-11-15 22:40:29 +08:00
|
|
|
for (i = 0; i < NR_IPI; i++) {
|
2020-09-25 23:22:00 +08:00
|
|
|
if (!ipi_desc[i])
|
|
|
|
continue;
|
|
|
|
|
2010-11-15 22:40:29 +08:00
|
|
|
seq_printf(p, "%*s%u: ", prec - 1, "IPI", i);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2012-12-04 04:13:03 +08:00
|
|
|
for_each_online_cpu(cpu)
|
2020-12-11 03:25:45 +08:00
|
|
|
seq_printf(p, "%10u ", irq_desc_kstat_cpu(ipi_desc[i], cpu));
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-11-15 22:40:29 +08:00
|
|
|
seq_printf(p, " %s\n", ipi_types[i]);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2014-07-26 04:05:31 +08:00
|
|
|
void arch_send_call_function_ipi_mask(const struct cpumask *mask)
|
|
|
|
{
|
|
|
|
smp_cross_call(mask, IPI_CALL_FUNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
void arch_send_wakeup_ipi_mask(const struct cpumask *mask)
|
|
|
|
{
|
|
|
|
smp_cross_call(mask, IPI_WAKEUP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void arch_send_call_function_single_ipi(int cpu)
|
|
|
|
{
|
2015-12-18 18:06:19 +08:00
|
|
|
smp_cross_call(cpumask_of(cpu), IPI_CALL_FUNC);
|
2014-07-26 04:05:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_IRQ_WORK
|
|
|
|
void arch_irq_work_raise(void)
|
|
|
|
{
|
2014-08-17 00:47:53 +08:00
|
|
|
if (arch_irq_work_has_interrupt())
|
2014-07-26 04:05:31 +08:00
|
|
|
smp_cross_call(cpumask_of(smp_processor_id()), IPI_IRQ_WORK);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-05-18 01:58:34 +08:00
|
|
|
#ifdef CONFIG_GENERIC_CLOCKEVENTS_BROADCAST
|
2012-10-30 20:13:42 +08:00
|
|
|
void tick_broadcast(const struct cpumask *mask)
|
2009-05-18 01:58:34 +08:00
|
|
|
{
|
2010-12-20 22:47:19 +08:00
|
|
|
smp_cross_call(mask, IPI_TIMER);
|
2009-05-18 01:58:34 +08:00
|
|
|
}
|
2010-07-26 20:19:43 +08:00
|
|
|
#endif
|
2009-05-18 01:58:34 +08:00
|
|
|
|
2009-07-03 21:44:46 +08:00
|
|
|
static DEFINE_RAW_SPINLOCK(stop_lock);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* ipi_cpu_stop - handle IPI from smp_send_stop()
|
|
|
|
*/
|
|
|
|
static void ipi_cpu_stop(unsigned int cpu)
|
|
|
|
{
|
ARM: 9257/1: Disable FIQs (but not IRQs) on CPUs shutdown paths
Currently the regular CPU shutdown path for ARM disables IRQs/FIQs
in the secondary CPUs - smp_send_stop() calls ipi_cpu_stop(), which
is responsible for that. IRQs are architecturally masked when we
take an interrupt, but FIQs are high priority than IRQs, hence they
aren't masked. With that said, it makes sense to disable FIQs here,
but there's no need for (re-)disabling IRQs.
More than that: there is an alternative path for disabling CPUs,
in the form of function crash_smp_send_stop(), which is used for
kexec/panic path. This function relies on a SMP call that also
triggers a busy-wait loop [at machine_crash_nonpanic_core()], but
without disabling FIQs. This might lead to odd scenarios, like
early interrupts in the boot of kexec'd kernel or even interrupts
in secondary "disabled" CPUs while the main one still works in the
panic path and assumes all secondary CPUs are (really!) off.
So, let's disable FIQs in both paths and *not* disable IRQs a second
time, since they are already masked in both paths by the architecture.
This way, we keep both CPU quiesce paths consistent and safe.
Cc: Marc Zyngier <maz@kernel.org>
Cc: Michael Kelley <mikelley@microsoft.com>
Signed-off-by: Guilherme G. Piccoli <gpiccoli@igalia.com>
Signed-off-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
2022-10-18 03:41:30 +08:00
|
|
|
local_fiq_disable();
|
|
|
|
|
2017-05-17 02:42:33 +08:00
|
|
|
if (system_state <= SYSTEM_RUNNING) {
|
2009-07-03 21:44:46 +08:00
|
|
|
raw_spin_lock(&stop_lock);
|
2014-10-28 19:26:42 +08:00
|
|
|
pr_crit("CPU%u: stopping\n", cpu);
|
2010-07-26 20:31:27 +08:00
|
|
|
dump_stack();
|
2009-07-03 21:44:46 +08:00
|
|
|
raw_spin_unlock(&stop_lock);
|
2010-07-26 20:31:27 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-05-28 21:16:52 +08:00
|
|
|
set_cpu_online(cpu, false);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
ARM: avoid Cortex-A9 livelock on tight dmb loops
machine_crash_nonpanic_core() does this:
while (1)
cpu_relax();
because the kernel has crashed, and we have no known safe way to deal
with the CPU. So, we place the CPU into an infinite loop which we
expect it to never exit - at least not until the system as a whole is
reset by some method.
In the absence of erratum 754327, this code assembles to:
b .
In other words, an infinite loop. When erratum 754327 is enabled,
this becomes:
1: dmb
b 1b
It has been observed that on some systems (eg, OMAP4) where, if a
crash is triggered, the system tries to kexec into the panic kernel,
but fails after taking the secondary CPU down - placing it into one
of these loops. This causes the system to livelock, and the most
noticable effect is the system stops after issuing:
Loading crashdump kernel...
to the system console.
The tested as working solution I came up with was to add wfe() to
these infinite loops thusly:
while (1) {
cpu_relax();
wfe();
}
which, without 754327 builds to:
1: wfe
b 1b
or with 754327 is enabled:
1: dmb
wfe
b 1b
Adding "wfe" does two things depending on the environment we're running
under:
- where we're running on bare metal, and the processor implements
"wfe", it stops us spinning endlessly in a loop where we're never
going to do any useful work.
- if we're running in a VM, it allows the CPU to be given back to the
hypervisor and rescheduled for other purposes (maybe a different VM)
rather than wasting CPU cycles inside a crashed VM.
However, in light of erratum 794072, Will Deacon wanted to see 10 nops
as well - which is reasonable to cover the case where we have erratum
754327 enabled _and_ we have a processor that doesn't implement the
wfe hint.
So, we now end up with:
1: wfe
b 1b
when erratum 754327 is disabled, or:
1: dmb
nop
nop
nop
nop
nop
nop
nop
nop
nop
nop
wfe
b 1b
when erratum 754327 is enabled. We also get the dmb + 10 nop
sequence elsewhere in the kernel, in terminating loops.
This is reasonable - it means we get the workaround for erratum
794072 when erratum 754327 is enabled, but still relinquish the dead
processor - either by placing it in a lower power mode when wfe is
implemented as such or by returning it to the hypervisior, or in the
case where wfe is a no-op, we use the workaround specified in erratum
794072 to avoid the problem.
These as two entirely orthogonal problems - the 10 nops addresses
erratum 794072, and the wfe is an optimisation that makes the system
more efficient when crashed either in terms of power consumption or
by allowing the host/other VMs to make use of the CPU.
I don't see any reason not to use kexec() inside a VM - it has the
potential to provide automated recovery from a failure of the VMs
kernel with the opportunity for saving a crashdump of the failure.
A panic() with a reboot timeout won't do that, and reading the
libvirt documentation, setting on_reboot to "preserve" won't either
(the documentation states "The preserve action for an on_reboot event
is treated as a destroy".) Surely it has to be a good thing to
avoiding having CPUs spinning inside a VM that is doing no useful
work.
Acked-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Russell King <rmk+kernel@armlinux.org.uk>
2018-04-10 18:35:36 +08:00
|
|
|
while (1) {
|
2005-04-17 06:20:36 +08:00
|
|
|
cpu_relax();
|
ARM: avoid Cortex-A9 livelock on tight dmb loops
machine_crash_nonpanic_core() does this:
while (1)
cpu_relax();
because the kernel has crashed, and we have no known safe way to deal
with the CPU. So, we place the CPU into an infinite loop which we
expect it to never exit - at least not until the system as a whole is
reset by some method.
In the absence of erratum 754327, this code assembles to:
b .
In other words, an infinite loop. When erratum 754327 is enabled,
this becomes:
1: dmb
b 1b
It has been observed that on some systems (eg, OMAP4) where, if a
crash is triggered, the system tries to kexec into the panic kernel,
but fails after taking the secondary CPU down - placing it into one
of these loops. This causes the system to livelock, and the most
noticable effect is the system stops after issuing:
Loading crashdump kernel...
to the system console.
The tested as working solution I came up with was to add wfe() to
these infinite loops thusly:
while (1) {
cpu_relax();
wfe();
}
which, without 754327 builds to:
1: wfe
b 1b
or with 754327 is enabled:
1: dmb
wfe
b 1b
Adding "wfe" does two things depending on the environment we're running
under:
- where we're running on bare metal, and the processor implements
"wfe", it stops us spinning endlessly in a loop where we're never
going to do any useful work.
- if we're running in a VM, it allows the CPU to be given back to the
hypervisor and rescheduled for other purposes (maybe a different VM)
rather than wasting CPU cycles inside a crashed VM.
However, in light of erratum 794072, Will Deacon wanted to see 10 nops
as well - which is reasonable to cover the case where we have erratum
754327 enabled _and_ we have a processor that doesn't implement the
wfe hint.
So, we now end up with:
1: wfe
b 1b
when erratum 754327 is disabled, or:
1: dmb
nop
nop
nop
nop
nop
nop
nop
nop
nop
nop
wfe
b 1b
when erratum 754327 is enabled. We also get the dmb + 10 nop
sequence elsewhere in the kernel, in terminating loops.
This is reasonable - it means we get the workaround for erratum
794072 when erratum 754327 is enabled, but still relinquish the dead
processor - either by placing it in a lower power mode when wfe is
implemented as such or by returning it to the hypervisior, or in the
case where wfe is a no-op, we use the workaround specified in erratum
794072 to avoid the problem.
These as two entirely orthogonal problems - the 10 nops addresses
erratum 794072, and the wfe is an optimisation that makes the system
more efficient when crashed either in terms of power consumption or
by allowing the host/other VMs to make use of the CPU.
I don't see any reason not to use kexec() inside a VM - it has the
potential to provide automated recovery from a failure of the VMs
kernel with the opportunity for saving a crashdump of the failure.
A panic() with a reboot timeout won't do that, and reading the
libvirt documentation, setting on_reboot to "preserve" won't either
(the documentation states "The preserve action for an on_reboot event
is treated as a destroy".) Surely it has to be a good thing to
avoiding having CPUs spinning inside a VM that is doing no useful
work.
Acked-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Russell King <rmk+kernel@armlinux.org.uk>
2018-04-10 18:35:36 +08:00
|
|
|
wfe();
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2012-11-28 10:54:41 +08:00
|
|
|
static DEFINE_PER_CPU(struct completion *, cpu_completion);
|
|
|
|
|
|
|
|
int register_ipi_completion(struct completion *completion, int cpu)
|
|
|
|
{
|
|
|
|
per_cpu(cpu_completion, cpu) = completion;
|
|
|
|
return IPI_COMPLETION;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ipi_complete(unsigned int cpu)
|
|
|
|
{
|
|
|
|
complete(per_cpu(cpu_completion, cpu));
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Main handler for inter-processor interrupts
|
|
|
|
*/
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
static void do_handle_IPI(int ipinr)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
|
|
|
unsigned int cpu = smp_processor_id();
|
|
|
|
|
2020-06-24 04:15:00 +08:00
|
|
|
if ((unsigned)ipinr < NR_IPI)
|
2023-01-13 03:43:37 +08:00
|
|
|
trace_ipi_entry(ipi_types[ipinr]);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-11-15 17:54:18 +08:00
|
|
|
switch (ipinr) {
|
2012-09-19 15:16:07 +08:00
|
|
|
case IPI_WAKEUP:
|
|
|
|
break;
|
|
|
|
|
2012-10-30 19:17:01 +08:00
|
|
|
#ifdef CONFIG_GENERIC_CLOCKEVENTS_BROADCAST
|
2010-11-15 17:54:18 +08:00
|
|
|
case IPI_TIMER:
|
2012-10-30 19:17:01 +08:00
|
|
|
tick_receive_broadcast();
|
2010-11-15 17:54:18 +08:00
|
|
|
break;
|
2012-10-30 19:17:01 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-11-15 17:54:18 +08:00
|
|
|
case IPI_RESCHEDULE:
|
2011-04-05 23:23:39 +08:00
|
|
|
scheduler_ipi();
|
2010-11-15 17:54:18 +08:00
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-11-15 17:54:18 +08:00
|
|
|
case IPI_CALL_FUNC:
|
|
|
|
generic_smp_call_function_interrupt();
|
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-11-15 17:54:18 +08:00
|
|
|
case IPI_CPU_STOP:
|
|
|
|
ipi_cpu_stop(cpu);
|
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-10-30 03:32:56 +08:00
|
|
|
#ifdef CONFIG_IRQ_WORK
|
|
|
|
case IPI_IRQ_WORK:
|
|
|
|
irq_work_run();
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2012-11-28 10:54:41 +08:00
|
|
|
case IPI_COMPLETION:
|
|
|
|
ipi_complete(cpu);
|
|
|
|
break;
|
|
|
|
|
2014-09-04 06:57:13 +08:00
|
|
|
case IPI_CPU_BACKTRACE:
|
2021-07-16 03:33:57 +08:00
|
|
|
printk_deferred_enter();
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
nmi_cpu_backtrace(get_irq_regs());
|
2021-07-16 03:33:57 +08:00
|
|
|
printk_deferred_exit();
|
2014-09-04 06:57:13 +08:00
|
|
|
break;
|
|
|
|
|
2010-11-15 17:54:18 +08:00
|
|
|
default:
|
2014-10-28 19:26:42 +08:00
|
|
|
pr_crit("CPU%u: Unknown IPI message 0x%x\n",
|
|
|
|
cpu, ipinr);
|
2010-11-15 17:54:18 +08:00
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2014-07-26 04:05:31 +08:00
|
|
|
|
|
|
|
if ((unsigned)ipinr < NR_IPI)
|
2023-01-13 03:43:37 +08:00
|
|
|
trace_ipi_exit(ipi_types[ipinr]);
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Legacy version, should go away once all irqchips have been converted */
|
|
|
|
void handle_IPI(int ipinr, struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
struct pt_regs *old_regs = set_irq_regs(regs);
|
|
|
|
|
|
|
|
irq_enter();
|
|
|
|
do_handle_IPI(ipinr);
|
|
|
|
irq_exit();
|
|
|
|
|
2006-10-25 20:59:16 +08:00
|
|
|
set_irq_regs(old_regs);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
static irqreturn_t ipi_handler(int irq, void *data)
|
|
|
|
{
|
|
|
|
do_handle_IPI(irq - ipi_irq_base);
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
2020-06-23 05:15:54 +08:00
|
|
|
static void smp_cross_call(const struct cpumask *target, unsigned int ipinr)
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
{
|
2023-01-13 03:43:37 +08:00
|
|
|
trace_ipi_raise(target, ipi_types[ipinr]);
|
2020-06-23 05:15:54 +08:00
|
|
|
__ipi_send_mask(ipi_desc[ipinr], target);
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ipi_setup(int cpu)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2020-06-23 05:15:54 +08:00
|
|
|
if (WARN_ON_ONCE(!ipi_irq_base))
|
ARM: Allow IPIs to be handled as normal interrupts
In order to deal with IPIs as normal interrupts, let's add
a new way to register them with the architecture code.
set_smp_ipi_range() takes a range of interrupts, and allows
the arch code to request them as if the were normal interrupts.
A standard handler is then called by the core IRQ code to deal
with the IPI.
This means that we don't need to call irq_enter/irq_exit, and
that we don't need to deal with set_irq_regs either. So let's
move the dispatcher into its own function, and leave handle_IPI()
as a compatibility function.
On the sending side, let's make use of ipi_send_mask, which
already exists for this purpose.
One of the major difference is that we end up, in some cases
(such as when performing IRQ time accounting on the scheduler
IPI), end up with nested irq_enter()/irq_exit() pairs.
Other than the (relatively small) overhead, there should be
no consequences to it (these pairs are designed to nest
correctly, and the accounting shouldn't be off).
Reviewed-by: Valentin Schneider <valentin.schneider@arm.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
2020-06-24 03:38:41 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < nr_ipi; i++)
|
|
|
|
enable_percpu_irq(ipi_irq_base + i, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void __init set_smp_ipi_range(int ipi_base, int n)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
WARN_ON(n < MAX_IPI);
|
|
|
|
nr_ipi = min(n, MAX_IPI);
|
|
|
|
|
|
|
|
for (i = 0; i < nr_ipi; i++) {
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = request_percpu_irq(ipi_base + i, ipi_handler,
|
|
|
|
"IPI", &irq_stat);
|
|
|
|
WARN_ON(err);
|
|
|
|
|
|
|
|
ipi_desc[i] = irq_to_desc(ipi_base + i);
|
|
|
|
irq_set_status_flags(ipi_base + i, IRQ_HIDDEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
ipi_irq_base = ipi_base;
|
|
|
|
|
|
|
|
/* Setup the boot CPU immediately */
|
|
|
|
ipi_setup(smp_processor_id());
|
|
|
|
}
|
|
|
|
|
2023-03-07 22:35:56 +08:00
|
|
|
void arch_smp_send_reschedule(int cpu)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2010-12-20 22:47:19 +08:00
|
|
|
smp_cross_call(cpumask_of(cpu), IPI_RESCHEDULE);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void smp_send_stop(void)
|
|
|
|
{
|
2010-12-02 17:53:54 +08:00
|
|
|
unsigned long timeout;
|
2012-04-27 19:51:43 +08:00
|
|
|
struct cpumask mask;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2012-04-27 19:51:43 +08:00
|
|
|
cpumask_copy(&mask, cpu_online_mask);
|
|
|
|
cpumask_clear_cpu(smp_processor_id(), &mask);
|
2012-07-28 22:19:55 +08:00
|
|
|
if (!cpumask_empty(&mask))
|
|
|
|
smp_cross_call(&mask, IPI_CPU_STOP);
|
2005-06-28 20:49:16 +08:00
|
|
|
|
2010-12-02 17:53:54 +08:00
|
|
|
/* Wait up to one second for other CPUs to stop */
|
|
|
|
timeout = USEC_PER_SEC;
|
|
|
|
while (num_online_cpus() > 1 && timeout--)
|
|
|
|
udelay(1);
|
2005-06-28 20:49:16 +08:00
|
|
|
|
2010-12-02 17:53:54 +08:00
|
|
|
if (num_online_cpus() > 1)
|
2014-09-17 03:41:43 +08:00
|
|
|
pr_warn("SMP: failed to stop secondary CPUs\n");
|
2005-06-28 20:49:16 +08:00
|
|
|
}
|
|
|
|
|
2018-11-02 18:51:31 +08:00
|
|
|
/* In case panic() and panic() called at the same time on CPU1 and CPU2,
|
|
|
|
* and CPU 1 calls panic_smp_self_stop() before crash_smp_send_stop()
|
|
|
|
* CPU1 can't receive the ipi irqs from CPU2, CPU1 will be always online,
|
|
|
|
* kdump fails. So split out the panic_smp_self_stop() and add
|
|
|
|
* set_cpu_online(smp_processor_id(), false).
|
|
|
|
*/
|
2023-04-13 07:49:35 +08:00
|
|
|
void __noreturn panic_smp_self_stop(void)
|
2018-11-02 18:51:31 +08:00
|
|
|
{
|
|
|
|
pr_debug("CPU %u will stop doing anything useful since another CPU has paniced\n",
|
|
|
|
smp_processor_id());
|
|
|
|
set_cpu_online(smp_processor_id(), false);
|
|
|
|
while (1)
|
|
|
|
cpu_relax();
|
|
|
|
}
|
|
|
|
|
2012-09-05 07:08:59 +08:00
|
|
|
#ifdef CONFIG_CPU_FREQ
|
|
|
|
|
|
|
|
static DEFINE_PER_CPU(unsigned long, l_p_j_ref);
|
|
|
|
static DEFINE_PER_CPU(unsigned long, l_p_j_ref_freq);
|
|
|
|
static unsigned long global_l_p_j_ref;
|
|
|
|
static unsigned long global_l_p_j_ref_freq;
|
|
|
|
|
|
|
|
static int cpufreq_callback(struct notifier_block *nb,
|
|
|
|
unsigned long val, void *data)
|
|
|
|
{
|
|
|
|
struct cpufreq_freqs *freq = data;
|
2019-04-29 17:33:58 +08:00
|
|
|
struct cpumask *cpus = freq->policy->cpus;
|
|
|
|
int cpu, first = cpumask_first(cpus);
|
|
|
|
unsigned int lpj;
|
2012-09-05 07:08:59 +08:00
|
|
|
|
|
|
|
if (freq->flags & CPUFREQ_CONST_LOOPS)
|
|
|
|
return NOTIFY_OK;
|
|
|
|
|
2019-04-29 17:33:58 +08:00
|
|
|
if (!per_cpu(l_p_j_ref, first)) {
|
|
|
|
for_each_cpu(cpu, cpus) {
|
|
|
|
per_cpu(l_p_j_ref, cpu) =
|
|
|
|
per_cpu(cpu_data, cpu).loops_per_jiffy;
|
|
|
|
per_cpu(l_p_j_ref_freq, cpu) = freq->old;
|
|
|
|
}
|
|
|
|
|
2012-09-05 07:08:59 +08:00
|
|
|
if (!global_l_p_j_ref) {
|
|
|
|
global_l_p_j_ref = loops_per_jiffy;
|
|
|
|
global_l_p_j_ref_freq = freq->old;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((val == CPUFREQ_PRECHANGE && freq->old < freq->new) ||
|
2014-03-19 13:54:58 +08:00
|
|
|
(val == CPUFREQ_POSTCHANGE && freq->old > freq->new)) {
|
2012-09-05 07:08:59 +08:00
|
|
|
loops_per_jiffy = cpufreq_scale(global_l_p_j_ref,
|
|
|
|
global_l_p_j_ref_freq,
|
|
|
|
freq->new);
|
2019-04-29 17:33:58 +08:00
|
|
|
|
|
|
|
lpj = cpufreq_scale(per_cpu(l_p_j_ref, first),
|
|
|
|
per_cpu(l_p_j_ref_freq, first), freq->new);
|
|
|
|
for_each_cpu(cpu, cpus)
|
|
|
|
per_cpu(cpu_data, cpu).loops_per_jiffy = lpj;
|
2012-09-05 07:08:59 +08:00
|
|
|
}
|
|
|
|
return NOTIFY_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct notifier_block cpufreq_notifier = {
|
|
|
|
.notifier_call = cpufreq_callback,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init register_cpufreq_notifier(void)
|
|
|
|
{
|
|
|
|
return cpufreq_register_notifier(&cpufreq_notifier,
|
|
|
|
CPUFREQ_TRANSITION_NOTIFIER);
|
|
|
|
}
|
|
|
|
core_initcall(register_cpufreq_notifier);
|
|
|
|
|
|
|
|
#endif
|
2014-09-04 06:57:13 +08:00
|
|
|
|
|
|
|
static void raise_nmi(cpumask_t *mask)
|
|
|
|
{
|
2020-06-23 05:15:54 +08:00
|
|
|
__ipi_send_mask(ipi_desc[IPI_CPU_BACKTRACE], mask);
|
2014-09-04 06:57:13 +08:00
|
|
|
}
|
|
|
|
|
2023-08-04 22:00:42 +08:00
|
|
|
void arch_trigger_cpumask_backtrace(const cpumask_t *mask, int exclude_cpu)
|
2014-09-04 06:57:13 +08:00
|
|
|
{
|
2023-08-04 22:00:42 +08:00
|
|
|
nmi_trigger_cpumask_backtrace(mask, exclude_cpu, raise_nmi);
|
2014-09-04 06:57:13 +08:00
|
|
|
}
|