xref: /OK3568_Linux_fs/kernel/drivers/pci/controller/pci-xgene.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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