Am Montag, den 31.03.2014, 11:30 +0100 schrieb Phil Edworthy: > Signed-off-by: Phil Edworthy <phil.edworthy@xxxxxxxxxxx> > Reviewed-by: Lucas Stach <l.stach@xxxxxxxxxxxxxx> > v6: > - Don't check MSI irq number is valid, as upper level checks this > - Change "Unexpected MSI" msg to debug level > - Reword "Unexpected MSI" comment so that it's one line > > v5: > - Return IRQ_NONE from MSI isr when there is no pending MSI > - Add additional interrupt bindings > --- > drivers/pci/host/pcie-rcar.c | 238 ++++++++++++++++++++++++++++++++++++++++++- > drivers/pci/host/pcie-rcar.h | 5 + > 2 files changed, 242 insertions(+), 1 deletion(-) > > diff --git a/drivers/pci/host/pcie-rcar.c b/drivers/pci/host/pcie-rcar.c > index c22c896..e3ce3d1 100644 > --- a/drivers/pci/host/pcie-rcar.c > +++ b/drivers/pci/host/pcie-rcar.c > @@ -15,8 +15,11 @@ > #include <linux/clk.h> > #include <linux/delay.h> > #include <linux/interrupt.h> > +#include <linux/irq.h> > +#include <linux/irqdomain.h> > #include <linux/kernel.h> > #include <linux/module.h> > +#include <linux/msi.h> > #include <linux/of_address.h> > #include <linux/of_irq.h> > #include <linux/of_pci.h> > @@ -28,6 +31,8 @@ > > #define DRV_NAME "rcar-pcie" > > +#define INT_PCI_MSI_NR 32 > + > #define RCONF(x) (PCICONF(0)+(x)) > #define RPMCAP(x) (PMCAP(0)+(x)) > #define REXPCAP(x) (EXPCAP(0)+(x)) > @@ -40,6 +45,21 @@ > #define PCI_MAX_RESOURCES 4 > #define MAX_NR_INBOUND_MAPS 6 > > +struct rcar_msi { > + DECLARE_BITMAP(used, INT_PCI_MSI_NR); > + struct irq_domain *domain; > + struct msi_chip chip; > + unsigned long pages; > + struct mutex lock; > + int irq1; > + int irq2; > +}; > + > +static inline struct rcar_msi *to_rcar_msi(struct msi_chip *chip) > +{ > + return container_of(chip, struct rcar_msi, chip); > +} > + > /* Structure representing the PCIe interface */ > struct rcar_pcie { > struct device *dev; > @@ -48,6 +68,7 @@ struct rcar_pcie { > u8 root_bus_nr; > struct clk *clk; > struct clk *bus_clk; > + struct rcar_msi msi; > }; > > static inline struct rcar_pcie *sys_to_pcie(struct pci_sys_data *sys) > @@ -292,6 +313,15 @@ static int rcar_pcie_setup(int nr, struct pci_sys_data *sys) > return 1; > } > > +static void rcar_pcie_add_bus(struct pci_bus *bus) > +{ > + if (IS_ENABLED(CONFIG_PCI_MSI)) { > + struct rcar_pcie *pcie = sys_to_pcie(bus->sysdata); > + > + bus->msi = &pcie->msi.chip; > + } > +} > + > static void __init rcar_pcie_enable(struct rcar_pcie *pcie) > { > struct platform_device *pdev = to_platform_device(pcie->dev); > @@ -301,6 +331,7 @@ static void __init rcar_pcie_enable(struct rcar_pcie *pcie) > .setup = rcar_pcie_setup, > .map_irq = of_irq_parse_and_map_pci, > .ops = &rcar_pcie_ops, > + .add_bus = rcar_pcie_add_bus, > }; > > pci_common_init_dev(&pdev->dev, &hw); > @@ -408,6 +439,10 @@ static int __init rcar_pcie_hw_init(struct rcar_pcie *pcie) > /* Enable MAC data scrambling. */ > rcar_rmw32(pcie, MACCTLR, SCRAMBLE_DISABLE, 0); > > + /* Enable MSI */ > + if (IS_ENABLED(CONFIG_PCI_MSI)) > + pci_write_reg(pcie, 0x101f0000, PCIEMSITXR); > + > /* Finish initialization - establish a PCI Express link */ > pci_write_reg(pcie, CFINIT, PCIETCTLR); > > @@ -461,11 +496,186 @@ static int __init rcar_pcie_hw_init_h1(struct rcar_pcie *pcie) > return -ETIMEDOUT; > } > > +static int rcar_msi_alloc(struct rcar_msi *chip) > +{ > + int msi; > + > + mutex_lock(&chip->lock); > + > + msi = find_first_zero_bit(chip->used, INT_PCI_MSI_NR); > + if (msi < INT_PCI_MSI_NR) > + set_bit(msi, chip->used); > + else > + msi = -ENOSPC; > + > + mutex_unlock(&chip->lock); > + > + return msi; > +} > + > +static void rcar_msi_free(struct rcar_msi *chip, unsigned long irq) > +{ > + struct device *dev = chip->chip.dev; > + > + mutex_lock(&chip->lock); > + clear_bit(irq, chip->used); > + mutex_unlock(&chip->lock); > +} > + > +static irqreturn_t rcar_pcie_msi_irq(int irq, void *data) > +{ > + struct rcar_pcie *pcie = data; > + struct rcar_msi *msi = &pcie->msi; > + unsigned long reg; > + > + reg = pci_read_reg(pcie, PCIEMSIFR); > + > + /* MSI & INTx share an interrupt - we only handle MSI here */ > + if (!reg) > + return IRQ_NONE; > + > + while (reg) { > + unsigned int index = find_first_bit(®, 32); > + unsigned int irq; > + > + /* clear the interrupt */ > + pci_write_reg(pcie, 1 << index, PCIEMSIFR); > + > + irq = irq_find_mapping(msi->domain, index); > + if (irq) { > + if (test_bit(index, msi->used)) > + generic_handle_irq(irq); > + else > + dev_info(pcie->dev, "unhandled MSI\n"); > + } else { > + /* Unknown MSI, just clear it */ > + dev_dbg(pcie->dev, "unexpected MSI\n"); > + } > + > + /* see if there's any more pending in this vector */ > + reg = pci_read_reg(pcie, PCIEMSIFR); > + } > + > + return IRQ_HANDLED; > +} > + > +static int rcar_msi_setup_irq(struct msi_chip *chip, struct pci_dev *pdev, > + struct msi_desc *desc) > +{ > + struct rcar_msi *msi = to_rcar_msi(chip); > + struct rcar_pcie *pcie = container_of(chip, struct rcar_pcie, msi.chip); > + struct msi_msg msg; > + unsigned int irq; > + int hwirq; > + > + hwirq = rcar_msi_alloc(msi); > + if (hwirq < 0) > + return hwirq; > + > + irq = irq_create_mapping(msi->domain, hwirq); > + if (!irq) { > + rcar_msi_free(msi, hwirq); > + return -EINVAL; > + } > + > + irq_set_msi_desc(irq, desc); > + > + msg.address_lo = pci_read_reg(pcie, PCIEMSIALR) & ~MSIFE; > + msg.address_hi = pci_read_reg(pcie, PCIEMSIAUR); > + msg.data = hwirq; > + > + write_msi_msg(irq, &msg); > + > + return 0; > +} > + > +static void rcar_msi_teardown_irq(struct msi_chip *chip, unsigned int irq) > +{ > + struct rcar_msi *msi = to_rcar_msi(chip); > + struct irq_data *d = irq_get_irq_data(irq); > + > + rcar_msi_free(msi, d->hwirq); > +} > + > +static struct irq_chip rcar_msi_irq_chip = { > + .name = "R-Car PCIe MSI", > + .irq_enable = unmask_msi_irq, > + .irq_disable = mask_msi_irq, > + .irq_mask = mask_msi_irq, > + .irq_unmask = unmask_msi_irq, > +}; > + > +static int rcar_msi_map(struct irq_domain *domain, unsigned int irq, > + irq_hw_number_t hwirq) > +{ > + irq_set_chip_and_handler(irq, &rcar_msi_irq_chip, handle_simple_irq); > + irq_set_chip_data(irq, domain->host_data); > + set_irq_flags(irq, IRQF_VALID); > + > + return 0; > +} > + > +static const struct irq_domain_ops msi_domain_ops = { > + .map = rcar_msi_map, > +}; > + > +static int rcar_pcie_enable_msi(struct rcar_pcie *pcie) > +{ > + struct platform_device *pdev = to_platform_device(pcie->dev); > + struct rcar_msi *msi = &pcie->msi; > + unsigned long base; > + int err; > + > + mutex_init(&msi->lock); > + > + msi->chip.dev = pcie->dev; > + msi->chip.setup_irq = rcar_msi_setup_irq; > + msi->chip.teardown_irq = rcar_msi_teardown_irq; > + > + msi->domain = irq_domain_add_linear(pcie->dev->of_node, INT_PCI_MSI_NR, > + &msi_domain_ops, &msi->chip); > + if (!msi->domain) { > + dev_err(&pdev->dev, "failed to create IRQ domain\n"); > + return -ENOMEM; > + } > + > + /* Two irqs are for MSI, but they are also used for non-MSI irqs */ > + err = devm_request_irq(&pdev->dev, msi->irq1, rcar_pcie_msi_irq, > + IRQF_SHARED, rcar_msi_irq_chip.name, pcie); > + if (err < 0) { > + dev_err(&pdev->dev, "failed to request IRQ: %d\n", err); > + goto err; > + } > + > + err = devm_request_irq(&pdev->dev, msi->irq2, rcar_pcie_msi_irq, > + IRQF_SHARED, rcar_msi_irq_chip.name, pcie); > + if (err < 0) { > + dev_err(&pdev->dev, "failed to request IRQ: %d\n", err); > + goto err; > + } > + > + /* setup MSI data target */ > + msi->pages = __get_free_pages(GFP_KERNEL, 0); > + base = virt_to_phys((void *)msi->pages); > + > + pci_write_reg(pcie, base | MSIFE, PCIEMSIALR); > + pci_write_reg(pcie, 0, PCIEMSIAUR); > + > + /* enable all MSI interrupts */ > + pci_write_reg(pcie, 0xffffffff, PCIEMSIIER); > + > + return 0; > + > +err: > + irq_domain_remove(msi->domain); > + return err; > +} > + > static int __init rcar_pcie_get_resources(struct platform_device *pdev, > struct rcar_pcie *pcie) > { > struct resource res; > - int err; > + int err, i; > > err = of_address_to_resource(pdev->dev.of_node, 0, &res); > if (err) > @@ -490,6 +700,22 @@ static int __init rcar_pcie_get_resources(struct platform_device *pdev, > if (err) > goto err_map_reg; > > + i = irq_of_parse_and_map(pdev->dev.of_node, 0); > + if (i < 0) { > + dev_err(pcie->dev, "cannot get platform resources for msi interrupt\n"); > + err = -ENOENT; > + goto err_map_reg; > + } > + pcie->msi.irq1 = i; > + > + i = irq_of_parse_and_map(pdev->dev.of_node, 1); > + if (i < 0) { > + dev_err(pcie->dev, "cannot get platform resources for msi interrupt\n"); > + err = -ENOENT; > + goto err_map_reg; > + } > + pcie->msi.irq2 = i; > + > pcie->base = devm_ioremap_resource(&pdev->dev, &res); > if (IS_ERR(pcie->base)) { > err = PTR_ERR(pcie->base); > @@ -657,6 +883,16 @@ static int __init rcar_pcie_probe(struct platform_device *pdev) > if (err) > return err; > > + if (IS_ENABLED(CONFIG_PCI_MSI)) { > + err = rcar_pcie_enable_msi(pcie); > + if (err < 0) { > + dev_err(&pdev->dev, > + "failed to enable MSI support: %d\n", > + err); > + return err; > + } > + } > + > of_id = of_match_device(rcar_pcie_of_match, pcie->dev); > if (!of_id || !of_id->data) > return -EINVAL; > diff --git a/drivers/pci/host/pcie-rcar.h b/drivers/pci/host/pcie-rcar.h > index 3dc026b..4f0c678 100644 > --- a/drivers/pci/host/pcie-rcar.h > +++ b/drivers/pci/host/pcie-rcar.h > @@ -13,6 +13,7 @@ > #define PCIEMSR 0x000028 > #define PCIEINTXR 0x000400 > #define PCIEPHYSR 0x0007f0 > +#define PCIEMSITXR 0x000840 > > /* Transfer control */ > #define PCIETCTLR 0x02000 > @@ -28,6 +29,10 @@ > #define PCIEPMSR 0x02034 > #define PCIEPMSCIER 0x02038 > #define PCIEMSIFR 0x02044 > +#define PCIEMSIALR 0x02048 > +#define MSIFE 1 > +#define PCIEMSIAUR 0x0204c > +#define PCIEMSIIER 0x02050 > > /* root port address */ > #define PCIEPRAR(x) (0x02080 + ((x) * 0x4)) -- Pengutronix e.K. | Lucas Stach | Industrial Linux Solutions | http://www.pengutronix.de/ | Peiner Str. 6-8, 31137 Hildesheim, Germany | Phone: +49-5121-206917-5076 | Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 | -- To unsubscribe from this list: send the line "unsubscribe linux-pci" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html