Commit 6d50bc26 authored by Yinghai Lu's avatar Yinghai Lu Committed by Ingo Molnar

x86: use 28 bits irq NR for pci msi/msix and ht

also print out irq no in /proc/interrups and /proc/stat in hex, so could
tell bus/dev/func.
Signed-off-by: default avatarYinghai Lu <yhlu.kernel@gmail.com>
Signed-off-by: default avatarIngo Molnar <mingo@elte.hu>
parent e420dfb4
...@@ -2520,17 +2520,21 @@ device_initcall(ioapic_init_sysfs); ...@@ -2520,17 +2520,21 @@ device_initcall(ioapic_init_sysfs);
/* /*
* Dynamic irq allocate and deallocation * Dynamic irq allocate and deallocation
*/ */
int create_irq(void) unsigned int create_irq_nr(unsigned int irq_want)
{ {
/* Allocate an unused irq */ /* Allocate an unused irq */
int irq; unsigned int irq;
int new; unsigned int new;
unsigned long flags; unsigned long flags;
struct irq_cfg *cfg_new; struct irq_cfg *cfg_new;
irq = -ENOSPC; #ifndef CONFIG_HAVE_SPARSE_IRQ
irq_want = nr_irqs - 1;
#endif
irq = 0;
spin_lock_irqsave(&vector_lock, flags); spin_lock_irqsave(&vector_lock, flags);
for (new = (nr_irqs - 1); new >= 0; new--) { for (new = irq_want; new > 0; new--) {
if (platform_legacy_irq(new)) if (platform_legacy_irq(new))
continue; continue;
cfg_new = irq_cfg(new); cfg_new = irq_cfg(new);
...@@ -2545,12 +2549,24 @@ int create_irq(void) ...@@ -2545,12 +2549,24 @@ int create_irq(void)
} }
spin_unlock_irqrestore(&vector_lock, flags); spin_unlock_irqrestore(&vector_lock, flags);
if (irq >= 0) { if (irq > 0) {
dynamic_irq_init(irq); dynamic_irq_init(irq);
} }
return irq; return irq;
} }
int create_irq(void)
{
int irq;
irq = create_irq_nr(nr_irqs - 1);
if (irq == 0)
irq = -1;
return irq;
}
void destroy_irq(unsigned int irq) void destroy_irq(unsigned int irq)
{ {
unsigned long flags; unsigned long flags;
...@@ -2803,13 +2819,29 @@ static int setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc, int irq) ...@@ -2803,13 +2819,29 @@ static int setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc, int irq)
return 0; return 0;
} }
int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc) static unsigned int build_irq_for_pci_dev(struct pci_dev *dev)
{ {
int irq, ret; unsigned int irq;
irq = dev->bus->number;
irq <<= 8;
irq |= dev->devfn;
irq <<= 12;
irq = create_irq();
if (irq < 0)
return irq; return irq;
}
int arch_setup_msi_irq(struct pci_dev *dev, struct msi_desc *desc)
{
unsigned int irq;
int ret;
unsigned int irq_want;
irq_want = build_irq_for_pci_dev(dev) + 0x100;
irq = create_irq_nr(irq_want);
if (irq == 0)
return -1;
#ifdef CONFIG_INTR_REMAP #ifdef CONFIG_INTR_REMAP
if (!intr_remapping_enabled) if (!intr_remapping_enabled)
...@@ -2836,18 +2868,22 @@ error: ...@@ -2836,18 +2868,22 @@ error:
int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) int arch_setup_msi_irqs(struct pci_dev *dev, int nvec, int type)
{ {
int irq, ret, sub_handle; unsigned int irq;
int ret, sub_handle;
struct msi_desc *desc; struct msi_desc *desc;
unsigned int irq_want;
#ifdef CONFIG_INTR_REMAP #ifdef CONFIG_INTR_REMAP
struct intel_iommu *iommu = 0; struct intel_iommu *iommu = 0;
int index = 0; int index = 0;
#endif #endif
irq_want = build_irq_for_pci_dev(dev) + 0x100;
sub_handle = 0; sub_handle = 0;
list_for_each_entry(desc, &dev->msi_list, list) { list_for_each_entry(desc, &dev->msi_list, list) {
irq = create_irq(); irq = create_irq_nr(irq_want--);
if (irq < 0) if (irq == 0)
return irq; return -1;
#ifdef CONFIG_INTR_REMAP #ifdef CONFIG_INTR_REMAP
if (!intr_remapping_enabled) if (!intr_remapping_enabled)
goto no_ir; goto no_ir;
......
...@@ -112,7 +112,7 @@ int show_interrupts(struct seq_file *p, void *v) ...@@ -112,7 +112,7 @@ int show_interrupts(struct seq_file *p, void *v)
action = desc->action; action = desc->action;
if (!action && !any_count) if (!action && !any_count)
goto skip; goto skip;
seq_printf(p, "%3d: ",i); seq_printf(p, "%#x: ",i);
#ifndef CONFIG_SMP #ifndef CONFIG_SMP
seq_printf(p, "%10u ", kstat_irqs(i)); seq_printf(p, "%10u ", kstat_irqs(i));
#else #else
......
...@@ -82,6 +82,18 @@ void unmask_ht_irq(unsigned int irq) ...@@ -82,6 +82,18 @@ void unmask_ht_irq(unsigned int irq)
write_ht_irq_msg(irq, &msg); write_ht_irq_msg(irq, &msg);
} }
static unsigned int build_irq_for_pci_dev(struct pci_dev *dev)
{
unsigned int irq;
irq = dev->bus->number;
irq <<= 8;
irq |= dev->devfn;
irq <<= 12;
return irq;
}
/** /**
* __ht_create_irq - create an irq and attach it to a device. * __ht_create_irq - create an irq and attach it to a device.
* @dev: The hypertransport device to find the irq capability on. * @dev: The hypertransport device to find the irq capability on.
...@@ -97,7 +109,8 @@ int __ht_create_irq(struct pci_dev *dev, int idx, ht_irq_update_t *update) ...@@ -97,7 +109,8 @@ int __ht_create_irq(struct pci_dev *dev, int idx, ht_irq_update_t *update)
u32 data; u32 data;
int max_irq; int max_irq;
int pos; int pos;
int irq; unsigned int irq;
unsigned int irq_want;
pos = pci_find_ht_capability(dev, HT_CAPTYPE_IRQ); pos = pci_find_ht_capability(dev, HT_CAPTYPE_IRQ);
if (!pos) if (!pos)
...@@ -125,8 +138,13 @@ int __ht_create_irq(struct pci_dev *dev, int idx, ht_irq_update_t *update) ...@@ -125,8 +138,13 @@ int __ht_create_irq(struct pci_dev *dev, int idx, ht_irq_update_t *update)
cfg->msg.address_lo = 0xffffffff; cfg->msg.address_lo = 0xffffffff;
cfg->msg.address_hi = 0xffffffff; cfg->msg.address_hi = 0xffffffff;
irq_want= build_irq_for_pci_dev(dev);
#ifdef CONFIG_HAVE_SPARSE_IRQ
irq = create_irq_nr(irq_want + idx);
#else
irq = create_irq(); irq = create_irq();
if (irq < 0) { #endif
if (irq == 0) {
kfree(cfg); kfree(cfg);
return -EBUSY; return -EBUSY;
} }
......
...@@ -589,7 +589,7 @@ static int show_stat(struct seq_file *p, void *v) ...@@ -589,7 +589,7 @@ static int show_stat(struct seq_file *p, void *v)
} }
#ifdef CONFIG_HAVE_SPARSE_IRQ #ifdef CONFIG_HAVE_SPARSE_IRQ
seq_printf(p, " %u:%u", j, per_irq_sum); seq_printf(p, " %#x:%u", j, per_irq_sum);
#else #else
seq_printf(p, " %u", per_irq_sum); seq_printf(p, " %u", per_irq_sum);
#endif #endif
......
...@@ -399,6 +399,7 @@ extern void set_irq_noprobe(unsigned int irq); ...@@ -399,6 +399,7 @@ extern void set_irq_noprobe(unsigned int irq);
extern void set_irq_probe(unsigned int irq); extern void set_irq_probe(unsigned int irq);
/* Handle dynamic irq creation and destruction */ /* Handle dynamic irq creation and destruction */
extern unsigned int create_irq_nr(unsigned int irq_want);
extern int create_irq(void); extern int create_irq(void);
extern void destroy_irq(unsigned int irq); extern void destroy_irq(unsigned int irq);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment