2 * PCIe RC driver for Synopsys DesignWare Core
4 * Copyright (C) 2015-2016 Synopsys, Inc. (www.synopsys.com)
6 * Authors: Joao Pinto <jpinto@synopsys.com>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
12 #include <linux/clk.h>
13 #include <linux/delay.h>
14 #include <linux/gpio.h>
15 #include <linux/interrupt.h>
16 #include <linux/kernel.h>
17 #include <linux/module.h>
18 #include <linux/of_gpio.h>
19 #include <linux/pci.h>
20 #include <linux/platform_device.h>
21 #include <linux/resource.h>
22 #include <linux/signal.h>
23 #include <linux/types.h>
25 #include "pcie-designware.h"
28 void __iomem
*mem_base
;
32 static irqreturn_t
dw_plat_pcie_msi_irq_handler(int irq
, void *arg
)
34 struct pcie_port
*pp
= arg
;
36 return dw_handle_msi_irq(pp
);
39 static void dw_plat_pcie_host_init(struct pcie_port
*pp
)
42 dw_pcie_wait_for_link(pp
);
44 if (IS_ENABLED(CONFIG_PCI_MSI
))
48 static struct pcie_host_ops dw_plat_pcie_host_ops
= {
49 .host_init
= dw_plat_pcie_host_init
,
52 static int dw_plat_add_pcie_port(struct pcie_port
*pp
,
53 struct platform_device
*pdev
)
57 pp
->irq
= platform_get_irq(pdev
, 1);
61 if (IS_ENABLED(CONFIG_PCI_MSI
)) {
62 pp
->msi_irq
= platform_get_irq(pdev
, 0);
66 ret
= devm_request_irq(&pdev
->dev
, pp
->msi_irq
,
67 dw_plat_pcie_msi_irq_handler
,
68 IRQF_SHARED
, "dw-plat-pcie-msi", pp
);
70 dev_err(&pdev
->dev
, "failed to request MSI IRQ\n");
76 pp
->ops
= &dw_plat_pcie_host_ops
;
78 ret
= dw_pcie_host_init(pp
);
80 dev_err(&pdev
->dev
, "failed to initialize host\n");
87 static int dw_plat_pcie_probe(struct platform_device
*pdev
)
89 struct dw_plat_pcie
*dw_plat_pcie
;
91 struct resource
*res
; /* Resource from DT */
94 dw_plat_pcie
= devm_kzalloc(&pdev
->dev
, sizeof(*dw_plat_pcie
),
99 pp
= &dw_plat_pcie
->pp
;
100 pp
->dev
= &pdev
->dev
;
102 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
106 dw_plat_pcie
->mem_base
= devm_ioremap_resource(&pdev
->dev
, res
);
107 if (IS_ERR(dw_plat_pcie
->mem_base
))
108 return PTR_ERR(dw_plat_pcie
->mem_base
);
110 pp
->dbi_base
= dw_plat_pcie
->mem_base
;
112 ret
= dw_plat_add_pcie_port(pp
, pdev
);
116 platform_set_drvdata(pdev
, dw_plat_pcie
);
120 static const struct of_device_id dw_plat_pcie_of_match
[] = {
121 { .compatible
= "snps,dw-pcie", },
124 MODULE_DEVICE_TABLE(of
, dw_plat_pcie_of_match
);
126 static struct platform_driver dw_plat_pcie_driver
= {
129 .of_match_table
= dw_plat_pcie_of_match
,
131 .probe
= dw_plat_pcie_probe
,
134 module_platform_driver(dw_plat_pcie_driver
);
136 MODULE_AUTHOR("Joao Pinto <Joao.Pinto@synopsys.com>");
137 MODULE_DESCRIPTION("Synopsys PCIe host controller glue platform driver");
138 MODULE_LICENSE("GPL v2");