Merge branch 'remotes/lorenzo/pci/microchip'
- Add missing semicolon after MODULE_DEVICE_TABLE() (Uwe Kleine-König) - Add chained_irq_enter()/chained_irq_exit() calls to mc_handle_msi() and mc_handle_intx() to avoid lost interrupts (Conor Dooley) - Fix interrupt handling race (Daire McNamara) * remotes/lorenzo/pci/microchip: PCI: microchip: Fix potential race in interrupt handling PCI: microchip: Add missing chained_irq_enter()/exit() calls PCI: microchip: Add a missing semicolon
This commit is contained in:
commit
5ae34e8f9e
@ -406,6 +406,7 @@ static void mc_pcie_enable_msi(struct mc_pcie *port, void __iomem *base)
|
|||||||
static void mc_handle_msi(struct irq_desc *desc)
|
static void mc_handle_msi(struct irq_desc *desc)
|
||||||
{
|
{
|
||||||
struct mc_pcie *port = irq_desc_get_handler_data(desc);
|
struct mc_pcie *port = irq_desc_get_handler_data(desc);
|
||||||
|
struct irq_chip *chip = irq_desc_get_chip(desc);
|
||||||
struct device *dev = port->dev;
|
struct device *dev = port->dev;
|
||||||
struct mc_msi *msi = &port->msi;
|
struct mc_msi *msi = &port->msi;
|
||||||
void __iomem *bridge_base_addr =
|
void __iomem *bridge_base_addr =
|
||||||
@ -414,8 +415,11 @@ static void mc_handle_msi(struct irq_desc *desc)
|
|||||||
u32 bit;
|
u32 bit;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
chained_irq_enter(chip, desc);
|
||||||
|
|
||||||
status = readl_relaxed(bridge_base_addr + ISTATUS_LOCAL);
|
status = readl_relaxed(bridge_base_addr + ISTATUS_LOCAL);
|
||||||
if (status & PM_MSI_INT_MSI_MASK) {
|
if (status & PM_MSI_INT_MSI_MASK) {
|
||||||
|
writel_relaxed(status & PM_MSI_INT_MSI_MASK, bridge_base_addr + ISTATUS_LOCAL);
|
||||||
status = readl_relaxed(bridge_base_addr + ISTATUS_MSI);
|
status = readl_relaxed(bridge_base_addr + ISTATUS_MSI);
|
||||||
for_each_set_bit(bit, &status, msi->num_vectors) {
|
for_each_set_bit(bit, &status, msi->num_vectors) {
|
||||||
ret = generic_handle_domain_irq(msi->dev_domain, bit);
|
ret = generic_handle_domain_irq(msi->dev_domain, bit);
|
||||||
@ -424,6 +428,8 @@ static void mc_handle_msi(struct irq_desc *desc)
|
|||||||
bit);
|
bit);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
chained_irq_exit(chip, desc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mc_msi_bottom_irq_ack(struct irq_data *data)
|
static void mc_msi_bottom_irq_ack(struct irq_data *data)
|
||||||
@ -432,13 +438,8 @@ static void mc_msi_bottom_irq_ack(struct irq_data *data)
|
|||||||
void __iomem *bridge_base_addr =
|
void __iomem *bridge_base_addr =
|
||||||
port->axi_base_addr + MC_PCIE_BRIDGE_ADDR;
|
port->axi_base_addr + MC_PCIE_BRIDGE_ADDR;
|
||||||
u32 bitpos = data->hwirq;
|
u32 bitpos = data->hwirq;
|
||||||
unsigned long status;
|
|
||||||
|
|
||||||
writel_relaxed(BIT(bitpos), bridge_base_addr + ISTATUS_MSI);
|
writel_relaxed(BIT(bitpos), bridge_base_addr + ISTATUS_MSI);
|
||||||
status = readl_relaxed(bridge_base_addr + ISTATUS_MSI);
|
|
||||||
if (!status)
|
|
||||||
writel_relaxed(BIT(PM_MSI_INT_MSI_SHIFT),
|
|
||||||
bridge_base_addr + ISTATUS_LOCAL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mc_compose_msi_msg(struct irq_data *data, struct msi_msg *msg)
|
static void mc_compose_msi_msg(struct irq_data *data, struct msi_msg *msg)
|
||||||
@ -563,6 +564,7 @@ static int mc_allocate_msi_domains(struct mc_pcie *port)
|
|||||||
static void mc_handle_intx(struct irq_desc *desc)
|
static void mc_handle_intx(struct irq_desc *desc)
|
||||||
{
|
{
|
||||||
struct mc_pcie *port = irq_desc_get_handler_data(desc);
|
struct mc_pcie *port = irq_desc_get_handler_data(desc);
|
||||||
|
struct irq_chip *chip = irq_desc_get_chip(desc);
|
||||||
struct device *dev = port->dev;
|
struct device *dev = port->dev;
|
||||||
void __iomem *bridge_base_addr =
|
void __iomem *bridge_base_addr =
|
||||||
port->axi_base_addr + MC_PCIE_BRIDGE_ADDR;
|
port->axi_base_addr + MC_PCIE_BRIDGE_ADDR;
|
||||||
@ -570,6 +572,8 @@ static void mc_handle_intx(struct irq_desc *desc)
|
|||||||
u32 bit;
|
u32 bit;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
chained_irq_enter(chip, desc);
|
||||||
|
|
||||||
status = readl_relaxed(bridge_base_addr + ISTATUS_LOCAL);
|
status = readl_relaxed(bridge_base_addr + ISTATUS_LOCAL);
|
||||||
if (status & PM_MSI_INT_INTX_MASK) {
|
if (status & PM_MSI_INT_INTX_MASK) {
|
||||||
status &= PM_MSI_INT_INTX_MASK;
|
status &= PM_MSI_INT_INTX_MASK;
|
||||||
@ -581,6 +585,8 @@ static void mc_handle_intx(struct irq_desc *desc)
|
|||||||
bit);
|
bit);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
chained_irq_exit(chip, desc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mc_ack_intx_irq(struct irq_data *data)
|
static void mc_ack_intx_irq(struct irq_data *data)
|
||||||
@ -1115,7 +1121,7 @@ static const struct of_device_id mc_pcie_of_match[] = {
|
|||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
|
|
||||||
MODULE_DEVICE_TABLE(of, mc_pcie_of_match)
|
MODULE_DEVICE_TABLE(of, mc_pcie_of_match);
|
||||||
|
|
||||||
static struct platform_driver mc_pcie_driver = {
|
static struct platform_driver mc_pcie_driver = {
|
||||||
.probe = pci_host_common_probe,
|
.probe = pci_host_common_probe,
|
||||||
|
Loading…
Reference in New Issue
Block a user