Merge branch 'x86-apic-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull x86 APIC updates from Thomas Gleixner: "This update provides a major overhaul of the APIC initialization and vector allocation code: - Unification of the APIC and interrupt mode setup which was scattered all over the place and was hard to follow. This also distangles the timer setup from the APIC initialization which brings a clear separation of functionality. Great detective work from Dou Lyiang! - Refactoring of the x86 vector allocation mechanism. The existing code was based on nested loops and rather convoluted APIC callbacks which had a horrible worst case behaviour and tried to serve all different use cases in one go. This led to quite odd hacks when supporting the new managed interupt facility for multiqueue devices and made it more or less impossible to deal with the vector space exhaustion which was a major roadblock for server hibernation. Aside of that the code dealing with cpu hotplug and the system vectors was disconnected from the actual vector management and allocation code, which made it hard to follow and maintain. Utilizing the new bitmap matrix allocator core mechanism, the new allocator and management code consolidates the handling of system vectors, legacy vectors, cpu hotplug mechanisms and the actual allocation which needs to be aware of system and legacy vectors and hotplug constraints into a single consistent entity. This has one visible change: The support for multi CPU targets of interrupts, which is only available on a certain subset of CPUs/APIC variants has been removed in favour of single interrupt targets. A proper analysis of the multi CPU target feature revealed that there is no real advantage as the vast majority of interrupts end up on the CPU with the lowest APIC id in the set of target CPUs anyway. That change was agreed on by the relevant folks and allowed to simplify the implementation significantly and to replace rather fragile constructs like the vector cleanup IPI with straight forward and solid code. Furthermore this allowed to cleanly separate the allocation details for legacy, normal and managed interrupts: * Legacy interrupts are not longer wasting 16 vectors unconditionally * Managed interrupts have now a guaranteed vector reservation, but the actual vector assignment happens when the interrupt is requested. It's guaranteed not to fail. * Normal interrupts no longer allocate vectors unconditionally when the interrupt is set up (IO/APIC init or MSI(X) enable). The mechanism has been switched to a best effort reservation mode. The actual allocation happens when the interrupt is requested. Contrary to managed interrupts the request can fail due to vector space exhaustion, but drivers must handle a fail of request_irq() anyway. When the interrupt is freed, the vector is handed back as well. This solves a long standing problem with large unconditional vector allocations for a certain class of enterprise devices which prevented server hibernation due to vector space exhaustion when the unused allocated vectors had to be migrated to CPU0 while unplugging all non boot CPUs. The code has been equipped with trace points and detailed debugfs information to aid analysis of the vector space" * 'x86-apic-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: (60 commits) x86/vector/msi: Select CONFIG_GENERIC_IRQ_RESERVATION_MODE PCI/MSI: Set MSI_FLAG_MUST_REACTIVATE in core code genirq: Add config option for reservation mode x86/vector: Use correct per cpu variable in free_moved_vector() x86/apic/vector: Ignore set_affinity call for inactive interrupts x86/apic: Fix spelling mistake: "symmectic" -> "symmetric" x86/apic: Use dead_cpu instead of current CPU when cleaning up ACPI/init: Invoke early ACPI initialization earlier x86/vector: Respect affinity mask in irq descriptor x86/irq: Simplify hotplug vector accounting x86/vector: Switch IOAPIC to global reservation mode x86/vector/msi: Switch to global reservation mode x86/vector: Handle managed interrupts proper x86/io_apic: Reevaluate vector configuration on activate() iommu/amd: Reevaluate vector configuration on activate() iommu/vt-d: Reevaluate vector configuration on activate() x86/apic/msi: Force reactivation of interrupts at startup time x86/vector: Untangle internal state from irq_cfg x86/vector: Compile SMP only code conditionally x86/apic: Remove unused callbacks ...
This commit is contained in:
@@ -7,7 +7,7 @@
|
||||
# In particualr, smp_apic_timer_interrupt() is called in random places.
|
||||
KCOV_INSTRUMENT := n
|
||||
|
||||
obj-$(CONFIG_X86_LOCAL_APIC) += apic.o apic_noop.o ipi.o vector.o
|
||||
obj-$(CONFIG_X86_LOCAL_APIC) += apic.o apic_common.o apic_noop.o ipi.o vector.o
|
||||
obj-y += hw_nmi.o
|
||||
|
||||
obj-$(CONFIG_X86_IO_APIC) += io_apic.o
|
||||
|
@@ -211,11 +211,7 @@ static inline int lapic_get_version(void)
|
||||
*/
|
||||
static inline int lapic_is_integrated(void)
|
||||
{
|
||||
#ifdef CONFIG_X86_64
|
||||
return 1;
|
||||
#else
|
||||
return APIC_INTEGRATED(lapic_get_version());
|
||||
#endif
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -298,14 +294,11 @@ int get_physical_broadcast(void)
|
||||
*/
|
||||
int lapic_get_maxlvt(void)
|
||||
{
|
||||
unsigned int v;
|
||||
|
||||
v = apic_read(APIC_LVR);
|
||||
/*
|
||||
* - we always have APIC integrated on 64bit mode
|
||||
* - 82489DXs do not report # of LVT entries
|
||||
*/
|
||||
return APIC_INTEGRATED(GET_APIC_VERSION(v)) ? GET_APIC_MAXLVT(v) : 2;
|
||||
return lapic_is_integrated() ? GET_APIC_MAXLVT(apic_read(APIC_LVR)) : 2;
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -1229,53 +1222,100 @@ void __init sync_Arb_IDs(void)
|
||||
APIC_INT_LEVELTRIG | APIC_DM_INIT);
|
||||
}
|
||||
|
||||
/*
|
||||
* An initial setup of the virtual wire mode.
|
||||
*/
|
||||
void __init init_bsp_APIC(void)
|
||||
enum apic_intr_mode_id apic_intr_mode;
|
||||
|
||||
static int __init apic_intr_mode_select(void)
|
||||
{
|
||||
unsigned int value;
|
||||
/* Check kernel option */
|
||||
if (disable_apic) {
|
||||
pr_info("APIC disabled via kernel command line\n");
|
||||
return APIC_PIC;
|
||||
}
|
||||
|
||||
/*
|
||||
* Don't do the setup now if we have a SMP BIOS as the
|
||||
* through-I/O-APIC virtual wire mode might be active.
|
||||
*/
|
||||
if (smp_found_config || !boot_cpu_has(X86_FEATURE_APIC))
|
||||
return;
|
||||
/* Check BIOS */
|
||||
#ifdef CONFIG_X86_64
|
||||
/* On 64-bit, the APIC must be integrated, Check local APIC only */
|
||||
if (!boot_cpu_has(X86_FEATURE_APIC)) {
|
||||
disable_apic = 1;
|
||||
pr_info("APIC disabled by BIOS\n");
|
||||
return APIC_PIC;
|
||||
}
|
||||
#else
|
||||
/* On 32-bit, the APIC may be integrated APIC or 82489DX */
|
||||
|
||||
/*
|
||||
* Do not trust the local APIC being empty at bootup.
|
||||
*/
|
||||
clear_local_APIC();
|
||||
/* Neither 82489DX nor integrated APIC ? */
|
||||
if (!boot_cpu_has(X86_FEATURE_APIC) && !smp_found_config) {
|
||||
disable_apic = 1;
|
||||
return APIC_PIC;
|
||||
}
|
||||
|
||||
/*
|
||||
* Enable APIC.
|
||||
*/
|
||||
value = apic_read(APIC_SPIV);
|
||||
value &= ~APIC_VECTOR_MASK;
|
||||
value |= APIC_SPIV_APIC_ENABLED;
|
||||
|
||||
#ifdef CONFIG_X86_32
|
||||
/* This bit is reserved on P4/Xeon and should be cleared */
|
||||
if ((boot_cpu_data.x86_vendor == X86_VENDOR_INTEL) &&
|
||||
(boot_cpu_data.x86 == 15))
|
||||
value &= ~APIC_SPIV_FOCUS_DISABLED;
|
||||
else
|
||||
/* If the BIOS pretends there is an integrated APIC ? */
|
||||
if (!boot_cpu_has(X86_FEATURE_APIC) &&
|
||||
APIC_INTEGRATED(boot_cpu_apic_version)) {
|
||||
disable_apic = 1;
|
||||
pr_err(FW_BUG "Local APIC %d not detected, force emulation\n",
|
||||
boot_cpu_physical_apicid);
|
||||
return APIC_PIC;
|
||||
}
|
||||
#endif
|
||||
value |= APIC_SPIV_FOCUS_DISABLED;
|
||||
value |= SPURIOUS_APIC_VECTOR;
|
||||
apic_write(APIC_SPIV, value);
|
||||
|
||||
/*
|
||||
* Set up the virtual wire mode.
|
||||
*/
|
||||
apic_write(APIC_LVT0, APIC_DM_EXTINT);
|
||||
value = APIC_DM_NMI;
|
||||
if (!lapic_is_integrated()) /* 82489DX */
|
||||
value |= APIC_LVT_LEVEL_TRIGGER;
|
||||
if (apic_extnmi == APIC_EXTNMI_NONE)
|
||||
value |= APIC_LVT_MASKED;
|
||||
apic_write(APIC_LVT1, value);
|
||||
/* Check MP table or ACPI MADT configuration */
|
||||
if (!smp_found_config) {
|
||||
disable_ioapic_support();
|
||||
if (!acpi_lapic) {
|
||||
pr_info("APIC: ACPI MADT or MP tables are not detected\n");
|
||||
return APIC_VIRTUAL_WIRE_NO_CONFIG;
|
||||
}
|
||||
return APIC_VIRTUAL_WIRE;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_SMP
|
||||
/* If SMP should be disabled, then really disable it! */
|
||||
if (!setup_max_cpus) {
|
||||
pr_info("APIC: SMP mode deactivated\n");
|
||||
return APIC_SYMMETRIC_IO_NO_ROUTING;
|
||||
}
|
||||
|
||||
if (read_apic_id() != boot_cpu_physical_apicid) {
|
||||
panic("Boot APIC ID in local APIC unexpected (%d vs %d)",
|
||||
read_apic_id(), boot_cpu_physical_apicid);
|
||||
/* Or can we switch back to PIC here? */
|
||||
}
|
||||
#endif
|
||||
|
||||
return APIC_SYMMETRIC_IO;
|
||||
}
|
||||
|
||||
/* Init the interrupt delivery mode for the BSP */
|
||||
void __init apic_intr_mode_init(void)
|
||||
{
|
||||
bool upmode = IS_ENABLED(CONFIG_UP_LATE_INIT);
|
||||
|
||||
apic_intr_mode = apic_intr_mode_select();
|
||||
|
||||
switch (apic_intr_mode) {
|
||||
case APIC_PIC:
|
||||
pr_info("APIC: Keep in PIC mode(8259)\n");
|
||||
return;
|
||||
case APIC_VIRTUAL_WIRE:
|
||||
pr_info("APIC: Switch to virtual wire mode setup\n");
|
||||
default_setup_apic_routing();
|
||||
break;
|
||||
case APIC_VIRTUAL_WIRE_NO_CONFIG:
|
||||
pr_info("APIC: Switch to virtual wire mode setup with no configuration\n");
|
||||
upmode = true;
|
||||
default_setup_apic_routing();
|
||||
break;
|
||||
case APIC_SYMMETRIC_IO:
|
||||
pr_info("APIC: Switch to symmetric I/O mode setup\n");
|
||||
default_setup_apic_routing();
|
||||
break;
|
||||
case APIC_SYMMETRIC_IO_NO_ROUTING:
|
||||
pr_info("APIC: Switch to symmetric I/O mode setup in no SMP routine\n");
|
||||
break;
|
||||
}
|
||||
|
||||
apic_bsp_setup(upmode);
|
||||
}
|
||||
|
||||
static void lapic_setup_esr(void)
|
||||
@@ -1499,7 +1539,9 @@ void setup_local_APIC(void)
|
||||
value = APIC_DM_NMI;
|
||||
else
|
||||
value = APIC_DM_NMI | APIC_LVT_MASKED;
|
||||
if (!lapic_is_integrated()) /* 82489DX */
|
||||
|
||||
/* Is 82489DX ? */
|
||||
if (!lapic_is_integrated())
|
||||
value |= APIC_LVT_LEVEL_TRIGGER;
|
||||
apic_write(APIC_LVT1, value);
|
||||
|
||||
@@ -1885,8 +1927,8 @@ void __init init_apic_mappings(void)
|
||||
* yeah -- we lie about apic_version
|
||||
* in case if apic was disabled via boot option
|
||||
* but it's not a problem for SMP compiled kernel
|
||||
* since smp_sanity_check is prepared for such a case
|
||||
* and disable smp mode
|
||||
* since apic_intr_mode_select is prepared for such
|
||||
* a case and disable smp mode
|
||||
*/
|
||||
boot_cpu_apic_version = GET_APIC_VERSION(apic_read(APIC_LVR));
|
||||
}
|
||||
@@ -2242,44 +2284,6 @@ int hard_smp_processor_id(void)
|
||||
return read_apic_id();
|
||||
}
|
||||
|
||||
void default_init_apic_ldr(void)
|
||||
{
|
||||
unsigned long val;
|
||||
|
||||
apic_write(APIC_DFR, APIC_DFR_VALUE);
|
||||
val = apic_read(APIC_LDR) & ~APIC_LDR_MASK;
|
||||
val |= SET_APIC_LOGICAL_ID(1UL << smp_processor_id());
|
||||
apic_write(APIC_LDR, val);
|
||||
}
|
||||
|
||||
int default_cpu_mask_to_apicid(const struct cpumask *mask,
|
||||
struct irq_data *irqdata,
|
||||
unsigned int *apicid)
|
||||
{
|
||||
unsigned int cpu = cpumask_first(mask);
|
||||
|
||||
if (cpu >= nr_cpu_ids)
|
||||
return -EINVAL;
|
||||
*apicid = per_cpu(x86_cpu_to_apicid, cpu);
|
||||
irq_data_update_effective_affinity(irqdata, cpumask_of(cpu));
|
||||
return 0;
|
||||
}
|
||||
|
||||
int flat_cpu_mask_to_apicid(const struct cpumask *mask,
|
||||
struct irq_data *irqdata,
|
||||
unsigned int *apicid)
|
||||
|
||||
{
|
||||
struct cpumask *effmsk = irq_data_get_effective_affinity_mask(irqdata);
|
||||
unsigned long cpu_mask = cpumask_bits(mask)[0] & APIC_ALL_CPUS;
|
||||
|
||||
if (!cpu_mask)
|
||||
return -EINVAL;
|
||||
*apicid = (unsigned int)cpu_mask;
|
||||
cpumask_bits(effmsk)[0] = cpu_mask;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Override the generic EOI implementation with an optimized version.
|
||||
* Only called during early boot when only one CPU is active and with
|
||||
@@ -2322,72 +2326,27 @@ static void __init apic_bsp_up_setup(void)
|
||||
* Returns:
|
||||
* apic_id of BSP APIC
|
||||
*/
|
||||
int __init apic_bsp_setup(bool upmode)
|
||||
void __init apic_bsp_setup(bool upmode)
|
||||
{
|
||||
int id;
|
||||
|
||||
connect_bsp_APIC();
|
||||
if (upmode)
|
||||
apic_bsp_up_setup();
|
||||
setup_local_APIC();
|
||||
|
||||
if (x2apic_mode)
|
||||
id = apic_read(APIC_LDR);
|
||||
else
|
||||
id = GET_APIC_LOGICAL_ID(apic_read(APIC_LDR));
|
||||
|
||||
enable_IO_APIC();
|
||||
end_local_APIC_setup();
|
||||
irq_remap_enable_fault_handling();
|
||||
setup_IO_APIC();
|
||||
/* Setup local timer */
|
||||
x86_init.timers.setup_percpu_clockev();
|
||||
return id;
|
||||
}
|
||||
|
||||
/*
|
||||
* This initializes the IO-APIC and APIC hardware if this is
|
||||
* a UP kernel.
|
||||
*/
|
||||
int __init APIC_init_uniprocessor(void)
|
||||
{
|
||||
if (disable_apic) {
|
||||
pr_info("Apic disabled\n");
|
||||
return -1;
|
||||
}
|
||||
#ifdef CONFIG_X86_64
|
||||
if (!boot_cpu_has(X86_FEATURE_APIC)) {
|
||||
disable_apic = 1;
|
||||
pr_info("Apic disabled by BIOS\n");
|
||||
return -1;
|
||||
}
|
||||
#else
|
||||
if (!smp_found_config && !boot_cpu_has(X86_FEATURE_APIC))
|
||||
return -1;
|
||||
|
||||
/*
|
||||
* Complain if the BIOS pretends there is one.
|
||||
*/
|
||||
if (!boot_cpu_has(X86_FEATURE_APIC) &&
|
||||
APIC_INTEGRATED(boot_cpu_apic_version)) {
|
||||
pr_err("BIOS bug, local APIC 0x%x not detected!...\n",
|
||||
boot_cpu_physical_apicid);
|
||||
return -1;
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!smp_found_config)
|
||||
disable_ioapic_support();
|
||||
|
||||
default_setup_apic_routing();
|
||||
apic_bsp_setup(true);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_UP_LATE_INIT
|
||||
void __init up_late_init(void)
|
||||
{
|
||||
APIC_init_uniprocessor();
|
||||
if (apic_intr_mode == APIC_PIC)
|
||||
return;
|
||||
|
||||
/* Setup local timer */
|
||||
x86_init.timers.setup_percpu_clockev();
|
||||
}
|
||||
#endif
|
||||
|
||||
|
46
arch/x86/kernel/apic/apic_common.c
Normal file
46
arch/x86/kernel/apic/apic_common.c
Normal file
@@ -0,0 +1,46 @@
|
||||
/*
|
||||
* Common functions shared between the various APIC flavours
|
||||
*
|
||||
* SPDX-License-Identifier: GPL-2.0
|
||||
*/
|
||||
#include <linux/irq.h>
|
||||
#include <asm/apic.h>
|
||||
|
||||
u32 apic_default_calc_apicid(unsigned int cpu)
|
||||
{
|
||||
return per_cpu(x86_cpu_to_apicid, cpu);
|
||||
}
|
||||
|
||||
u32 apic_flat_calc_apicid(unsigned int cpu)
|
||||
{
|
||||
return 1U << cpu;
|
||||
}
|
||||
|
||||
bool default_check_apicid_used(physid_mask_t *map, int apicid)
|
||||
{
|
||||
return physid_isset(apicid, *map);
|
||||
}
|
||||
|
||||
void default_ioapic_phys_id_map(physid_mask_t *phys_map, physid_mask_t *retmap)
|
||||
{
|
||||
*retmap = *phys_map;
|
||||
}
|
||||
|
||||
int default_cpu_present_to_apicid(int mps_cpu)
|
||||
{
|
||||
if (mps_cpu < nr_cpu_ids && cpu_present(mps_cpu))
|
||||
return (int)per_cpu(x86_bios_cpu_apicid, mps_cpu);
|
||||
else
|
||||
return BAD_APICID;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(default_cpu_present_to_apicid);
|
||||
|
||||
int default_check_phys_apicid_present(int phys_apicid)
|
||||
{
|
||||
return physid_isset(phys_apicid, phys_cpu_present_map);
|
||||
}
|
||||
|
||||
int default_apic_id_valid(int apicid)
|
||||
{
|
||||
return (apicid < 255);
|
||||
}
|
@@ -119,7 +119,7 @@ static unsigned int flat_get_apic_id(unsigned long x)
|
||||
return (x >> 24) & 0xFF;
|
||||
}
|
||||
|
||||
static unsigned long set_apic_id(unsigned int id)
|
||||
static u32 set_apic_id(unsigned int id)
|
||||
{
|
||||
return (id & 0xFF) << 24;
|
||||
}
|
||||
@@ -154,12 +154,10 @@ static struct apic apic_flat __ro_after_init = {
|
||||
.irq_delivery_mode = dest_LowestPrio,
|
||||
.irq_dest_mode = 1, /* logical */
|
||||
|
||||
.target_cpus = online_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = APIC_DEST_LOGICAL,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = flat_vector_allocation_domain,
|
||||
.init_apic_ldr = flat_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = NULL,
|
||||
@@ -172,7 +170,7 @@ static struct apic apic_flat __ro_after_init = {
|
||||
.get_apic_id = flat_get_apic_id,
|
||||
.set_apic_id = set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = flat_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_flat_calc_apicid,
|
||||
|
||||
.send_IPI = default_send_IPI_single,
|
||||
.send_IPI_mask = flat_send_IPI_mask,
|
||||
@@ -249,12 +247,10 @@ static struct apic apic_physflat __ro_after_init = {
|
||||
.irq_delivery_mode = dest_Fixed,
|
||||
.irq_dest_mode = 0, /* physical */
|
||||
|
||||
.target_cpus = online_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = 0,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = default_vector_allocation_domain,
|
||||
/* not needed, but shouldn't hurt: */
|
||||
.init_apic_ldr = flat_init_apic_ldr,
|
||||
|
||||
@@ -268,7 +264,7 @@ static struct apic apic_physflat __ro_after_init = {
|
||||
.get_apic_id = flat_get_apic_id,
|
||||
.set_apic_id = set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = default_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_default_calc_apicid,
|
||||
|
||||
.send_IPI = default_send_IPI_single_phys,
|
||||
.send_IPI_mask = default_send_IPI_mask_sequence_phys,
|
||||
|
@@ -84,20 +84,6 @@ static int noop_apic_id_registered(void)
|
||||
return physid_isset(0, phys_cpu_present_map);
|
||||
}
|
||||
|
||||
static const struct cpumask *noop_target_cpus(void)
|
||||
{
|
||||
/* only BSP here */
|
||||
return cpumask_of(0);
|
||||
}
|
||||
|
||||
static void noop_vector_allocation_domain(int cpu, struct cpumask *retmask,
|
||||
const struct cpumask *mask)
|
||||
{
|
||||
if (cpu != 0)
|
||||
pr_warning("APIC: Vector allocated for non-BSP cpu\n");
|
||||
cpumask_copy(retmask, cpumask_of(cpu));
|
||||
}
|
||||
|
||||
static u32 noop_apic_read(u32 reg)
|
||||
{
|
||||
WARN_ON_ONCE(boot_cpu_has(X86_FEATURE_APIC) && !disable_apic);
|
||||
@@ -109,6 +95,13 @@ static void noop_apic_write(u32 reg, u32 v)
|
||||
WARN_ON_ONCE(boot_cpu_has(X86_FEATURE_APIC) && !disable_apic);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_X86_32
|
||||
static int noop_x86_32_early_logical_apicid(int cpu)
|
||||
{
|
||||
return BAD_APICID;
|
||||
}
|
||||
#endif
|
||||
|
||||
struct apic apic_noop __ro_after_init = {
|
||||
.name = "noop",
|
||||
.probe = noop_probe,
|
||||
@@ -121,12 +114,10 @@ struct apic apic_noop __ro_after_init = {
|
||||
/* logical delivery broadcast to all CPUs: */
|
||||
.irq_dest_mode = 1,
|
||||
|
||||
.target_cpus = noop_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = APIC_DEST_LOGICAL,
|
||||
.check_apicid_used = default_check_apicid_used,
|
||||
|
||||
.vector_allocation_domain = noop_vector_allocation_domain,
|
||||
.init_apic_ldr = noop_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = default_ioapic_phys_id_map,
|
||||
@@ -142,7 +133,7 @@ struct apic apic_noop __ro_after_init = {
|
||||
.get_apic_id = noop_get_apic_id,
|
||||
.set_apic_id = NULL,
|
||||
|
||||
.cpu_mask_to_apicid = flat_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_flat_calc_apicid,
|
||||
|
||||
.send_IPI = noop_send_IPI,
|
||||
.send_IPI_mask = noop_send_IPI_mask,
|
||||
|
@@ -38,7 +38,7 @@ static unsigned int numachip1_get_apic_id(unsigned long x)
|
||||
return id;
|
||||
}
|
||||
|
||||
static unsigned long numachip1_set_apic_id(unsigned int id)
|
||||
static u32 numachip1_set_apic_id(unsigned int id)
|
||||
{
|
||||
return (id & 0xff) << 24;
|
||||
}
|
||||
@@ -51,7 +51,7 @@ static unsigned int numachip2_get_apic_id(unsigned long x)
|
||||
return ((mcfg >> (28 - 8)) & 0xfff00) | (x >> 24);
|
||||
}
|
||||
|
||||
static unsigned long numachip2_set_apic_id(unsigned int id)
|
||||
static u32 numachip2_set_apic_id(unsigned int id)
|
||||
{
|
||||
return id << 24;
|
||||
}
|
||||
@@ -249,12 +249,10 @@ static const struct apic apic_numachip1 __refconst = {
|
||||
.irq_delivery_mode = dest_Fixed,
|
||||
.irq_dest_mode = 0, /* physical */
|
||||
|
||||
.target_cpus = online_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = 0,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = default_vector_allocation_domain,
|
||||
.init_apic_ldr = flat_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = NULL,
|
||||
@@ -267,7 +265,7 @@ static const struct apic apic_numachip1 __refconst = {
|
||||
.get_apic_id = numachip1_get_apic_id,
|
||||
.set_apic_id = numachip1_set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = default_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_default_calc_apicid,
|
||||
|
||||
.send_IPI = numachip_send_IPI_one,
|
||||
.send_IPI_mask = numachip_send_IPI_mask,
|
||||
@@ -300,12 +298,10 @@ static const struct apic apic_numachip2 __refconst = {
|
||||
.irq_delivery_mode = dest_Fixed,
|
||||
.irq_dest_mode = 0, /* physical */
|
||||
|
||||
.target_cpus = online_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = 0,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = default_vector_allocation_domain,
|
||||
.init_apic_ldr = flat_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = NULL,
|
||||
@@ -318,7 +314,7 @@ static const struct apic apic_numachip2 __refconst = {
|
||||
.get_apic_id = numachip2_get_apic_id,
|
||||
.set_apic_id = numachip2_set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = default_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_default_calc_apicid,
|
||||
|
||||
.send_IPI = numachip_send_IPI_one,
|
||||
.send_IPI_mask = numachip_send_IPI_mask,
|
||||
|
@@ -27,9 +27,9 @@ static int bigsmp_apic_id_registered(void)
|
||||
return 1;
|
||||
}
|
||||
|
||||
static unsigned long bigsmp_check_apicid_used(physid_mask_t *map, int apicid)
|
||||
static bool bigsmp_check_apicid_used(physid_mask_t *map, int apicid)
|
||||
{
|
||||
return 0;
|
||||
return false;
|
||||
}
|
||||
|
||||
static int bigsmp_early_logical_apicid(int cpu)
|
||||
@@ -155,12 +155,10 @@ static struct apic apic_bigsmp __ro_after_init = {
|
||||
/* phys delivery to target CPU: */
|
||||
.irq_dest_mode = 0,
|
||||
|
||||
.target_cpus = default_target_cpus,
|
||||
.disable_esr = 1,
|
||||
.dest_logical = 0,
|
||||
.check_apicid_used = bigsmp_check_apicid_used,
|
||||
|
||||
.vector_allocation_domain = default_vector_allocation_domain,
|
||||
.init_apic_ldr = bigsmp_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = bigsmp_ioapic_phys_id_map,
|
||||
@@ -173,7 +171,7 @@ static struct apic apic_bigsmp __ro_after_init = {
|
||||
.get_apic_id = bigsmp_get_apic_id,
|
||||
.set_apic_id = NULL,
|
||||
|
||||
.cpu_mask_to_apicid = default_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_default_calc_apicid,
|
||||
|
||||
.send_IPI = default_send_IPI_single_phys,
|
||||
.send_IPI_mask = default_send_IPI_mask_sequence_phys,
|
||||
|
@@ -1014,6 +1014,7 @@ static int alloc_isa_irq_from_domain(struct irq_domain *domain,
|
||||
info->ioapic_pin))
|
||||
return -ENOMEM;
|
||||
} else {
|
||||
info->flags |= X86_IRQ_ALLOC_LEGACY;
|
||||
irq = __irq_domain_alloc_irqs(domain, irq, 1, node, info, true,
|
||||
NULL);
|
||||
if (irq >= 0) {
|
||||
@@ -1586,6 +1587,43 @@ static int __init notimercheck(char *s)
|
||||
}
|
||||
__setup("no_timer_check", notimercheck);
|
||||
|
||||
static void __init delay_with_tsc(void)
|
||||
{
|
||||
unsigned long long start, now;
|
||||
unsigned long end = jiffies + 4;
|
||||
|
||||
start = rdtsc();
|
||||
|
||||
/*
|
||||
* We don't know the TSC frequency yet, but waiting for
|
||||
* 40000000000/HZ TSC cycles is safe:
|
||||
* 4 GHz == 10 jiffies
|
||||
* 1 GHz == 40 jiffies
|
||||
*/
|
||||
do {
|
||||
rep_nop();
|
||||
now = rdtsc();
|
||||
} while ((now - start) < 40000000000UL / HZ &&
|
||||
time_before_eq(jiffies, end));
|
||||
}
|
||||
|
||||
static void __init delay_without_tsc(void)
|
||||
{
|
||||
unsigned long end = jiffies + 4;
|
||||
int band = 1;
|
||||
|
||||
/*
|
||||
* We don't know any frequency yet, but waiting for
|
||||
* 40940000000/HZ cycles is safe:
|
||||
* 4 GHz == 10 jiffies
|
||||
* 1 GHz == 40 jiffies
|
||||
* 1 << 1 + 1 << 2 +...+ 1 << 11 = 4094
|
||||
*/
|
||||
do {
|
||||
__delay(((1U << band++) * 10000000UL) / HZ);
|
||||
} while (band < 12 && time_before_eq(jiffies, end));
|
||||
}
|
||||
|
||||
/*
|
||||
* There is a nasty bug in some older SMP boards, their mptable lies
|
||||
* about the timer IRQ. We do the following to work around the situation:
|
||||
@@ -1604,8 +1642,12 @@ static int __init timer_irq_works(void)
|
||||
|
||||
local_save_flags(flags);
|
||||
local_irq_enable();
|
||||
/* Let ten ticks pass... */
|
||||
mdelay((10 * 1000) / HZ);
|
||||
|
||||
if (boot_cpu_has(X86_FEATURE_TSC))
|
||||
delay_with_tsc();
|
||||
else
|
||||
delay_without_tsc();
|
||||
|
||||
local_irq_restore(flags);
|
||||
|
||||
/*
|
||||
@@ -1821,26 +1863,36 @@ static void ioapic_ir_ack_level(struct irq_data *irq_data)
|
||||
eoi_ioapic_pin(data->entry.vector, data);
|
||||
}
|
||||
|
||||
static void ioapic_configure_entry(struct irq_data *irqd)
|
||||
{
|
||||
struct mp_chip_data *mpd = irqd->chip_data;
|
||||
struct irq_cfg *cfg = irqd_cfg(irqd);
|
||||
struct irq_pin_list *entry;
|
||||
|
||||
/*
|
||||
* Only update when the parent is the vector domain, don't touch it
|
||||
* if the parent is the remapping domain. Check the installed
|
||||
* ioapic chip to verify that.
|
||||
*/
|
||||
if (irqd->chip == &ioapic_chip) {
|
||||
mpd->entry.dest = cfg->dest_apicid;
|
||||
mpd->entry.vector = cfg->vector;
|
||||
}
|
||||
for_each_irq_pin(entry, mpd->irq_2_pin)
|
||||
__ioapic_write_entry(entry->apic, entry->pin, mpd->entry);
|
||||
}
|
||||
|
||||
static int ioapic_set_affinity(struct irq_data *irq_data,
|
||||
const struct cpumask *mask, bool force)
|
||||
{
|
||||
struct irq_data *parent = irq_data->parent_data;
|
||||
struct mp_chip_data *data = irq_data->chip_data;
|
||||
struct irq_pin_list *entry;
|
||||
struct irq_cfg *cfg;
|
||||
unsigned long flags;
|
||||
int ret;
|
||||
|
||||
ret = parent->chip->irq_set_affinity(parent, mask, force);
|
||||
raw_spin_lock_irqsave(&ioapic_lock, flags);
|
||||
if (ret >= 0 && ret != IRQ_SET_MASK_OK_DONE) {
|
||||
cfg = irqd_cfg(irq_data);
|
||||
data->entry.dest = cfg->dest_apicid;
|
||||
data->entry.vector = cfg->vector;
|
||||
for_each_irq_pin(entry, data->irq_2_pin)
|
||||
__ioapic_write_entry(entry->apic, entry->pin,
|
||||
data->entry);
|
||||
}
|
||||
if (ret >= 0 && ret != IRQ_SET_MASK_OK_DONE)
|
||||
ioapic_configure_entry(irq_data);
|
||||
raw_spin_unlock_irqrestore(&ioapic_lock, flags);
|
||||
|
||||
return ret;
|
||||
@@ -2513,52 +2565,9 @@ int acpi_get_override_irq(u32 gsi, int *trigger, int *polarity)
|
||||
}
|
||||
|
||||
/*
|
||||
* This function currently is only a helper for the i386 smp boot process where
|
||||
* we need to reprogram the ioredtbls to cater for the cpus which have come online
|
||||
* so mask in all cases should simply be apic->target_cpus()
|
||||
* This function updates target affinity of IOAPIC interrupts to include
|
||||
* the CPUs which came online during SMP bringup.
|
||||
*/
|
||||
#ifdef CONFIG_SMP
|
||||
void __init setup_ioapic_dest(void)
|
||||
{
|
||||
int pin, ioapic, irq, irq_entry;
|
||||
const struct cpumask *mask;
|
||||
struct irq_desc *desc;
|
||||
struct irq_data *idata;
|
||||
struct irq_chip *chip;
|
||||
|
||||
if (skip_ioapic_setup == 1)
|
||||
return;
|
||||
|
||||
for_each_ioapic_pin(ioapic, pin) {
|
||||
irq_entry = find_irq_entry(ioapic, pin, mp_INT);
|
||||
if (irq_entry == -1)
|
||||
continue;
|
||||
|
||||
irq = pin_2_irq(irq_entry, ioapic, pin, 0);
|
||||
if (irq < 0 || !mp_init_irq_at_boot(ioapic, irq))
|
||||
continue;
|
||||
|
||||
desc = irq_to_desc(irq);
|
||||
raw_spin_lock_irq(&desc->lock);
|
||||
idata = irq_desc_get_irq_data(desc);
|
||||
|
||||
/*
|
||||
* Honour affinities which have been set in early boot
|
||||
*/
|
||||
if (!irqd_can_balance(idata) || irqd_affinity_was_set(idata))
|
||||
mask = irq_data_get_affinity_mask(idata);
|
||||
else
|
||||
mask = apic->target_cpus();
|
||||
|
||||
chip = irq_data_get_irq_chip(idata);
|
||||
/* Might be lapic_chip for irq 0 */
|
||||
if (chip->irq_set_affinity)
|
||||
chip->irq_set_affinity(idata, mask, false);
|
||||
raw_spin_unlock_irq(&desc->lock);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#define IOAPIC_RESOURCE_NAME_SIZE 11
|
||||
|
||||
static struct resource *ioapic_resources;
|
||||
@@ -2982,12 +2991,9 @@ int mp_irqdomain_activate(struct irq_domain *domain,
|
||||
struct irq_data *irq_data, bool early)
|
||||
{
|
||||
unsigned long flags;
|
||||
struct irq_pin_list *entry;
|
||||
struct mp_chip_data *data = irq_data->chip_data;
|
||||
|
||||
raw_spin_lock_irqsave(&ioapic_lock, flags);
|
||||
for_each_irq_pin(entry, data->irq_2_pin)
|
||||
__ioapic_write_entry(entry->apic, entry->pin, data->entry);
|
||||
ioapic_configure_entry(irq_data);
|
||||
raw_spin_unlock_irqrestore(&ioapic_lock, flags);
|
||||
return 0;
|
||||
}
|
||||
|
@@ -66,6 +66,31 @@ static void setup_apic_flat_routing(void)
|
||||
#endif
|
||||
}
|
||||
|
||||
static int default_apic_id_registered(void)
|
||||
{
|
||||
return physid_isset(read_apic_id(), phys_cpu_present_map);
|
||||
}
|
||||
|
||||
/*
|
||||
* Set up the logical destination ID. Intel recommends to set DFR, LDR and
|
||||
* TPR before enabling an APIC. See e.g. "AP-388 82489DX User's Manual"
|
||||
* (Intel document number 292116).
|
||||
*/
|
||||
static void default_init_apic_ldr(void)
|
||||
{
|
||||
unsigned long val;
|
||||
|
||||
apic_write(APIC_DFR, APIC_DFR_VALUE);
|
||||
val = apic_read(APIC_LDR) & ~APIC_LDR_MASK;
|
||||
val |= SET_APIC_LOGICAL_ID(1UL << smp_processor_id());
|
||||
apic_write(APIC_LDR, val);
|
||||
}
|
||||
|
||||
static int default_phys_pkg_id(int cpuid_apic, int index_msb)
|
||||
{
|
||||
return cpuid_apic >> index_msb;
|
||||
}
|
||||
|
||||
/* should be called last. */
|
||||
static int probe_default(void)
|
||||
{
|
||||
@@ -84,12 +109,10 @@ static struct apic apic_default __ro_after_init = {
|
||||
/* logical delivery broadcast to all CPUs: */
|
||||
.irq_dest_mode = 1,
|
||||
|
||||
.target_cpus = default_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = APIC_DEST_LOGICAL,
|
||||
.check_apicid_used = default_check_apicid_used,
|
||||
|
||||
.vector_allocation_domain = flat_vector_allocation_domain,
|
||||
.init_apic_ldr = default_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = default_ioapic_phys_id_map,
|
||||
@@ -102,7 +125,7 @@ static struct apic apic_default __ro_after_init = {
|
||||
.get_apic_id = default_get_apic_id,
|
||||
.set_apic_id = NULL,
|
||||
|
||||
.cpu_mask_to_apicid = flat_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_flat_calc_apicid,
|
||||
|
||||
.send_IPI = default_send_IPI_single,
|
||||
.send_IPI_mask = default_send_IPI_mask_logical,
|
||||
|
File diff suppressed because it is too large
Load Diff
9
arch/x86/kernel/apic/x2apic.h
Normal file
9
arch/x86/kernel/apic/x2apic.h
Normal file
@@ -0,0 +1,9 @@
|
||||
/* Common bits for X2APIC cluster/physical modes. */
|
||||
|
||||
int x2apic_apic_id_valid(int apicid);
|
||||
int x2apic_apic_id_registered(void);
|
||||
void __x2apic_send_IPI_dest(unsigned int apicid, int vector, unsigned int dest);
|
||||
unsigned int x2apic_get_apic_id(unsigned long id);
|
||||
u32 x2apic_set_apic_id(unsigned int id);
|
||||
int x2apic_phys_pkg_id(int initial_apicid, int index_msb);
|
||||
void x2apic_send_IPI_self(int vector);
|
@@ -9,22 +9,24 @@
|
||||
#include <linux/cpu.h>
|
||||
|
||||
#include <asm/smp.h>
|
||||
#include <asm/x2apic.h>
|
||||
#include "x2apic.h"
|
||||
|
||||
struct cluster_mask {
|
||||
unsigned int clusterid;
|
||||
int node;
|
||||
struct cpumask mask;
|
||||
};
|
||||
|
||||
static DEFINE_PER_CPU(u32, x86_cpu_to_logical_apicid);
|
||||
static DEFINE_PER_CPU(cpumask_var_t, cpus_in_cluster);
|
||||
static DEFINE_PER_CPU(cpumask_var_t, ipi_mask);
|
||||
static DEFINE_PER_CPU(struct cluster_mask *, cluster_masks);
|
||||
static struct cluster_mask *cluster_hotplug_mask;
|
||||
|
||||
static int x2apic_acpi_madt_oem_check(char *oem_id, char *oem_table_id)
|
||||
{
|
||||
return x2apic_enabled();
|
||||
}
|
||||
|
||||
static inline u32 x2apic_cluster(int cpu)
|
||||
{
|
||||
return per_cpu(x86_cpu_to_logical_apicid, cpu) >> 16;
|
||||
}
|
||||
|
||||
static void x2apic_send_IPI(int cpu, int vector)
|
||||
{
|
||||
u32 dest = per_cpu(x86_cpu_to_logical_apicid, cpu);
|
||||
@@ -36,49 +38,34 @@ static void x2apic_send_IPI(int cpu, int vector)
|
||||
static void
|
||||
__x2apic_send_IPI_mask(const struct cpumask *mask, int vector, int apic_dest)
|
||||
{
|
||||
struct cpumask *cpus_in_cluster_ptr;
|
||||
struct cpumask *ipi_mask_ptr;
|
||||
unsigned int cpu, this_cpu;
|
||||
unsigned int cpu, clustercpu;
|
||||
struct cpumask *tmpmsk;
|
||||
unsigned long flags;
|
||||
u32 dest;
|
||||
|
||||
x2apic_wrmsr_fence();
|
||||
|
||||
local_irq_save(flags);
|
||||
|
||||
this_cpu = smp_processor_id();
|
||||
tmpmsk = this_cpu_cpumask_var_ptr(ipi_mask);
|
||||
cpumask_copy(tmpmsk, mask);
|
||||
/* If IPI should not be sent to self, clear current CPU */
|
||||
if (apic_dest != APIC_DEST_ALLINC)
|
||||
cpumask_clear_cpu(smp_processor_id(), tmpmsk);
|
||||
|
||||
/*
|
||||
* We are to modify mask, so we need an own copy
|
||||
* and be sure it's manipulated with irq off.
|
||||
*/
|
||||
ipi_mask_ptr = this_cpu_cpumask_var_ptr(ipi_mask);
|
||||
cpumask_copy(ipi_mask_ptr, mask);
|
||||
/* Collapse cpus in a cluster so a single IPI per cluster is sent */
|
||||
for_each_cpu(cpu, tmpmsk) {
|
||||
struct cluster_mask *cmsk = per_cpu(cluster_masks, cpu);
|
||||
|
||||
/*
|
||||
* The idea is to send one IPI per cluster.
|
||||
*/
|
||||
for_each_cpu(cpu, ipi_mask_ptr) {
|
||||
unsigned long i;
|
||||
|
||||
cpus_in_cluster_ptr = per_cpu(cpus_in_cluster, cpu);
|
||||
dest = 0;
|
||||
|
||||
/* Collect cpus in cluster. */
|
||||
for_each_cpu_and(i, ipi_mask_ptr, cpus_in_cluster_ptr) {
|
||||
if (apic_dest == APIC_DEST_ALLINC || i != this_cpu)
|
||||
dest |= per_cpu(x86_cpu_to_logical_apicid, i);
|
||||
}
|
||||
for_each_cpu_and(clustercpu, tmpmsk, &cmsk->mask)
|
||||
dest |= per_cpu(x86_cpu_to_logical_apicid, clustercpu);
|
||||
|
||||
if (!dest)
|
||||
continue;
|
||||
|
||||
__x2apic_send_IPI_dest(dest, vector, apic->dest_logical);
|
||||
/*
|
||||
* Cluster sibling cpus should be discared now so
|
||||
* we would not send IPI them second time.
|
||||
*/
|
||||
cpumask_andnot(ipi_mask_ptr, ipi_mask_ptr, cpus_in_cluster_ptr);
|
||||
/* Remove cluster CPUs from tmpmask */
|
||||
cpumask_andnot(tmpmsk, tmpmsk, &cmsk->mask);
|
||||
}
|
||||
|
||||
local_irq_restore(flags);
|
||||
@@ -105,125 +92,90 @@ static void x2apic_send_IPI_all(int vector)
|
||||
__x2apic_send_IPI_mask(cpu_online_mask, vector, APIC_DEST_ALLINC);
|
||||
}
|
||||
|
||||
static int
|
||||
x2apic_cpu_mask_to_apicid(const struct cpumask *mask, struct irq_data *irqdata,
|
||||
unsigned int *apicid)
|
||||
static u32 x2apic_calc_apicid(unsigned int cpu)
|
||||
{
|
||||
struct cpumask *effmsk = irq_data_get_effective_affinity_mask(irqdata);
|
||||
unsigned int cpu;
|
||||
u32 dest = 0;
|
||||
u16 cluster;
|
||||
|
||||
cpu = cpumask_first(mask);
|
||||
if (cpu >= nr_cpu_ids)
|
||||
return -EINVAL;
|
||||
|
||||
dest = per_cpu(x86_cpu_to_logical_apicid, cpu);
|
||||
cluster = x2apic_cluster(cpu);
|
||||
|
||||
cpumask_clear(effmsk);
|
||||
for_each_cpu(cpu, mask) {
|
||||
if (cluster != x2apic_cluster(cpu))
|
||||
continue;
|
||||
dest |= per_cpu(x86_cpu_to_logical_apicid, cpu);
|
||||
cpumask_set_cpu(cpu, effmsk);
|
||||
}
|
||||
|
||||
*apicid = dest;
|
||||
return 0;
|
||||
return per_cpu(x86_cpu_to_logical_apicid, cpu);
|
||||
}
|
||||
|
||||
static void init_x2apic_ldr(void)
|
||||
{
|
||||
unsigned int this_cpu = smp_processor_id();
|
||||
struct cluster_mask *cmsk = this_cpu_read(cluster_masks);
|
||||
u32 cluster, apicid = apic_read(APIC_LDR);
|
||||
unsigned int cpu;
|
||||
|
||||
per_cpu(x86_cpu_to_logical_apicid, this_cpu) = apic_read(APIC_LDR);
|
||||
this_cpu_write(x86_cpu_to_logical_apicid, apicid);
|
||||
|
||||
cpumask_set_cpu(this_cpu, per_cpu(cpus_in_cluster, this_cpu));
|
||||
if (cmsk)
|
||||
goto update;
|
||||
|
||||
cluster = apicid >> 16;
|
||||
for_each_online_cpu(cpu) {
|
||||
if (x2apic_cluster(this_cpu) != x2apic_cluster(cpu))
|
||||
continue;
|
||||
cpumask_set_cpu(this_cpu, per_cpu(cpus_in_cluster, cpu));
|
||||
cpumask_set_cpu(cpu, per_cpu(cpus_in_cluster, this_cpu));
|
||||
cmsk = per_cpu(cluster_masks, cpu);
|
||||
/* Matching cluster found. Link and update it. */
|
||||
if (cmsk && cmsk->clusterid == cluster)
|
||||
goto update;
|
||||
}
|
||||
cmsk = cluster_hotplug_mask;
|
||||
cluster_hotplug_mask = NULL;
|
||||
update:
|
||||
this_cpu_write(cluster_masks, cmsk);
|
||||
cpumask_set_cpu(smp_processor_id(), &cmsk->mask);
|
||||
}
|
||||
|
||||
/*
|
||||
* At CPU state changes, update the x2apic cluster sibling info.
|
||||
*/
|
||||
static int x2apic_prepare_cpu(unsigned int cpu)
|
||||
static int alloc_clustermask(unsigned int cpu, int node)
|
||||
{
|
||||
if (!zalloc_cpumask_var(&per_cpu(cpus_in_cluster, cpu), GFP_KERNEL))
|
||||
return -ENOMEM;
|
||||
|
||||
if (!zalloc_cpumask_var(&per_cpu(ipi_mask, cpu), GFP_KERNEL)) {
|
||||
free_cpumask_var(per_cpu(cpus_in_cluster, cpu));
|
||||
return -ENOMEM;
|
||||
if (per_cpu(cluster_masks, cpu))
|
||||
return 0;
|
||||
/*
|
||||
* If a hotplug spare mask exists, check whether it's on the right
|
||||
* node. If not, free it and allocate a new one.
|
||||
*/
|
||||
if (cluster_hotplug_mask) {
|
||||
if (cluster_hotplug_mask->node == node)
|
||||
return 0;
|
||||
kfree(cluster_hotplug_mask);
|
||||
}
|
||||
|
||||
cluster_hotplug_mask = kzalloc_node(sizeof(*cluster_hotplug_mask),
|
||||
GFP_KERNEL, node);
|
||||
if (!cluster_hotplug_mask)
|
||||
return -ENOMEM;
|
||||
cluster_hotplug_mask->node = node;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int x2apic_dead_cpu(unsigned int this_cpu)
|
||||
static int x2apic_prepare_cpu(unsigned int cpu)
|
||||
{
|
||||
int cpu;
|
||||
if (alloc_clustermask(cpu, cpu_to_node(cpu)) < 0)
|
||||
return -ENOMEM;
|
||||
if (!zalloc_cpumask_var(&per_cpu(ipi_mask, cpu), GFP_KERNEL))
|
||||
return -ENOMEM;
|
||||
return 0;
|
||||
}
|
||||
|
||||
for_each_online_cpu(cpu) {
|
||||
if (x2apic_cluster(this_cpu) != x2apic_cluster(cpu))
|
||||
continue;
|
||||
cpumask_clear_cpu(this_cpu, per_cpu(cpus_in_cluster, cpu));
|
||||
cpumask_clear_cpu(cpu, per_cpu(cpus_in_cluster, this_cpu));
|
||||
}
|
||||
free_cpumask_var(per_cpu(cpus_in_cluster, this_cpu));
|
||||
free_cpumask_var(per_cpu(ipi_mask, this_cpu));
|
||||
static int x2apic_dead_cpu(unsigned int dead_cpu)
|
||||
{
|
||||
struct cluster_mask *cmsk = per_cpu(cluster_masks, dead_cpu);
|
||||
|
||||
cpumask_clear_cpu(dead_cpu, &cmsk->mask);
|
||||
free_cpumask_var(per_cpu(ipi_mask, dead_cpu));
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int x2apic_cluster_probe(void)
|
||||
{
|
||||
int cpu = smp_processor_id();
|
||||
int ret;
|
||||
|
||||
if (!x2apic_mode)
|
||||
return 0;
|
||||
|
||||
ret = cpuhp_setup_state(CPUHP_X2APIC_PREPARE, "x86/x2apic:prepare",
|
||||
x2apic_prepare_cpu, x2apic_dead_cpu);
|
||||
if (ret < 0) {
|
||||
if (cpuhp_setup_state(CPUHP_X2APIC_PREPARE, "x86/x2apic:prepare",
|
||||
x2apic_prepare_cpu, x2apic_dead_cpu) < 0) {
|
||||
pr_err("Failed to register X2APIC_PREPARE\n");
|
||||
return 0;
|
||||
}
|
||||
cpumask_set_cpu(cpu, per_cpu(cpus_in_cluster, cpu));
|
||||
init_x2apic_ldr();
|
||||
return 1;
|
||||
}
|
||||
|
||||
static const struct cpumask *x2apic_cluster_target_cpus(void)
|
||||
{
|
||||
return cpu_all_mask;
|
||||
}
|
||||
|
||||
/*
|
||||
* Each x2apic cluster is an allocation domain.
|
||||
*/
|
||||
static void cluster_vector_allocation_domain(int cpu, struct cpumask *retmask,
|
||||
const struct cpumask *mask)
|
||||
{
|
||||
/*
|
||||
* To minimize vector pressure, default case of boot, device bringup
|
||||
* etc will use a single cpu for the interrupt destination.
|
||||
*
|
||||
* On explicit migration requests coming from irqbalance etc,
|
||||
* interrupts will be routed to the x2apic cluster (cluster-id
|
||||
* derived from the first cpu in the mask) members specified
|
||||
* in the mask.
|
||||
*/
|
||||
if (mask == x2apic_cluster_target_cpus())
|
||||
cpumask_copy(retmask, cpumask_of(cpu));
|
||||
else
|
||||
cpumask_and(retmask, mask, per_cpu(cpus_in_cluster, cpu));
|
||||
}
|
||||
|
||||
static struct apic apic_x2apic_cluster __ro_after_init = {
|
||||
|
||||
.name = "cluster x2apic",
|
||||
@@ -235,12 +187,10 @@ static struct apic apic_x2apic_cluster __ro_after_init = {
|
||||
.irq_delivery_mode = dest_LowestPrio,
|
||||
.irq_dest_mode = 1, /* logical */
|
||||
|
||||
.target_cpus = x2apic_cluster_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = APIC_DEST_LOGICAL,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = cluster_vector_allocation_domain,
|
||||
.init_apic_ldr = init_x2apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = NULL,
|
||||
@@ -253,7 +203,7 @@ static struct apic apic_x2apic_cluster __ro_after_init = {
|
||||
.get_apic_id = x2apic_get_apic_id,
|
||||
.set_apic_id = x2apic_set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = x2apic_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = x2apic_calc_apicid,
|
||||
|
||||
.send_IPI = x2apic_send_IPI,
|
||||
.send_IPI_mask = x2apic_send_IPI_mask,
|
||||
|
@@ -7,7 +7,8 @@
|
||||
#include <linux/dmar.h>
|
||||
|
||||
#include <asm/smp.h>
|
||||
#include <asm/x2apic.h>
|
||||
#include <asm/ipi.h>
|
||||
#include "x2apic.h"
|
||||
|
||||
int x2apic_phys;
|
||||
|
||||
@@ -99,6 +100,43 @@ static int x2apic_phys_probe(void)
|
||||
return apic == &apic_x2apic_phys;
|
||||
}
|
||||
|
||||
/* Common x2apic functions, also used by x2apic_cluster */
|
||||
int x2apic_apic_id_valid(int apicid)
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
int x2apic_apic_id_registered(void)
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
void __x2apic_send_IPI_dest(unsigned int apicid, int vector, unsigned int dest)
|
||||
{
|
||||
unsigned long cfg = __prepare_ICR(0, vector, dest);
|
||||
native_x2apic_icr_write(cfg, apicid);
|
||||
}
|
||||
|
||||
unsigned int x2apic_get_apic_id(unsigned long id)
|
||||
{
|
||||
return id;
|
||||
}
|
||||
|
||||
u32 x2apic_set_apic_id(unsigned int id)
|
||||
{
|
||||
return id;
|
||||
}
|
||||
|
||||
int x2apic_phys_pkg_id(int initial_apicid, int index_msb)
|
||||
{
|
||||
return initial_apicid >> index_msb;
|
||||
}
|
||||
|
||||
void x2apic_send_IPI_self(int vector)
|
||||
{
|
||||
apic_write(APIC_SELF_IPI, vector);
|
||||
}
|
||||
|
||||
static struct apic apic_x2apic_phys __ro_after_init = {
|
||||
|
||||
.name = "physical x2apic",
|
||||
@@ -110,12 +148,10 @@ static struct apic apic_x2apic_phys __ro_after_init = {
|
||||
.irq_delivery_mode = dest_Fixed,
|
||||
.irq_dest_mode = 0, /* physical */
|
||||
|
||||
.target_cpus = online_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = 0,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = default_vector_allocation_domain,
|
||||
.init_apic_ldr = init_x2apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = NULL,
|
||||
@@ -128,7 +164,7 @@ static struct apic apic_x2apic_phys __ro_after_init = {
|
||||
.get_apic_id = x2apic_get_apic_id,
|
||||
.set_apic_id = x2apic_set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = default_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_default_calc_apicid,
|
||||
|
||||
.send_IPI = x2apic_send_IPI,
|
||||
.send_IPI_mask = x2apic_send_IPI_mask,
|
||||
|
@@ -525,16 +525,9 @@ static void uv_init_apic_ldr(void)
|
||||
{
|
||||
}
|
||||
|
||||
static int
|
||||
uv_cpu_mask_to_apicid(const struct cpumask *mask, struct irq_data *irqdata,
|
||||
unsigned int *apicid)
|
||||
static u32 apic_uv_calc_apicid(unsigned int cpu)
|
||||
{
|
||||
int ret = default_cpu_mask_to_apicid(mask, irqdata, apicid);
|
||||
|
||||
if (!ret)
|
||||
*apicid |= uv_apicid_hibits;
|
||||
|
||||
return ret;
|
||||
return apic_default_calc_apicid(cpu) | uv_apicid_hibits;
|
||||
}
|
||||
|
||||
static unsigned int x2apic_get_apic_id(unsigned long x)
|
||||
@@ -547,7 +540,7 @@ static unsigned int x2apic_get_apic_id(unsigned long x)
|
||||
return id;
|
||||
}
|
||||
|
||||
static unsigned long set_apic_id(unsigned int id)
|
||||
static u32 set_apic_id(unsigned int id)
|
||||
{
|
||||
/* CHECKME: Do we need to mask out the xapic extra bits? */
|
||||
return id;
|
||||
@@ -584,12 +577,10 @@ static struct apic apic_x2apic_uv_x __ro_after_init = {
|
||||
.irq_delivery_mode = dest_Fixed,
|
||||
.irq_dest_mode = 0, /* Physical */
|
||||
|
||||
.target_cpus = online_target_cpus,
|
||||
.disable_esr = 0,
|
||||
.dest_logical = APIC_DEST_LOGICAL,
|
||||
.check_apicid_used = NULL,
|
||||
|
||||
.vector_allocation_domain = default_vector_allocation_domain,
|
||||
.init_apic_ldr = uv_init_apic_ldr,
|
||||
|
||||
.ioapic_phys_id_map = NULL,
|
||||
@@ -602,7 +593,7 @@ static struct apic apic_x2apic_uv_x __ro_after_init = {
|
||||
.get_apic_id = x2apic_get_apic_id,
|
||||
.set_apic_id = set_apic_id,
|
||||
|
||||
.cpu_mask_to_apicid = uv_cpu_mask_to_apicid,
|
||||
.calc_dest_apicid = apic_uv_calc_apicid,
|
||||
|
||||
.send_IPI = uv_send_IPI_one,
|
||||
.send_IPI_mask = uv_send_IPI_mask,
|
||||
|
@@ -114,6 +114,7 @@ static void make_8259A_irq(unsigned int irq)
|
||||
io_apic_irqs &= ~(1<<irq);
|
||||
irq_set_chip_and_handler(irq, &i8259A_chip, handle_level_irq);
|
||||
enable_irq(irq);
|
||||
lapic_assign_legacy_vector(irq, true);
|
||||
}
|
||||
|
||||
/*
|
||||
|
@@ -223,7 +223,7 @@ idt_setup_from_table(gate_desc *idt, const struct idt_data *t, int size, bool sy
|
||||
idt_init_desc(&desc, t);
|
||||
write_idt_entry(idt, t->vector, &desc);
|
||||
if (sys)
|
||||
set_bit(t->vector, used_vectors);
|
||||
set_bit(t->vector, system_vectors);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -311,14 +311,14 @@ void __init idt_setup_apic_and_irq_gates(void)
|
||||
|
||||
idt_setup_from_table(idt_table, apic_idts, ARRAY_SIZE(apic_idts), true);
|
||||
|
||||
for_each_clear_bit_from(i, used_vectors, FIRST_SYSTEM_VECTOR) {
|
||||
for_each_clear_bit_from(i, system_vectors, FIRST_SYSTEM_VECTOR) {
|
||||
entry = irq_entries_start + 8 * (i - FIRST_EXTERNAL_VECTOR);
|
||||
set_intr_gate(i, entry);
|
||||
}
|
||||
|
||||
for_each_clear_bit_from(i, used_vectors, NR_VECTORS) {
|
||||
for_each_clear_bit_from(i, system_vectors, NR_VECTORS) {
|
||||
#ifdef CONFIG_X86_LOCAL_APIC
|
||||
set_bit(i, used_vectors);
|
||||
set_bit(i, system_vectors);
|
||||
set_intr_gate(i, spurious_interrupt);
|
||||
#else
|
||||
entry = irq_entries_start + 8 * (i - FIRST_EXTERNAL_VECTOR);
|
||||
@@ -356,7 +356,7 @@ void idt_invalidate(void *addr)
|
||||
|
||||
void __init update_intr_gate(unsigned int n, const void *addr)
|
||||
{
|
||||
if (WARN_ON_ONCE(!test_bit(n, used_vectors)))
|
||||
if (WARN_ON_ONCE(!test_bit(n, system_vectors)))
|
||||
return;
|
||||
set_intr_gate(n, addr);
|
||||
}
|
||||
@@ -364,6 +364,6 @@ void __init update_intr_gate(unsigned int n, const void *addr)
|
||||
void alloc_intr_gate(unsigned int n, const void *addr)
|
||||
{
|
||||
BUG_ON(n < FIRST_SYSTEM_VECTOR);
|
||||
if (!test_and_set_bit(n, used_vectors))
|
||||
if (!test_and_set_bit(n, system_vectors))
|
||||
set_intr_gate(n, addr);
|
||||
}
|
||||
|
@@ -134,7 +134,7 @@ int arch_show_interrupts(struct seq_file *p, int prec)
|
||||
seq_puts(p, " Machine check polls\n");
|
||||
#endif
|
||||
#if IS_ENABLED(CONFIG_HYPERV) || defined(CONFIG_XEN)
|
||||
if (test_bit(HYPERVISOR_CALLBACK_VECTOR, used_vectors)) {
|
||||
if (test_bit(HYPERVISOR_CALLBACK_VECTOR, system_vectors)) {
|
||||
seq_printf(p, "%*s: ", prec, "HYP");
|
||||
for_each_online_cpu(j)
|
||||
seq_printf(p, "%10u ",
|
||||
@@ -333,105 +333,6 @@ __visible void smp_kvm_posted_intr_nested_ipi(struct pt_regs *regs)
|
||||
|
||||
|
||||
#ifdef CONFIG_HOTPLUG_CPU
|
||||
|
||||
/* These two declarations are only used in check_irq_vectors_for_cpu_disable()
|
||||
* below, which is protected by stop_machine(). Putting them on the stack
|
||||
* results in a stack frame overflow. Dynamically allocating could result in a
|
||||
* failure so declare these two cpumasks as global.
|
||||
*/
|
||||
static struct cpumask affinity_new, online_new;
|
||||
|
||||
/*
|
||||
* This cpu is going to be removed and its vectors migrated to the remaining
|
||||
* online cpus. Check to see if there are enough vectors in the remaining cpus.
|
||||
* This function is protected by stop_machine().
|
||||
*/
|
||||
int check_irq_vectors_for_cpu_disable(void)
|
||||
{
|
||||
unsigned int this_cpu, vector, this_count, count;
|
||||
struct irq_desc *desc;
|
||||
struct irq_data *data;
|
||||
int cpu;
|
||||
|
||||
this_cpu = smp_processor_id();
|
||||
cpumask_copy(&online_new, cpu_online_mask);
|
||||
cpumask_clear_cpu(this_cpu, &online_new);
|
||||
|
||||
this_count = 0;
|
||||
for (vector = FIRST_EXTERNAL_VECTOR; vector < NR_VECTORS; vector++) {
|
||||
desc = __this_cpu_read(vector_irq[vector]);
|
||||
if (IS_ERR_OR_NULL(desc))
|
||||
continue;
|
||||
/*
|
||||
* Protect against concurrent action removal, affinity
|
||||
* changes etc.
|
||||
*/
|
||||
raw_spin_lock(&desc->lock);
|
||||
data = irq_desc_get_irq_data(desc);
|
||||
cpumask_copy(&affinity_new,
|
||||
irq_data_get_affinity_mask(data));
|
||||
cpumask_clear_cpu(this_cpu, &affinity_new);
|
||||
|
||||
/* Do not count inactive or per-cpu irqs. */
|
||||
if (!irq_desc_has_action(desc) || irqd_is_per_cpu(data)) {
|
||||
raw_spin_unlock(&desc->lock);
|
||||
continue;
|
||||
}
|
||||
|
||||
raw_spin_unlock(&desc->lock);
|
||||
/*
|
||||
* A single irq may be mapped to multiple cpu's
|
||||
* vector_irq[] (for example IOAPIC cluster mode). In
|
||||
* this case we have two possibilities:
|
||||
*
|
||||
* 1) the resulting affinity mask is empty; that is
|
||||
* this the down'd cpu is the last cpu in the irq's
|
||||
* affinity mask, or
|
||||
*
|
||||
* 2) the resulting affinity mask is no longer a
|
||||
* subset of the online cpus but the affinity mask is
|
||||
* not zero; that is the down'd cpu is the last online
|
||||
* cpu in a user set affinity mask.
|
||||
*/
|
||||
if (cpumask_empty(&affinity_new) ||
|
||||
!cpumask_subset(&affinity_new, &online_new))
|
||||
this_count++;
|
||||
}
|
||||
/* No need to check any further. */
|
||||
if (!this_count)
|
||||
return 0;
|
||||
|
||||
count = 0;
|
||||
for_each_online_cpu(cpu) {
|
||||
if (cpu == this_cpu)
|
||||
continue;
|
||||
/*
|
||||
* We scan from FIRST_EXTERNAL_VECTOR to first system
|
||||
* vector. If the vector is marked in the used vectors
|
||||
* bitmap or an irq is assigned to it, we don't count
|
||||
* it as available.
|
||||
*
|
||||
* As this is an inaccurate snapshot anyway, we can do
|
||||
* this w/o holding vector_lock.
|
||||
*/
|
||||
for (vector = FIRST_EXTERNAL_VECTOR;
|
||||
vector < FIRST_SYSTEM_VECTOR; vector++) {
|
||||
if (!test_bit(vector, used_vectors) &&
|
||||
IS_ERR_OR_NULL(per_cpu(vector_irq, cpu)[vector])) {
|
||||
if (++count == this_count)
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (count < this_count) {
|
||||
pr_warn("CPU %d disable failed: CPU has %u vectors assigned and there are only %u available.\n",
|
||||
this_cpu, this_count, count);
|
||||
return -ERANGE;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* A cpu has been removed from cpu_online_mask. Reset irq affinities. */
|
||||
void fixup_irqs(void)
|
||||
{
|
||||
|
@@ -61,9 +61,6 @@ void __init init_ISA_irqs(void)
|
||||
struct irq_chip *chip = legacy_pic->chip;
|
||||
int i;
|
||||
|
||||
#if defined(CONFIG_X86_64) || defined(CONFIG_X86_LOCAL_APIC)
|
||||
init_bsp_APIC();
|
||||
#endif
|
||||
legacy_pic->init(0);
|
||||
|
||||
for (i = 0; i < nr_legacy_irqs(); i++)
|
||||
@@ -94,6 +91,7 @@ void __init native_init_IRQ(void)
|
||||
x86_init.irqs.pre_vector_init();
|
||||
|
||||
idt_setup_apic_and_irq_gates();
|
||||
lapic_assign_system_vectors();
|
||||
|
||||
if (!acpi_ioapic && !of_ioapic && nr_legacy_irqs())
|
||||
setup_irq(2, &irq2);
|
||||
|
@@ -136,18 +136,6 @@ RESERVE_BRK(dmi_alloc, 65536);
|
||||
static __initdata unsigned long _brk_start = (unsigned long)__brk_base;
|
||||
unsigned long _brk_end = (unsigned long)__brk_base;
|
||||
|
||||
#ifdef CONFIG_X86_64
|
||||
int default_cpu_present_to_apicid(int mps_cpu)
|
||||
{
|
||||
return __default_cpu_present_to_apicid(mps_cpu);
|
||||
}
|
||||
|
||||
int default_check_phys_apicid_present(int phys_apicid)
|
||||
{
|
||||
return __default_check_phys_apicid_present(phys_apicid);
|
||||
}
|
||||
#endif
|
||||
|
||||
struct boot_params boot_params;
|
||||
|
||||
/*
|
||||
|
@@ -256,14 +256,14 @@ static void notrace start_secondary(void *unused)
|
||||
check_tsc_sync_target();
|
||||
|
||||
/*
|
||||
* Lock vector_lock and initialize the vectors on this cpu
|
||||
* before setting the cpu online. We must set it online with
|
||||
* vector_lock held to prevent a concurrent setup/teardown
|
||||
* from seeing a half valid vector space.
|
||||
* Lock vector_lock, set CPU online and bring the vector
|
||||
* allocator online. Online must be set with vector_lock held
|
||||
* to prevent a concurrent irq setup/teardown from seeing a
|
||||
* half valid vector space.
|
||||
*/
|
||||
lock_vector_lock();
|
||||
setup_vector_irq(smp_processor_id());
|
||||
set_cpu_online(smp_processor_id(), true);
|
||||
lapic_online();
|
||||
unlock_vector_lock();
|
||||
cpu_set_state_online(smp_processor_id());
|
||||
x86_platform.nmi_init();
|
||||
@@ -1191,17 +1191,10 @@ static __init void disable_smp(void)
|
||||
cpumask_set_cpu(0, topology_core_cpumask(0));
|
||||
}
|
||||
|
||||
enum {
|
||||
SMP_OK,
|
||||
SMP_NO_CONFIG,
|
||||
SMP_NO_APIC,
|
||||
SMP_FORCE_UP,
|
||||
};
|
||||
|
||||
/*
|
||||
* Various sanity checks.
|
||||
*/
|
||||
static int __init smp_sanity_check(unsigned max_cpus)
|
||||
static void __init smp_sanity_check(void)
|
||||
{
|
||||
preempt_disable();
|
||||
|
||||
@@ -1238,16 +1231,6 @@ static int __init smp_sanity_check(unsigned max_cpus)
|
||||
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) {
|
||||
preempt_enable();
|
||||
pr_notice("SMP motherboard not detected\n");
|
||||
return SMP_NO_CONFIG;
|
||||
}
|
||||
|
||||
/*
|
||||
* Should not be necessary because the MP table should list the boot
|
||||
* CPU too, but we do it for the sake of robustness anyway.
|
||||
@@ -1258,29 +1241,6 @@ static int __init smp_sanity_check(unsigned max_cpus)
|
||||
physid_set(hard_smp_processor_id(), phys_cpu_present_map);
|
||||
}
|
||||
preempt_enable();
|
||||
|
||||
/*
|
||||
* If we couldn't find a local APIC, then get out of here now!
|
||||
*/
|
||||
if (APIC_INTEGRATED(boot_cpu_apic_version) &&
|
||||
!boot_cpu_has(X86_FEATURE_APIC)) {
|
||||
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 (tell your hw vendor)\n");
|
||||
}
|
||||
return SMP_NO_APIC;
|
||||
}
|
||||
|
||||
/*
|
||||
* If SMP should be disabled, then really disable it!
|
||||
*/
|
||||
if (!max_cpus) {
|
||||
pr_info("SMP mode deactivated\n");
|
||||
return SMP_FORCE_UP;
|
||||
}
|
||||
|
||||
return SMP_OK;
|
||||
}
|
||||
|
||||
static void __init smp_cpu_index_default(void)
|
||||
@@ -1295,9 +1255,18 @@ static void __init smp_cpu_index_default(void)
|
||||
}
|
||||
}
|
||||
|
||||
static void __init smp_get_logical_apicid(void)
|
||||
{
|
||||
if (x2apic_mode)
|
||||
cpu0_logical_apicid = apic_read(APIC_LDR);
|
||||
else
|
||||
cpu0_logical_apicid = GET_APIC_LOGICAL_ID(apic_read(APIC_LDR));
|
||||
}
|
||||
|
||||
/*
|
||||
* Prepare for SMP bootup. The MP table or ACPI has been read
|
||||
* earlier. Just do some sanity checking here and enable APIC mode.
|
||||
* Prepare for SMP bootup.
|
||||
* @max_cpus: configured maximum number of CPUs, It is a legacy parameter
|
||||
* for common interface support.
|
||||
*/
|
||||
void __init native_smp_prepare_cpus(unsigned int max_cpus)
|
||||
{
|
||||
@@ -1329,31 +1298,27 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus)
|
||||
|
||||
set_cpu_sibling_map(0);
|
||||
|
||||
switch (smp_sanity_check(max_cpus)) {
|
||||
case SMP_NO_CONFIG:
|
||||
disable_smp();
|
||||
if (APIC_init_uniprocessor())
|
||||
pr_notice("Local APIC not detected. Using dummy APIC emulation.\n");
|
||||
return;
|
||||
case SMP_NO_APIC:
|
||||
smp_sanity_check();
|
||||
|
||||
switch (apic_intr_mode) {
|
||||
case APIC_PIC:
|
||||
case APIC_VIRTUAL_WIRE_NO_CONFIG:
|
||||
disable_smp();
|
||||
return;
|
||||
case SMP_FORCE_UP:
|
||||
case APIC_SYMMETRIC_IO_NO_ROUTING:
|
||||
disable_smp();
|
||||
apic_bsp_setup(false);
|
||||
/* Setup local timer */
|
||||
x86_init.timers.setup_percpu_clockev();
|
||||
return;
|
||||
case SMP_OK:
|
||||
case APIC_VIRTUAL_WIRE:
|
||||
case APIC_SYMMETRIC_IO:
|
||||
break;
|
||||
}
|
||||
|
||||
if (read_apic_id() != boot_cpu_physical_apicid) {
|
||||
panic("Boot APIC ID in local APIC unexpected (%d vs %d)",
|
||||
read_apic_id(), boot_cpu_physical_apicid);
|
||||
/* Or can we switch back to PIC here? */
|
||||
}
|
||||
/* Setup local timer */
|
||||
x86_init.timers.setup_percpu_clockev();
|
||||
|
||||
default_setup_apic_routing();
|
||||
cpu0_logical_apicid = apic_bsp_setup(false);
|
||||
smp_get_logical_apicid();
|
||||
|
||||
pr_info("CPU0: ");
|
||||
print_cpu_info(&cpu_data(0));
|
||||
@@ -1398,7 +1363,6 @@ void __init native_smp_cpus_done(unsigned int max_cpus)
|
||||
|
||||
nmi_selftest();
|
||||
impress_friends();
|
||||
setup_ioapic_dest();
|
||||
mtrr_aps_init();
|
||||
}
|
||||
|
||||
@@ -1557,13 +1521,14 @@ void cpu_disable_common(void)
|
||||
remove_cpu_from_maps(cpu);
|
||||
unlock_vector_lock();
|
||||
fixup_irqs();
|
||||
lapic_offline();
|
||||
}
|
||||
|
||||
int native_cpu_disable(void)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = check_irq_vectors_for_cpu_disable();
|
||||
ret = lapic_can_unplug_cpu();
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
|
@@ -85,6 +85,11 @@ void __init hpet_time_init(void)
|
||||
static __init void x86_late_time_init(void)
|
||||
{
|
||||
x86_init.timers.timer_init();
|
||||
/*
|
||||
* After PIT/HPET timers init, select and setup
|
||||
* the final interrupt mode for delivering IRQs.
|
||||
*/
|
||||
x86_init.irqs.intr_mode_init();
|
||||
tsc_init();
|
||||
}
|
||||
|
||||
|
@@ -72,7 +72,7 @@
|
||||
#include <asm/proto.h>
|
||||
#endif
|
||||
|
||||
DECLARE_BITMAP(used_vectors, NR_VECTORS);
|
||||
DECLARE_BITMAP(system_vectors, NR_VECTORS);
|
||||
|
||||
static inline void cond_local_irq_enable(struct pt_regs *regs)
|
||||
{
|
||||
|
@@ -26,9 +26,6 @@
|
||||
|
||||
#define TOPOLOGY_REGISTER_OFFSET 0x10
|
||||
|
||||
/* Flag below is initialized once during vSMP PCI initialization. */
|
||||
static int irq_routing_comply = 1;
|
||||
|
||||
#if defined CONFIG_PCI && defined CONFIG_PARAVIRT
|
||||
/*
|
||||
* Interrupt control on vSMPowered systems:
|
||||
@@ -105,9 +102,6 @@ static void __init set_vsmp_pv_ops(void)
|
||||
if (cap & ctl & BIT(8)) {
|
||||
ctl &= ~BIT(8);
|
||||
|
||||
/* Interrupt routing set to ignore */
|
||||
irq_routing_comply = 0;
|
||||
|
||||
#ifdef CONFIG_PROC_FS
|
||||
/* Don't let users change irq affinity via procfs */
|
||||
no_irq_affinity = 1;
|
||||
@@ -211,23 +205,10 @@ static int apicid_phys_pkg_id(int initial_apic_id, int index_msb)
|
||||
return hard_smp_processor_id() >> index_msb;
|
||||
}
|
||||
|
||||
/*
|
||||
* In vSMP, all cpus should be capable of handling interrupts, regardless of
|
||||
* the APIC used.
|
||||
*/
|
||||
static void fill_vector_allocation_domain(int cpu, struct cpumask *retmask,
|
||||
const struct cpumask *mask)
|
||||
{
|
||||
cpumask_setall(retmask);
|
||||
}
|
||||
|
||||
static void vsmp_apic_post_init(void)
|
||||
{
|
||||
/* need to update phys_pkg_id */
|
||||
apic->phys_pkg_id = apicid_phys_pkg_id;
|
||||
|
||||
if (!irq_routing_comply)
|
||||
apic->vector_allocation_domain = fill_vector_allocation_domain;
|
||||
}
|
||||
|
||||
void __init vsmp_init(void)
|
||||
|
@@ -57,6 +57,7 @@ struct x86_init_ops x86_init __initdata = {
|
||||
.pre_vector_init = init_ISA_irqs,
|
||||
.intr_init = native_init_IRQ,
|
||||
.trap_init = x86_init_noop,
|
||||
.intr_mode_init = apic_intr_mode_init
|
||||
},
|
||||
|
||||
.oem = {
|
||||
|
Reference in New Issue
Block a user