aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/pci/host/pcie-tango.c
diff options
context:
space:
mode:
authorMarc Gonzalez <marc_gonzalez@sigmadesigns.com>2017-06-20 04:17:40 -0400
committerBjorn Helgaas <bhelgaas@google.com>2017-07-07 14:41:28 -0400
commit5e14e9fac308daf5607362f879e6de67e0b8dd5b (patch)
treefc4b9d4e7185344265cba0f6aad01747cad8400c /drivers/pci/host/pcie-tango.c
parentfdd620bd05e08572601bb89116727bb23cf7ce4e (diff)
PCI: tango: Add Sigma Designs Tango SMP8759 PCIe host bridge support
This driver is required to work around several hardware bugs in the PCIe controller. The SMP8759 does not support legacy interrupts or IO space. Signed-off-by: Marc Gonzalez <marc_gonzalez@sigmadesigns.com> [bhelgaas: add CONFIG_BROKEN dependency, various cleanups] Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Diffstat (limited to 'drivers/pci/host/pcie-tango.c')
-rw-r--r--drivers/pci/host/pcie-tango.c141
1 files changed, 141 insertions, 0 deletions
diff --git a/drivers/pci/host/pcie-tango.c b/drivers/pci/host/pcie-tango.c
new file mode 100644
index 000000000000..6bbb81f06a53
--- /dev/null
+++ b/drivers/pci/host/pcie-tango.c
@@ -0,0 +1,141 @@
1#include <linux/pci-ecam.h>
2#include <linux/delay.h>
3#include <linux/of.h>
4
5#define SMP8759_MUX 0x48
6#define SMP8759_TEST_OUT 0x74
7
8struct tango_pcie {
9 void __iomem *base;
10};
11
12static int smp8759_config_read(struct pci_bus *bus, unsigned int devfn,
13 int where, int size, u32 *val)
14{
15 struct pci_config_window *cfg = bus->sysdata;
16 struct tango_pcie *pcie = dev_get_drvdata(cfg->parent);
17 int ret;
18
19 /* Reads in configuration space outside devfn 0 return garbage */
20 if (devfn != 0)
21 return PCIBIOS_FUNC_NOT_SUPPORTED;
22
23 /*
24 * PCI config and MMIO accesses are muxed. Linux doesn't have a
25 * mutual exclusion mechanism for config vs. MMIO accesses, so
26 * concurrent accesses may cause corruption.
27 */
28 writel_relaxed(1, pcie->base + SMP8759_MUX);
29 ret = pci_generic_config_read(bus, devfn, where, size, val);
30 writel_relaxed(0, pcie->base + SMP8759_MUX);
31
32 return ret;
33}
34
35static int smp8759_config_write(struct pci_bus *bus, unsigned int devfn,
36 int where, int size, u32 val)
37{
38 struct pci_config_window *cfg = bus->sysdata;
39 struct tango_pcie *pcie = dev_get_drvdata(cfg->parent);
40 int ret;
41
42 writel_relaxed(1, pcie->base + SMP8759_MUX);
43 ret = pci_generic_config_write(bus, devfn, where, size, val);
44 writel_relaxed(0, pcie->base + SMP8759_MUX);
45
46 return ret;
47}
48
49static struct pci_ecam_ops smp8759_ecam_ops = {
50 .bus_shift = 20,
51 .pci_ops = {
52 .map_bus = pci_ecam_map_bus,
53 .read = smp8759_config_read,
54 .write = smp8759_config_write,
55 }
56};
57
58static int tango_pcie_link_up(struct tango_pcie *pcie)
59{
60 void __iomem *test_out = pcie->base + SMP8759_TEST_OUT;
61 int i;
62
63 writel_relaxed(16, test_out);
64 for (i = 0; i < 10; ++i) {
65 u32 ltssm_state = readl_relaxed(test_out) >> 8;
66 if ((ltssm_state & 0x1f) == 0xf) /* L0 */
67 return 1;
68 usleep_range(3000, 4000);
69 }
70
71 return 0;
72}
73
74static int tango_pcie_probe(struct platform_device *pdev)
75{
76 struct device *dev = &pdev->dev;
77 struct tango_pcie *pcie;
78 struct resource *res;
79 int ret;
80
81 dev_warn(dev, "simultaneous PCI config and MMIO accesses may cause data corruption\n");
82 add_taint(TAINT_CRAP, LOCKDEP_STILL_OK);
83
84 pcie = devm_kzalloc(dev, sizeof(*pcie), GFP_KERNEL);
85 if (!pcie)
86 return -ENOMEM;
87
88 res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
89 pcie->base = devm_ioremap_resource(dev, res);
90 if (IS_ERR(pcie->base))
91 return PTR_ERR(pcie->base);
92
93 platform_set_drvdata(pdev, pcie);
94
95 if (!tango_pcie_link_up(pcie))
96 return -ENODEV;
97
98 return pci_host_common_probe(pdev, &smp8759_ecam_ops);
99}
100
101static const struct of_device_id tango_pcie_ids[] = {
102 { .compatible = "sigma,smp8759-pcie" },
103 { },
104};
105
106static struct platform_driver tango_pcie_driver = {
107 .probe = tango_pcie_probe,
108 .driver = {
109 .name = KBUILD_MODNAME,
110 .of_match_table = tango_pcie_ids,
111 .suppress_bind_attrs = true,
112 },
113};
114builtin_platform_driver(tango_pcie_driver);
115
116/*
117 * The root complex advertises the wrong device class.
118 * Header Type 1 is for PCI-to-PCI bridges.
119 */
120static void tango_fixup_class(struct pci_dev *dev)
121{
122 dev->class = PCI_CLASS_BRIDGE_PCI << 8;
123}
124DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_SIGMA, 0x0024, tango_fixup_class);
125DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_SIGMA, 0x0028, tango_fixup_class);
126
127/*
128 * The root complex exposes a "fake" BAR, which is used to filter
129 * bus-to-system accesses. Only accesses within the range defined by this
130 * BAR are forwarded to the host, others are ignored.
131 *
132 * By default, the DMA framework expects an identity mapping, and DRAM0 is
133 * mapped at 0x80000000.
134 */
135static void tango_fixup_bar(struct pci_dev *dev)
136{
137 dev->non_compliant_bars = true;
138 pci_write_config_dword(dev, PCI_BASE_ADDRESS_0, 0x80000000);
139}
140DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_SIGMA, 0x0024, tango_fixup_bar);
141DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_SIGMA, 0x0028, tango_fixup_bar);