a31e58e129
Some of the APIC incarnations are operating in lowest priority delivery
mode. This worked as long as the vector management code allocated the same
vector on all possible CPUs for each interrupt.
Lowest priority delivery mode does not necessarily respect the affinity
setting and may redirect to some other online CPU. This was documented
somewhere in the old code and the conversion to single target delivery
missed to update the delivery mode of the affected APIC drivers which
results in spurious interrupts on some of the affected CPU/Chipset
combinations.
Switch the APIC drivers over to Fixed delivery mode and remove all
leftovers of lowest priority delivery mode.
Switching to Fixed delivery mode is not a problem on these CPUs because the
kernel already uses Fixed delivery mode for IPIs. The reason for this is
that th SDM explicitely forbids lowest prio mode for IPIs. The reason is
obvious: If the irq routing does not honor destination targets in lowest
prio mode then an IPI targeted at CPU1 might end up on CPU0, which would be
a fatal problem in many cases.
As a consequence of this change, the apic::irq_delivery_mode field is now
pointless, but this needs to be cleaned up in a separate patch.
Fixes: fdba46ffb4
("x86/apic: Get rid of multi CPU affinity")
Reported-by: vcaputo@pengaru.com
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: vcaputo@pengaru.com
Cc: Pavel Machek <pavel@ucw.cz>
Link: https://lkml.kernel.org/r/alpine.DEB.2.20.1712281140440.1688@nanos
389 lines
9.6 KiB
C
389 lines
9.6 KiB
C
/*
|
|
* Support of MSI, HPET and DMAR interrupts.
|
|
*
|
|
* Copyright (C) 1997, 1998, 1999, 2000, 2009 Ingo Molnar, Hajnalka Szabo
|
|
* Moved from arch/x86/kernel/apic/io_apic.c.
|
|
* Jiang Liu <jiang.liu@linux.intel.com>
|
|
* Convert to hierarchical irqdomain
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
* published by the Free Software Foundation.
|
|
*/
|
|
#include <linux/mm.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/pci.h>
|
|
#include <linux/dmar.h>
|
|
#include <linux/hpet.h>
|
|
#include <linux/msi.h>
|
|
#include <asm/irqdomain.h>
|
|
#include <asm/msidef.h>
|
|
#include <asm/hpet.h>
|
|
#include <asm/hw_irq.h>
|
|
#include <asm/apic.h>
|
|
#include <asm/irq_remapping.h>
|
|
|
|
static struct irq_domain *msi_default_domain;
|
|
|
|
static void irq_msi_compose_msg(struct irq_data *data, struct msi_msg *msg)
|
|
{
|
|
struct irq_cfg *cfg = irqd_cfg(data);
|
|
|
|
msg->address_hi = MSI_ADDR_BASE_HI;
|
|
|
|
if (x2apic_enabled())
|
|
msg->address_hi |= MSI_ADDR_EXT_DEST_ID(cfg->dest_apicid);
|
|
|
|
msg->address_lo =
|
|
MSI_ADDR_BASE_LO |
|
|
((apic->irq_dest_mode == 0) ?
|
|
MSI_ADDR_DEST_MODE_PHYSICAL :
|
|
MSI_ADDR_DEST_MODE_LOGICAL) |
|
|
MSI_ADDR_REDIRECTION_CPU |
|
|
MSI_ADDR_DEST_ID(cfg->dest_apicid);
|
|
|
|
msg->data =
|
|
MSI_DATA_TRIGGER_EDGE |
|
|
MSI_DATA_LEVEL_ASSERT |
|
|
MSI_DATA_DELIVERY_FIXED |
|
|
MSI_DATA_VECTOR(cfg->vector);
|
|
}
|
|
|
|
/*
|
|
* IRQ Chip for MSI PCI/PCI-X/PCI-Express Devices,
|
|
* which implement the MSI or MSI-X Capability Structure.
|
|
*/
|
|
static struct irq_chip pci_msi_controller = {
|
|
.name = "PCI-MSI",
|
|
.irq_unmask = pci_msi_unmask_irq,
|
|
.irq_mask = pci_msi_mask_irq,
|
|
.irq_ack = irq_chip_ack_parent,
|
|
.irq_retrigger = irq_chip_retrigger_hierarchy,
|
|
.irq_compose_msi_msg = irq_msi_compose_msg,
|
|
.flags = IRQCHIP_SKIP_SET_WAKE,
|
|
};
|
|
|
|
int native_setup_msi_irqs(struct pci_dev *dev, int nvec, int type)
|
|
{
|
|
struct irq_domain *domain;
|
|
struct irq_alloc_info info;
|
|
|
|
init_irq_alloc_info(&info, NULL);
|
|
info.type = X86_IRQ_ALLOC_TYPE_MSI;
|
|
info.msi_dev = dev;
|
|
|
|
domain = irq_remapping_get_irq_domain(&info);
|
|
if (domain == NULL)
|
|
domain = msi_default_domain;
|
|
if (domain == NULL)
|
|
return -ENOSYS;
|
|
|
|
return msi_domain_alloc_irqs(domain, &dev->dev, nvec);
|
|
}
|
|
|
|
void native_teardown_msi_irq(unsigned int irq)
|
|
{
|
|
irq_domain_free_irqs(irq, 1);
|
|
}
|
|
|
|
static irq_hw_number_t pci_msi_get_hwirq(struct msi_domain_info *info,
|
|
msi_alloc_info_t *arg)
|
|
{
|
|
return arg->msi_hwirq;
|
|
}
|
|
|
|
int pci_msi_prepare(struct irq_domain *domain, struct device *dev, int nvec,
|
|
msi_alloc_info_t *arg)
|
|
{
|
|
struct pci_dev *pdev = to_pci_dev(dev);
|
|
struct msi_desc *desc = first_pci_msi_entry(pdev);
|
|
|
|
init_irq_alloc_info(arg, NULL);
|
|
arg->msi_dev = pdev;
|
|
if (desc->msi_attrib.is_msix) {
|
|
arg->type = X86_IRQ_ALLOC_TYPE_MSIX;
|
|
} else {
|
|
arg->type = X86_IRQ_ALLOC_TYPE_MSI;
|
|
arg->flags |= X86_IRQ_ALLOC_CONTIGUOUS_VECTORS;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(pci_msi_prepare);
|
|
|
|
void pci_msi_set_desc(msi_alloc_info_t *arg, struct msi_desc *desc)
|
|
{
|
|
arg->msi_hwirq = pci_msi_domain_calc_hwirq(arg->msi_dev, desc);
|
|
}
|
|
EXPORT_SYMBOL_GPL(pci_msi_set_desc);
|
|
|
|
static struct msi_domain_ops pci_msi_domain_ops = {
|
|
.get_hwirq = pci_msi_get_hwirq,
|
|
.msi_prepare = pci_msi_prepare,
|
|
.set_desc = pci_msi_set_desc,
|
|
};
|
|
|
|
static struct msi_domain_info pci_msi_domain_info = {
|
|
.flags = MSI_FLAG_USE_DEF_DOM_OPS | MSI_FLAG_USE_DEF_CHIP_OPS |
|
|
MSI_FLAG_PCI_MSIX,
|
|
.ops = &pci_msi_domain_ops,
|
|
.chip = &pci_msi_controller,
|
|
.handler = handle_edge_irq,
|
|
.handler_name = "edge",
|
|
};
|
|
|
|
void __init arch_init_msi_domain(struct irq_domain *parent)
|
|
{
|
|
struct fwnode_handle *fn;
|
|
|
|
if (disable_apic)
|
|
return;
|
|
|
|
fn = irq_domain_alloc_named_fwnode("PCI-MSI");
|
|
if (fn) {
|
|
msi_default_domain =
|
|
pci_msi_create_irq_domain(fn, &pci_msi_domain_info,
|
|
parent);
|
|
irq_domain_free_fwnode(fn);
|
|
}
|
|
if (!msi_default_domain)
|
|
pr_warn("failed to initialize irqdomain for MSI/MSI-x.\n");
|
|
}
|
|
|
|
#ifdef CONFIG_IRQ_REMAP
|
|
static struct irq_chip pci_msi_ir_controller = {
|
|
.name = "IR-PCI-MSI",
|
|
.irq_unmask = pci_msi_unmask_irq,
|
|
.irq_mask = pci_msi_mask_irq,
|
|
.irq_ack = irq_chip_ack_parent,
|
|
.irq_retrigger = irq_chip_retrigger_hierarchy,
|
|
.irq_set_vcpu_affinity = irq_chip_set_vcpu_affinity_parent,
|
|
.flags = IRQCHIP_SKIP_SET_WAKE,
|
|
};
|
|
|
|
static struct msi_domain_info pci_msi_ir_domain_info = {
|
|
.flags = MSI_FLAG_USE_DEF_DOM_OPS | MSI_FLAG_USE_DEF_CHIP_OPS |
|
|
MSI_FLAG_MULTI_PCI_MSI | MSI_FLAG_PCI_MSIX,
|
|
.ops = &pci_msi_domain_ops,
|
|
.chip = &pci_msi_ir_controller,
|
|
.handler = handle_edge_irq,
|
|
.handler_name = "edge",
|
|
};
|
|
|
|
struct irq_domain *arch_create_remap_msi_irq_domain(struct irq_domain *parent,
|
|
const char *name, int id)
|
|
{
|
|
struct fwnode_handle *fn;
|
|
struct irq_domain *d;
|
|
|
|
fn = irq_domain_alloc_named_id_fwnode(name, id);
|
|
if (!fn)
|
|
return NULL;
|
|
d = pci_msi_create_irq_domain(fn, &pci_msi_ir_domain_info, parent);
|
|
irq_domain_free_fwnode(fn);
|
|
return d;
|
|
}
|
|
#endif
|
|
|
|
#ifdef CONFIG_DMAR_TABLE
|
|
static void dmar_msi_write_msg(struct irq_data *data, struct msi_msg *msg)
|
|
{
|
|
dmar_msi_write(data->irq, msg);
|
|
}
|
|
|
|
static struct irq_chip dmar_msi_controller = {
|
|
.name = "DMAR-MSI",
|
|
.irq_unmask = dmar_msi_unmask,
|
|
.irq_mask = dmar_msi_mask,
|
|
.irq_ack = irq_chip_ack_parent,
|
|
.irq_set_affinity = msi_domain_set_affinity,
|
|
.irq_retrigger = irq_chip_retrigger_hierarchy,
|
|
.irq_compose_msi_msg = irq_msi_compose_msg,
|
|
.irq_write_msi_msg = dmar_msi_write_msg,
|
|
.flags = IRQCHIP_SKIP_SET_WAKE,
|
|
};
|
|
|
|
static irq_hw_number_t dmar_msi_get_hwirq(struct msi_domain_info *info,
|
|
msi_alloc_info_t *arg)
|
|
{
|
|
return arg->dmar_id;
|
|
}
|
|
|
|
static int dmar_msi_init(struct irq_domain *domain,
|
|
struct msi_domain_info *info, unsigned int virq,
|
|
irq_hw_number_t hwirq, msi_alloc_info_t *arg)
|
|
{
|
|
irq_domain_set_info(domain, virq, arg->dmar_id, info->chip, NULL,
|
|
handle_edge_irq, arg->dmar_data, "edge");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct msi_domain_ops dmar_msi_domain_ops = {
|
|
.get_hwirq = dmar_msi_get_hwirq,
|
|
.msi_init = dmar_msi_init,
|
|
};
|
|
|
|
static struct msi_domain_info dmar_msi_domain_info = {
|
|
.ops = &dmar_msi_domain_ops,
|
|
.chip = &dmar_msi_controller,
|
|
};
|
|
|
|
static struct irq_domain *dmar_get_irq_domain(void)
|
|
{
|
|
static struct irq_domain *dmar_domain;
|
|
static DEFINE_MUTEX(dmar_lock);
|
|
struct fwnode_handle *fn;
|
|
|
|
mutex_lock(&dmar_lock);
|
|
if (dmar_domain)
|
|
goto out;
|
|
|
|
fn = irq_domain_alloc_named_fwnode("DMAR-MSI");
|
|
if (fn) {
|
|
dmar_domain = msi_create_irq_domain(fn, &dmar_msi_domain_info,
|
|
x86_vector_domain);
|
|
irq_domain_free_fwnode(fn);
|
|
}
|
|
out:
|
|
mutex_unlock(&dmar_lock);
|
|
return dmar_domain;
|
|
}
|
|
|
|
int dmar_alloc_hwirq(int id, int node, void *arg)
|
|
{
|
|
struct irq_domain *domain = dmar_get_irq_domain();
|
|
struct irq_alloc_info info;
|
|
|
|
if (!domain)
|
|
return -1;
|
|
|
|
init_irq_alloc_info(&info, NULL);
|
|
info.type = X86_IRQ_ALLOC_TYPE_DMAR;
|
|
info.dmar_id = id;
|
|
info.dmar_data = arg;
|
|
|
|
return irq_domain_alloc_irqs(domain, 1, node, &info);
|
|
}
|
|
|
|
void dmar_free_hwirq(int irq)
|
|
{
|
|
irq_domain_free_irqs(irq, 1);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* MSI message composition
|
|
*/
|
|
#ifdef CONFIG_HPET_TIMER
|
|
static inline int hpet_dev_id(struct irq_domain *domain)
|
|
{
|
|
struct msi_domain_info *info = msi_get_domain_info(domain);
|
|
|
|
return (int)(long)info->data;
|
|
}
|
|
|
|
static void hpet_msi_write_msg(struct irq_data *data, struct msi_msg *msg)
|
|
{
|
|
hpet_msi_write(irq_data_get_irq_handler_data(data), msg);
|
|
}
|
|
|
|
static struct irq_chip hpet_msi_controller __ro_after_init = {
|
|
.name = "HPET-MSI",
|
|
.irq_unmask = hpet_msi_unmask,
|
|
.irq_mask = hpet_msi_mask,
|
|
.irq_ack = irq_chip_ack_parent,
|
|
.irq_set_affinity = msi_domain_set_affinity,
|
|
.irq_retrigger = irq_chip_retrigger_hierarchy,
|
|
.irq_compose_msi_msg = irq_msi_compose_msg,
|
|
.irq_write_msi_msg = hpet_msi_write_msg,
|
|
.flags = IRQCHIP_SKIP_SET_WAKE,
|
|
};
|
|
|
|
static irq_hw_number_t hpet_msi_get_hwirq(struct msi_domain_info *info,
|
|
msi_alloc_info_t *arg)
|
|
{
|
|
return arg->hpet_index;
|
|
}
|
|
|
|
static int hpet_msi_init(struct irq_domain *domain,
|
|
struct msi_domain_info *info, unsigned int virq,
|
|
irq_hw_number_t hwirq, msi_alloc_info_t *arg)
|
|
{
|
|
irq_set_status_flags(virq, IRQ_MOVE_PCNTXT);
|
|
irq_domain_set_info(domain, virq, arg->hpet_index, info->chip, NULL,
|
|
handle_edge_irq, arg->hpet_data, "edge");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void hpet_msi_free(struct irq_domain *domain,
|
|
struct msi_domain_info *info, unsigned int virq)
|
|
{
|
|
irq_clear_status_flags(virq, IRQ_MOVE_PCNTXT);
|
|
}
|
|
|
|
static struct msi_domain_ops hpet_msi_domain_ops = {
|
|
.get_hwirq = hpet_msi_get_hwirq,
|
|
.msi_init = hpet_msi_init,
|
|
.msi_free = hpet_msi_free,
|
|
};
|
|
|
|
static struct msi_domain_info hpet_msi_domain_info = {
|
|
.ops = &hpet_msi_domain_ops,
|
|
.chip = &hpet_msi_controller,
|
|
};
|
|
|
|
struct irq_domain *hpet_create_irq_domain(int hpet_id)
|
|
{
|
|
struct msi_domain_info *domain_info;
|
|
struct irq_domain *parent, *d;
|
|
struct irq_alloc_info info;
|
|
struct fwnode_handle *fn;
|
|
|
|
if (x86_vector_domain == NULL)
|
|
return NULL;
|
|
|
|
domain_info = kzalloc(sizeof(*domain_info), GFP_KERNEL);
|
|
if (!domain_info)
|
|
return NULL;
|
|
|
|
*domain_info = hpet_msi_domain_info;
|
|
domain_info->data = (void *)(long)hpet_id;
|
|
|
|
init_irq_alloc_info(&info, NULL);
|
|
info.type = X86_IRQ_ALLOC_TYPE_HPET;
|
|
info.hpet_id = hpet_id;
|
|
parent = irq_remapping_get_ir_irq_domain(&info);
|
|
if (parent == NULL)
|
|
parent = x86_vector_domain;
|
|
else
|
|
hpet_msi_controller.name = "IR-HPET-MSI";
|
|
|
|
fn = irq_domain_alloc_named_id_fwnode(hpet_msi_controller.name,
|
|
hpet_id);
|
|
if (!fn) {
|
|
kfree(domain_info);
|
|
return NULL;
|
|
}
|
|
|
|
d = msi_create_irq_domain(fn, domain_info, parent);
|
|
irq_domain_free_fwnode(fn);
|
|
return d;
|
|
}
|
|
|
|
int hpet_assign_irq(struct irq_domain *domain, struct hpet_dev *dev,
|
|
int dev_num)
|
|
{
|
|
struct irq_alloc_info info;
|
|
|
|
init_irq_alloc_info(&info, NULL);
|
|
info.type = X86_IRQ_ALLOC_TYPE_HPET;
|
|
info.hpet_data = dev;
|
|
info.hpet_id = hpet_dev_id(domain);
|
|
info.hpet_index = dev_num;
|
|
|
|
return irq_domain_alloc_irqs(domain, 1, NUMA_NO_NODE, &info);
|
|
}
|
|
#endif
|