1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun /**
3*4882a593Smuzhiyun * APM X-Gene PCIe Driver
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (c) 2014 Applied Micro Circuits Corporation.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Author: Tanmay Inamdar <tinamdar@apm.com>.
8*4882a593Smuzhiyun */
9*4882a593Smuzhiyun #include <linux/clk.h>
10*4882a593Smuzhiyun #include <linux/delay.h>
11*4882a593Smuzhiyun #include <linux/io.h>
12*4882a593Smuzhiyun #include <linux/jiffies.h>
13*4882a593Smuzhiyun #include <linux/memblock.h>
14*4882a593Smuzhiyun #include <linux/init.h>
15*4882a593Smuzhiyun #include <linux/of.h>
16*4882a593Smuzhiyun #include <linux/of_address.h>
17*4882a593Smuzhiyun #include <linux/of_irq.h>
18*4882a593Smuzhiyun #include <linux/of_pci.h>
19*4882a593Smuzhiyun #include <linux/pci.h>
20*4882a593Smuzhiyun #include <linux/pci-acpi.h>
21*4882a593Smuzhiyun #include <linux/pci-ecam.h>
22*4882a593Smuzhiyun #include <linux/platform_device.h>
23*4882a593Smuzhiyun #include <linux/slab.h>
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun #include "../pci.h"
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #define PCIECORE_CTLANDSTATUS 0x50
28*4882a593Smuzhiyun #define PIM1_1L 0x80
29*4882a593Smuzhiyun #define IBAR2 0x98
30*4882a593Smuzhiyun #define IR2MSK 0x9c
31*4882a593Smuzhiyun #define PIM2_1L 0xa0
32*4882a593Smuzhiyun #define IBAR3L 0xb4
33*4882a593Smuzhiyun #define IR3MSKL 0xbc
34*4882a593Smuzhiyun #define PIM3_1L 0xc4
35*4882a593Smuzhiyun #define OMR1BARL 0x100
36*4882a593Smuzhiyun #define OMR2BARL 0x118
37*4882a593Smuzhiyun #define OMR3BARL 0x130
38*4882a593Smuzhiyun #define CFGBARL 0x154
39*4882a593Smuzhiyun #define CFGBARH 0x158
40*4882a593Smuzhiyun #define CFGCTL 0x15c
41*4882a593Smuzhiyun #define RTDID 0x160
42*4882a593Smuzhiyun #define BRIDGE_CFG_0 0x2000
43*4882a593Smuzhiyun #define BRIDGE_CFG_4 0x2010
44*4882a593Smuzhiyun #define BRIDGE_STATUS_0 0x2600
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun #define LINK_UP_MASK 0x00000100
47*4882a593Smuzhiyun #define AXI_EP_CFG_ACCESS 0x10000
48*4882a593Smuzhiyun #define EN_COHERENCY 0xF0000000
49*4882a593Smuzhiyun #define EN_REG 0x00000001
50*4882a593Smuzhiyun #define OB_LO_IO 0x00000002
51*4882a593Smuzhiyun #define XGENE_PCIE_VENDORID 0x10E8
52*4882a593Smuzhiyun #define XGENE_PCIE_DEVICEID 0xE004
53*4882a593Smuzhiyun #define SZ_1T (SZ_1G*1024ULL)
54*4882a593Smuzhiyun #define PIPE_PHY_RATE_RD(src) ((0xc000 & (u32)(src)) >> 0xe)
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun #define XGENE_V1_PCI_EXP_CAP 0x40
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun /* PCIe IP version */
59*4882a593Smuzhiyun #define XGENE_PCIE_IP_VER_UNKN 0
60*4882a593Smuzhiyun #define XGENE_PCIE_IP_VER_1 1
61*4882a593Smuzhiyun #define XGENE_PCIE_IP_VER_2 2
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun #if defined(CONFIG_PCI_XGENE) || (defined(CONFIG_ACPI) && defined(CONFIG_PCI_QUIRKS))
64*4882a593Smuzhiyun struct xgene_pcie_port {
65*4882a593Smuzhiyun struct device_node *node;
66*4882a593Smuzhiyun struct device *dev;
67*4882a593Smuzhiyun struct clk *clk;
68*4882a593Smuzhiyun void __iomem *csr_base;
69*4882a593Smuzhiyun void __iomem *cfg_base;
70*4882a593Smuzhiyun unsigned long cfg_addr;
71*4882a593Smuzhiyun bool link_up;
72*4882a593Smuzhiyun u32 version;
73*4882a593Smuzhiyun };
74*4882a593Smuzhiyun
xgene_pcie_readl(struct xgene_pcie_port * port,u32 reg)75*4882a593Smuzhiyun static u32 xgene_pcie_readl(struct xgene_pcie_port *port, u32 reg)
76*4882a593Smuzhiyun {
77*4882a593Smuzhiyun return readl(port->csr_base + reg);
78*4882a593Smuzhiyun }
79*4882a593Smuzhiyun
xgene_pcie_writel(struct xgene_pcie_port * port,u32 reg,u32 val)80*4882a593Smuzhiyun static void xgene_pcie_writel(struct xgene_pcie_port *port, u32 reg, u32 val)
81*4882a593Smuzhiyun {
82*4882a593Smuzhiyun writel(val, port->csr_base + reg);
83*4882a593Smuzhiyun }
84*4882a593Smuzhiyun
pcie_bar_low_val(u32 addr,u32 flags)85*4882a593Smuzhiyun static inline u32 pcie_bar_low_val(u32 addr, u32 flags)
86*4882a593Smuzhiyun {
87*4882a593Smuzhiyun return (addr & PCI_BASE_ADDRESS_MEM_MASK) | flags;
88*4882a593Smuzhiyun }
89*4882a593Smuzhiyun
pcie_bus_to_port(struct pci_bus * bus)90*4882a593Smuzhiyun static inline struct xgene_pcie_port *pcie_bus_to_port(struct pci_bus *bus)
91*4882a593Smuzhiyun {
92*4882a593Smuzhiyun struct pci_config_window *cfg;
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun if (acpi_disabled)
95*4882a593Smuzhiyun return (struct xgene_pcie_port *)(bus->sysdata);
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun cfg = bus->sysdata;
98*4882a593Smuzhiyun return (struct xgene_pcie_port *)(cfg->priv);
99*4882a593Smuzhiyun }
100*4882a593Smuzhiyun
101*4882a593Smuzhiyun /*
102*4882a593Smuzhiyun * When the address bit [17:16] is 2'b01, the Configuration access will be
103*4882a593Smuzhiyun * treated as Type 1 and it will be forwarded to external PCIe device.
104*4882a593Smuzhiyun */
xgene_pcie_get_cfg_base(struct pci_bus * bus)105*4882a593Smuzhiyun static void __iomem *xgene_pcie_get_cfg_base(struct pci_bus *bus)
106*4882a593Smuzhiyun {
107*4882a593Smuzhiyun struct xgene_pcie_port *port = pcie_bus_to_port(bus);
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun if (bus->number >= (bus->primary + 1))
110*4882a593Smuzhiyun return port->cfg_base + AXI_EP_CFG_ACCESS;
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun return port->cfg_base;
113*4882a593Smuzhiyun }
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun /*
116*4882a593Smuzhiyun * For Configuration request, RTDID register is used as Bus Number,
117*4882a593Smuzhiyun * Device Number and Function number of the header fields.
118*4882a593Smuzhiyun */
xgene_pcie_set_rtdid_reg(struct pci_bus * bus,uint devfn)119*4882a593Smuzhiyun static void xgene_pcie_set_rtdid_reg(struct pci_bus *bus, uint devfn)
120*4882a593Smuzhiyun {
121*4882a593Smuzhiyun struct xgene_pcie_port *port = pcie_bus_to_port(bus);
122*4882a593Smuzhiyun unsigned int b, d, f;
123*4882a593Smuzhiyun u32 rtdid_val = 0;
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun b = bus->number;
126*4882a593Smuzhiyun d = PCI_SLOT(devfn);
127*4882a593Smuzhiyun f = PCI_FUNC(devfn);
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun if (!pci_is_root_bus(bus))
130*4882a593Smuzhiyun rtdid_val = (b << 8) | (d << 3) | f;
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun xgene_pcie_writel(port, RTDID, rtdid_val);
133*4882a593Smuzhiyun /* read the register back to ensure flush */
134*4882a593Smuzhiyun xgene_pcie_readl(port, RTDID);
135*4882a593Smuzhiyun }
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun /*
138*4882a593Smuzhiyun * X-Gene PCIe port uses BAR0-BAR1 of RC's configuration space as
139*4882a593Smuzhiyun * the translation from PCI bus to native BUS. Entire DDR region
140*4882a593Smuzhiyun * is mapped into PCIe space using these registers, so it can be
141*4882a593Smuzhiyun * reached by DMA from EP devices. The BAR0/1 of bridge should be
142*4882a593Smuzhiyun * hidden during enumeration to avoid the sizing and resource allocation
143*4882a593Smuzhiyun * by PCIe core.
144*4882a593Smuzhiyun */
xgene_pcie_hide_rc_bars(struct pci_bus * bus,int offset)145*4882a593Smuzhiyun static bool xgene_pcie_hide_rc_bars(struct pci_bus *bus, int offset)
146*4882a593Smuzhiyun {
147*4882a593Smuzhiyun if (pci_is_root_bus(bus) && ((offset == PCI_BASE_ADDRESS_0) ||
148*4882a593Smuzhiyun (offset == PCI_BASE_ADDRESS_1)))
149*4882a593Smuzhiyun return true;
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun return false;
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun
xgene_pcie_map_bus(struct pci_bus * bus,unsigned int devfn,int offset)154*4882a593Smuzhiyun static void __iomem *xgene_pcie_map_bus(struct pci_bus *bus, unsigned int devfn,
155*4882a593Smuzhiyun int offset)
156*4882a593Smuzhiyun {
157*4882a593Smuzhiyun if ((pci_is_root_bus(bus) && devfn != 0) ||
158*4882a593Smuzhiyun xgene_pcie_hide_rc_bars(bus, offset))
159*4882a593Smuzhiyun return NULL;
160*4882a593Smuzhiyun
161*4882a593Smuzhiyun xgene_pcie_set_rtdid_reg(bus, devfn);
162*4882a593Smuzhiyun return xgene_pcie_get_cfg_base(bus) + offset;
163*4882a593Smuzhiyun }
164*4882a593Smuzhiyun
xgene_pcie_config_read32(struct pci_bus * bus,unsigned int devfn,int where,int size,u32 * val)165*4882a593Smuzhiyun static int xgene_pcie_config_read32(struct pci_bus *bus, unsigned int devfn,
166*4882a593Smuzhiyun int where, int size, u32 *val)
167*4882a593Smuzhiyun {
168*4882a593Smuzhiyun struct xgene_pcie_port *port = pcie_bus_to_port(bus);
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun if (pci_generic_config_read32(bus, devfn, where & ~0x3, 4, val) !=
171*4882a593Smuzhiyun PCIBIOS_SUCCESSFUL)
172*4882a593Smuzhiyun return PCIBIOS_DEVICE_NOT_FOUND;
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun /*
175*4882a593Smuzhiyun * The v1 controller has a bug in its Configuration Request
176*4882a593Smuzhiyun * Retry Status (CRS) logic: when CRS is enabled and we read the
177*4882a593Smuzhiyun * Vendor and Device ID of a non-existent device, the controller
178*4882a593Smuzhiyun * fabricates return data of 0xFFFF0001 ("device exists but is not
179*4882a593Smuzhiyun * ready") instead of 0xFFFFFFFF ("device does not exist"). This
180*4882a593Smuzhiyun * causes the PCI core to retry the read until it times out.
181*4882a593Smuzhiyun * Avoid this by not claiming to support CRS.
182*4882a593Smuzhiyun */
183*4882a593Smuzhiyun if (pci_is_root_bus(bus) && (port->version == XGENE_PCIE_IP_VER_1) &&
184*4882a593Smuzhiyun ((where & ~0x3) == XGENE_V1_PCI_EXP_CAP + PCI_EXP_RTCTL))
185*4882a593Smuzhiyun *val &= ~(PCI_EXP_RTCAP_CRSVIS << 16);
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun if (size <= 2)
188*4882a593Smuzhiyun *val = (*val >> (8 * (where & 3))) & ((1 << (size * 8)) - 1);
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun return PCIBIOS_SUCCESSFUL;
191*4882a593Smuzhiyun }
192*4882a593Smuzhiyun #endif
193*4882a593Smuzhiyun
194*4882a593Smuzhiyun #if defined(CONFIG_ACPI) && defined(CONFIG_PCI_QUIRKS)
xgene_get_csr_resource(struct acpi_device * adev,struct resource * res)195*4882a593Smuzhiyun static int xgene_get_csr_resource(struct acpi_device *adev,
196*4882a593Smuzhiyun struct resource *res)
197*4882a593Smuzhiyun {
198*4882a593Smuzhiyun struct device *dev = &adev->dev;
199*4882a593Smuzhiyun struct resource_entry *entry;
200*4882a593Smuzhiyun struct list_head list;
201*4882a593Smuzhiyun unsigned long flags;
202*4882a593Smuzhiyun int ret;
203*4882a593Smuzhiyun
204*4882a593Smuzhiyun INIT_LIST_HEAD(&list);
205*4882a593Smuzhiyun flags = IORESOURCE_MEM;
206*4882a593Smuzhiyun ret = acpi_dev_get_resources(adev, &list,
207*4882a593Smuzhiyun acpi_dev_filter_resource_type_cb,
208*4882a593Smuzhiyun (void *) flags);
209*4882a593Smuzhiyun if (ret < 0) {
210*4882a593Smuzhiyun dev_err(dev, "failed to parse _CRS method, error code %d\n",
211*4882a593Smuzhiyun ret);
212*4882a593Smuzhiyun return ret;
213*4882a593Smuzhiyun }
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun if (ret == 0) {
216*4882a593Smuzhiyun dev_err(dev, "no IO and memory resources present in _CRS\n");
217*4882a593Smuzhiyun return -EINVAL;
218*4882a593Smuzhiyun }
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun entry = list_first_entry(&list, struct resource_entry, node);
221*4882a593Smuzhiyun *res = *entry->res;
222*4882a593Smuzhiyun acpi_dev_free_resource_list(&list);
223*4882a593Smuzhiyun return 0;
224*4882a593Smuzhiyun }
225*4882a593Smuzhiyun
xgene_pcie_ecam_init(struct pci_config_window * cfg,u32 ipversion)226*4882a593Smuzhiyun static int xgene_pcie_ecam_init(struct pci_config_window *cfg, u32 ipversion)
227*4882a593Smuzhiyun {
228*4882a593Smuzhiyun struct device *dev = cfg->parent;
229*4882a593Smuzhiyun struct acpi_device *adev = to_acpi_device(dev);
230*4882a593Smuzhiyun struct xgene_pcie_port *port;
231*4882a593Smuzhiyun struct resource csr;
232*4882a593Smuzhiyun int ret;
233*4882a593Smuzhiyun
234*4882a593Smuzhiyun port = devm_kzalloc(dev, sizeof(*port), GFP_KERNEL);
235*4882a593Smuzhiyun if (!port)
236*4882a593Smuzhiyun return -ENOMEM;
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun ret = xgene_get_csr_resource(adev, &csr);
239*4882a593Smuzhiyun if (ret) {
240*4882a593Smuzhiyun dev_err(dev, "can't get CSR resource\n");
241*4882a593Smuzhiyun return ret;
242*4882a593Smuzhiyun }
243*4882a593Smuzhiyun port->csr_base = devm_pci_remap_cfg_resource(dev, &csr);
244*4882a593Smuzhiyun if (IS_ERR(port->csr_base))
245*4882a593Smuzhiyun return PTR_ERR(port->csr_base);
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun port->cfg_base = cfg->win;
248*4882a593Smuzhiyun port->version = ipversion;
249*4882a593Smuzhiyun
250*4882a593Smuzhiyun cfg->priv = port;
251*4882a593Smuzhiyun return 0;
252*4882a593Smuzhiyun }
253*4882a593Smuzhiyun
xgene_v1_pcie_ecam_init(struct pci_config_window * cfg)254*4882a593Smuzhiyun static int xgene_v1_pcie_ecam_init(struct pci_config_window *cfg)
255*4882a593Smuzhiyun {
256*4882a593Smuzhiyun return xgene_pcie_ecam_init(cfg, XGENE_PCIE_IP_VER_1);
257*4882a593Smuzhiyun }
258*4882a593Smuzhiyun
259*4882a593Smuzhiyun const struct pci_ecam_ops xgene_v1_pcie_ecam_ops = {
260*4882a593Smuzhiyun .bus_shift = 16,
261*4882a593Smuzhiyun .init = xgene_v1_pcie_ecam_init,
262*4882a593Smuzhiyun .pci_ops = {
263*4882a593Smuzhiyun .map_bus = xgene_pcie_map_bus,
264*4882a593Smuzhiyun .read = xgene_pcie_config_read32,
265*4882a593Smuzhiyun .write = pci_generic_config_write,
266*4882a593Smuzhiyun }
267*4882a593Smuzhiyun };
268*4882a593Smuzhiyun
xgene_v2_pcie_ecam_init(struct pci_config_window * cfg)269*4882a593Smuzhiyun static int xgene_v2_pcie_ecam_init(struct pci_config_window *cfg)
270*4882a593Smuzhiyun {
271*4882a593Smuzhiyun return xgene_pcie_ecam_init(cfg, XGENE_PCIE_IP_VER_2);
272*4882a593Smuzhiyun }
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun const struct pci_ecam_ops xgene_v2_pcie_ecam_ops = {
275*4882a593Smuzhiyun .bus_shift = 16,
276*4882a593Smuzhiyun .init = xgene_v2_pcie_ecam_init,
277*4882a593Smuzhiyun .pci_ops = {
278*4882a593Smuzhiyun .map_bus = xgene_pcie_map_bus,
279*4882a593Smuzhiyun .read = xgene_pcie_config_read32,
280*4882a593Smuzhiyun .write = pci_generic_config_write,
281*4882a593Smuzhiyun }
282*4882a593Smuzhiyun };
283*4882a593Smuzhiyun #endif
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun #if defined(CONFIG_PCI_XGENE)
xgene_pcie_set_ib_mask(struct xgene_pcie_port * port,u32 addr,u32 flags,u64 size)286*4882a593Smuzhiyun static u64 xgene_pcie_set_ib_mask(struct xgene_pcie_port *port, u32 addr,
287*4882a593Smuzhiyun u32 flags, u64 size)
288*4882a593Smuzhiyun {
289*4882a593Smuzhiyun u64 mask = (~(size - 1) & PCI_BASE_ADDRESS_MEM_MASK) | flags;
290*4882a593Smuzhiyun u32 val32 = 0;
291*4882a593Smuzhiyun u32 val;
292*4882a593Smuzhiyun
293*4882a593Smuzhiyun val32 = xgene_pcie_readl(port, addr);
294*4882a593Smuzhiyun val = (val32 & 0x0000ffff) | (lower_32_bits(mask) << 16);
295*4882a593Smuzhiyun xgene_pcie_writel(port, addr, val);
296*4882a593Smuzhiyun
297*4882a593Smuzhiyun val32 = xgene_pcie_readl(port, addr + 0x04);
298*4882a593Smuzhiyun val = (val32 & 0xffff0000) | (lower_32_bits(mask) >> 16);
299*4882a593Smuzhiyun xgene_pcie_writel(port, addr + 0x04, val);
300*4882a593Smuzhiyun
301*4882a593Smuzhiyun val32 = xgene_pcie_readl(port, addr + 0x04);
302*4882a593Smuzhiyun val = (val32 & 0x0000ffff) | (upper_32_bits(mask) << 16);
303*4882a593Smuzhiyun xgene_pcie_writel(port, addr + 0x04, val);
304*4882a593Smuzhiyun
305*4882a593Smuzhiyun val32 = xgene_pcie_readl(port, addr + 0x08);
306*4882a593Smuzhiyun val = (val32 & 0xffff0000) | (upper_32_bits(mask) >> 16);
307*4882a593Smuzhiyun xgene_pcie_writel(port, addr + 0x08, val);
308*4882a593Smuzhiyun
309*4882a593Smuzhiyun return mask;
310*4882a593Smuzhiyun }
311*4882a593Smuzhiyun
xgene_pcie_linkup(struct xgene_pcie_port * port,u32 * lanes,u32 * speed)312*4882a593Smuzhiyun static void xgene_pcie_linkup(struct xgene_pcie_port *port,
313*4882a593Smuzhiyun u32 *lanes, u32 *speed)
314*4882a593Smuzhiyun {
315*4882a593Smuzhiyun u32 val32;
316*4882a593Smuzhiyun
317*4882a593Smuzhiyun port->link_up = false;
318*4882a593Smuzhiyun val32 = xgene_pcie_readl(port, PCIECORE_CTLANDSTATUS);
319*4882a593Smuzhiyun if (val32 & LINK_UP_MASK) {
320*4882a593Smuzhiyun port->link_up = true;
321*4882a593Smuzhiyun *speed = PIPE_PHY_RATE_RD(val32);
322*4882a593Smuzhiyun val32 = xgene_pcie_readl(port, BRIDGE_STATUS_0);
323*4882a593Smuzhiyun *lanes = val32 >> 26;
324*4882a593Smuzhiyun }
325*4882a593Smuzhiyun }
326*4882a593Smuzhiyun
xgene_pcie_init_port(struct xgene_pcie_port * port)327*4882a593Smuzhiyun static int xgene_pcie_init_port(struct xgene_pcie_port *port)
328*4882a593Smuzhiyun {
329*4882a593Smuzhiyun struct device *dev = port->dev;
330*4882a593Smuzhiyun int rc;
331*4882a593Smuzhiyun
332*4882a593Smuzhiyun port->clk = clk_get(dev, NULL);
333*4882a593Smuzhiyun if (IS_ERR(port->clk)) {
334*4882a593Smuzhiyun dev_err(dev, "clock not available\n");
335*4882a593Smuzhiyun return -ENODEV;
336*4882a593Smuzhiyun }
337*4882a593Smuzhiyun
338*4882a593Smuzhiyun rc = clk_prepare_enable(port->clk);
339*4882a593Smuzhiyun if (rc) {
340*4882a593Smuzhiyun dev_err(dev, "clock enable failed\n");
341*4882a593Smuzhiyun return rc;
342*4882a593Smuzhiyun }
343*4882a593Smuzhiyun
344*4882a593Smuzhiyun return 0;
345*4882a593Smuzhiyun }
346*4882a593Smuzhiyun
xgene_pcie_map_reg(struct xgene_pcie_port * port,struct platform_device * pdev)347*4882a593Smuzhiyun static int xgene_pcie_map_reg(struct xgene_pcie_port *port,
348*4882a593Smuzhiyun struct platform_device *pdev)
349*4882a593Smuzhiyun {
350*4882a593Smuzhiyun struct device *dev = port->dev;
351*4882a593Smuzhiyun struct resource *res;
352*4882a593Smuzhiyun
353*4882a593Smuzhiyun res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "csr");
354*4882a593Smuzhiyun port->csr_base = devm_pci_remap_cfg_resource(dev, res);
355*4882a593Smuzhiyun if (IS_ERR(port->csr_base))
356*4882a593Smuzhiyun return PTR_ERR(port->csr_base);
357*4882a593Smuzhiyun
358*4882a593Smuzhiyun res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "cfg");
359*4882a593Smuzhiyun port->cfg_base = devm_ioremap_resource(dev, res);
360*4882a593Smuzhiyun if (IS_ERR(port->cfg_base))
361*4882a593Smuzhiyun return PTR_ERR(port->cfg_base);
362*4882a593Smuzhiyun port->cfg_addr = res->start;
363*4882a593Smuzhiyun
364*4882a593Smuzhiyun return 0;
365*4882a593Smuzhiyun }
366*4882a593Smuzhiyun
xgene_pcie_setup_ob_reg(struct xgene_pcie_port * port,struct resource * res,u32 offset,u64 cpu_addr,u64 pci_addr)367*4882a593Smuzhiyun static void xgene_pcie_setup_ob_reg(struct xgene_pcie_port *port,
368*4882a593Smuzhiyun struct resource *res, u32 offset,
369*4882a593Smuzhiyun u64 cpu_addr, u64 pci_addr)
370*4882a593Smuzhiyun {
371*4882a593Smuzhiyun struct device *dev = port->dev;
372*4882a593Smuzhiyun resource_size_t size = resource_size(res);
373*4882a593Smuzhiyun u64 restype = resource_type(res);
374*4882a593Smuzhiyun u64 mask = 0;
375*4882a593Smuzhiyun u32 min_size;
376*4882a593Smuzhiyun u32 flag = EN_REG;
377*4882a593Smuzhiyun
378*4882a593Smuzhiyun if (restype == IORESOURCE_MEM) {
379*4882a593Smuzhiyun min_size = SZ_128M;
380*4882a593Smuzhiyun } else {
381*4882a593Smuzhiyun min_size = 128;
382*4882a593Smuzhiyun flag |= OB_LO_IO;
383*4882a593Smuzhiyun }
384*4882a593Smuzhiyun
385*4882a593Smuzhiyun if (size >= min_size)
386*4882a593Smuzhiyun mask = ~(size - 1) | flag;
387*4882a593Smuzhiyun else
388*4882a593Smuzhiyun dev_warn(dev, "res size 0x%llx less than minimum 0x%x\n",
389*4882a593Smuzhiyun (u64)size, min_size);
390*4882a593Smuzhiyun
391*4882a593Smuzhiyun xgene_pcie_writel(port, offset, lower_32_bits(cpu_addr));
392*4882a593Smuzhiyun xgene_pcie_writel(port, offset + 0x04, upper_32_bits(cpu_addr));
393*4882a593Smuzhiyun xgene_pcie_writel(port, offset + 0x08, lower_32_bits(mask));
394*4882a593Smuzhiyun xgene_pcie_writel(port, offset + 0x0c, upper_32_bits(mask));
395*4882a593Smuzhiyun xgene_pcie_writel(port, offset + 0x10, lower_32_bits(pci_addr));
396*4882a593Smuzhiyun xgene_pcie_writel(port, offset + 0x14, upper_32_bits(pci_addr));
397*4882a593Smuzhiyun }
398*4882a593Smuzhiyun
xgene_pcie_setup_cfg_reg(struct xgene_pcie_port * port)399*4882a593Smuzhiyun static void xgene_pcie_setup_cfg_reg(struct xgene_pcie_port *port)
400*4882a593Smuzhiyun {
401*4882a593Smuzhiyun u64 addr = port->cfg_addr;
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun xgene_pcie_writel(port, CFGBARL, lower_32_bits(addr));
404*4882a593Smuzhiyun xgene_pcie_writel(port, CFGBARH, upper_32_bits(addr));
405*4882a593Smuzhiyun xgene_pcie_writel(port, CFGCTL, EN_REG);
406*4882a593Smuzhiyun }
407*4882a593Smuzhiyun
xgene_pcie_map_ranges(struct xgene_pcie_port * port)408*4882a593Smuzhiyun static int xgene_pcie_map_ranges(struct xgene_pcie_port *port)
409*4882a593Smuzhiyun {
410*4882a593Smuzhiyun struct pci_host_bridge *bridge = pci_host_bridge_from_priv(port);
411*4882a593Smuzhiyun struct resource_entry *window;
412*4882a593Smuzhiyun struct device *dev = port->dev;
413*4882a593Smuzhiyun
414*4882a593Smuzhiyun resource_list_for_each_entry(window, &bridge->windows) {
415*4882a593Smuzhiyun struct resource *res = window->res;
416*4882a593Smuzhiyun u64 restype = resource_type(res);
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun dev_dbg(dev, "%pR\n", res);
419*4882a593Smuzhiyun
420*4882a593Smuzhiyun switch (restype) {
421*4882a593Smuzhiyun case IORESOURCE_IO:
422*4882a593Smuzhiyun xgene_pcie_setup_ob_reg(port, res, OMR3BARL,
423*4882a593Smuzhiyun pci_pio_to_address(res->start),
424*4882a593Smuzhiyun res->start - window->offset);
425*4882a593Smuzhiyun break;
426*4882a593Smuzhiyun case IORESOURCE_MEM:
427*4882a593Smuzhiyun if (res->flags & IORESOURCE_PREFETCH)
428*4882a593Smuzhiyun xgene_pcie_setup_ob_reg(port, res, OMR2BARL,
429*4882a593Smuzhiyun res->start,
430*4882a593Smuzhiyun res->start -
431*4882a593Smuzhiyun window->offset);
432*4882a593Smuzhiyun else
433*4882a593Smuzhiyun xgene_pcie_setup_ob_reg(port, res, OMR1BARL,
434*4882a593Smuzhiyun res->start,
435*4882a593Smuzhiyun res->start -
436*4882a593Smuzhiyun window->offset);
437*4882a593Smuzhiyun break;
438*4882a593Smuzhiyun case IORESOURCE_BUS:
439*4882a593Smuzhiyun break;
440*4882a593Smuzhiyun default:
441*4882a593Smuzhiyun dev_err(dev, "invalid resource %pR\n", res);
442*4882a593Smuzhiyun return -EINVAL;
443*4882a593Smuzhiyun }
444*4882a593Smuzhiyun }
445*4882a593Smuzhiyun xgene_pcie_setup_cfg_reg(port);
446*4882a593Smuzhiyun return 0;
447*4882a593Smuzhiyun }
448*4882a593Smuzhiyun
xgene_pcie_setup_pims(struct xgene_pcie_port * port,u32 pim_reg,u64 pim,u64 size)449*4882a593Smuzhiyun static void xgene_pcie_setup_pims(struct xgene_pcie_port *port, u32 pim_reg,
450*4882a593Smuzhiyun u64 pim, u64 size)
451*4882a593Smuzhiyun {
452*4882a593Smuzhiyun xgene_pcie_writel(port, pim_reg, lower_32_bits(pim));
453*4882a593Smuzhiyun xgene_pcie_writel(port, pim_reg + 0x04,
454*4882a593Smuzhiyun upper_32_bits(pim) | EN_COHERENCY);
455*4882a593Smuzhiyun xgene_pcie_writel(port, pim_reg + 0x10, lower_32_bits(size));
456*4882a593Smuzhiyun xgene_pcie_writel(port, pim_reg + 0x14, upper_32_bits(size));
457*4882a593Smuzhiyun }
458*4882a593Smuzhiyun
459*4882a593Smuzhiyun /*
460*4882a593Smuzhiyun * X-Gene PCIe support maximum 3 inbound memory regions
461*4882a593Smuzhiyun * This function helps to select a region based on size of region
462*4882a593Smuzhiyun */
xgene_pcie_select_ib_reg(u8 * ib_reg_mask,u64 size)463*4882a593Smuzhiyun static int xgene_pcie_select_ib_reg(u8 *ib_reg_mask, u64 size)
464*4882a593Smuzhiyun {
465*4882a593Smuzhiyun if ((size > 4) && (size < SZ_16M) && !(*ib_reg_mask & (1 << 1))) {
466*4882a593Smuzhiyun *ib_reg_mask |= (1 << 1);
467*4882a593Smuzhiyun return 1;
468*4882a593Smuzhiyun }
469*4882a593Smuzhiyun
470*4882a593Smuzhiyun if ((size > SZ_1K) && (size < SZ_1T) && !(*ib_reg_mask & (1 << 0))) {
471*4882a593Smuzhiyun *ib_reg_mask |= (1 << 0);
472*4882a593Smuzhiyun return 0;
473*4882a593Smuzhiyun }
474*4882a593Smuzhiyun
475*4882a593Smuzhiyun if ((size > SZ_1M) && (size < SZ_1T) && !(*ib_reg_mask & (1 << 2))) {
476*4882a593Smuzhiyun *ib_reg_mask |= (1 << 2);
477*4882a593Smuzhiyun return 2;
478*4882a593Smuzhiyun }
479*4882a593Smuzhiyun
480*4882a593Smuzhiyun return -EINVAL;
481*4882a593Smuzhiyun }
482*4882a593Smuzhiyun
xgene_pcie_setup_ib_reg(struct xgene_pcie_port * port,struct of_pci_range * range,u8 * ib_reg_mask)483*4882a593Smuzhiyun static void xgene_pcie_setup_ib_reg(struct xgene_pcie_port *port,
484*4882a593Smuzhiyun struct of_pci_range *range, u8 *ib_reg_mask)
485*4882a593Smuzhiyun {
486*4882a593Smuzhiyun void __iomem *cfg_base = port->cfg_base;
487*4882a593Smuzhiyun struct device *dev = port->dev;
488*4882a593Smuzhiyun void *bar_addr;
489*4882a593Smuzhiyun u32 pim_reg;
490*4882a593Smuzhiyun u64 cpu_addr = range->cpu_addr;
491*4882a593Smuzhiyun u64 pci_addr = range->pci_addr;
492*4882a593Smuzhiyun u64 size = range->size;
493*4882a593Smuzhiyun u64 mask = ~(size - 1) | EN_REG;
494*4882a593Smuzhiyun u32 flags = PCI_BASE_ADDRESS_MEM_TYPE_64;
495*4882a593Smuzhiyun u32 bar_low;
496*4882a593Smuzhiyun int region;
497*4882a593Smuzhiyun
498*4882a593Smuzhiyun region = xgene_pcie_select_ib_reg(ib_reg_mask, range->size);
499*4882a593Smuzhiyun if (region < 0) {
500*4882a593Smuzhiyun dev_warn(dev, "invalid pcie dma-range config\n");
501*4882a593Smuzhiyun return;
502*4882a593Smuzhiyun }
503*4882a593Smuzhiyun
504*4882a593Smuzhiyun if (range->flags & IORESOURCE_PREFETCH)
505*4882a593Smuzhiyun flags |= PCI_BASE_ADDRESS_MEM_PREFETCH;
506*4882a593Smuzhiyun
507*4882a593Smuzhiyun bar_low = pcie_bar_low_val((u32)cpu_addr, flags);
508*4882a593Smuzhiyun switch (region) {
509*4882a593Smuzhiyun case 0:
510*4882a593Smuzhiyun xgene_pcie_set_ib_mask(port, BRIDGE_CFG_4, flags, size);
511*4882a593Smuzhiyun bar_addr = cfg_base + PCI_BASE_ADDRESS_0;
512*4882a593Smuzhiyun writel(bar_low, bar_addr);
513*4882a593Smuzhiyun writel(upper_32_bits(cpu_addr), bar_addr + 0x4);
514*4882a593Smuzhiyun pim_reg = PIM1_1L;
515*4882a593Smuzhiyun break;
516*4882a593Smuzhiyun case 1:
517*4882a593Smuzhiyun xgene_pcie_writel(port, IBAR2, bar_low);
518*4882a593Smuzhiyun xgene_pcie_writel(port, IR2MSK, lower_32_bits(mask));
519*4882a593Smuzhiyun pim_reg = PIM2_1L;
520*4882a593Smuzhiyun break;
521*4882a593Smuzhiyun case 2:
522*4882a593Smuzhiyun xgene_pcie_writel(port, IBAR3L, bar_low);
523*4882a593Smuzhiyun xgene_pcie_writel(port, IBAR3L + 0x4, upper_32_bits(cpu_addr));
524*4882a593Smuzhiyun xgene_pcie_writel(port, IR3MSKL, lower_32_bits(mask));
525*4882a593Smuzhiyun xgene_pcie_writel(port, IR3MSKL + 0x4, upper_32_bits(mask));
526*4882a593Smuzhiyun pim_reg = PIM3_1L;
527*4882a593Smuzhiyun break;
528*4882a593Smuzhiyun }
529*4882a593Smuzhiyun
530*4882a593Smuzhiyun xgene_pcie_setup_pims(port, pim_reg, pci_addr, ~(size - 1));
531*4882a593Smuzhiyun }
532*4882a593Smuzhiyun
xgene_pcie_parse_map_dma_ranges(struct xgene_pcie_port * port)533*4882a593Smuzhiyun static int xgene_pcie_parse_map_dma_ranges(struct xgene_pcie_port *port)
534*4882a593Smuzhiyun {
535*4882a593Smuzhiyun struct device_node *np = port->node;
536*4882a593Smuzhiyun struct of_pci_range range;
537*4882a593Smuzhiyun struct of_pci_range_parser parser;
538*4882a593Smuzhiyun struct device *dev = port->dev;
539*4882a593Smuzhiyun u8 ib_reg_mask = 0;
540*4882a593Smuzhiyun
541*4882a593Smuzhiyun if (of_pci_dma_range_parser_init(&parser, np)) {
542*4882a593Smuzhiyun dev_err(dev, "missing dma-ranges property\n");
543*4882a593Smuzhiyun return -EINVAL;
544*4882a593Smuzhiyun }
545*4882a593Smuzhiyun
546*4882a593Smuzhiyun /* Get the dma-ranges from DT */
547*4882a593Smuzhiyun for_each_of_pci_range(&parser, &range) {
548*4882a593Smuzhiyun u64 end = range.cpu_addr + range.size - 1;
549*4882a593Smuzhiyun
550*4882a593Smuzhiyun dev_dbg(dev, "0x%08x 0x%016llx..0x%016llx -> 0x%016llx\n",
551*4882a593Smuzhiyun range.flags, range.cpu_addr, end, range.pci_addr);
552*4882a593Smuzhiyun xgene_pcie_setup_ib_reg(port, &range, &ib_reg_mask);
553*4882a593Smuzhiyun }
554*4882a593Smuzhiyun return 0;
555*4882a593Smuzhiyun }
556*4882a593Smuzhiyun
557*4882a593Smuzhiyun /* clear BAR configuration which was done by firmware */
xgene_pcie_clear_config(struct xgene_pcie_port * port)558*4882a593Smuzhiyun static void xgene_pcie_clear_config(struct xgene_pcie_port *port)
559*4882a593Smuzhiyun {
560*4882a593Smuzhiyun int i;
561*4882a593Smuzhiyun
562*4882a593Smuzhiyun for (i = PIM1_1L; i <= CFGCTL; i += 4)
563*4882a593Smuzhiyun xgene_pcie_writel(port, i, 0);
564*4882a593Smuzhiyun }
565*4882a593Smuzhiyun
xgene_pcie_setup(struct xgene_pcie_port * port)566*4882a593Smuzhiyun static int xgene_pcie_setup(struct xgene_pcie_port *port)
567*4882a593Smuzhiyun {
568*4882a593Smuzhiyun struct device *dev = port->dev;
569*4882a593Smuzhiyun u32 val, lanes = 0, speed = 0;
570*4882a593Smuzhiyun int ret;
571*4882a593Smuzhiyun
572*4882a593Smuzhiyun xgene_pcie_clear_config(port);
573*4882a593Smuzhiyun
574*4882a593Smuzhiyun /* setup the vendor and device IDs correctly */
575*4882a593Smuzhiyun val = (XGENE_PCIE_DEVICEID << 16) | XGENE_PCIE_VENDORID;
576*4882a593Smuzhiyun xgene_pcie_writel(port, BRIDGE_CFG_0, val);
577*4882a593Smuzhiyun
578*4882a593Smuzhiyun ret = xgene_pcie_map_ranges(port);
579*4882a593Smuzhiyun if (ret)
580*4882a593Smuzhiyun return ret;
581*4882a593Smuzhiyun
582*4882a593Smuzhiyun ret = xgene_pcie_parse_map_dma_ranges(port);
583*4882a593Smuzhiyun if (ret)
584*4882a593Smuzhiyun return ret;
585*4882a593Smuzhiyun
586*4882a593Smuzhiyun xgene_pcie_linkup(port, &lanes, &speed);
587*4882a593Smuzhiyun if (!port->link_up)
588*4882a593Smuzhiyun dev_info(dev, "(rc) link down\n");
589*4882a593Smuzhiyun else
590*4882a593Smuzhiyun dev_info(dev, "(rc) x%d gen-%d link up\n", lanes, speed + 1);
591*4882a593Smuzhiyun return 0;
592*4882a593Smuzhiyun }
593*4882a593Smuzhiyun
594*4882a593Smuzhiyun static struct pci_ops xgene_pcie_ops = {
595*4882a593Smuzhiyun .map_bus = xgene_pcie_map_bus,
596*4882a593Smuzhiyun .read = xgene_pcie_config_read32,
597*4882a593Smuzhiyun .write = pci_generic_config_write32,
598*4882a593Smuzhiyun };
599*4882a593Smuzhiyun
xgene_pcie_probe(struct platform_device * pdev)600*4882a593Smuzhiyun static int xgene_pcie_probe(struct platform_device *pdev)
601*4882a593Smuzhiyun {
602*4882a593Smuzhiyun struct device *dev = &pdev->dev;
603*4882a593Smuzhiyun struct device_node *dn = dev->of_node;
604*4882a593Smuzhiyun struct xgene_pcie_port *port;
605*4882a593Smuzhiyun struct pci_host_bridge *bridge;
606*4882a593Smuzhiyun int ret;
607*4882a593Smuzhiyun
608*4882a593Smuzhiyun bridge = devm_pci_alloc_host_bridge(dev, sizeof(*port));
609*4882a593Smuzhiyun if (!bridge)
610*4882a593Smuzhiyun return -ENOMEM;
611*4882a593Smuzhiyun
612*4882a593Smuzhiyun port = pci_host_bridge_priv(bridge);
613*4882a593Smuzhiyun
614*4882a593Smuzhiyun port->node = of_node_get(dn);
615*4882a593Smuzhiyun port->dev = dev;
616*4882a593Smuzhiyun
617*4882a593Smuzhiyun port->version = XGENE_PCIE_IP_VER_UNKN;
618*4882a593Smuzhiyun if (of_device_is_compatible(port->node, "apm,xgene-pcie"))
619*4882a593Smuzhiyun port->version = XGENE_PCIE_IP_VER_1;
620*4882a593Smuzhiyun
621*4882a593Smuzhiyun ret = xgene_pcie_map_reg(port, pdev);
622*4882a593Smuzhiyun if (ret)
623*4882a593Smuzhiyun return ret;
624*4882a593Smuzhiyun
625*4882a593Smuzhiyun ret = xgene_pcie_init_port(port);
626*4882a593Smuzhiyun if (ret)
627*4882a593Smuzhiyun return ret;
628*4882a593Smuzhiyun
629*4882a593Smuzhiyun ret = xgene_pcie_setup(port);
630*4882a593Smuzhiyun if (ret)
631*4882a593Smuzhiyun return ret;
632*4882a593Smuzhiyun
633*4882a593Smuzhiyun bridge->sysdata = port;
634*4882a593Smuzhiyun bridge->ops = &xgene_pcie_ops;
635*4882a593Smuzhiyun
636*4882a593Smuzhiyun return pci_host_probe(bridge);
637*4882a593Smuzhiyun }
638*4882a593Smuzhiyun
639*4882a593Smuzhiyun static const struct of_device_id xgene_pcie_match_table[] = {
640*4882a593Smuzhiyun {.compatible = "apm,xgene-pcie",},
641*4882a593Smuzhiyun {},
642*4882a593Smuzhiyun };
643*4882a593Smuzhiyun
644*4882a593Smuzhiyun static struct platform_driver xgene_pcie_driver = {
645*4882a593Smuzhiyun .driver = {
646*4882a593Smuzhiyun .name = "xgene-pcie",
647*4882a593Smuzhiyun .of_match_table = of_match_ptr(xgene_pcie_match_table),
648*4882a593Smuzhiyun .suppress_bind_attrs = true,
649*4882a593Smuzhiyun },
650*4882a593Smuzhiyun .probe = xgene_pcie_probe,
651*4882a593Smuzhiyun };
652*4882a593Smuzhiyun builtin_platform_driver(xgene_pcie_driver);
653*4882a593Smuzhiyun #endif
654