Commit dbed12da authored by Bjorn Helgaas's avatar Bjorn Helgaas Committed by Len Brown

[ACPI] PNPACPI IRQ workaround for HP workstations

Move pcibios_penalize_isa_irq() to pnpacpi_parse_allocated_irqresource().
Previously we passed the GSI, not the IRQ, and we did it even if parsing
the IRQ resource failed.

Parse IRQ descriptors that contain multiple interrupts.  This violates the
spec (in _CRS, only one interrupt per descriptor is allowed), but some
firmware, e.g., HP rx7620 and rx8620 descriptions of HPET, has this bug.
Signed-off-by: default avatarBjorn Helgaas <bjorn.helgaas@hp.com>
Cc: Adam Belay <ambx1@neo.rr.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent 5f0110f2
...@@ -73,25 +73,35 @@ static void decode_irq_flags(int flag, int *edge_level, int *active_high_low) ...@@ -73,25 +73,35 @@ static void decode_irq_flags(int flag, int *edge_level, int *active_high_low)
} }
static void static void
pnpacpi_parse_allocated_irqresource(struct pnp_resource_table * res, int irq) pnpacpi_parse_allocated_irqresource(struct pnp_resource_table * res, u32 gsi,
int edge_level, int active_high_low)
{ {
int i = 0; int i = 0;
int irq;
if (!valid_IRQ(gsi))
return;
while (!(res->irq_resource[i].flags & IORESOURCE_UNSET) && while (!(res->irq_resource[i].flags & IORESOURCE_UNSET) &&
i < PNP_MAX_IRQ) i < PNP_MAX_IRQ)
i++; i++;
if (i < PNP_MAX_IRQ) { if (i >= PNP_MAX_IRQ)
res->irq_resource[i].flags = IORESOURCE_IRQ; //Also clears _UNSET flag return;
res->irq_resource[i].flags = IORESOURCE_IRQ; // Also clears _UNSET flag
irq = acpi_register_gsi(gsi, edge_level, active_high_low);
if (irq < 0) { if (irq < 0) {
res->irq_resource[i].flags |= IORESOURCE_DISABLED; res->irq_resource[i].flags |= IORESOURCE_DISABLED;
return; return;
} }
res->irq_resource[i].start =(unsigned long) irq;
res->irq_resource[i].end = (unsigned long) irq; res->irq_resource[i].start = irq;
} res->irq_resource[i].end = irq;
pcibios_penalize_isa_irq(irq, 1);
} }
static void static void
pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table * res, int dma) pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table * res, u32 dma)
{ {
int i = 0; int i = 0;
while (i < PNP_MAX_DMA && while (i < PNP_MAX_DMA &&
...@@ -103,14 +113,14 @@ pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table * res, int dma) ...@@ -103,14 +113,14 @@ pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table * res, int dma)
res->dma_resource[i].flags |= IORESOURCE_DISABLED; res->dma_resource[i].flags |= IORESOURCE_DISABLED;
return; return;
} }
res->dma_resource[i].start =(unsigned long) dma; res->dma_resource[i].start = dma;
res->dma_resource[i].end = (unsigned long) dma; res->dma_resource[i].end = dma;
} }
} }
static void static void
pnpacpi_parse_allocated_ioresource(struct pnp_resource_table * res, pnpacpi_parse_allocated_ioresource(struct pnp_resource_table * res,
int io, int len) u32 io, u32 len)
{ {
int i = 0; int i = 0;
while (!(res->port_resource[i].flags & IORESOURCE_UNSET) && while (!(res->port_resource[i].flags & IORESOURCE_UNSET) &&
...@@ -122,14 +132,14 @@ pnpacpi_parse_allocated_ioresource(struct pnp_resource_table * res, ...@@ -122,14 +132,14 @@ pnpacpi_parse_allocated_ioresource(struct pnp_resource_table * res,
res->port_resource[i].flags |= IORESOURCE_DISABLED; res->port_resource[i].flags |= IORESOURCE_DISABLED;
return; return;
} }
res->port_resource[i].start = (unsigned long) io; res->port_resource[i].start = io;
res->port_resource[i].end = (unsigned long)(io + len - 1); res->port_resource[i].end = io + len - 1;
} }
} }
static void static void
pnpacpi_parse_allocated_memresource(struct pnp_resource_table * res, pnpacpi_parse_allocated_memresource(struct pnp_resource_table * res,
int mem, int len) u64 mem, u64 len)
{ {
int i = 0; int i = 0;
while (!(res->mem_resource[i].flags & IORESOURCE_UNSET) && while (!(res->mem_resource[i].flags & IORESOURCE_UNSET) &&
...@@ -141,8 +151,8 @@ pnpacpi_parse_allocated_memresource(struct pnp_resource_table * res, ...@@ -141,8 +151,8 @@ pnpacpi_parse_allocated_memresource(struct pnp_resource_table * res,
res->mem_resource[i].flags |= IORESOURCE_DISABLED; res->mem_resource[i].flags |= IORESOURCE_DISABLED;
return; return;
} }
res->mem_resource[i].start = (unsigned long) mem; res->mem_resource[i].start = mem;
res->mem_resource[i].end = (unsigned long)(mem + len - 1); res->mem_resource[i].end = mem + len - 1;
} }
} }
...@@ -151,27 +161,28 @@ static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, ...@@ -151,27 +161,28 @@ static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res,
void *data) void *data)
{ {
struct pnp_resource_table * res_table = (struct pnp_resource_table *)data; struct pnp_resource_table * res_table = (struct pnp_resource_table *)data;
int i;
switch (res->id) { switch (res->id) {
case ACPI_RSTYPE_IRQ: case ACPI_RSTYPE_IRQ:
if ((res->data.irq.number_of_interrupts > 0) && /*
valid_IRQ(res->data.irq.interrupts[0])) { * Per spec, only one interrupt per descriptor is allowed in
* _CRS, but some firmware violates this, so parse them all.
*/
for (i = 0; i < res->data.irq.number_of_interrupts; i++) {
pnpacpi_parse_allocated_irqresource(res_table, pnpacpi_parse_allocated_irqresource(res_table,
acpi_register_gsi(res->data.irq.interrupts[0], res->data.irq.interrupts[i],
res->data.irq.edge_level, res->data.irq.edge_level,
res->data.irq.active_high_low)); res->data.irq.active_high_low);
pcibios_penalize_isa_irq(res->data.irq.interrupts[0], 1);
} }
break; break;
case ACPI_RSTYPE_EXT_IRQ: case ACPI_RSTYPE_EXT_IRQ:
if ((res->data.extended_irq.number_of_interrupts > 0) && for (i = 0; i < res->data.extended_irq.number_of_interrupts; i++) {
valid_IRQ(res->data.extended_irq.interrupts[0])) {
pnpacpi_parse_allocated_irqresource(res_table, pnpacpi_parse_allocated_irqresource(res_table,
acpi_register_gsi(res->data.extended_irq.interrupts[0], res->data.extended_irq.interrupts[i],
res->data.extended_irq.edge_level, res->data.extended_irq.edge_level,
res->data.extended_irq.active_high_low)); res->data.extended_irq.active_high_low);
pcibios_penalize_isa_irq(res->data.extended_irq.interrupts[0], 1);
} }
break; break;
case ACPI_RSTYPE_DMA: case ACPI_RSTYPE_DMA:
......
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