2008-03-20 01:26:14 +08:00
|
|
|
/*
|
|
|
|
* x86 SMP booting functions
|
|
|
|
*
|
2009-01-05 22:08:04 +08:00
|
|
|
* (c) 1995 Alan Cox, Building #3 <alan@lxorguk.ukuu.org.uk>
|
2009-01-31 09:03:42 +08:00
|
|
|
* (c) 1998, 1999, 2000, 2009 Ingo Molnar <mingo@redhat.com>
|
2008-03-20 01:26:14 +08:00
|
|
|
* Copyright 2001 Andi Kleen, SuSE Labs.
|
|
|
|
*
|
|
|
|
* Much of the core SMP work is based on previous work by Thomas Radke, to
|
|
|
|
* whom a great many thanks are extended.
|
|
|
|
*
|
|
|
|
* Thanks to Intel for making available several different Pentium,
|
|
|
|
* Pentium Pro and Pentium-II/Xeon MP machines.
|
|
|
|
* Original development of Linux SMP code supported by Caldera.
|
|
|
|
*
|
|
|
|
* This code is released under the GNU General Public License version 2 or
|
|
|
|
* later.
|
|
|
|
*
|
|
|
|
* Fixes
|
|
|
|
* Felix Koop : NR_CPUS used properly
|
|
|
|
* Jose Renau : Handle single CPU case.
|
|
|
|
* Alan Cox : By repeated request 8) - Total BogoMIPS report.
|
|
|
|
* Greg Wright : Fix for kernel stacks panic.
|
|
|
|
* Erich Boleyn : MP v1.4 and additional changes.
|
|
|
|
* Matthias Sattler : Changes for 2.1 kernel map.
|
|
|
|
* Michel Lespinasse : Changes for 2.1 kernel map.
|
|
|
|
* Michael Chastain : Change trampoline.S to gnu as.
|
|
|
|
* Alan Cox : Dumb bug: 'B' step PPro's are fine
|
|
|
|
* Ingo Molnar : Added APIC timers, based on code
|
|
|
|
* from Jose Renau
|
|
|
|
* Ingo Molnar : various cleanups and rewrites
|
|
|
|
* Tigran Aivazian : fixed "0.00 in /proc/uptime on SMP" bug.
|
|
|
|
* Maciej W. Rozycki : Bits for genuine 82489DX APICs
|
|
|
|
* Andi Kleen : Changed for SMP boot into long mode.
|
|
|
|
* Martin J. Bligh : Added support for multi-quad systems
|
|
|
|
* Dave Jones : Report invalid combinations of Athlon CPUs.
|
|
|
|
* Rusty Russell : Hacked into shape for new "hotplug" boot process.
|
|
|
|
* Andi Kleen : Converted to new state machine.
|
|
|
|
* Ashok Raj : CPU hotplug support
|
|
|
|
* Glauber Costa : i386 and x86_64 integration
|
|
|
|
*/
|
|
|
|
|
2008-03-04 01:12:42 +08:00
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/smp.h>
|
2008-03-04 01:12:58 +08:00
|
|
|
#include <linux/module.h>
|
2008-03-04 01:13:03 +08:00
|
|
|
#include <linux/sched.h>
|
2008-03-04 01:13:07 +08:00
|
|
|
#include <linux/percpu.h>
|
2008-03-04 01:13:12 +08:00
|
|
|
#include <linux/bootmem.h>
|
2008-03-20 01:25:59 +08:00
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/nmi.h>
|
2009-09-02 09:25:07 +08:00
|
|
|
#include <linux/tboot.h>
|
2009-09-17 22:36:43 +08:00
|
|
|
#include <linux/stackprotector.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/gfp.h>
|
2008-03-04 01:13:07 +08:00
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
#include <asm/acpi.h>
|
2008-03-20 01:25:59 +08:00
|
|
|
#include <asm/desc.h>
|
2008-03-04 01:13:07 +08:00
|
|
|
#include <asm/nmi.h>
|
|
|
|
#include <asm/irq.h>
|
2008-09-24 05:26:42 +08:00
|
|
|
#include <asm/idle.h>
|
2008-04-11 05:28:10 +08:00
|
|
|
#include <asm/trampoline.h>
|
2008-03-04 01:13:07 +08:00
|
|
|
#include <asm/cpu.h>
|
|
|
|
#include <asm/numa.h>
|
2008-03-20 01:25:59 +08:00
|
|
|
#include <asm/pgtable.h>
|
|
|
|
#include <asm/tlbflush.h>
|
|
|
|
#include <asm/mtrr.h>
|
2008-03-20 01:26:00 +08:00
|
|
|
#include <asm/vmi.h>
|
2009-02-17 20:58:15 +08:00
|
|
|
#include <asm/apic.h>
|
x86: fix wakeup_cpu with numaq/es7000, v2
Impact: fix secondary-CPU wakeup/init path with numaq and es7000
While looking at wakeup_secondary_cpu for WAKE_SECONDARY_VIA_NMI:
|#ifdef WAKE_SECONDARY_VIA_NMI
|/*
| * Poke the other CPU in the eye via NMI to wake it up. Remember that the normal
| * INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
| * won't ... remember to clear down the APIC, etc later.
| */
|static int __devinit
|wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip)
|{
| unsigned long send_status, accept_status = 0;
| int maxlvt;
|...
| if (APIC_INTEGRATED(apic_version[phys_apicid])) {
| maxlvt = lapic_get_maxlvt();
I noticed that there is no warning about undefined phys_apicid...
because WAKE_SECONDARY_VIA_NMI and WAKE_SECONDARY_VIA_INIT can not be
defined at the same time. So NUMAQ is using wrong wakeup_secondary_cpu.
WAKE_SECONDARY_VIA_NMI, WAKE_SECONDARY_VIA_INIT and
WAKE_SECONDARY_VIA_MIP are variants of a weird and fragile
preprocessor-driven "HAL" mechanisms to specify the kind of secondary-CPU
wakeup strategy a given x86 kernel will use.
The vast majority of systems want to use INIT for secondary wakeup - NUMAQ
uses an NMI, (old-style-) ES7000 uses 'MIP' (a firmware driven in-memory
flag to let secondaries continue).
So convert these mechanisms to x86_quirks and add a
->wakeup_secondary_cpu() method to specify the rare exception
to the sane default.
Extend genapic accordingly as well, for 32-bit.
While looking further, I noticed that functions in wakecup.h for numaq
and es7000 are different to the default in mach_wakecpu.h - but smpboot.c
will only use default mach_wakecpu.h with smphook.h.
So we need to add mach_wakecpu.h for mach_generic, to properly support
numaq and es7000, and vectorize the following SMP init methods:
int trampoline_phys_low;
int trampoline_phys_high;
void (*wait_for_init_deassert)(atomic_t *deassert);
void (*smp_callin_clear_local_apic)(void);
void (*store_NMI_vector)(unsigned short *high, unsigned short *low);
void (*restore_NMI_vector)(unsigned short *high, unsigned short *low);
void (*inquire_remote_apic)(int apicid);
Signed-off-by: Yinghai Lu <yinghai@kernel.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-11-16 19:12:49 +08:00
|
|
|
#include <asm/setup.h>
|
2009-01-21 16:26:06 +08:00
|
|
|
#include <asm/uv/uv.h>
|
2008-03-20 01:25:59 +08:00
|
|
|
#include <linux/mc146818rtc.h>
|
2008-03-04 01:12:42 +08:00
|
|
|
|
2009-01-29 02:34:09 +08:00
|
|
|
#include <asm/smpboot_hooks.h>
|
2009-11-10 03:27:04 +08:00
|
|
|
#include <asm/i8259.h>
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-04-05 03:41:00 +08:00
|
|
|
#ifdef CONFIG_X86_32
|
2008-03-20 01:26:14 +08:00
|
|
|
u8 apicid_2_node[MAX_APICID];
|
2008-03-20 01:26:13 +08:00
|
|
|
#endif
|
|
|
|
|
2008-03-20 01:26:01 +08:00
|
|
|
/* State of each CPU */
|
|
|
|
DEFINE_PER_CPU(int, cpu_state) = { 0 };
|
|
|
|
|
2008-03-20 01:25:59 +08:00
|
|
|
/* Store all idle threads, this can be reused instead of creating
|
|
|
|
* a new thread. Also avoids complicated thread destroy functionality
|
|
|
|
* for idle threads.
|
|
|
|
*/
|
|
|
|
#ifdef CONFIG_HOTPLUG_CPU
|
|
|
|
/*
|
|
|
|
* Needed only for CONFIG_HOTPLUG_CPU because __cpuinitdata is
|
|
|
|
* removed after init for !CONFIG_HOTPLUG_CPU.
|
|
|
|
*/
|
|
|
|
static DEFINE_PER_CPU(struct task_struct *, idle_thread_array);
|
|
|
|
#define get_idle_for_cpu(x) (per_cpu(idle_thread_array, x))
|
|
|
|
#define set_idle_for_cpu(x, p) (per_cpu(idle_thread_array, x) = (p))
|
2010-08-20 02:10:29 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We need this for trampoline_base protection from concurrent accesses when
|
|
|
|
* off- and onlining cores wildly.
|
|
|
|
*/
|
|
|
|
static DEFINE_MUTEX(x86_cpu_hotplug_driver_mutex);
|
|
|
|
|
|
|
|
void cpu_hotplug_driver_lock()
|
|
|
|
{
|
|
|
|
mutex_lock(&x86_cpu_hotplug_driver_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cpu_hotplug_driver_unlock()
|
|
|
|
{
|
|
|
|
mutex_unlock(&x86_cpu_hotplug_driver_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t arch_cpu_probe(const char *buf, size_t count) { return -1; }
|
|
|
|
ssize_t arch_cpu_release(const char *buf, size_t count) { return -1; }
|
2008-03-20 01:25:59 +08:00
|
|
|
#else
|
2008-07-25 13:22:53 +08:00
|
|
|
static struct task_struct *idle_thread_array[NR_CPUS] __cpuinitdata ;
|
2008-03-20 01:25:59 +08:00
|
|
|
#define get_idle_for_cpu(x) (idle_thread_array[(x)])
|
|
|
|
#define set_idle_for_cpu(x, p) (idle_thread_array[(x)] = (p))
|
|
|
|
#endif
|
2008-03-20 01:25:53 +08:00
|
|
|
|
2008-03-04 01:12:58 +08:00
|
|
|
/* Number of siblings per CPU package */
|
|
|
|
int smp_num_siblings = 1;
|
|
|
|
EXPORT_SYMBOL(smp_num_siblings);
|
|
|
|
|
|
|
|
/* Last level cache ID of each logical CPU */
|
|
|
|
DEFINE_PER_CPU(u16, cpu_llc_id) = BAD_APICID;
|
|
|
|
|
|
|
|
/* representing HT siblings of each logical CPU */
|
2009-03-13 12:19:50 +08:00
|
|
|
DEFINE_PER_CPU(cpumask_var_t, cpu_sibling_map);
|
2008-03-04 01:12:58 +08:00
|
|
|
EXPORT_PER_CPU_SYMBOL(cpu_sibling_map);
|
|
|
|
|
|
|
|
/* representing HT and core siblings of each logical CPU */
|
2009-03-13 12:19:50 +08:00
|
|
|
DEFINE_PER_CPU(cpumask_var_t, cpu_core_map);
|
2008-03-04 01:12:58 +08:00
|
|
|
EXPORT_PER_CPU_SYMBOL(cpu_core_map);
|
|
|
|
|
|
|
|
/* Per CPU bogomips and other parameters */
|
|
|
|
DEFINE_PER_CPU_SHARED_ALIGNED(struct cpuinfo_x86, cpu_info);
|
|
|
|
EXPORT_PER_CPU_SYMBOL(cpu_info);
|
2008-03-04 01:13:02 +08:00
|
|
|
|
2009-02-26 12:50:49 +08:00
|
|
|
atomic_t init_deasserted;
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-03-20 01:25:56 +08:00
|
|
|
#if defined(CONFIG_NUMA) && defined(CONFIG_X86_32)
|
|
|
|
/* which node each logical CPU is on */
|
|
|
|
int cpu_to_node_map[NR_CPUS] __read_mostly = { [0 ... NR_CPUS-1] = 0 };
|
|
|
|
EXPORT_SYMBOL(cpu_to_node_map);
|
|
|
|
|
|
|
|
/* set up a mapping between cpu and node. */
|
|
|
|
static void map_cpu_to_node(int cpu, int node)
|
|
|
|
{
|
|
|
|
printk(KERN_INFO "Mapping cpu %d to node %d\n", cpu, node);
|
2009-03-13 12:19:53 +08:00
|
|
|
cpumask_set_cpu(cpu, node_to_cpumask_map[node]);
|
2008-03-20 01:25:56 +08:00
|
|
|
cpu_to_node_map[cpu] = node;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* undo a mapping between cpu and node. */
|
|
|
|
static void unmap_cpu_to_node(int cpu)
|
|
|
|
{
|
|
|
|
int node;
|
|
|
|
|
|
|
|
printk(KERN_INFO "Unmapping cpu %d from all nodes\n", cpu);
|
|
|
|
for (node = 0; node < MAX_NUMNODES; node++)
|
2009-03-13 12:19:53 +08:00
|
|
|
cpumask_clear_cpu(cpu, node_to_cpumask_map[node]);
|
2008-03-20 01:25:56 +08:00
|
|
|
cpu_to_node_map[cpu] = 0;
|
|
|
|
}
|
|
|
|
#else /* !(CONFIG_NUMA && CONFIG_X86_32) */
|
|
|
|
#define map_cpu_to_node(cpu, node) ({})
|
|
|
|
#define unmap_cpu_to_node(cpu) ({})
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_X86_32
|
2008-07-11 02:16:49 +08:00
|
|
|
static int boot_cpu_logical_apicid;
|
|
|
|
|
2008-03-20 01:25:56 +08:00
|
|
|
u8 cpu_2_logical_apicid[NR_CPUS] __read_mostly =
|
|
|
|
{ [0 ... NR_CPUS-1] = BAD_APICID };
|
|
|
|
|
2008-04-23 19:20:56 +08:00
|
|
|
static void map_cpu_to_logical_apicid(void)
|
2008-03-20 01:25:56 +08:00
|
|
|
{
|
|
|
|
int cpu = smp_processor_id();
|
|
|
|
int apicid = logical_smp_processor_id();
|
2009-01-28 13:50:47 +08:00
|
|
|
int node = apic->apicid_to_node(apicid);
|
2008-03-20 01:25:56 +08:00
|
|
|
|
|
|
|
if (!node_online(node))
|
|
|
|
node = first_online_node;
|
|
|
|
|
|
|
|
cpu_2_logical_apicid[cpu] = apicid;
|
|
|
|
map_cpu_to_node(cpu, node);
|
|
|
|
}
|
|
|
|
|
2008-06-05 02:35:03 +08:00
|
|
|
void numa_remove_cpu(int cpu)
|
2008-03-20 01:25:56 +08:00
|
|
|
{
|
|
|
|
cpu_2_logical_apicid[cpu] = BAD_APICID;
|
|
|
|
unmap_cpu_to_node(cpu);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define map_cpu_to_logical_apicid() do {} while (0)
|
|
|
|
#endif
|
|
|
|
|
2008-03-20 01:25:59 +08:00
|
|
|
/*
|
|
|
|
* Report back to the Boot Processor.
|
|
|
|
* Running on AP.
|
|
|
|
*/
|
2008-04-23 19:20:56 +08:00
|
|
|
static void __cpuinit smp_callin(void)
|
2008-03-20 01:25:59 +08:00
|
|
|
{
|
|
|
|
int cpuid, phys_id;
|
|
|
|
unsigned long timeout;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If waken up by an INIT in an 82489DX configuration
|
|
|
|
* we may get here before an INIT-deassert IPI reaches
|
|
|
|
* our local APIC. We have to wait for the IPI or we'll
|
|
|
|
* lock up on an APIC access.
|
|
|
|
*/
|
2009-01-28 23:21:32 +08:00
|
|
|
if (apic->wait_for_init_deassert)
|
|
|
|
apic->wait_for_init_deassert(&init_deasserted);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* (This works even if the APIC is not enabled.)
|
|
|
|
*/
|
2008-07-12 09:44:16 +08:00
|
|
|
phys_id = read_apic_id();
|
2008-03-20 01:25:59 +08:00
|
|
|
cpuid = smp_processor_id();
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_test_cpu(cpuid, cpu_callin_mask)) {
|
2008-03-20 01:25:59 +08:00
|
|
|
panic("%s: phys CPU#%d, CPU#%d already present??\n", __func__,
|
|
|
|
phys_id, cpuid);
|
|
|
|
}
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("CPU#%d (phys ID: %d) waiting for CALLOUT\n", cpuid, phys_id);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* STARTUP IPIs are fragile beasts as they might sometimes
|
|
|
|
* trigger some glue motherboard logic. Complete APIC bus
|
|
|
|
* silence for 1 second, this overestimates the time the
|
|
|
|
* boot CPU is spending to send the up to 2 STARTUP IPIs
|
|
|
|
* by a factor of two. This should be enough.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Waiting 2s total for startup (udelay is not yet working)
|
|
|
|
*/
|
|
|
|
timeout = jiffies + 2*HZ;
|
|
|
|
while (time_before(jiffies, timeout)) {
|
|
|
|
/*
|
|
|
|
* Has the boot CPU finished it's STARTUP sequence?
|
|
|
|
*/
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_test_cpu(cpuid, cpu_callout_mask))
|
2008-03-20 01:25:59 +08:00
|
|
|
break;
|
|
|
|
cpu_relax();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!time_before(jiffies, timeout)) {
|
|
|
|
panic("%s: CPU%d started up but did not get a callout!\n",
|
|
|
|
__func__, cpuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* the boot CPU has finished the init stage and is spinning
|
|
|
|
* on callin_map until we finish. We are free to set up this
|
|
|
|
* CPU, first the APIC. (this is probably redundant on most
|
|
|
|
* boards)
|
|
|
|
*/
|
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("CALLIN, before setup_local_APIC().\n");
|
2009-01-28 23:31:52 +08:00
|
|
|
if (apic->smp_callin_clear_local_apic)
|
|
|
|
apic->smp_callin_clear_local_apic();
|
2008-03-20 01:25:59 +08:00
|
|
|
setup_local_APIC();
|
|
|
|
end_local_APIC_setup();
|
|
|
|
map_cpu_to_logical_apicid();
|
|
|
|
|
2010-01-30 03:42:21 +08:00
|
|
|
/*
|
|
|
|
* Need to setup vector mappings before we enable interrupts.
|
|
|
|
*/
|
2010-03-16 06:33:06 +08:00
|
|
|
setup_vector_irq(smp_processor_id());
|
2008-03-20 01:25:59 +08:00
|
|
|
/*
|
|
|
|
* Get our bogomips.
|
|
|
|
*
|
|
|
|
* Need to enable IRQs because it can take longer and then
|
|
|
|
* the NMI watchdog might kill us.
|
|
|
|
*/
|
|
|
|
local_irq_enable();
|
|
|
|
calibrate_delay();
|
|
|
|
local_irq_disable();
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Stack at about %p\n", &cpuid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Save our processor parameters
|
|
|
|
*/
|
|
|
|
smp_store_cpu_info(cpuid);
|
|
|
|
|
2010-03-24 02:30:52 +08:00
|
|
|
notify_cpu_starting(cpuid);
|
|
|
|
|
2008-03-20 01:25:59 +08:00
|
|
|
/*
|
|
|
|
* Allow the master to continue.
|
|
|
|
*/
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_set_cpu(cpuid, cpu_callin_mask);
|
2008-03-20 01:25:59 +08:00
|
|
|
}
|
|
|
|
|
2008-03-20 01:26:00 +08:00
|
|
|
/*
|
|
|
|
* Activate a secondary processor.
|
|
|
|
*/
|
2008-12-25 06:30:02 +08:00
|
|
|
notrace static void __cpuinit start_secondary(void *unused)
|
2008-03-20 01:26:00 +08:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Don't put *anything* before cpu_init(), SMP booting is too
|
|
|
|
* fragile that we want to limit the things done here to the
|
|
|
|
* most necessary things.
|
|
|
|
*/
|
x86-32: Separate 1:1 pagetables from swapper_pg_dir
This patch fixes machine crashes which occur when heavily exercising the
CPU hotplug codepaths on a 32-bit kernel. These crashes are caused by
AMD Erratum 383 and result in a fatal machine check exception. Here's
the scenario:
1. On 32-bit, the swapper_pg_dir page table is used as the initial page
table for booting a secondary CPU.
2. To make this work, swapper_pg_dir needs a direct mapping of physical
memory in it (the low mappings). By adding those low, large page (2M)
mappings (PAE kernel), we create the necessary conditions for Erratum
383 to occur.
3. Other CPUs which do not participate in the off- and onlining game may
use swapper_pg_dir while the low mappings are present (when leave_mm is
called). For all steps below, the CPU referred to is a CPU that is using
swapper_pg_dir, and not the CPU which is being onlined.
4. The presence of the low mappings in swapper_pg_dir can result
in TLB entries for addresses below __PAGE_OFFSET to be established
speculatively. These TLB entries are marked global and large.
5. When the CPU with such TLB entry switches to another page table, this
TLB entry remains because it is global.
6. The process then generates an access to an address covered by the
above TLB entry but there is a permission mismatch - the TLB entry
covers a large global page not accessible to userspace.
7. Due to this permission mismatch a new 4kb, user TLB entry gets
established. Further, Erratum 383 provides for a small window of time
where both TLB entries are present. This results in an uncorrectable
machine check exception signalling a TLB multimatch which panics the
machine.
There are two ways to fix this issue:
1. Always do a global TLB flush when a new cr3 is loaded and the
old page table was swapper_pg_dir. I consider this a hack hard
to understand and with performance implications
2. Do not use swapper_pg_dir to boot secondary CPUs like 64-bit
does.
This patch implements solution 2. It introduces a trampoline_pg_dir
which has the same layout as swapper_pg_dir with low_mappings. This page
table is used as the initial page table of the booting CPU. Later in the
bringup process, it switches to swapper_pg_dir and does a global TLB
flush. This fixes the crashes in our test cases.
-v2: switch to swapper_pg_dir right after entering start_secondary() so
that we are able to access percpu data which might not be mapped in the
trampoline page table.
Signed-off-by: Joerg Roedel <joerg.roedel@amd.com>
LKML-Reference: <20100816123833.GB28147@aftab>
Signed-off-by: Borislav Petkov <borislav.petkov@amd.com>
Signed-off-by: H. Peter Anvin <hpa@zytor.com>
2010-08-16 20:38:33 +08:00
|
|
|
|
|
|
|
#ifdef CONFIG_X86_32
|
|
|
|
/*
|
|
|
|
* Switch away from the trampoline page-table
|
|
|
|
*
|
|
|
|
* Do this before cpu_init() because it needs to access per-cpu
|
|
|
|
* data which may not be mapped in the trampoline page-table.
|
|
|
|
*/
|
|
|
|
load_cr3(swapper_pg_dir);
|
|
|
|
__flush_tlb_all();
|
|
|
|
#endif
|
|
|
|
|
2008-03-20 01:26:00 +08:00
|
|
|
vmi_bringup();
|
|
|
|
cpu_init();
|
|
|
|
preempt_disable();
|
|
|
|
smp_callin();
|
|
|
|
|
|
|
|
/* otherwise gcc will move up smp_processor_id before the cpu_init */
|
|
|
|
barrier();
|
|
|
|
/*
|
|
|
|
* Check TSC synchronization with the BP:
|
|
|
|
*/
|
|
|
|
check_tsc_sync_target();
|
|
|
|
|
|
|
|
if (nmi_watchdog == NMI_IO_APIC) {
|
2009-11-10 03:27:04 +08:00
|
|
|
legacy_pic->chip->mask(0);
|
2008-03-20 01:26:00 +08:00
|
|
|
enable_NMI_through_LVT0();
|
2009-11-10 03:27:04 +08:00
|
|
|
legacy_pic->chip->unmask(0);
|
2008-03-20 01:26:00 +08:00
|
|
|
}
|
|
|
|
|
2009-03-13 12:19:54 +08:00
|
|
|
/* This must be done before setting cpu_online_mask */
|
2008-03-20 01:26:00 +08:00
|
|
|
set_cpu_sibling_map(raw_smp_processor_id());
|
|
|
|
wmb();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We need to hold call_lock, so there is no inconsistency
|
|
|
|
* between the time smp_call_function() determines number of
|
|
|
|
* IPI recipients, and the time when the determination is made
|
|
|
|
* for which cpus receive the IPI. Holding this
|
|
|
|
* lock helps us to not include this cpu in a currently in progress
|
|
|
|
* smp_call_function().
|
2008-08-10 06:09:02 +08:00
|
|
|
*
|
|
|
|
* We need to hold vector_lock so there the set of online cpus
|
|
|
|
* does not change while we are assigning vectors to cpus. Holding
|
|
|
|
* this lock ensures we don't half assign or remove an irq from a cpu.
|
2008-03-20 01:26:00 +08:00
|
|
|
*/
|
2008-09-07 17:29:58 +08:00
|
|
|
ipi_call_lock();
|
2008-08-10 06:09:02 +08:00
|
|
|
lock_vector_lock();
|
2009-01-04 21:18:03 +08:00
|
|
|
set_cpu_online(smp_processor_id(), true);
|
2008-08-10 06:09:02 +08:00
|
|
|
unlock_vector_lock();
|
2008-09-07 17:29:58 +08:00
|
|
|
ipi_call_unlock();
|
2008-03-20 01:26:00 +08:00
|
|
|
per_cpu(cpu_state, smp_processor_id()) = CPU_ONLINE;
|
2010-02-27 00:49:12 +08:00
|
|
|
x86_platform.nmi_init();
|
2008-03-20 01:26:00 +08:00
|
|
|
|
2008-09-07 17:29:58 +08:00
|
|
|
/* enable local interrupts */
|
|
|
|
local_irq_enable();
|
|
|
|
|
2009-09-17 22:36:43 +08:00
|
|
|
/* to prevent fake stack check failure in clock setup */
|
|
|
|
boot_init_stack_canary();
|
2008-09-07 17:29:58 +08:00
|
|
|
|
2009-08-19 18:35:53 +08:00
|
|
|
x86_cpuinit.setup_percpu_clockev();
|
2008-03-20 01:26:00 +08:00
|
|
|
|
|
|
|
wmb();
|
|
|
|
cpu_idle();
|
|
|
|
}
|
|
|
|
|
2009-03-13 12:19:53 +08:00
|
|
|
#ifdef CONFIG_CPUMASK_OFFSTACK
|
|
|
|
/* In this case, llc_shared_map is a pointer to a cpumask. */
|
|
|
|
static inline void copy_cpuinfo_x86(struct cpuinfo_x86 *dst,
|
|
|
|
const struct cpuinfo_x86 *src)
|
|
|
|
{
|
|
|
|
struct cpumask *llc = dst->llc_shared_map;
|
|
|
|
*dst = *src;
|
|
|
|
dst->llc_shared_map = llc;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static inline void copy_cpuinfo_x86(struct cpuinfo_x86 *dst,
|
|
|
|
const struct cpuinfo_x86 *src)
|
|
|
|
{
|
|
|
|
*dst = *src;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_CPUMASK_OFFSTACK */
|
|
|
|
|
2008-03-20 01:25:05 +08:00
|
|
|
/*
|
|
|
|
* The bootstrap kernel entry code has set these up. Save them for
|
|
|
|
* a given CPU
|
|
|
|
*/
|
|
|
|
|
|
|
|
void __cpuinit smp_store_cpu_info(int id)
|
|
|
|
{
|
|
|
|
struct cpuinfo_x86 *c = &cpu_data(id);
|
|
|
|
|
2009-03-13 12:19:53 +08:00
|
|
|
copy_cpuinfo_x86(c, &boot_cpu_data);
|
2008-03-20 01:25:05 +08:00
|
|
|
c->cpu_index = id;
|
|
|
|
if (id != 0)
|
|
|
|
identify_secondary_cpu(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-04 01:13:02 +08:00
|
|
|
void __cpuinit set_cpu_sibling_map(int cpu)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct cpuinfo_x86 *c = &cpu_data(cpu);
|
|
|
|
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_set_cpu(cpu, cpu_sibling_setup_mask);
|
2008-03-04 01:13:02 +08:00
|
|
|
|
|
|
|
if (smp_num_siblings > 1) {
|
2009-01-04 21:18:03 +08:00
|
|
|
for_each_cpu(i, cpu_sibling_setup_mask) {
|
|
|
|
struct cpuinfo_x86 *o = &cpu_data(i);
|
|
|
|
|
|
|
|
if (c->phys_proc_id == o->phys_proc_id &&
|
|
|
|
c->cpu_core_id == o->cpu_core_id) {
|
|
|
|
cpumask_set_cpu(i, cpu_sibling_mask(cpu));
|
|
|
|
cpumask_set_cpu(cpu, cpu_sibling_mask(i));
|
|
|
|
cpumask_set_cpu(i, cpu_core_mask(cpu));
|
|
|
|
cpumask_set_cpu(cpu, cpu_core_mask(i));
|
2009-03-13 12:19:53 +08:00
|
|
|
cpumask_set_cpu(i, c->llc_shared_map);
|
|
|
|
cpumask_set_cpu(cpu, o->llc_shared_map);
|
2008-03-04 01:13:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_set_cpu(cpu, cpu_sibling_mask(cpu));
|
2008-03-04 01:13:02 +08:00
|
|
|
}
|
|
|
|
|
2009-03-13 12:19:53 +08:00
|
|
|
cpumask_set_cpu(cpu, c->llc_shared_map);
|
2008-03-04 01:13:02 +08:00
|
|
|
|
|
|
|
if (current_cpu_data.x86_max_cores == 1) {
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_copy(cpu_core_mask(cpu), cpu_sibling_mask(cpu));
|
2008-03-04 01:13:02 +08:00
|
|
|
c->booted_cores = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-04 21:18:03 +08:00
|
|
|
for_each_cpu(i, cpu_sibling_setup_mask) {
|
2008-03-04 01:13:02 +08:00
|
|
|
if (per_cpu(cpu_llc_id, cpu) != BAD_APICID &&
|
|
|
|
per_cpu(cpu_llc_id, cpu) == per_cpu(cpu_llc_id, i)) {
|
2009-03-13 12:19:53 +08:00
|
|
|
cpumask_set_cpu(i, c->llc_shared_map);
|
|
|
|
cpumask_set_cpu(cpu, cpu_data(i).llc_shared_map);
|
2008-03-04 01:13:02 +08:00
|
|
|
}
|
|
|
|
if (c->phys_proc_id == cpu_data(i).phys_proc_id) {
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_set_cpu(i, cpu_core_mask(cpu));
|
|
|
|
cpumask_set_cpu(cpu, cpu_core_mask(i));
|
2008-03-04 01:13:02 +08:00
|
|
|
/*
|
|
|
|
* Does this new cpu bringup a new core?
|
|
|
|
*/
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_weight(cpu_sibling_mask(cpu)) == 1) {
|
2008-03-04 01:13:02 +08:00
|
|
|
/*
|
|
|
|
* for each core in package, increment
|
|
|
|
* the booted_cores for this new cpu
|
|
|
|
*/
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_first(cpu_sibling_mask(i)) == i)
|
2008-03-04 01:13:02 +08:00
|
|
|
c->booted_cores++;
|
|
|
|
/*
|
|
|
|
* increment the core count for all
|
|
|
|
* the other cpus in this package
|
|
|
|
*/
|
|
|
|
if (i != cpu)
|
|
|
|
cpu_data(i).booted_cores++;
|
|
|
|
} else if (i != cpu && !c->booted_cores)
|
|
|
|
c->booted_cores = cpu_data(i).booted_cores;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-04 01:13:03 +08:00
|
|
|
/* maps the cpu to the sched domain representing multi-core */
|
2008-12-26 19:53:41 +08:00
|
|
|
const struct cpumask *cpu_coregroup_mask(int cpu)
|
2008-03-04 01:13:03 +08:00
|
|
|
{
|
|
|
|
struct cpuinfo_x86 *c = &cpu_data(cpu);
|
|
|
|
/*
|
|
|
|
* For perf, we return last level cache shared map.
|
|
|
|
* And for power savings, we return cpu_core_map
|
|
|
|
*/
|
2009-09-03 15:44:28 +08:00
|
|
|
if ((sched_mc_power_savings || sched_smt_power_savings) &&
|
|
|
|
!(cpu_has(c, X86_FEATURE_AMD_DCM)))
|
2009-01-04 21:18:03 +08:00
|
|
|
return cpu_core_mask(cpu);
|
2008-03-04 01:13:03 +08:00
|
|
|
else
|
2009-03-13 12:19:53 +08:00
|
|
|
return c->llc_shared_map;
|
2008-12-26 19:53:41 +08:00
|
|
|
}
|
|
|
|
|
2008-04-23 19:20:56 +08:00
|
|
|
static void impress_friends(void)
|
2008-03-20 01:25:27 +08:00
|
|
|
{
|
|
|
|
int cpu;
|
|
|
|
unsigned long bogosum = 0;
|
|
|
|
/*
|
|
|
|
* Allow the user to impress friends.
|
|
|
|
*/
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Before bogomips.\n");
|
2008-03-20 01:25:27 +08:00
|
|
|
for_each_possible_cpu(cpu)
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_test_cpu(cpu, cpu_callout_mask))
|
2008-03-20 01:25:27 +08:00
|
|
|
bogosum += cpu_data(cpu).loops_per_jiffy;
|
|
|
|
printk(KERN_INFO
|
|
|
|
"Total of %d processors activated (%lu.%02lu BogoMIPS).\n",
|
2008-03-20 01:25:29 +08:00
|
|
|
num_online_cpus(),
|
2008-03-20 01:25:27 +08:00
|
|
|
bogosum/(500000/HZ),
|
|
|
|
(bogosum/(5000/HZ))%100);
|
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Before bogocount - setting activated=1.\n");
|
2008-03-20 01:25:27 +08:00
|
|
|
}
|
|
|
|
|
x86: fix wakeup_cpu with numaq/es7000, v2
Impact: fix secondary-CPU wakeup/init path with numaq and es7000
While looking at wakeup_secondary_cpu for WAKE_SECONDARY_VIA_NMI:
|#ifdef WAKE_SECONDARY_VIA_NMI
|/*
| * Poke the other CPU in the eye via NMI to wake it up. Remember that the normal
| * INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
| * won't ... remember to clear down the APIC, etc later.
| */
|static int __devinit
|wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip)
|{
| unsigned long send_status, accept_status = 0;
| int maxlvt;
|...
| if (APIC_INTEGRATED(apic_version[phys_apicid])) {
| maxlvt = lapic_get_maxlvt();
I noticed that there is no warning about undefined phys_apicid...
because WAKE_SECONDARY_VIA_NMI and WAKE_SECONDARY_VIA_INIT can not be
defined at the same time. So NUMAQ is using wrong wakeup_secondary_cpu.
WAKE_SECONDARY_VIA_NMI, WAKE_SECONDARY_VIA_INIT and
WAKE_SECONDARY_VIA_MIP are variants of a weird and fragile
preprocessor-driven "HAL" mechanisms to specify the kind of secondary-CPU
wakeup strategy a given x86 kernel will use.
The vast majority of systems want to use INIT for secondary wakeup - NUMAQ
uses an NMI, (old-style-) ES7000 uses 'MIP' (a firmware driven in-memory
flag to let secondaries continue).
So convert these mechanisms to x86_quirks and add a
->wakeup_secondary_cpu() method to specify the rare exception
to the sane default.
Extend genapic accordingly as well, for 32-bit.
While looking further, I noticed that functions in wakecup.h for numaq
and es7000 are different to the default in mach_wakecpu.h - but smpboot.c
will only use default mach_wakecpu.h with smphook.h.
So we need to add mach_wakecpu.h for mach_generic, to properly support
numaq and es7000, and vectorize the following SMP init methods:
int trampoline_phys_low;
int trampoline_phys_high;
void (*wait_for_init_deassert)(atomic_t *deassert);
void (*smp_callin_clear_local_apic)(void);
void (*store_NMI_vector)(unsigned short *high, unsigned short *low);
void (*restore_NMI_vector)(unsigned short *high, unsigned short *low);
void (*inquire_remote_apic)(int apicid);
Signed-off-by: Yinghai Lu <yinghai@kernel.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-11-16 19:12:49 +08:00
|
|
|
void __inquire_remote_apic(int apicid)
|
2008-03-20 01:25:59 +08:00
|
|
|
{
|
|
|
|
unsigned i, regs[] = { APIC_ID >> 4, APIC_LVR >> 4, APIC_SPIV >> 4 };
|
|
|
|
char *names[] = { "ID", "VERSION", "SPIV" };
|
|
|
|
int timeout;
|
|
|
|
u32 status;
|
|
|
|
|
2008-09-11 12:56:46 +08:00
|
|
|
printk(KERN_INFO "Inquiring remote APIC 0x%x...\n", apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(regs); i++) {
|
2008-09-11 12:56:46 +08:00
|
|
|
printk(KERN_INFO "... APIC 0x%x %s: ", apicid, names[i]);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait for idle.
|
|
|
|
*/
|
|
|
|
status = safe_apic_wait_icr_idle();
|
|
|
|
if (status)
|
|
|
|
printk(KERN_CONT
|
|
|
|
"a previous APIC delivery may have failed\n");
|
|
|
|
|
2008-07-11 02:16:49 +08:00
|
|
|
apic_icr_write(APIC_DM_REMRD | regs[i], apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
timeout = 0;
|
|
|
|
do {
|
|
|
|
udelay(100);
|
|
|
|
status = apic_read(APIC_ICR) & APIC_ICR_RR_MASK;
|
|
|
|
} while (status == APIC_ICR_RR_INPROG && timeout++ < 1000);
|
|
|
|
|
|
|
|
switch (status) {
|
|
|
|
case APIC_ICR_RR_VALID:
|
|
|
|
status = apic_read(APIC_RRR);
|
|
|
|
printk(KERN_CONT "%08x\n", status);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printk(KERN_CONT "failed\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Poke the other CPU in the eye via NMI to wake it up. Remember that the normal
|
|
|
|
* INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
|
|
|
|
* won't ... remember to clear down the APIC, etc later.
|
|
|
|
*/
|
2009-04-19 03:45:28 +08:00
|
|
|
int __cpuinit
|
x86: fix wakeup_cpu with numaq/es7000, v2
Impact: fix secondary-CPU wakeup/init path with numaq and es7000
While looking at wakeup_secondary_cpu for WAKE_SECONDARY_VIA_NMI:
|#ifdef WAKE_SECONDARY_VIA_NMI
|/*
| * Poke the other CPU in the eye via NMI to wake it up. Remember that the normal
| * INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
| * won't ... remember to clear down the APIC, etc later.
| */
|static int __devinit
|wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip)
|{
| unsigned long send_status, accept_status = 0;
| int maxlvt;
|...
| if (APIC_INTEGRATED(apic_version[phys_apicid])) {
| maxlvt = lapic_get_maxlvt();
I noticed that there is no warning about undefined phys_apicid...
because WAKE_SECONDARY_VIA_NMI and WAKE_SECONDARY_VIA_INIT can not be
defined at the same time. So NUMAQ is using wrong wakeup_secondary_cpu.
WAKE_SECONDARY_VIA_NMI, WAKE_SECONDARY_VIA_INIT and
WAKE_SECONDARY_VIA_MIP are variants of a weird and fragile
preprocessor-driven "HAL" mechanisms to specify the kind of secondary-CPU
wakeup strategy a given x86 kernel will use.
The vast majority of systems want to use INIT for secondary wakeup - NUMAQ
uses an NMI, (old-style-) ES7000 uses 'MIP' (a firmware driven in-memory
flag to let secondaries continue).
So convert these mechanisms to x86_quirks and add a
->wakeup_secondary_cpu() method to specify the rare exception
to the sane default.
Extend genapic accordingly as well, for 32-bit.
While looking further, I noticed that functions in wakecup.h for numaq
and es7000 are different to the default in mach_wakecpu.h - but smpboot.c
will only use default mach_wakecpu.h with smphook.h.
So we need to add mach_wakecpu.h for mach_generic, to properly support
numaq and es7000, and vectorize the following SMP init methods:
int trampoline_phys_low;
int trampoline_phys_high;
void (*wait_for_init_deassert)(atomic_t *deassert);
void (*smp_callin_clear_local_apic)(void);
void (*store_NMI_vector)(unsigned short *high, unsigned short *low);
void (*restore_NMI_vector)(unsigned short *high, unsigned short *low);
void (*inquire_remote_apic)(int apicid);
Signed-off-by: Yinghai Lu <yinghai@kernel.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-11-16 19:12:49 +08:00
|
|
|
wakeup_secondary_cpu_via_nmi(int logical_apicid, unsigned long start_eip)
|
2008-03-20 01:25:59 +08:00
|
|
|
{
|
|
|
|
unsigned long send_status, accept_status = 0;
|
|
|
|
int maxlvt;
|
|
|
|
|
|
|
|
/* Target chip */
|
|
|
|
/* Boot on the stack */
|
|
|
|
/* Kick the second */
|
2009-01-28 12:29:25 +08:00
|
|
|
apic_icr_write(APIC_DM_NMI | apic->dest_logical, logical_apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Waiting for send to finish...\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
send_status = safe_apic_wait_icr_idle();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Give the other CPU some time to accept the IPI.
|
|
|
|
*/
|
|
|
|
udelay(200);
|
x86: fix wakeup_cpu with numaq/es7000, v2
Impact: fix secondary-CPU wakeup/init path with numaq and es7000
While looking at wakeup_secondary_cpu for WAKE_SECONDARY_VIA_NMI:
|#ifdef WAKE_SECONDARY_VIA_NMI
|/*
| * Poke the other CPU in the eye via NMI to wake it up. Remember that the normal
| * INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
| * won't ... remember to clear down the APIC, etc later.
| */
|static int __devinit
|wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip)
|{
| unsigned long send_status, accept_status = 0;
| int maxlvt;
|...
| if (APIC_INTEGRATED(apic_version[phys_apicid])) {
| maxlvt = lapic_get_maxlvt();
I noticed that there is no warning about undefined phys_apicid...
because WAKE_SECONDARY_VIA_NMI and WAKE_SECONDARY_VIA_INIT can not be
defined at the same time. So NUMAQ is using wrong wakeup_secondary_cpu.
WAKE_SECONDARY_VIA_NMI, WAKE_SECONDARY_VIA_INIT and
WAKE_SECONDARY_VIA_MIP are variants of a weird and fragile
preprocessor-driven "HAL" mechanisms to specify the kind of secondary-CPU
wakeup strategy a given x86 kernel will use.
The vast majority of systems want to use INIT for secondary wakeup - NUMAQ
uses an NMI, (old-style-) ES7000 uses 'MIP' (a firmware driven in-memory
flag to let secondaries continue).
So convert these mechanisms to x86_quirks and add a
->wakeup_secondary_cpu() method to specify the rare exception
to the sane default.
Extend genapic accordingly as well, for 32-bit.
While looking further, I noticed that functions in wakecup.h for numaq
and es7000 are different to the default in mach_wakecpu.h - but smpboot.c
will only use default mach_wakecpu.h with smphook.h.
So we need to add mach_wakecpu.h for mach_generic, to properly support
numaq and es7000, and vectorize the following SMP init methods:
int trampoline_phys_low;
int trampoline_phys_high;
void (*wait_for_init_deassert)(atomic_t *deassert);
void (*smp_callin_clear_local_apic)(void);
void (*store_NMI_vector)(unsigned short *high, unsigned short *low);
void (*restore_NMI_vector)(unsigned short *high, unsigned short *low);
void (*inquire_remote_apic)(int apicid);
Signed-off-by: Yinghai Lu <yinghai@kernel.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-11-16 19:12:49 +08:00
|
|
|
if (APIC_INTEGRATED(apic_version[boot_cpu_physical_apicid])) {
|
2008-09-15 01:58:49 +08:00
|
|
|
maxlvt = lapic_get_maxlvt();
|
|
|
|
if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */
|
|
|
|
apic_write(APIC_ESR, 0);
|
|
|
|
accept_status = (apic_read(APIC_ESR) & 0xEF);
|
|
|
|
}
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("NMI sent.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
if (send_status)
|
|
|
|
printk(KERN_ERR "APIC never delivered???\n");
|
|
|
|
if (accept_status)
|
|
|
|
printk(KERN_ERR "APIC delivery error (%lx).\n", accept_status);
|
|
|
|
|
|
|
|
return (send_status | accept_status);
|
|
|
|
}
|
|
|
|
|
2009-04-19 03:45:28 +08:00
|
|
|
static int __cpuinit
|
x86: fix wakeup_cpu with numaq/es7000, v2
Impact: fix secondary-CPU wakeup/init path with numaq and es7000
While looking at wakeup_secondary_cpu for WAKE_SECONDARY_VIA_NMI:
|#ifdef WAKE_SECONDARY_VIA_NMI
|/*
| * Poke the other CPU in the eye via NMI to wake it up. Remember that the normal
| * INIT, INIT, STARTUP sequence will reset the chip hard for us, and this
| * won't ... remember to clear down the APIC, etc later.
| */
|static int __devinit
|wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip)
|{
| unsigned long send_status, accept_status = 0;
| int maxlvt;
|...
| if (APIC_INTEGRATED(apic_version[phys_apicid])) {
| maxlvt = lapic_get_maxlvt();
I noticed that there is no warning about undefined phys_apicid...
because WAKE_SECONDARY_VIA_NMI and WAKE_SECONDARY_VIA_INIT can not be
defined at the same time. So NUMAQ is using wrong wakeup_secondary_cpu.
WAKE_SECONDARY_VIA_NMI, WAKE_SECONDARY_VIA_INIT and
WAKE_SECONDARY_VIA_MIP are variants of a weird and fragile
preprocessor-driven "HAL" mechanisms to specify the kind of secondary-CPU
wakeup strategy a given x86 kernel will use.
The vast majority of systems want to use INIT for secondary wakeup - NUMAQ
uses an NMI, (old-style-) ES7000 uses 'MIP' (a firmware driven in-memory
flag to let secondaries continue).
So convert these mechanisms to x86_quirks and add a
->wakeup_secondary_cpu() method to specify the rare exception
to the sane default.
Extend genapic accordingly as well, for 32-bit.
While looking further, I noticed that functions in wakecup.h for numaq
and es7000 are different to the default in mach_wakecpu.h - but smpboot.c
will only use default mach_wakecpu.h with smphook.h.
So we need to add mach_wakecpu.h for mach_generic, to properly support
numaq and es7000, and vectorize the following SMP init methods:
int trampoline_phys_low;
int trampoline_phys_high;
void (*wait_for_init_deassert)(atomic_t *deassert);
void (*smp_callin_clear_local_apic)(void);
void (*store_NMI_vector)(unsigned short *high, unsigned short *low);
void (*restore_NMI_vector)(unsigned short *high, unsigned short *low);
void (*inquire_remote_apic)(int apicid);
Signed-off-by: Yinghai Lu <yinghai@kernel.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-11-16 19:12:49 +08:00
|
|
|
wakeup_secondary_cpu_via_init(int phys_apicid, unsigned long start_eip)
|
2008-03-20 01:25:59 +08:00
|
|
|
{
|
|
|
|
unsigned long send_status, accept_status = 0;
|
|
|
|
int maxlvt, num_starts, j;
|
|
|
|
|
x86: APIC: remove apic_write_around(); use alternatives
Use alternatives to select the workaround for the 11AP Pentium erratum
for the affected steppings on the fly rather than build time. Remove the
X86_GOOD_APIC configuration option and replace all the calls to
apic_write_around() with plain apic_write(), protecting accesses to the
ESR as appropriate due to the 3AP Pentium erratum. Remove
apic_read_around() and all its invocations altogether as not needed.
Remove apic_write_atomic() and all its implementing backends. The use of
ASM_OUTPUT2() is not strictly needed for input constraints, but I have
used it for readability's sake.
I had the feeling no one else was brave enough to do it, so I went ahead
and here it is. Verified by checking the generated assembly and tested
with both a 32-bit and a 64-bit configuration, also with the 11AP
"feature" forced on and verified with gdb on /proc/kcore to work as
expected (as an 11AP machines are quite hard to get hands on these days).
Some script complained about the use of "volatile", but apic_write() needs
it for the same reason and is effectively a replacement for writel(), so I
have disregarded it.
I am not sure what the policy wrt defconfig files is, they are generated
and there is risk of a conflict resulting from an unrelated change, so I
have left changes to them out. The option will get removed from them at
the next run.
Some testing with machines other than mine will be needed to avoid some
stupid mistake, but despite its volume, the change is not really that
intrusive, so I am fairly confident that because it works for me, it will
everywhere.
Signed-off-by: Maciej W. Rozycki <macro@linux-mips.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-07-17 02:15:30 +08:00
|
|
|
maxlvt = lapic_get_maxlvt();
|
|
|
|
|
2008-03-20 01:25:59 +08:00
|
|
|
/*
|
|
|
|
* Be paranoid about clearing APIC errors.
|
|
|
|
*/
|
|
|
|
if (APIC_INTEGRATED(apic_version[phys_apicid])) {
|
x86: APIC: remove apic_write_around(); use alternatives
Use alternatives to select the workaround for the 11AP Pentium erratum
for the affected steppings on the fly rather than build time. Remove the
X86_GOOD_APIC configuration option and replace all the calls to
apic_write_around() with plain apic_write(), protecting accesses to the
ESR as appropriate due to the 3AP Pentium erratum. Remove
apic_read_around() and all its invocations altogether as not needed.
Remove apic_write_atomic() and all its implementing backends. The use of
ASM_OUTPUT2() is not strictly needed for input constraints, but I have
used it for readability's sake.
I had the feeling no one else was brave enough to do it, so I went ahead
and here it is. Verified by checking the generated assembly and tested
with both a 32-bit and a 64-bit configuration, also with the 11AP
"feature" forced on and verified with gdb on /proc/kcore to work as
expected (as an 11AP machines are quite hard to get hands on these days).
Some script complained about the use of "volatile", but apic_write() needs
it for the same reason and is effectively a replacement for writel(), so I
have disregarded it.
I am not sure what the policy wrt defconfig files is, they are generated
and there is risk of a conflict resulting from an unrelated change, so I
have left changes to them out. The option will get removed from them at
the next run.
Some testing with machines other than mine will be needed to avoid some
stupid mistake, but despite its volume, the change is not really that
intrusive, so I am fairly confident that because it works for me, it will
everywhere.
Signed-off-by: Maciej W. Rozycki <macro@linux-mips.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-07-17 02:15:30 +08:00
|
|
|
if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */
|
|
|
|
apic_write(APIC_ESR, 0);
|
2008-03-20 01:25:59 +08:00
|
|
|
apic_read(APIC_ESR);
|
|
|
|
}
|
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Asserting INIT.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Turn INIT on target chip
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Send IPI
|
|
|
|
*/
|
2008-07-11 02:16:49 +08:00
|
|
|
apic_icr_write(APIC_INT_LEVELTRIG | APIC_INT_ASSERT | APIC_DM_INIT,
|
|
|
|
phys_apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Waiting for send to finish...\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
send_status = safe_apic_wait_icr_idle();
|
|
|
|
|
|
|
|
mdelay(10);
|
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Deasserting INIT.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/* Target chip */
|
|
|
|
/* Send IPI */
|
2008-07-11 02:16:49 +08:00
|
|
|
apic_icr_write(APIC_INT_LEVELTRIG | APIC_DM_INIT, phys_apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Waiting for send to finish...\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
send_status = safe_apic_wait_icr_idle();
|
|
|
|
|
|
|
|
mb();
|
|
|
|
atomic_set(&init_deasserted, 1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Should we send STARTUP IPIs ?
|
|
|
|
*
|
|
|
|
* Determine this based on the APIC version.
|
|
|
|
* If we don't have an integrated APIC, don't send the STARTUP IPIs.
|
|
|
|
*/
|
|
|
|
if (APIC_INTEGRATED(apic_version[phys_apicid]))
|
|
|
|
num_starts = 2;
|
|
|
|
else
|
|
|
|
num_starts = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Paravirt / VMI wants a startup IPI hook here to set up the
|
|
|
|
* target processor state.
|
|
|
|
*/
|
|
|
|
startup_ipi_hook(phys_apicid, (unsigned long) start_secondary,
|
|
|
|
(unsigned long)stack_start.sp);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Run STARTUP IPI loop.
|
|
|
|
*/
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("#startup loops: %d.\n", num_starts);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
for (j = 1; j <= num_starts; j++) {
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Sending STARTUP #%d.\n", j);
|
x86: APIC: remove apic_write_around(); use alternatives
Use alternatives to select the workaround for the 11AP Pentium erratum
for the affected steppings on the fly rather than build time. Remove the
X86_GOOD_APIC configuration option and replace all the calls to
apic_write_around() with plain apic_write(), protecting accesses to the
ESR as appropriate due to the 3AP Pentium erratum. Remove
apic_read_around() and all its invocations altogether as not needed.
Remove apic_write_atomic() and all its implementing backends. The use of
ASM_OUTPUT2() is not strictly needed for input constraints, but I have
used it for readability's sake.
I had the feeling no one else was brave enough to do it, so I went ahead
and here it is. Verified by checking the generated assembly and tested
with both a 32-bit and a 64-bit configuration, also with the 11AP
"feature" forced on and verified with gdb on /proc/kcore to work as
expected (as an 11AP machines are quite hard to get hands on these days).
Some script complained about the use of "volatile", but apic_write() needs
it for the same reason and is effectively a replacement for writel(), so I
have disregarded it.
I am not sure what the policy wrt defconfig files is, they are generated
and there is risk of a conflict resulting from an unrelated change, so I
have left changes to them out. The option will get removed from them at
the next run.
Some testing with machines other than mine will be needed to avoid some
stupid mistake, but despite its volume, the change is not really that
intrusive, so I am fairly confident that because it works for me, it will
everywhere.
Signed-off-by: Maciej W. Rozycki <macro@linux-mips.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-07-17 02:15:30 +08:00
|
|
|
if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */
|
|
|
|
apic_write(APIC_ESR, 0);
|
2008-03-20 01:25:59 +08:00
|
|
|
apic_read(APIC_ESR);
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("After apic_write.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* STARTUP IPI
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Target chip */
|
|
|
|
/* Boot on the stack */
|
|
|
|
/* Kick the second */
|
2008-07-11 02:16:49 +08:00
|
|
|
apic_icr_write(APIC_DM_STARTUP | (start_eip >> 12),
|
|
|
|
phys_apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Give the other CPU some time to accept the IPI.
|
|
|
|
*/
|
|
|
|
udelay(300);
|
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Startup point 1.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Waiting for send to finish...\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
send_status = safe_apic_wait_icr_idle();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Give the other CPU some time to accept the IPI.
|
|
|
|
*/
|
|
|
|
udelay(200);
|
x86: APIC: remove apic_write_around(); use alternatives
Use alternatives to select the workaround for the 11AP Pentium erratum
for the affected steppings on the fly rather than build time. Remove the
X86_GOOD_APIC configuration option and replace all the calls to
apic_write_around() with plain apic_write(), protecting accesses to the
ESR as appropriate due to the 3AP Pentium erratum. Remove
apic_read_around() and all its invocations altogether as not needed.
Remove apic_write_atomic() and all its implementing backends. The use of
ASM_OUTPUT2() is not strictly needed for input constraints, but I have
used it for readability's sake.
I had the feeling no one else was brave enough to do it, so I went ahead
and here it is. Verified by checking the generated assembly and tested
with both a 32-bit and a 64-bit configuration, also with the 11AP
"feature" forced on and verified with gdb on /proc/kcore to work as
expected (as an 11AP machines are quite hard to get hands on these days).
Some script complained about the use of "volatile", but apic_write() needs
it for the same reason and is effectively a replacement for writel(), so I
have disregarded it.
I am not sure what the policy wrt defconfig files is, they are generated
and there is risk of a conflict resulting from an unrelated change, so I
have left changes to them out. The option will get removed from them at
the next run.
Some testing with machines other than mine will be needed to avoid some
stupid mistake, but despite its volume, the change is not really that
intrusive, so I am fairly confident that because it works for me, it will
everywhere.
Signed-off-by: Maciej W. Rozycki <macro@linux-mips.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-07-17 02:15:30 +08:00
|
|
|
if (maxlvt > 3) /* Due to the Pentium erratum 3AP. */
|
2008-03-20 01:25:59 +08:00
|
|
|
apic_write(APIC_ESR, 0);
|
|
|
|
accept_status = (apic_read(APIC_ESR) & 0xEF);
|
|
|
|
if (send_status || accept_status)
|
|
|
|
break;
|
|
|
|
}
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("After Startup.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
if (send_status)
|
|
|
|
printk(KERN_ERR "APIC never delivered???\n");
|
|
|
|
if (accept_status)
|
|
|
|
printk(KERN_ERR "APIC delivery error (%lx).\n", accept_status);
|
|
|
|
|
|
|
|
return (send_status | accept_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct create_idle {
|
|
|
|
struct work_struct work;
|
|
|
|
struct task_struct *idle;
|
|
|
|
struct completion done;
|
|
|
|
int cpu;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void __cpuinit do_fork_idle(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct create_idle *c_idle =
|
|
|
|
container_of(work, struct create_idle, work);
|
|
|
|
|
|
|
|
c_idle->idle = fork_idle(c_idle->cpu);
|
|
|
|
complete(&c_idle->done);
|
|
|
|
}
|
|
|
|
|
2009-12-11 09:19:36 +08:00
|
|
|
/* reduce the number of lines printed when booting a large cpu count system */
|
|
|
|
static void __cpuinit announce_cpu(int cpu, int apicid)
|
|
|
|
{
|
|
|
|
static int current_node = -1;
|
2010-06-02 03:04:55 +08:00
|
|
|
int node = early_cpu_to_node(cpu);
|
2009-12-11 09:19:36 +08:00
|
|
|
|
|
|
|
if (system_state == SYSTEM_BOOTING) {
|
|
|
|
if (node != current_node) {
|
|
|
|
if (current_node > (-1))
|
|
|
|
pr_cont(" Ok.\n");
|
|
|
|
current_node = node;
|
|
|
|
pr_info("Booting Node %3d, Processors ", node);
|
|
|
|
}
|
|
|
|
pr_cont(" #%d%s", cpu, cpu == (nr_cpu_ids - 1) ? " Ok.\n" : "");
|
|
|
|
return;
|
|
|
|
} else
|
|
|
|
pr_info("Booting Node %d Processor %d APIC 0x%x\n",
|
|
|
|
node, cpu, apicid);
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:25:59 +08:00
|
|
|
/*
|
|
|
|
* NOTE - on most systems this is a PHYSICAL apic ID, but on multiquad
|
|
|
|
* (ie clustered apic addressing mode), this is a LOGICAL apic ID.
|
2009-02-26 20:51:40 +08:00
|
|
|
* Returns zero if CPU booted OK, else error code from
|
|
|
|
* ->wakeup_secondary_cpu.
|
2008-03-20 01:25:59 +08:00
|
|
|
*/
|
2009-02-17 23:22:09 +08:00
|
|
|
static int __cpuinit do_boot_cpu(int apicid, int cpu)
|
2008-03-20 01:25:59 +08:00
|
|
|
{
|
|
|
|
unsigned long boot_error = 0;
|
|
|
|
unsigned long start_ip;
|
2009-02-17 23:22:09 +08:00
|
|
|
int timeout;
|
2008-03-20 01:25:59 +08:00
|
|
|
struct create_idle c_idle = {
|
2009-02-17 23:22:09 +08:00
|
|
|
.cpu = cpu,
|
|
|
|
.done = COMPLETION_INITIALIZER_ONSTACK(c_idle.done),
|
2008-03-20 01:25:59 +08:00
|
|
|
};
|
2009-02-17 23:22:09 +08:00
|
|
|
|
2009-11-16 00:09:48 +08:00
|
|
|
INIT_WORK_ON_STACK(&c_idle.work, do_fork_idle);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
alternatives_smp_switch(1);
|
|
|
|
|
|
|
|
c_idle.idle = get_idle_for_cpu(cpu);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We can't use kernel_thread since we must avoid to
|
|
|
|
* reschedule the child.
|
|
|
|
*/
|
|
|
|
if (c_idle.idle) {
|
|
|
|
c_idle.idle->thread.sp = (unsigned long) (((struct pt_regs *)
|
|
|
|
(THREAD_SIZE + task_stack_page(c_idle.idle))) - 1);
|
|
|
|
init_idle(c_idle.idle, cpu);
|
|
|
|
goto do_rest;
|
|
|
|
}
|
|
|
|
|
2010-08-10 08:20:33 +08:00
|
|
|
schedule_work(&c_idle.work);
|
|
|
|
wait_for_completion(&c_idle.done);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
if (IS_ERR(c_idle.idle)) {
|
|
|
|
printk("failed fork for CPU %d\n", cpu);
|
2009-11-16 00:09:48 +08:00
|
|
|
destroy_work_on_stack(&c_idle.work);
|
2008-03-20 01:25:59 +08:00
|
|
|
return PTR_ERR(c_idle.idle);
|
|
|
|
}
|
|
|
|
|
|
|
|
set_idle_for_cpu(cpu, c_idle.idle);
|
|
|
|
do_rest:
|
|
|
|
per_cpu(current_task, cpu) = c_idle.idle;
|
2009-01-18 23:38:58 +08:00
|
|
|
#ifdef CONFIG_X86_32
|
2008-03-20 01:25:59 +08:00
|
|
|
/* Stack for startup_32 can be just as for start_secondary onwards */
|
|
|
|
irq_ctx_init(cpu);
|
x86-32: Separate 1:1 pagetables from swapper_pg_dir
This patch fixes machine crashes which occur when heavily exercising the
CPU hotplug codepaths on a 32-bit kernel. These crashes are caused by
AMD Erratum 383 and result in a fatal machine check exception. Here's
the scenario:
1. On 32-bit, the swapper_pg_dir page table is used as the initial page
table for booting a secondary CPU.
2. To make this work, swapper_pg_dir needs a direct mapping of physical
memory in it (the low mappings). By adding those low, large page (2M)
mappings (PAE kernel), we create the necessary conditions for Erratum
383 to occur.
3. Other CPUs which do not participate in the off- and onlining game may
use swapper_pg_dir while the low mappings are present (when leave_mm is
called). For all steps below, the CPU referred to is a CPU that is using
swapper_pg_dir, and not the CPU which is being onlined.
4. The presence of the low mappings in swapper_pg_dir can result
in TLB entries for addresses below __PAGE_OFFSET to be established
speculatively. These TLB entries are marked global and large.
5. When the CPU with such TLB entry switches to another page table, this
TLB entry remains because it is global.
6. The process then generates an access to an address covered by the
above TLB entry but there is a permission mismatch - the TLB entry
covers a large global page not accessible to userspace.
7. Due to this permission mismatch a new 4kb, user TLB entry gets
established. Further, Erratum 383 provides for a small window of time
where both TLB entries are present. This results in an uncorrectable
machine check exception signalling a TLB multimatch which panics the
machine.
There are two ways to fix this issue:
1. Always do a global TLB flush when a new cr3 is loaded and the
old page table was swapper_pg_dir. I consider this a hack hard
to understand and with performance implications
2. Do not use swapper_pg_dir to boot secondary CPUs like 64-bit
does.
This patch implements solution 2. It introduces a trampoline_pg_dir
which has the same layout as swapper_pg_dir with low_mappings. This page
table is used as the initial page table of the booting CPU. Later in the
bringup process, it switches to swapper_pg_dir and does a global TLB
flush. This fixes the crashes in our test cases.
-v2: switch to swapper_pg_dir right after entering start_secondary() so
that we are able to access percpu data which might not be mapped in the
trampoline page table.
Signed-off-by: Joerg Roedel <joerg.roedel@amd.com>
LKML-Reference: <20100816123833.GB28147@aftab>
Signed-off-by: Borislav Petkov <borislav.petkov@amd.com>
Signed-off-by: H. Peter Anvin <hpa@zytor.com>
2010-08-16 20:38:33 +08:00
|
|
|
initial_page_table = __pa(&trampoline_pg_dir);
|
2008-03-20 01:25:59 +08:00
|
|
|
#else
|
|
|
|
clear_tsk_thread_flag(c_idle.idle, TIF_FORK);
|
2009-01-13 19:41:35 +08:00
|
|
|
initial_gs = per_cpu_offset(cpu);
|
2009-01-18 23:38:58 +08:00
|
|
|
per_cpu(kernel_stack, cpu) =
|
|
|
|
(unsigned long)task_stack_page(c_idle.idle) -
|
|
|
|
KERNEL_STACK_OFFSET + THREAD_SIZE;
|
2008-03-20 01:25:59 +08:00
|
|
|
#endif
|
2008-05-29 07:19:53 +08:00
|
|
|
early_gdt_descr.address = (unsigned long)get_cpu_gdt_table(cpu);
|
2008-05-29 00:01:54 +08:00
|
|
|
initial_code = (unsigned long)start_secondary;
|
2008-05-28 09:22:54 +08:00
|
|
|
stack_start.sp = (void *) c_idle.idle->thread.sp;
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/* start_ip had better be page-aligned! */
|
|
|
|
start_ip = setup_trampoline();
|
|
|
|
|
2009-12-11 09:19:36 +08:00
|
|
|
/* So we see what's up */
|
|
|
|
announce_cpu(cpu, apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This grunge runs the startup process for
|
|
|
|
* the targeted processor.
|
|
|
|
*/
|
|
|
|
|
|
|
|
atomic_set(&init_deasserted, 0);
|
|
|
|
|
2008-04-17 00:45:15 +08:00
|
|
|
if (get_uv_system_type() != UV_NON_UNIQUE_APIC) {
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Setting warm reset code and vector.\n");
|
2008-03-20 01:25:59 +08:00
|
|
|
|
2008-04-17 00:45:15 +08:00
|
|
|
smpboot_setup_warm_reset_vector(start_ip);
|
|
|
|
/*
|
|
|
|
* Be paranoid about clearing APIC errors.
|
2008-10-22 22:00:09 +08:00
|
|
|
*/
|
|
|
|
if (APIC_INTEGRATED(apic_version[boot_cpu_physical_apicid])) {
|
|
|
|
apic_write(APIC_ESR, 0);
|
|
|
|
apic_read(APIC_ESR);
|
|
|
|
}
|
2008-04-17 00:45:15 +08:00
|
|
|
}
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
2009-02-26 20:51:40 +08:00
|
|
|
* Kick the secondary CPU. Use the method in the APIC driver
|
|
|
|
* if it's defined - or use an INIT boot APIC message otherwise:
|
2008-03-20 01:25:59 +08:00
|
|
|
*/
|
2009-02-26 20:51:40 +08:00
|
|
|
if (apic->wakeup_secondary_cpu)
|
|
|
|
boot_error = apic->wakeup_secondary_cpu(apicid, start_ip);
|
|
|
|
else
|
|
|
|
boot_error = wakeup_secondary_cpu_via_init(apicid, start_ip);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
if (!boot_error) {
|
|
|
|
/*
|
|
|
|
* allow APs to start initializing.
|
|
|
|
*/
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Before Callout %d.\n", cpu);
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_set_cpu(cpu, cpu_callout_mask);
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("After Callout %d.\n", cpu);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait 5s total for a response
|
|
|
|
*/
|
|
|
|
for (timeout = 0; timeout < 50000; timeout++) {
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_test_cpu(cpu, cpu_callin_mask))
|
2008-03-20 01:25:59 +08:00
|
|
|
break; /* It has booted */
|
|
|
|
udelay(100);
|
2010-07-31 02:46:42 +08:00
|
|
|
/*
|
|
|
|
* Allow other tasks to run while we wait for the
|
|
|
|
* AP to come online. This also gives a chance
|
|
|
|
* for the MTRR work(triggered by the AP coming online)
|
|
|
|
* to be completed in the stop machine context.
|
|
|
|
*/
|
|
|
|
schedule();
|
2008-03-20 01:25:59 +08:00
|
|
|
}
|
|
|
|
|
2009-12-11 09:19:36 +08:00
|
|
|
if (cpumask_test_cpu(cpu, cpu_callin_mask))
|
|
|
|
pr_debug("CPU%d: has booted.\n", cpu);
|
|
|
|
else {
|
2008-03-20 01:25:59 +08:00
|
|
|
boot_error = 1;
|
|
|
|
if (*((volatile unsigned char *)trampoline_base)
|
|
|
|
== 0xA5)
|
|
|
|
/* trampoline started but...? */
|
2009-12-11 09:19:36 +08:00
|
|
|
pr_err("CPU%d: Stuck ??\n", cpu);
|
2008-03-20 01:25:59 +08:00
|
|
|
else
|
|
|
|
/* trampoline code not run */
|
2009-12-11 09:19:36 +08:00
|
|
|
pr_err("CPU%d: Not responding.\n", cpu);
|
2009-01-28 23:31:52 +08:00
|
|
|
if (apic->inquire_remote_apic)
|
|
|
|
apic->inquire_remote_apic(apicid);
|
2008-03-20 01:25:59 +08:00
|
|
|
}
|
|
|
|
}
|
2009-01-13 19:41:35 +08:00
|
|
|
|
2008-03-20 01:25:59 +08:00
|
|
|
if (boot_error) {
|
|
|
|
/* Try to put things back the way they were before ... */
|
x86: cleanup early per cpu variables/accesses v4
* Introduce a new PER_CPU macro called "EARLY_PER_CPU". This is
used by some per_cpu variables that are initialized and accessed
before there are per_cpu areas allocated.
["Early" in respect to per_cpu variables is "earlier than the per_cpu
areas have been setup".]
This patchset adds these new macros:
DEFINE_EARLY_PER_CPU(_type, _name, _initvalue)
EXPORT_EARLY_PER_CPU_SYMBOL(_name)
DECLARE_EARLY_PER_CPU(_type, _name)
early_per_cpu_ptr(_name)
early_per_cpu_map(_name, _idx)
early_per_cpu(_name, _cpu)
The DEFINE macro defines the per_cpu variable as well as the early
map and pointer. It also initializes the per_cpu variable and map
elements to "_initvalue". The early_* macros provide access to
the initial map (usually setup during system init) and the early
pointer. This pointer is initialized to point to the early map
but is then NULL'ed when the actual per_cpu areas are setup. After
that the per_cpu variable is the correct access to the variable.
The early_per_cpu() macro is not very efficient but does show how to
access the variable if you have a function that can be called both
"early" and "late". It tests the early ptr to be NULL, and if not
then it's still valid. Otherwise, the per_cpu variable is used
instead:
#define early_per_cpu(_name, _cpu) \
(early_per_cpu_ptr(_name) ? \
early_per_cpu_ptr(_name)[_cpu] : \
per_cpu(_name, _cpu))
A better method is to actually check the pointer manually. In the
case below, numa_set_node can be called both "early" and "late":
void __cpuinit numa_set_node(int cpu, int node)
{
int *cpu_to_node_map = early_per_cpu_ptr(x86_cpu_to_node_map);
if (cpu_to_node_map)
cpu_to_node_map[cpu] = node;
else
per_cpu(x86_cpu_to_node_map, cpu) = node;
}
* Add a flag "arch_provides_topology_pointers" that indicates pointers
to topology cpumask_t maps are available. Otherwise, use the function
returning the cpumask_t value. This is useful if cpumask_t set size
is very large to avoid copying data on to/off of the stack.
* The coverage of CONFIG_DEBUG_PER_CPU_MAPS has been increased while
the non-debug case has been optimized a bit.
* Remove an unreferenced compiler warning in drivers/base/topology.c
* Clean up #ifdef in setup.c
For inclusion into sched-devel/latest tree.
Based on:
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6.git
+ sched-devel/latest .../mingo/linux-2.6-sched-devel.git
Signed-off-by: Mike Travis <travis@sgi.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
2008-05-13 03:21:12 +08:00
|
|
|
numa_remove_cpu(cpu); /* was set by numa_add_cpu */
|
2009-01-04 21:18:03 +08:00
|
|
|
|
|
|
|
/* was set by do_boot_cpu() */
|
|
|
|
cpumask_clear_cpu(cpu, cpu_callout_mask);
|
|
|
|
|
|
|
|
/* was set by cpu_init() */
|
|
|
|
cpumask_clear_cpu(cpu, cpu_initialized_mask);
|
|
|
|
|
|
|
|
set_cpu_present(cpu, false);
|
2008-03-20 01:25:59 +08:00
|
|
|
per_cpu(x86_cpu_to_apicid, cpu) = BAD_APICID;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* mark "stuck" area as not stuck */
|
|
|
|
*((volatile unsigned long *)trampoline_base) = 0;
|
|
|
|
|
2009-04-04 08:15:53 +08:00
|
|
|
if (get_uv_system_type() != UV_NON_UNIQUE_APIC) {
|
|
|
|
/*
|
|
|
|
* Cleanup possible dangling ends...
|
|
|
|
*/
|
|
|
|
smpboot_restore_warm_reset_vector();
|
|
|
|
}
|
2008-04-08 02:38:33 +08:00
|
|
|
|
2009-11-16 00:09:48 +08:00
|
|
|
destroy_work_on_stack(&c_idle.work);
|
2008-03-20 01:25:59 +08:00
|
|
|
return boot_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
int __cpuinit native_cpu_up(unsigned int cpu)
|
|
|
|
{
|
2009-01-28 13:50:47 +08:00
|
|
|
int apicid = apic->cpu_present_to_apicid(cpu);
|
2008-03-20 01:25:59 +08:00
|
|
|
unsigned long flags;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
WARN_ON(irqs_disabled());
|
|
|
|
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("++++++++++++++++++++=_---CPU UP %u\n", cpu);
|
2008-03-20 01:25:59 +08:00
|
|
|
|
|
|
|
if (apicid == BAD_APICID || apicid == boot_cpu_physical_apicid ||
|
|
|
|
!physid_isset(apicid, phys_cpu_present_map)) {
|
|
|
|
printk(KERN_ERR "%s: bad cpu %d\n", __func__, cpu);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Already booted CPU?
|
|
|
|
*/
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_test_cpu(cpu, cpu_callin_mask)) {
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("do_boot_cpu %d Already started\n", cpu);
|
2008-03-20 01:25:59 +08:00
|
|
|
return -ENOSYS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Save current MTRR state in case it was changed since early boot
|
|
|
|
* (e.g. by the ACPI SMI) to initialize new CPUs with MTRRs in sync:
|
|
|
|
*/
|
|
|
|
mtrr_save_state();
|
|
|
|
|
|
|
|
per_cpu(cpu_state, cpu) = CPU_UP_PREPARE;
|
|
|
|
|
|
|
|
err = do_boot_cpu(apicid, cpu);
|
x86: fix app crashes after SMP resume
After resume on a 2cpu laptop, kernel builds collapse with a sed hang,
sh or make segfault (often on 20295564), real-time signal to cc1 etc.
Several hurdles to jump, but a manually-assisted bisect led to -rc1's
d2bcbad5f3ad38a1c09861bca7e252dde7bb8259 x86: do not zap_low_mappings
in __smp_prepare_cpus. Though the low mappings were removed at bootup,
they were left behind (with Global flags helping to keep them in TLB)
after resume or cpu online, causing the crashes seen.
Reinstate zap_low_mappings (with local __flush_tlb_all) for each cpu_up
on x86_32. This used to be serialized by smp_commenced_mask: that's now
gone, but a low_mappings flag will do. No need for native_smp_cpus_done
to repeat the zap: let mem_init zap BSP's low mappings just like on UP.
(In passing, fix error code from native_cpu_up: do_boot_cpu returns a
variety of diagnostic values, Dprintk what it says but convert to -EIO.
And save_pg_dir separately before zap_low_mappings: doesn't matter now,
but zapping twice in succession wiped out resume's swsusp_pg_dir.)
That worked well on the duo and one quad, but wouldn't boot 3rd or 4th
cpu on P4 Xeon, oopsing just after unlock_ipi_call_lock. The TLB flush
IPI now being sent reveals a long-standing bug: the booting cpu has its
APIC readied in smp_callin at the top of start_secondary, but isn't put
into the cpu_online_map until just before that unlock_ipi_call_lock.
So native_smp_call_function_mask to online cpus would send_IPI_allbutself,
including the cpu just coming up, though it has been excluded from the
count to wait for: by the time it handles the IPI, the call data on
native_smp_call_function_mask's stack may well have been overwritten.
So fall back to send_IPI_mask while cpu_online_map does not match
cpu_callout_map: perhaps there's a better APICological fix to be
made at the start_secondary end, but I wouldn't know that.
Signed-off-by: Hugh Dickins <hugh@veritas.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-05-13 21:26:57 +08:00
|
|
|
|
|
|
|
if (err) {
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("do_boot_cpu failed %d\n", err);
|
x86: fix app crashes after SMP resume
After resume on a 2cpu laptop, kernel builds collapse with a sed hang,
sh or make segfault (often on 20295564), real-time signal to cc1 etc.
Several hurdles to jump, but a manually-assisted bisect led to -rc1's
d2bcbad5f3ad38a1c09861bca7e252dde7bb8259 x86: do not zap_low_mappings
in __smp_prepare_cpus. Though the low mappings were removed at bootup,
they were left behind (with Global flags helping to keep them in TLB)
after resume or cpu online, causing the crashes seen.
Reinstate zap_low_mappings (with local __flush_tlb_all) for each cpu_up
on x86_32. This used to be serialized by smp_commenced_mask: that's now
gone, but a low_mappings flag will do. No need for native_smp_cpus_done
to repeat the zap: let mem_init zap BSP's low mappings just like on UP.
(In passing, fix error code from native_cpu_up: do_boot_cpu returns a
variety of diagnostic values, Dprintk what it says but convert to -EIO.
And save_pg_dir separately before zap_low_mappings: doesn't matter now,
but zapping twice in succession wiped out resume's swsusp_pg_dir.)
That worked well on the duo and one quad, but wouldn't boot 3rd or 4th
cpu on P4 Xeon, oopsing just after unlock_ipi_call_lock. The TLB flush
IPI now being sent reveals a long-standing bug: the booting cpu has its
APIC readied in smp_callin at the top of start_secondary, but isn't put
into the cpu_online_map until just before that unlock_ipi_call_lock.
So native_smp_call_function_mask to online cpus would send_IPI_allbutself,
including the cpu just coming up, though it has been excluded from the
count to wait for: by the time it handles the IPI, the call data on
native_smp_call_function_mask's stack may well have been overwritten.
So fall back to send_IPI_mask while cpu_online_map does not match
cpu_callout_map: perhaps there's a better APICological fix to be
made at the start_secondary end, but I wouldn't know that.
Signed-off-by: Hugh Dickins <hugh@veritas.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-05-13 21:26:57 +08:00
|
|
|
return -EIO;
|
2008-03-20 01:25:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check TSC synchronization with the AP (keep irqs disabled
|
|
|
|
* while doing so):
|
|
|
|
*/
|
|
|
|
local_irq_save(flags);
|
|
|
|
check_tsc_sync_source(cpu);
|
|
|
|
local_irq_restore(flags);
|
|
|
|
|
2008-04-19 22:55:17 +08:00
|
|
|
while (!cpu_online(cpu)) {
|
2008-03-20 01:25:59 +08:00
|
|
|
cpu_relax();
|
|
|
|
touch_nmi_watchdog();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
/*
|
|
|
|
* Fall back to non SMP mode after errors.
|
|
|
|
*
|
|
|
|
* RED-PEN audit/test this more. I bet there is more state messed up here.
|
|
|
|
*/
|
|
|
|
static __init void disable_smp(void)
|
|
|
|
{
|
2009-03-13 12:19:54 +08:00
|
|
|
init_cpu_present(cpumask_of(0));
|
|
|
|
init_cpu_possible(cpumask_of(0));
|
2008-03-20 01:26:11 +08:00
|
|
|
smpboot_clear_io_apic_irqs();
|
2008-05-29 08:09:53 +08:00
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
if (smp_found_config)
|
2008-06-20 10:51:05 +08:00
|
|
|
physid_set_mask_of_physid(boot_cpu_physical_apicid, &phys_cpu_present_map);
|
2008-03-20 01:26:11 +08:00
|
|
|
else
|
2008-06-20 10:51:05 +08:00
|
|
|
physid_set_mask_of_physid(0, &phys_cpu_present_map);
|
2008-03-20 01:26:11 +08:00
|
|
|
map_cpu_to_logical_apicid();
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_set_cpu(0, cpu_sibling_mask(0));
|
|
|
|
cpumask_set_cpu(0, cpu_core_mask(0));
|
2008-03-20 01:26:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Various sanity checks.
|
|
|
|
*/
|
|
|
|
static int __init smp_sanity_check(unsigned max_cpus)
|
|
|
|
{
|
2008-03-29 03:12:16 +08:00
|
|
|
preempt_disable();
|
2008-08-14 17:16:30 +08:00
|
|
|
|
2009-01-30 11:30:04 +08:00
|
|
|
#if !defined(CONFIG_X86_BIGSMP) && defined(CONFIG_X86_32)
|
2008-08-14 17:16:30 +08:00
|
|
|
if (def_to_bigsmp && nr_cpu_ids > 8) {
|
|
|
|
unsigned int cpu;
|
|
|
|
unsigned nr;
|
|
|
|
|
|
|
|
printk(KERN_WARNING
|
|
|
|
"More than 8 CPUs detected - skipping them.\n"
|
2009-01-30 06:19:22 +08:00
|
|
|
"Use CONFIG_X86_BIGSMP.\n");
|
2008-08-14 17:16:30 +08:00
|
|
|
|
|
|
|
nr = 0;
|
|
|
|
for_each_present_cpu(cpu) {
|
|
|
|
if (nr >= 8)
|
2009-01-04 21:18:03 +08:00
|
|
|
set_cpu_present(cpu, false);
|
2008-08-14 17:16:30 +08:00
|
|
|
nr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
nr = 0;
|
|
|
|
for_each_possible_cpu(cpu) {
|
|
|
|
if (nr >= 8)
|
2009-01-04 21:18:03 +08:00
|
|
|
set_cpu_possible(cpu, false);
|
2008-08-14 17:16:30 +08:00
|
|
|
nr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
nr_cpu_ids = 8;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
if (!physid_isset(hard_smp_processor_id(), phys_cpu_present_map)) {
|
2008-12-05 19:42:20 +08:00
|
|
|
printk(KERN_WARNING
|
|
|
|
"weird, boot CPU (#%d) not listed by the BIOS.\n",
|
|
|
|
hard_smp_processor_id());
|
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
physid_set(hard_smp_processor_id(), phys_cpu_present_map);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we couldn't find an SMP configuration at boot time,
|
|
|
|
* get out of here now!
|
|
|
|
*/
|
|
|
|
if (!smp_found_config && !acpi_lapic) {
|
2008-03-29 03:12:16 +08:00
|
|
|
preempt_enable();
|
2008-03-20 01:26:11 +08:00
|
|
|
printk(KERN_NOTICE "SMP motherboard not detected.\n");
|
|
|
|
disable_smp();
|
|
|
|
if (APIC_init_uniprocessor())
|
|
|
|
printk(KERN_NOTICE "Local APIC not detected."
|
|
|
|
" Using dummy APIC emulation.\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Should not be necessary because the MP table should list the boot
|
|
|
|
* CPU too, but we do it for the sake of robustness anyway.
|
|
|
|
*/
|
2009-01-28 19:43:18 +08:00
|
|
|
if (!apic->check_phys_apicid_present(boot_cpu_physical_apicid)) {
|
2008-03-20 01:26:11 +08:00
|
|
|
printk(KERN_NOTICE
|
|
|
|
"weird, boot CPU (#%d) not listed by the BIOS.\n",
|
|
|
|
boot_cpu_physical_apicid);
|
|
|
|
physid_set(hard_smp_processor_id(), phys_cpu_present_map);
|
|
|
|
}
|
2008-03-29 03:12:16 +08:00
|
|
|
preempt_enable();
|
2008-03-20 01:26:11 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If we couldn't find a local APIC, then get out of here now!
|
|
|
|
*/
|
|
|
|
if (APIC_INTEGRATED(apic_version[boot_cpu_physical_apicid]) &&
|
|
|
|
!cpu_has_apic) {
|
2009-06-07 20:48:40 +08:00
|
|
|
if (!disable_apic) {
|
|
|
|
pr_err("BIOS bug, local APIC #%d not detected!...\n",
|
|
|
|
boot_cpu_physical_apicid);
|
|
|
|
pr_err("... forcing use of dummy APIC emulation."
|
2008-03-20 01:26:11 +08:00
|
|
|
"(tell your hw vendor)\n");
|
2009-06-07 20:48:40 +08:00
|
|
|
}
|
2008-03-20 01:26:11 +08:00
|
|
|
smpboot_clear_io_apic();
|
2009-01-31 10:36:17 +08:00
|
|
|
arch_disable_smp_support();
|
2008-03-20 01:26:11 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
verify_local_APIC();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If SMP should be disabled, then really disable it!
|
|
|
|
*/
|
|
|
|
if (!max_cpus) {
|
2008-05-22 05:09:43 +08:00
|
|
|
printk(KERN_INFO "SMP mode deactivated.\n");
|
2008-03-20 01:26:11 +08:00
|
|
|
smpboot_clear_io_apic();
|
2008-06-06 10:28:02 +08:00
|
|
|
|
|
|
|
localise_nmi_watchdog();
|
|
|
|
|
2008-04-22 05:14:44 +08:00
|
|
|
connect_bsp_APIC();
|
|
|
|
setup_local_APIC();
|
|
|
|
end_local_APIC_setup();
|
2008-03-20 01:26:11 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __init smp_cpu_index_default(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct cpuinfo_x86 *c;
|
|
|
|
|
2008-04-19 22:55:17 +08:00
|
|
|
for_each_possible_cpu(i) {
|
2008-03-20 01:26:11 +08:00
|
|
|
c = &cpu_data(i);
|
|
|
|
/* mark all to hotplug */
|
2009-01-01 10:08:46 +08:00
|
|
|
c->cpu_index = nr_cpu_ids;
|
2008-03-20 01:26:11 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Prepare for SMP bootup. The MP table or ACPI has been read
|
|
|
|
* earlier. Just do some sanity checking here and enable APIC mode.
|
|
|
|
*/
|
|
|
|
void __init native_smp_prepare_cpus(unsigned int max_cpus)
|
|
|
|
{
|
2009-03-13 12:19:50 +08:00
|
|
|
unsigned int i;
|
|
|
|
|
2008-05-12 21:44:38 +08:00
|
|
|
preempt_disable();
|
2008-03-20 01:26:11 +08:00
|
|
|
smp_cpu_index_default();
|
|
|
|
current_cpu_data = boot_cpu_data;
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_copy(cpu_callin_mask, cpumask_of(0));
|
2008-03-20 01:26:11 +08:00
|
|
|
mb();
|
|
|
|
/*
|
|
|
|
* Setup boot CPU information
|
|
|
|
*/
|
|
|
|
smp_store_cpu_info(0); /* Final full version of the data */
|
2008-07-11 02:16:49 +08:00
|
|
|
#ifdef CONFIG_X86_32
|
2008-03-20 01:26:11 +08:00
|
|
|
boot_cpu_logical_apicid = logical_smp_processor_id();
|
2008-07-11 02:16:49 +08:00
|
|
|
#endif
|
2008-03-20 01:26:11 +08:00
|
|
|
current_thread_info()->cpu = 0; /* needed? */
|
2009-03-13 12:19:50 +08:00
|
|
|
for_each_possible_cpu(i) {
|
2009-06-15 14:58:26 +08:00
|
|
|
zalloc_cpumask_var(&per_cpu(cpu_sibling_map, i), GFP_KERNEL);
|
|
|
|
zalloc_cpumask_var(&per_cpu(cpu_core_map, i), GFP_KERNEL);
|
|
|
|
zalloc_cpumask_var(&cpu_data(i).llc_shared_map, GFP_KERNEL);
|
2009-03-13 12:19:50 +08:00
|
|
|
}
|
2008-03-20 01:26:11 +08:00
|
|
|
set_cpu_sibling_map(0);
|
|
|
|
|
2008-07-11 02:16:58 +08:00
|
|
|
enable_IR_x2apic();
|
2009-01-28 13:50:47 +08:00
|
|
|
default_setup_apic_routing();
|
2008-07-11 02:16:58 +08:00
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
if (smp_sanity_check(max_cpus) < 0) {
|
|
|
|
printk(KERN_INFO "SMP disabled\n");
|
|
|
|
disable_smp();
|
2008-05-12 21:44:38 +08:00
|
|
|
goto out;
|
2008-03-20 01:26:11 +08:00
|
|
|
}
|
|
|
|
|
2008-03-29 03:12:16 +08:00
|
|
|
preempt_disable();
|
2008-07-12 09:44:16 +08:00
|
|
|
if (read_apic_id() != boot_cpu_physical_apicid) {
|
2008-03-20 01:26:11 +08:00
|
|
|
panic("Boot APIC ID in local APIC unexpected (%d vs %d)",
|
2008-07-12 09:44:16 +08:00
|
|
|
read_apic_id(), boot_cpu_physical_apicid);
|
2008-03-20 01:26:11 +08:00
|
|
|
/* Or can we switch back to PIC here? */
|
|
|
|
}
|
2008-03-29 03:12:16 +08:00
|
|
|
preempt_enable();
|
2008-03-20 01:26:11 +08:00
|
|
|
|
|
|
|
connect_bsp_APIC();
|
2008-05-29 00:38:28 +08:00
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
/*
|
|
|
|
* Switch from PIC to APIC mode.
|
|
|
|
*/
|
|
|
|
setup_local_APIC();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enable IO APIC before setting up error vector
|
|
|
|
*/
|
|
|
|
if (!skip_ioapic_setup && nr_ioapics)
|
|
|
|
enable_IO_APIC();
|
2009-02-15 15:57:28 +08:00
|
|
|
|
2008-03-20 01:26:11 +08:00
|
|
|
end_local_APIC_setup();
|
|
|
|
|
|
|
|
map_cpu_to_logical_apicid();
|
|
|
|
|
2009-01-28 19:43:18 +08:00
|
|
|
if (apic->setup_portio_remap)
|
|
|
|
apic->setup_portio_remap();
|
2008-03-20 01:26:11 +08:00
|
|
|
|
|
|
|
smpboot_setup_io_apic();
|
|
|
|
/*
|
|
|
|
* Set up local APIC timer on boot CPU.
|
|
|
|
*/
|
|
|
|
|
|
|
|
printk(KERN_INFO "CPU%d: ", 0);
|
|
|
|
print_cpu_info(&cpu_data(0));
|
2009-08-19 18:35:53 +08:00
|
|
|
x86_init.timers.setup_percpu_clockev();
|
2008-08-22 02:49:05 +08:00
|
|
|
|
|
|
|
if (is_uv_system())
|
|
|
|
uv_system_init();
|
2009-08-20 09:05:36 +08:00
|
|
|
|
|
|
|
set_mtrr_aps_delayed_init();
|
2008-05-12 21:44:38 +08:00
|
|
|
out:
|
|
|
|
preempt_enable();
|
2008-03-20 01:26:11 +08:00
|
|
|
}
|
2009-08-20 09:05:36 +08:00
|
|
|
|
|
|
|
void arch_enable_nonboot_cpus_begin(void)
|
|
|
|
{
|
|
|
|
set_mtrr_aps_delayed_init();
|
|
|
|
}
|
|
|
|
|
|
|
|
void arch_enable_nonboot_cpus_end(void)
|
|
|
|
{
|
|
|
|
mtrr_aps_init();
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:26:01 +08:00
|
|
|
/*
|
|
|
|
* Early setup to make printk work.
|
|
|
|
*/
|
|
|
|
void __init native_smp_prepare_boot_cpu(void)
|
|
|
|
{
|
|
|
|
int me = smp_processor_id();
|
2009-01-30 16:47:53 +08:00
|
|
|
switch_to_new_gdt(me);
|
2009-01-04 21:18:03 +08:00
|
|
|
/* already set me in cpu_online_mask in boot_cpu_init() */
|
|
|
|
cpumask_set_cpu(me, cpu_callout_mask);
|
2008-03-20 01:26:01 +08:00
|
|
|
per_cpu(cpu_state, me) = CPU_ONLINE;
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:26:02 +08:00
|
|
|
void __init native_smp_cpus_done(unsigned int max_cpus)
|
|
|
|
{
|
2008-07-22 03:35:38 +08:00
|
|
|
pr_debug("Boot done.\n");
|
2008-03-20 01:26:02 +08:00
|
|
|
|
|
|
|
impress_friends();
|
|
|
|
#ifdef CONFIG_X86_IO_APIC
|
|
|
|
setup_ioapic_dest();
|
|
|
|
#endif
|
|
|
|
check_nmi_watchdog();
|
2009-08-20 09:05:36 +08:00
|
|
|
mtrr_aps_init();
|
2008-03-20 01:26:02 +08:00
|
|
|
}
|
|
|
|
|
2008-12-18 07:21:39 +08:00
|
|
|
static int __initdata setup_possible_cpus = -1;
|
|
|
|
static int __init _setup_possible_cpus(char *str)
|
|
|
|
{
|
|
|
|
get_option(&str, &setup_possible_cpus);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
early_param("possible_cpus", _setup_possible_cpus);
|
|
|
|
|
|
|
|
|
2008-03-04 01:12:42 +08:00
|
|
|
/*
|
2009-03-13 12:19:54 +08:00
|
|
|
* cpu_possible_mask should be static, it cannot change as cpu's
|
2008-03-04 01:12:42 +08:00
|
|
|
* are onlined, or offlined. The reason is per-cpu data-structures
|
|
|
|
* are allocated by some modules at init time, and dont expect to
|
|
|
|
* do this dynamically on cpu arrival/departure.
|
2009-03-13 12:19:54 +08:00
|
|
|
* cpu_present_mask on the other hand can change dynamically.
|
2008-03-04 01:12:42 +08:00
|
|
|
* In case when cpu_hotplug is not compiled, then we resort to current
|
|
|
|
* behaviour, which is cpu_possible == cpu_present.
|
|
|
|
* - Ashok Raj
|
|
|
|
*
|
|
|
|
* Three ways to find out the number of additional hotplug CPUs:
|
|
|
|
* - If the BIOS specified disabled CPUs in ACPI/mptables use that.
|
2008-12-18 07:21:39 +08:00
|
|
|
* - The user can overwrite it with possible_cpus=NUM
|
2008-03-04 01:12:42 +08:00
|
|
|
* - Otherwise don't reserve additional CPUs.
|
|
|
|
* We do this because additional CPUs waste a lot of memory.
|
|
|
|
* -AK
|
|
|
|
*/
|
|
|
|
__init void prefill_possible_map(void)
|
|
|
|
{
|
2008-10-05 23:51:52 +08:00
|
|
|
int i, possible;
|
2008-03-04 01:12:42 +08:00
|
|
|
|
2008-07-03 09:54:40 +08:00
|
|
|
/* no processor from mptable or madt */
|
|
|
|
if (!num_processors)
|
|
|
|
num_processors = 1;
|
|
|
|
|
2010-05-25 03:13:17 +08:00
|
|
|
i = setup_max_cpus ?: 1;
|
|
|
|
if (setup_possible_cpus == -1) {
|
|
|
|
possible = num_processors;
|
|
|
|
#ifdef CONFIG_HOTPLUG_CPU
|
|
|
|
if (setup_max_cpus)
|
|
|
|
possible += disabled_cpus;
|
|
|
|
#else
|
|
|
|
if (possible > i)
|
|
|
|
possible = i;
|
|
|
|
#endif
|
|
|
|
} else
|
2008-12-18 07:21:39 +08:00
|
|
|
possible = setup_possible_cpus;
|
|
|
|
|
2009-01-01 10:08:45 +08:00
|
|
|
total_cpus = max_t(int, possible, num_processors + disabled_cpus);
|
|
|
|
|
2010-02-10 17:20:37 +08:00
|
|
|
/* nr_cpu_ids could be reduced via nr_cpus= */
|
|
|
|
if (possible > nr_cpu_ids) {
|
2008-12-18 07:21:39 +08:00
|
|
|
printk(KERN_WARNING
|
|
|
|
"%d Processors exceeds NR_CPUS limit of %d\n",
|
2010-02-10 17:20:37 +08:00
|
|
|
possible, nr_cpu_ids);
|
|
|
|
possible = nr_cpu_ids;
|
2008-12-18 07:21:39 +08:00
|
|
|
}
|
2008-03-04 01:12:42 +08:00
|
|
|
|
2010-05-25 03:13:17 +08:00
|
|
|
#ifdef CONFIG_HOTPLUG_CPU
|
|
|
|
if (!setup_max_cpus)
|
|
|
|
#endif
|
|
|
|
if (possible > i) {
|
|
|
|
printk(KERN_WARNING
|
|
|
|
"%d Processors exceeds max_cpus limit of %u\n",
|
|
|
|
possible, setup_max_cpus);
|
|
|
|
possible = i;
|
|
|
|
}
|
|
|
|
|
2008-03-04 01:12:42 +08:00
|
|
|
printk(KERN_INFO "SMP: Allowing %d CPUs, %d hotplug CPUs\n",
|
|
|
|
possible, max_t(int, possible - num_processors, 0));
|
|
|
|
|
|
|
|
for (i = 0; i < possible; i++)
|
2009-01-04 21:18:03 +08:00
|
|
|
set_cpu_possible(i, true);
|
2010-05-25 03:13:17 +08:00
|
|
|
for (; i < NR_CPUS; i++)
|
|
|
|
set_cpu_possible(i, false);
|
2008-05-13 03:21:13 +08:00
|
|
|
|
|
|
|
nr_cpu_ids = possible;
|
2008-03-04 01:12:42 +08:00
|
|
|
}
|
2008-03-04 01:13:07 +08:00
|
|
|
|
2008-09-30 06:29:42 +08:00
|
|
|
#ifdef CONFIG_HOTPLUG_CPU
|
|
|
|
|
|
|
|
static void remove_siblinginfo(int cpu)
|
|
|
|
{
|
|
|
|
int sibling;
|
|
|
|
struct cpuinfo_x86 *c = &cpu_data(cpu);
|
|
|
|
|
2009-01-04 21:18:03 +08:00
|
|
|
for_each_cpu(sibling, cpu_core_mask(cpu)) {
|
|
|
|
cpumask_clear_cpu(cpu, cpu_core_mask(sibling));
|
2008-09-30 06:29:42 +08:00
|
|
|
/*/
|
|
|
|
* last thread sibling in this cpu core going down
|
|
|
|
*/
|
2009-01-04 21:18:03 +08:00
|
|
|
if (cpumask_weight(cpu_sibling_mask(cpu)) == 1)
|
2008-09-30 06:29:42 +08:00
|
|
|
cpu_data(sibling).booted_cores--;
|
|
|
|
}
|
|
|
|
|
2009-01-04 21:18:03 +08:00
|
|
|
for_each_cpu(sibling, cpu_sibling_mask(cpu))
|
|
|
|
cpumask_clear_cpu(cpu, cpu_sibling_mask(sibling));
|
|
|
|
cpumask_clear(cpu_sibling_mask(cpu));
|
|
|
|
cpumask_clear(cpu_core_mask(cpu));
|
2008-09-30 06:29:42 +08:00
|
|
|
c->phys_proc_id = 0;
|
|
|
|
c->cpu_core_id = 0;
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_clear_cpu(cpu, cpu_sibling_setup_mask);
|
2008-09-30 06:29:42 +08:00
|
|
|
}
|
|
|
|
|
2008-03-04 01:13:07 +08:00
|
|
|
static void __ref remove_cpu_from_maps(int cpu)
|
|
|
|
{
|
2009-01-04 21:18:03 +08:00
|
|
|
set_cpu_online(cpu, false);
|
|
|
|
cpumask_clear_cpu(cpu, cpu_callout_mask);
|
|
|
|
cpumask_clear_cpu(cpu, cpu_callin_mask);
|
2008-03-04 01:13:07 +08:00
|
|
|
/* was set by cpu_init() */
|
2009-01-04 21:18:03 +08:00
|
|
|
cpumask_clear_cpu(cpu, cpu_initialized_mask);
|
x86: cleanup early per cpu variables/accesses v4
* Introduce a new PER_CPU macro called "EARLY_PER_CPU". This is
used by some per_cpu variables that are initialized and accessed
before there are per_cpu areas allocated.
["Early" in respect to per_cpu variables is "earlier than the per_cpu
areas have been setup".]
This patchset adds these new macros:
DEFINE_EARLY_PER_CPU(_type, _name, _initvalue)
EXPORT_EARLY_PER_CPU_SYMBOL(_name)
DECLARE_EARLY_PER_CPU(_type, _name)
early_per_cpu_ptr(_name)
early_per_cpu_map(_name, _idx)
early_per_cpu(_name, _cpu)
The DEFINE macro defines the per_cpu variable as well as the early
map and pointer. It also initializes the per_cpu variable and map
elements to "_initvalue". The early_* macros provide access to
the initial map (usually setup during system init) and the early
pointer. This pointer is initialized to point to the early map
but is then NULL'ed when the actual per_cpu areas are setup. After
that the per_cpu variable is the correct access to the variable.
The early_per_cpu() macro is not very efficient but does show how to
access the variable if you have a function that can be called both
"early" and "late". It tests the early ptr to be NULL, and if not
then it's still valid. Otherwise, the per_cpu variable is used
instead:
#define early_per_cpu(_name, _cpu) \
(early_per_cpu_ptr(_name) ? \
early_per_cpu_ptr(_name)[_cpu] : \
per_cpu(_name, _cpu))
A better method is to actually check the pointer manually. In the
case below, numa_set_node can be called both "early" and "late":
void __cpuinit numa_set_node(int cpu, int node)
{
int *cpu_to_node_map = early_per_cpu_ptr(x86_cpu_to_node_map);
if (cpu_to_node_map)
cpu_to_node_map[cpu] = node;
else
per_cpu(x86_cpu_to_node_map, cpu) = node;
}
* Add a flag "arch_provides_topology_pointers" that indicates pointers
to topology cpumask_t maps are available. Otherwise, use the function
returning the cpumask_t value. This is useful if cpumask_t set size
is very large to avoid copying data on to/off of the stack.
* The coverage of CONFIG_DEBUG_PER_CPU_MAPS has been increased while
the non-debug case has been optimized a bit.
* Remove an unreferenced compiler warning in drivers/base/topology.c
* Clean up #ifdef in setup.c
For inclusion into sched-devel/latest tree.
Based on:
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6.git
+ sched-devel/latest .../mingo/linux-2.6-sched-devel.git
Signed-off-by: Mike Travis <travis@sgi.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
2008-05-13 03:21:12 +08:00
|
|
|
numa_remove_cpu(cpu);
|
2008-03-04 01:13:07 +08:00
|
|
|
}
|
|
|
|
|
2008-08-22 18:52:14 +08:00
|
|
|
void cpu_disable_common(void)
|
2008-03-04 01:13:07 +08:00
|
|
|
{
|
|
|
|
int cpu = smp_processor_id();
|
|
|
|
|
|
|
|
remove_siblinginfo(cpu);
|
|
|
|
|
|
|
|
/* It's now safe to remove this processor from the online map */
|
2008-08-10 06:09:02 +08:00
|
|
|
lock_vector_lock();
|
2008-03-04 01:13:07 +08:00
|
|
|
remove_cpu_from_maps(cpu);
|
2008-08-10 06:09:02 +08:00
|
|
|
unlock_vector_lock();
|
2008-12-17 09:33:58 +08:00
|
|
|
fixup_irqs();
|
2008-08-22 18:52:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int native_cpu_disable(void)
|
|
|
|
{
|
|
|
|
int cpu = smp_processor_id();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Perhaps use cpufreq to drop frequency, but that could go
|
|
|
|
* into generic code.
|
|
|
|
*
|
|
|
|
* We won't take down the boot processor on i386 due to some
|
|
|
|
* interrupts only being able to be serviced by the BSP.
|
|
|
|
* Especially so if we're not using an IOAPIC -zwane
|
|
|
|
*/
|
|
|
|
if (cpu == 0)
|
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
if (nmi_watchdog == NMI_LOCAL_APIC)
|
|
|
|
stop_apic_nmi_watchdog(NULL);
|
|
|
|
clear_local_APIC();
|
|
|
|
|
|
|
|
cpu_disable_common();
|
2008-03-04 01:13:07 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-08-22 18:52:11 +08:00
|
|
|
void native_cpu_die(unsigned int cpu)
|
2008-03-04 01:13:07 +08:00
|
|
|
{
|
|
|
|
/* We don't do anything here: idle task is faking death itself. */
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < 10; i++) {
|
|
|
|
/* They ack this in play_dead by setting CPU_DEAD */
|
|
|
|
if (per_cpu(cpu_state, cpu) == CPU_DEAD) {
|
2009-12-11 09:19:36 +08:00
|
|
|
if (system_state == SYSTEM_RUNNING)
|
|
|
|
pr_info("CPU %u is now offline\n", cpu);
|
|
|
|
|
2008-03-04 01:13:07 +08:00
|
|
|
if (1 == num_online_cpus())
|
|
|
|
alternatives_smp_switch(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
msleep(100);
|
|
|
|
}
|
2009-12-11 09:19:36 +08:00
|
|
|
pr_err("CPU %u didn't die...\n", cpu);
|
2008-03-04 01:13:07 +08:00
|
|
|
}
|
2008-08-22 18:52:13 +08:00
|
|
|
|
|
|
|
void play_dead_common(void)
|
|
|
|
{
|
|
|
|
idle_task_exit();
|
|
|
|
reset_lazy_tlbstate();
|
|
|
|
irq_ctx_exit(raw_smp_processor_id());
|
2008-09-24 05:26:42 +08:00
|
|
|
c1e_remove_cpu(raw_smp_processor_id());
|
2008-08-22 18:52:13 +08:00
|
|
|
|
|
|
|
mb();
|
|
|
|
/* Ack it */
|
|
|
|
__get_cpu_var(cpu_state) = CPU_DEAD;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* With physical CPU hotplug, we should halt the cpu
|
|
|
|
*/
|
|
|
|
local_irq_disable();
|
|
|
|
}
|
|
|
|
|
|
|
|
void native_play_dead(void)
|
|
|
|
{
|
|
|
|
play_dead_common();
|
2009-07-01 10:31:07 +08:00
|
|
|
tboot_shutdown(TB_SHUTDOWN_WFS);
|
2008-08-22 18:52:13 +08:00
|
|
|
wbinvd_halt();
|
|
|
|
}
|
|
|
|
|
2008-03-04 01:13:07 +08:00
|
|
|
#else /* ... !CONFIG_HOTPLUG_CPU */
|
2008-08-22 18:52:11 +08:00
|
|
|
int native_cpu_disable(void)
|
2008-03-04 01:13:07 +08:00
|
|
|
{
|
|
|
|
return -ENOSYS;
|
|
|
|
}
|
|
|
|
|
2008-08-22 18:52:11 +08:00
|
|
|
void native_cpu_die(unsigned int cpu)
|
2008-03-04 01:13:07 +08:00
|
|
|
{
|
|
|
|
/* We said "no" in __cpu_disable */
|
|
|
|
BUG();
|
|
|
|
}
|
2008-08-22 18:52:13 +08:00
|
|
|
|
|
|
|
void native_play_dead(void)
|
|
|
|
{
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
|
2008-03-04 01:12:42 +08:00
|
|
|
#endif
|