Commit | Line | Data |
---|---|---|
1fb37a81 RJ |
1 | /* |
2 | * Copyright (C) 2014 Hauke Mehrtens <hauke@hauke-m.de> | |
be908d21 | 3 | * Copyright (C) 2015 Broadcom Corporation |
1fb37a81 RJ |
4 | * |
5 | * This program is free software; you can redistribute it and/or | |
6 | * modify it under the terms of the GNU General Public License as | |
7 | * published by the Free Software Foundation version 2. | |
8 | * | |
9 | * This program is distributed "as is" WITHOUT ANY WARRANTY of any | |
10 | * kind, whether express or implied; without even the implied warranty | |
11 | * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 | * GNU General Public License for more details. | |
13 | */ | |
14 | ||
15 | #include <linux/kernel.h> | |
16 | #include <linux/pci.h> | |
17 | #include <linux/msi.h> | |
18 | #include <linux/clk.h> | |
19 | #include <linux/module.h> | |
20 | #include <linux/mbus.h> | |
21 | #include <linux/slab.h> | |
22 | #include <linux/delay.h> | |
23 | #include <linux/interrupt.h> | |
787b3c4f | 24 | #include <linux/irqchip/arm-gic-v3.h> |
1fb37a81 RJ |
25 | #include <linux/platform_device.h> |
26 | #include <linux/of_address.h> | |
27 | #include <linux/of_pci.h> | |
28 | #include <linux/of_irq.h> | |
29 | #include <linux/of_platform.h> | |
30 | #include <linux/phy/phy.h> | |
31 | ||
32 | #include "pcie-iproc.h" | |
33 | ||
199ff141 RJ |
34 | #define EP_PERST_SOURCE_SELECT_SHIFT 2 |
35 | #define EP_PERST_SOURCE_SELECT BIT(EP_PERST_SOURCE_SELECT_SHIFT) | |
1fb37a81 RJ |
36 | #define EP_MODE_SURVIVE_PERST_SHIFT 1 |
37 | #define EP_MODE_SURVIVE_PERST BIT(EP_MODE_SURVIVE_PERST_SHIFT) | |
38 | #define RC_PCIE_RST_OUTPUT_SHIFT 0 | |
39 | #define RC_PCIE_RST_OUTPUT BIT(RC_PCIE_RST_OUTPUT_SHIFT) | |
943ebae7 | 40 | #define PAXC_RESET_MASK 0x7f |
1fb37a81 | 41 | |
787b3c4f RJ |
42 | #define GIC_V3_CFG_SHIFT 0 |
43 | #define GIC_V3_CFG BIT(GIC_V3_CFG_SHIFT) | |
44 | ||
45 | #define MSI_ENABLE_CFG_SHIFT 0 | |
46 | #define MSI_ENABLE_CFG BIT(MSI_ENABLE_CFG_SHIFT) | |
47 | ||
1fb37a81 RJ |
48 | #define CFG_IND_ADDR_MASK 0x00001ffc |
49 | ||
1fb37a81 RJ |
50 | #define CFG_ADDR_BUS_NUM_SHIFT 20 |
51 | #define CFG_ADDR_BUS_NUM_MASK 0x0ff00000 | |
52 | #define CFG_ADDR_DEV_NUM_SHIFT 15 | |
53 | #define CFG_ADDR_DEV_NUM_MASK 0x000f8000 | |
54 | #define CFG_ADDR_FUNC_NUM_SHIFT 12 | |
55 | #define CFG_ADDR_FUNC_NUM_MASK 0x00007000 | |
56 | #define CFG_ADDR_REG_NUM_SHIFT 2 | |
57 | #define CFG_ADDR_REG_NUM_MASK 0x00000ffc | |
58 | #define CFG_ADDR_CFG_TYPE_SHIFT 0 | |
59 | #define CFG_ADDR_CFG_TYPE_MASK 0x00000003 | |
60 | ||
1fb37a81 RJ |
61 | #define SYS_RC_INTX_MASK 0xf |
62 | ||
aaf22ab4 RJ |
63 | #define PCIE_PHYLINKUP_SHIFT 3 |
64 | #define PCIE_PHYLINKUP BIT(PCIE_PHYLINKUP_SHIFT) | |
65 | #define PCIE_DL_ACTIVE_SHIFT 2 | |
66 | #define PCIE_DL_ACTIVE BIT(PCIE_DL_ACTIVE_SHIFT) | |
67 | ||
538928fd RJ |
68 | #define APB_ERR_EN_SHIFT 0 |
69 | #define APB_ERR_EN BIT(APB_ERR_EN_SHIFT) | |
70 | ||
4213e15c RJ |
71 | /* derive the enum index of the outbound/inbound mapping registers */ |
72 | #define MAP_REG(base_reg, index) ((base_reg) + (index) * 2) | |
73 | ||
74 | /* | |
75 | * Maximum number of outbound mapping window sizes that can be supported by any | |
76 | * OARR/OMAP mapping pair | |
77 | */ | |
78 | #define MAX_NUM_OB_WINDOW_SIZES 4 | |
79 | ||
e99a187b RJ |
80 | #define OARR_VALID_SHIFT 0 |
81 | #define OARR_VALID BIT(OARR_VALID_SHIFT) | |
82 | #define OARR_SIZE_CFG_SHIFT 1 | |
e99a187b | 83 | |
dd9d4e74 RJ |
84 | /* |
85 | * Maximum number of inbound mapping region sizes that can be supported by an | |
86 | * IARR | |
87 | */ | |
88 | #define MAX_NUM_IB_REGION_SIZES 9 | |
89 | ||
90 | #define IMAP_VALID_SHIFT 0 | |
91 | #define IMAP_VALID BIT(IMAP_VALID_SHIFT) | |
92 | ||
e3a1698b BH |
93 | #define PCI_EXP_CAP 0xac |
94 | ||
943ebae7 RJ |
95 | #define IPROC_PCIE_REG_INVALID 0xffff |
96 | ||
4213e15c RJ |
97 | /** |
98 | * iProc PCIe outbound mapping controller specific parameters | |
99 | * | |
100 | * @window_sizes: list of supported outbound mapping window sizes in MB | |
101 | * @nr_sizes: number of supported outbound mapping window sizes | |
102 | */ | |
103 | struct iproc_pcie_ob_map { | |
104 | resource_size_t window_sizes[MAX_NUM_OB_WINDOW_SIZES]; | |
105 | unsigned int nr_sizes; | |
106 | }; | |
107 | ||
108 | static const struct iproc_pcie_ob_map paxb_ob_map[] = { | |
109 | { | |
110 | /* OARR0/OMAP0 */ | |
111 | .window_sizes = { 128, 256 }, | |
112 | .nr_sizes = 2, | |
113 | }, | |
114 | { | |
115 | /* OARR1/OMAP1 */ | |
116 | .window_sizes = { 128, 256 }, | |
117 | .nr_sizes = 2, | |
118 | }, | |
119 | }; | |
120 | ||
c7c44527 RJ |
121 | static const struct iproc_pcie_ob_map paxb_v2_ob_map[] = { |
122 | { | |
123 | /* OARR0/OMAP0 */ | |
124 | .window_sizes = { 128, 256 }, | |
125 | .nr_sizes = 2, | |
126 | }, | |
127 | { | |
128 | /* OARR1/OMAP1 */ | |
129 | .window_sizes = { 128, 256 }, | |
130 | .nr_sizes = 2, | |
131 | }, | |
132 | { | |
133 | /* OARR2/OMAP2 */ | |
134 | .window_sizes = { 128, 256, 512, 1024 }, | |
135 | .nr_sizes = 4, | |
136 | }, | |
137 | { | |
138 | /* OARR3/OMAP3 */ | |
139 | .window_sizes = { 128, 256, 512, 1024 }, | |
140 | .nr_sizes = 4, | |
141 | }, | |
142 | }; | |
143 | ||
dd9d4e74 RJ |
144 | /** |
145 | * iProc PCIe inbound mapping type | |
146 | */ | |
147 | enum iproc_pcie_ib_map_type { | |
148 | /* for DDR memory */ | |
149 | IPROC_PCIE_IB_MAP_MEM = 0, | |
150 | ||
151 | /* for device I/O memory */ | |
152 | IPROC_PCIE_IB_MAP_IO, | |
153 | ||
154 | /* invalid or unused */ | |
155 | IPROC_PCIE_IB_MAP_INVALID | |
156 | }; | |
157 | ||
158 | /** | |
159 | * iProc PCIe inbound mapping controller specific parameters | |
160 | * | |
161 | * @type: inbound mapping region type | |
162 | * @size_unit: inbound mapping region size unit, could be SZ_1K, SZ_1M, or | |
163 | * SZ_1G | |
164 | * @region_sizes: list of supported inbound mapping region sizes in KB, MB, or | |
165 | * GB, depedning on the size unit | |
166 | * @nr_sizes: number of supported inbound mapping region sizes | |
167 | * @nr_windows: number of supported inbound mapping windows for the region | |
168 | * @imap_addr_offset: register offset between the upper and lower 32-bit | |
169 | * IMAP address registers | |
170 | * @imap_window_offset: register offset between each IMAP window | |
171 | */ | |
172 | struct iproc_pcie_ib_map { | |
173 | enum iproc_pcie_ib_map_type type; | |
174 | unsigned int size_unit; | |
175 | resource_size_t region_sizes[MAX_NUM_IB_REGION_SIZES]; | |
176 | unsigned int nr_sizes; | |
177 | unsigned int nr_windows; | |
178 | u16 imap_addr_offset; | |
179 | u16 imap_window_offset; | |
180 | }; | |
181 | ||
c7c44527 RJ |
182 | static const struct iproc_pcie_ib_map paxb_v2_ib_map[] = { |
183 | { | |
184 | /* IARR0/IMAP0 */ | |
185 | .type = IPROC_PCIE_IB_MAP_IO, | |
186 | .size_unit = SZ_1K, | |
187 | .region_sizes = { 32 }, | |
188 | .nr_sizes = 1, | |
189 | .nr_windows = 8, | |
190 | .imap_addr_offset = 0x40, | |
191 | .imap_window_offset = 0x4, | |
192 | }, | |
193 | { | |
194 | /* IARR1/IMAP1 (currently unused) */ | |
195 | .type = IPROC_PCIE_IB_MAP_INVALID, | |
196 | }, | |
197 | { | |
198 | /* IARR2/IMAP2 */ | |
199 | .type = IPROC_PCIE_IB_MAP_MEM, | |
200 | .size_unit = SZ_1M, | |
201 | .region_sizes = { 64, 128, 256, 512, 1024, 2048, 4096, 8192, | |
202 | 16384 }, | |
203 | .nr_sizes = 9, | |
204 | .nr_windows = 1, | |
205 | .imap_addr_offset = 0x4, | |
206 | .imap_window_offset = 0x8, | |
207 | }, | |
208 | { | |
209 | /* IARR3/IMAP3 */ | |
210 | .type = IPROC_PCIE_IB_MAP_MEM, | |
211 | .size_unit = SZ_1G, | |
212 | .region_sizes = { 1, 2, 4, 8, 16, 32 }, | |
213 | .nr_sizes = 6, | |
214 | .nr_windows = 8, | |
215 | .imap_addr_offset = 0x4, | |
216 | .imap_window_offset = 0x8, | |
217 | }, | |
218 | { | |
219 | /* IARR4/IMAP4 */ | |
220 | .type = IPROC_PCIE_IB_MAP_MEM, | |
221 | .size_unit = SZ_1G, | |
222 | .region_sizes = { 32, 64, 128, 256, 512 }, | |
223 | .nr_sizes = 5, | |
224 | .nr_windows = 8, | |
225 | .imap_addr_offset = 0x4, | |
226 | .imap_window_offset = 0x8, | |
227 | }, | |
228 | }; | |
229 | ||
06324ede RJ |
230 | /* |
231 | * iProc PCIe host registers | |
232 | */ | |
943ebae7 | 233 | enum iproc_pcie_reg { |
06324ede | 234 | /* clock/reset signal control */ |
943ebae7 | 235 | IPROC_PCIE_CLK_CTRL = 0, |
06324ede | 236 | |
787b3c4f RJ |
237 | /* |
238 | * To allow MSI to be steered to an external MSI controller (e.g., ARM | |
239 | * GICv3 ITS) | |
240 | */ | |
241 | IPROC_PCIE_MSI_GIC_MODE, | |
242 | ||
243 | /* | |
244 | * IPROC_PCIE_MSI_BASE_ADDR and IPROC_PCIE_MSI_WINDOW_SIZE define the | |
245 | * window where the MSI posted writes are written, for the writes to be | |
246 | * interpreted as MSI writes. | |
247 | */ | |
248 | IPROC_PCIE_MSI_BASE_ADDR, | |
249 | IPROC_PCIE_MSI_WINDOW_SIZE, | |
250 | ||
251 | /* | |
252 | * To hold the address of the register where the MSI writes are | |
253 | * programed. When ARM GICv3 ITS is used, this should be programmed | |
254 | * with the address of the GITS_TRANSLATER register. | |
255 | */ | |
256 | IPROC_PCIE_MSI_ADDR_LO, | |
257 | IPROC_PCIE_MSI_ADDR_HI, | |
258 | ||
259 | /* enable MSI */ | |
260 | IPROC_PCIE_MSI_EN_CFG, | |
261 | ||
06324ede | 262 | /* allow access to root complex configuration space */ |
943ebae7 RJ |
263 | IPROC_PCIE_CFG_IND_ADDR, |
264 | IPROC_PCIE_CFG_IND_DATA, | |
06324ede RJ |
265 | |
266 | /* allow access to device configuration space */ | |
943ebae7 RJ |
267 | IPROC_PCIE_CFG_ADDR, |
268 | IPROC_PCIE_CFG_DATA, | |
06324ede RJ |
269 | |
270 | /* enable INTx */ | |
943ebae7 | 271 | IPROC_PCIE_INTX_EN, |
06324ede RJ |
272 | |
273 | /* outbound address mapping */ | |
4213e15c RJ |
274 | IPROC_PCIE_OARR0, |
275 | IPROC_PCIE_OMAP0, | |
276 | IPROC_PCIE_OARR1, | |
277 | IPROC_PCIE_OMAP1, | |
278 | IPROC_PCIE_OARR2, | |
279 | IPROC_PCIE_OMAP2, | |
280 | IPROC_PCIE_OARR3, | |
281 | IPROC_PCIE_OMAP3, | |
06324ede | 282 | |
dd9d4e74 RJ |
283 | /* inbound address mapping */ |
284 | IPROC_PCIE_IARR0, | |
285 | IPROC_PCIE_IMAP0, | |
286 | IPROC_PCIE_IARR1, | |
287 | IPROC_PCIE_IMAP1, | |
288 | IPROC_PCIE_IARR2, | |
289 | IPROC_PCIE_IMAP2, | |
290 | IPROC_PCIE_IARR3, | |
291 | IPROC_PCIE_IMAP3, | |
292 | IPROC_PCIE_IARR4, | |
293 | IPROC_PCIE_IMAP4, | |
294 | ||
06324ede | 295 | /* link status */ |
943ebae7 | 296 | IPROC_PCIE_LINK_STATUS, |
06324ede | 297 | |
538928fd RJ |
298 | /* enable APB error for unsupported requests */ |
299 | IPROC_PCIE_APB_ERR_EN, | |
300 | ||
06324ede RJ |
301 | /* total number of core registers */ |
302 | IPROC_PCIE_MAX_NUM_REG, | |
943ebae7 RJ |
303 | }; |
304 | ||
404349c5 RJ |
305 | /* iProc PCIe PAXB BCMA registers */ |
306 | static const u16 iproc_pcie_reg_paxb_bcma[] = { | |
307 | [IPROC_PCIE_CLK_CTRL] = 0x000, | |
308 | [IPROC_PCIE_CFG_IND_ADDR] = 0x120, | |
309 | [IPROC_PCIE_CFG_IND_DATA] = 0x124, | |
310 | [IPROC_PCIE_CFG_ADDR] = 0x1f8, | |
311 | [IPROC_PCIE_CFG_DATA] = 0x1fc, | |
312 | [IPROC_PCIE_INTX_EN] = 0x330, | |
313 | [IPROC_PCIE_LINK_STATUS] = 0xf0c, | |
314 | }; | |
315 | ||
943ebae7 RJ |
316 | /* iProc PCIe PAXB registers */ |
317 | static const u16 iproc_pcie_reg_paxb[] = { | |
4213e15c RJ |
318 | [IPROC_PCIE_CLK_CTRL] = 0x000, |
319 | [IPROC_PCIE_CFG_IND_ADDR] = 0x120, | |
320 | [IPROC_PCIE_CFG_IND_DATA] = 0x124, | |
321 | [IPROC_PCIE_CFG_ADDR] = 0x1f8, | |
322 | [IPROC_PCIE_CFG_DATA] = 0x1fc, | |
323 | [IPROC_PCIE_INTX_EN] = 0x330, | |
324 | [IPROC_PCIE_OARR0] = 0xd20, | |
325 | [IPROC_PCIE_OMAP0] = 0xd40, | |
326 | [IPROC_PCIE_OARR1] = 0xd28, | |
327 | [IPROC_PCIE_OMAP1] = 0xd48, | |
328 | [IPROC_PCIE_LINK_STATUS] = 0xf0c, | |
329 | [IPROC_PCIE_APB_ERR_EN] = 0xf40, | |
943ebae7 RJ |
330 | }; |
331 | ||
c7c44527 RJ |
332 | /* iProc PCIe PAXB v2 registers */ |
333 | static const u16 iproc_pcie_reg_paxb_v2[] = { | |
334 | [IPROC_PCIE_CLK_CTRL] = 0x000, | |
335 | [IPROC_PCIE_CFG_IND_ADDR] = 0x120, | |
336 | [IPROC_PCIE_CFG_IND_DATA] = 0x124, | |
337 | [IPROC_PCIE_CFG_ADDR] = 0x1f8, | |
338 | [IPROC_PCIE_CFG_DATA] = 0x1fc, | |
339 | [IPROC_PCIE_INTX_EN] = 0x330, | |
340 | [IPROC_PCIE_OARR0] = 0xd20, | |
341 | [IPROC_PCIE_OMAP0] = 0xd40, | |
342 | [IPROC_PCIE_OARR1] = 0xd28, | |
343 | [IPROC_PCIE_OMAP1] = 0xd48, | |
344 | [IPROC_PCIE_OARR2] = 0xd60, | |
345 | [IPROC_PCIE_OMAP2] = 0xd68, | |
346 | [IPROC_PCIE_OARR3] = 0xdf0, | |
347 | [IPROC_PCIE_OMAP3] = 0xdf8, | |
348 | [IPROC_PCIE_IARR0] = 0xd00, | |
349 | [IPROC_PCIE_IMAP0] = 0xc00, | |
350 | [IPROC_PCIE_IARR2] = 0xd10, | |
351 | [IPROC_PCIE_IMAP2] = 0xcc0, | |
352 | [IPROC_PCIE_IARR3] = 0xe00, | |
353 | [IPROC_PCIE_IMAP3] = 0xe08, | |
354 | [IPROC_PCIE_IARR4] = 0xe68, | |
355 | [IPROC_PCIE_IMAP4] = 0xe70, | |
356 | [IPROC_PCIE_LINK_STATUS] = 0xf0c, | |
357 | [IPROC_PCIE_APB_ERR_EN] = 0xf40, | |
358 | }; | |
359 | ||
943ebae7 RJ |
360 | /* iProc PCIe PAXC v1 registers */ |
361 | static const u16 iproc_pcie_reg_paxc[] = { | |
4213e15c RJ |
362 | [IPROC_PCIE_CLK_CTRL] = 0x000, |
363 | [IPROC_PCIE_CFG_IND_ADDR] = 0x1f0, | |
364 | [IPROC_PCIE_CFG_IND_DATA] = 0x1f4, | |
365 | [IPROC_PCIE_CFG_ADDR] = 0x1f8, | |
366 | [IPROC_PCIE_CFG_DATA] = 0x1fc, | |
943ebae7 | 367 | }; |
e99a187b | 368 | |
787b3c4f RJ |
369 | /* iProc PCIe PAXC v2 registers */ |
370 | static const u16 iproc_pcie_reg_paxc_v2[] = { | |
371 | [IPROC_PCIE_MSI_GIC_MODE] = 0x050, | |
372 | [IPROC_PCIE_MSI_BASE_ADDR] = 0x074, | |
373 | [IPROC_PCIE_MSI_WINDOW_SIZE] = 0x078, | |
374 | [IPROC_PCIE_MSI_ADDR_LO] = 0x07c, | |
375 | [IPROC_PCIE_MSI_ADDR_HI] = 0x080, | |
376 | [IPROC_PCIE_MSI_EN_CFG] = 0x09c, | |
377 | [IPROC_PCIE_CFG_IND_ADDR] = 0x1f0, | |
378 | [IPROC_PCIE_CFG_IND_DATA] = 0x1f4, | |
379 | [IPROC_PCIE_CFG_ADDR] = 0x1f8, | |
380 | [IPROC_PCIE_CFG_DATA] = 0x1fc, | |
381 | }; | |
382 | ||
8d9bfe37 | 383 | static inline struct iproc_pcie *iproc_data(struct pci_bus *bus) |
1fb37a81 | 384 | { |
8d9bfe37 RJ |
385 | struct iproc_pcie *pcie; |
386 | #ifdef CONFIG_ARM | |
387 | struct pci_sys_data *sys = bus->sysdata; | |
388 | ||
389 | pcie = sys->private_data; | |
390 | #else | |
391 | pcie = bus->sysdata; | |
392 | #endif | |
393 | return pcie; | |
1fb37a81 RJ |
394 | } |
395 | ||
943ebae7 RJ |
396 | static inline bool iproc_pcie_reg_is_invalid(u16 reg_offset) |
397 | { | |
398 | return !!(reg_offset == IPROC_PCIE_REG_INVALID); | |
399 | } | |
400 | ||
401 | static inline u16 iproc_pcie_reg_offset(struct iproc_pcie *pcie, | |
402 | enum iproc_pcie_reg reg) | |
403 | { | |
404 | return pcie->reg_offsets[reg]; | |
405 | } | |
406 | ||
407 | static inline u32 iproc_pcie_read_reg(struct iproc_pcie *pcie, | |
408 | enum iproc_pcie_reg reg) | |
409 | { | |
410 | u16 offset = iproc_pcie_reg_offset(pcie, reg); | |
411 | ||
412 | if (iproc_pcie_reg_is_invalid(offset)) | |
413 | return 0; | |
414 | ||
415 | return readl(pcie->base + offset); | |
416 | } | |
417 | ||
418 | static inline void iproc_pcie_write_reg(struct iproc_pcie *pcie, | |
419 | enum iproc_pcie_reg reg, u32 val) | |
420 | { | |
421 | u16 offset = iproc_pcie_reg_offset(pcie, reg); | |
422 | ||
423 | if (iproc_pcie_reg_is_invalid(offset)) | |
424 | return; | |
425 | ||
426 | writel(val, pcie->base + offset); | |
427 | } | |
428 | ||
538928fd RJ |
429 | /** |
430 | * APB error forwarding can be disabled during access of configuration | |
431 | * registers of the endpoint device, to prevent unsupported requests | |
432 | * (typically seen during enumeration with multi-function devices) from | |
433 | * triggering a system exception. | |
434 | */ | |
435 | static inline void iproc_pcie_apb_err_disable(struct pci_bus *bus, | |
436 | bool disable) | |
437 | { | |
438 | struct iproc_pcie *pcie = iproc_data(bus); | |
439 | u32 val; | |
440 | ||
441 | if (bus->number && pcie->has_apb_err_disable) { | |
442 | val = iproc_pcie_read_reg(pcie, IPROC_PCIE_APB_ERR_EN); | |
443 | if (disable) | |
444 | val &= ~APB_ERR_EN; | |
445 | else | |
446 | val |= APB_ERR_EN; | |
447 | iproc_pcie_write_reg(pcie, IPROC_PCIE_APB_ERR_EN, val); | |
448 | } | |
449 | } | |
450 | ||
1fb37a81 RJ |
451 | /** |
452 | * Note access to the configuration registers are protected at the higher layer | |
453 | * by 'pci_lock' in drivers/pci/access.c | |
454 | */ | |
022adcfc LP |
455 | static void __iomem *iproc_pcie_map_cfg_bus(struct iproc_pcie *pcie, |
456 | int busno, | |
1fb37a81 RJ |
457 | unsigned int devfn, |
458 | int where) | |
459 | { | |
1fb37a81 RJ |
460 | unsigned slot = PCI_SLOT(devfn); |
461 | unsigned fn = PCI_FUNC(devfn); | |
1fb37a81 | 462 | u32 val; |
943ebae7 RJ |
463 | u16 offset; |
464 | ||
1fb37a81 RJ |
465 | /* root complex access */ |
466 | if (busno == 0) { | |
46560388 RJ |
467 | if (slot > 0 || fn > 0) |
468 | return NULL; | |
469 | ||
943ebae7 RJ |
470 | iproc_pcie_write_reg(pcie, IPROC_PCIE_CFG_IND_ADDR, |
471 | where & CFG_IND_ADDR_MASK); | |
472 | offset = iproc_pcie_reg_offset(pcie, IPROC_PCIE_CFG_IND_DATA); | |
473 | if (iproc_pcie_reg_is_invalid(offset)) | |
1fb37a81 | 474 | return NULL; |
943ebae7 RJ |
475 | else |
476 | return (pcie->base + offset); | |
1fb37a81 RJ |
477 | } |
478 | ||
46560388 RJ |
479 | /* |
480 | * PAXC is connected to an internally emulated EP within the SoC. It | |
481 | * allows only one device. | |
482 | */ | |
06324ede | 483 | if (pcie->ep_is_internal) |
46560388 RJ |
484 | if (slot > 0) |
485 | return NULL; | |
486 | ||
1fb37a81 RJ |
487 | /* EP device access */ |
488 | val = (busno << CFG_ADDR_BUS_NUM_SHIFT) | | |
489 | (slot << CFG_ADDR_DEV_NUM_SHIFT) | | |
490 | (fn << CFG_ADDR_FUNC_NUM_SHIFT) | | |
491 | (where & CFG_ADDR_REG_NUM_MASK) | | |
492 | (1 & CFG_ADDR_CFG_TYPE_MASK); | |
943ebae7 RJ |
493 | iproc_pcie_write_reg(pcie, IPROC_PCIE_CFG_ADDR, val); |
494 | offset = iproc_pcie_reg_offset(pcie, IPROC_PCIE_CFG_DATA); | |
495 | if (iproc_pcie_reg_is_invalid(offset)) | |
496 | return NULL; | |
497 | else | |
498 | return (pcie->base + offset); | |
1fb37a81 RJ |
499 | } |
500 | ||
022adcfc LP |
501 | static void __iomem *iproc_pcie_bus_map_cfg_bus(struct pci_bus *bus, |
502 | unsigned int devfn, | |
503 | int where) | |
504 | { | |
505 | return iproc_pcie_map_cfg_bus(iproc_data(bus), bus->number, devfn, | |
506 | where); | |
507 | } | |
508 | ||
509 | static int iproc_pci_raw_config_read32(struct iproc_pcie *pcie, | |
510 | unsigned int devfn, int where, | |
511 | int size, u32 *val) | |
512 | { | |
513 | void __iomem *addr; | |
514 | ||
515 | addr = iproc_pcie_map_cfg_bus(pcie, 0, devfn, where & ~0x3); | |
516 | if (!addr) { | |
517 | *val = ~0; | |
518 | return PCIBIOS_DEVICE_NOT_FOUND; | |
519 | } | |
520 | ||
521 | *val = readl(addr); | |
522 | ||
523 | if (size <= 2) | |
524 | *val = (*val >> (8 * (where & 3))) & ((1 << (size * 8)) - 1); | |
525 | ||
526 | return PCIBIOS_SUCCESSFUL; | |
527 | } | |
528 | ||
529 | static int iproc_pci_raw_config_write32(struct iproc_pcie *pcie, | |
530 | unsigned int devfn, int where, | |
531 | int size, u32 val) | |
532 | { | |
533 | void __iomem *addr; | |
534 | u32 mask, tmp; | |
535 | ||
536 | addr = iproc_pcie_map_cfg_bus(pcie, 0, devfn, where & ~0x3); | |
537 | if (!addr) | |
538 | return PCIBIOS_DEVICE_NOT_FOUND; | |
539 | ||
540 | if (size == 4) { | |
541 | writel(val, addr); | |
542 | return PCIBIOS_SUCCESSFUL; | |
543 | } | |
544 | ||
545 | mask = ~(((1 << (size * 8)) - 1) << ((where & 0x3) * 8)); | |
546 | tmp = readl(addr) & mask; | |
547 | tmp |= val << ((where & 0x3) * 8); | |
548 | writel(tmp, addr); | |
549 | ||
550 | return PCIBIOS_SUCCESSFUL; | |
551 | } | |
552 | ||
538928fd RJ |
553 | static int iproc_pcie_config_read32(struct pci_bus *bus, unsigned int devfn, |
554 | int where, int size, u32 *val) | |
555 | { | |
556 | int ret; | |
557 | ||
558 | iproc_pcie_apb_err_disable(bus, true); | |
559 | ret = pci_generic_config_read32(bus, devfn, where, size, val); | |
560 | iproc_pcie_apb_err_disable(bus, false); | |
561 | ||
562 | return ret; | |
563 | } | |
564 | ||
565 | static int iproc_pcie_config_write32(struct pci_bus *bus, unsigned int devfn, | |
566 | int where, int size, u32 val) | |
567 | { | |
568 | int ret; | |
569 | ||
570 | iproc_pcie_apb_err_disable(bus, true); | |
571 | ret = pci_generic_config_write32(bus, devfn, where, size, val); | |
572 | iproc_pcie_apb_err_disable(bus, false); | |
573 | ||
574 | return ret; | |
575 | } | |
576 | ||
1fb37a81 | 577 | static struct pci_ops iproc_pcie_ops = { |
022adcfc | 578 | .map_bus = iproc_pcie_bus_map_cfg_bus, |
538928fd RJ |
579 | .read = iproc_pcie_config_read32, |
580 | .write = iproc_pcie_config_write32, | |
1fb37a81 RJ |
581 | }; |
582 | ||
583 | static void iproc_pcie_reset(struct iproc_pcie *pcie) | |
584 | { | |
585 | u32 val; | |
586 | ||
7cbd50d2 RJ |
587 | /* |
588 | * PAXC and the internal emulated endpoint device downstream should not | |
589 | * be reset. If firmware has been loaded on the endpoint device at an | |
590 | * earlier boot stage, reset here causes issues. | |
591 | */ | |
592 | if (pcie->ep_is_internal) | |
943ebae7 | 593 | return; |
943ebae7 | 594 | |
1fb37a81 | 595 | /* |
199ff141 RJ |
596 | * Select perst_b signal as reset source. Put the device into reset, |
597 | * and then bring it out of reset | |
1fb37a81 | 598 | */ |
943ebae7 | 599 | val = iproc_pcie_read_reg(pcie, IPROC_PCIE_CLK_CTRL); |
199ff141 RJ |
600 | val &= ~EP_PERST_SOURCE_SELECT & ~EP_MODE_SURVIVE_PERST & |
601 | ~RC_PCIE_RST_OUTPUT; | |
943ebae7 | 602 | iproc_pcie_write_reg(pcie, IPROC_PCIE_CLK_CTRL, val); |
1fb37a81 | 603 | udelay(250); |
199ff141 RJ |
604 | |
605 | val |= RC_PCIE_RST_OUTPUT; | |
943ebae7 | 606 | iproc_pcie_write_reg(pcie, IPROC_PCIE_CLK_CTRL, val); |
199ff141 | 607 | msleep(100); |
1fb37a81 RJ |
608 | } |
609 | ||
022adcfc | 610 | static int iproc_pcie_check_link(struct iproc_pcie *pcie) |
1fb37a81 | 611 | { |
786aeccb | 612 | struct device *dev = pcie->dev; |
022adcfc LP |
613 | u32 hdr_type, link_ctrl, link_status, class, val; |
614 | u16 pos = PCI_EXP_CAP; | |
aaf22ab4 RJ |
615 | bool link_is_active = false; |
616 | ||
943ebae7 RJ |
617 | /* |
618 | * PAXC connects to emulated endpoint devices directly and does not | |
619 | * have a Serdes. Therefore skip the link detection logic here. | |
620 | */ | |
06324ede | 621 | if (pcie->ep_is_internal) |
943ebae7 RJ |
622 | return 0; |
623 | ||
624 | val = iproc_pcie_read_reg(pcie, IPROC_PCIE_LINK_STATUS); | |
aaf22ab4 | 625 | if (!(val & PCIE_PHYLINKUP) || !(val & PCIE_DL_ACTIVE)) { |
786aeccb | 626 | dev_err(dev, "PHY or data link is INACTIVE!\n"); |
aaf22ab4 RJ |
627 | return -ENODEV; |
628 | } | |
1fb37a81 RJ |
629 | |
630 | /* make sure we are not in EP mode */ | |
022adcfc | 631 | iproc_pci_raw_config_read32(pcie, 0, PCI_HEADER_TYPE, 1, &hdr_type); |
1fb37a81 | 632 | if ((hdr_type & 0x7f) != PCI_HEADER_TYPE_BRIDGE) { |
786aeccb | 633 | dev_err(dev, "in EP mode, hdr=%#02x\n", hdr_type); |
1fb37a81 RJ |
634 | return -EFAULT; |
635 | } | |
636 | ||
637 | /* force class to PCI_CLASS_BRIDGE_PCI (0x0604) */ | |
aaf22ab4 RJ |
638 | #define PCI_BRIDGE_CTRL_REG_OFFSET 0x43c |
639 | #define PCI_CLASS_BRIDGE_MASK 0xffff00 | |
640 | #define PCI_CLASS_BRIDGE_SHIFT 8 | |
022adcfc LP |
641 | iproc_pci_raw_config_read32(pcie, 0, PCI_BRIDGE_CTRL_REG_OFFSET, |
642 | 4, &class); | |
aaf22ab4 RJ |
643 | class &= ~PCI_CLASS_BRIDGE_MASK; |
644 | class |= (PCI_CLASS_BRIDGE_PCI << PCI_CLASS_BRIDGE_SHIFT); | |
022adcfc LP |
645 | iproc_pci_raw_config_write32(pcie, 0, PCI_BRIDGE_CTRL_REG_OFFSET, |
646 | 4, class); | |
1fb37a81 RJ |
647 | |
648 | /* check link status to see if link is active */ | |
022adcfc LP |
649 | iproc_pci_raw_config_read32(pcie, 0, pos + PCI_EXP_LNKSTA, |
650 | 2, &link_status); | |
1fb37a81 | 651 | if (link_status & PCI_EXP_LNKSTA_NLW) |
aaf22ab4 | 652 | link_is_active = true; |
1fb37a81 RJ |
653 | |
654 | if (!link_is_active) { | |
655 | /* try GEN 1 link speed */ | |
1fb37a81 RJ |
656 | #define PCI_TARGET_LINK_SPEED_MASK 0xf |
657 | #define PCI_TARGET_LINK_SPEED_GEN2 0x2 | |
658 | #define PCI_TARGET_LINK_SPEED_GEN1 0x1 | |
022adcfc LP |
659 | iproc_pci_raw_config_read32(pcie, 0, |
660 | pos + PCI_EXP_LNKCTL2, 4, | |
1fb37a81 RJ |
661 | &link_ctrl); |
662 | if ((link_ctrl & PCI_TARGET_LINK_SPEED_MASK) == | |
663 | PCI_TARGET_LINK_SPEED_GEN2) { | |
664 | link_ctrl &= ~PCI_TARGET_LINK_SPEED_MASK; | |
665 | link_ctrl |= PCI_TARGET_LINK_SPEED_GEN1; | |
022adcfc LP |
666 | iproc_pci_raw_config_write32(pcie, 0, |
667 | pos + PCI_EXP_LNKCTL2, | |
668 | 4, link_ctrl); | |
1fb37a81 RJ |
669 | msleep(100); |
670 | ||
022adcfc LP |
671 | iproc_pci_raw_config_read32(pcie, 0, |
672 | pos + PCI_EXP_LNKSTA, | |
673 | 2, &link_status); | |
1fb37a81 | 674 | if (link_status & PCI_EXP_LNKSTA_NLW) |
aaf22ab4 | 675 | link_is_active = true; |
1fb37a81 RJ |
676 | } |
677 | } | |
678 | ||
786aeccb | 679 | dev_info(dev, "link: %s\n", link_is_active ? "UP" : "DOWN"); |
1fb37a81 RJ |
680 | |
681 | return link_is_active ? 0 : -ENODEV; | |
682 | } | |
683 | ||
684 | static void iproc_pcie_enable(struct iproc_pcie *pcie) | |
685 | { | |
943ebae7 | 686 | iproc_pcie_write_reg(pcie, IPROC_PCIE_INTX_EN, SYS_RC_INTX_MASK); |
1fb37a81 RJ |
687 | } |
688 | ||
4213e15c RJ |
689 | static inline bool iproc_pcie_ob_is_valid(struct iproc_pcie *pcie, |
690 | int window_idx) | |
691 | { | |
692 | u32 val; | |
693 | ||
694 | val = iproc_pcie_read_reg(pcie, MAP_REG(IPROC_PCIE_OARR0, window_idx)); | |
695 | ||
696 | return !!(val & OARR_VALID); | |
697 | } | |
698 | ||
699 | static inline int iproc_pcie_ob_write(struct iproc_pcie *pcie, int window_idx, | |
700 | int size_idx, u64 axi_addr, u64 pci_addr) | |
701 | { | |
702 | struct device *dev = pcie->dev; | |
703 | u16 oarr_offset, omap_offset; | |
704 | ||
705 | /* | |
706 | * Derive the OARR/OMAP offset from the first pair (OARR0/OMAP0) based | |
707 | * on window index. | |
708 | */ | |
709 | oarr_offset = iproc_pcie_reg_offset(pcie, MAP_REG(IPROC_PCIE_OARR0, | |
710 | window_idx)); | |
711 | omap_offset = iproc_pcie_reg_offset(pcie, MAP_REG(IPROC_PCIE_OMAP0, | |
712 | window_idx)); | |
713 | if (iproc_pcie_reg_is_invalid(oarr_offset) || | |
714 | iproc_pcie_reg_is_invalid(omap_offset)) | |
715 | return -EINVAL; | |
716 | ||
717 | /* | |
718 | * Program the OARR registers. The upper 32-bit OARR register is | |
719 | * always right after the lower 32-bit OARR register. | |
720 | */ | |
721 | writel(lower_32_bits(axi_addr) | (size_idx << OARR_SIZE_CFG_SHIFT) | | |
722 | OARR_VALID, pcie->base + oarr_offset); | |
723 | writel(upper_32_bits(axi_addr), pcie->base + oarr_offset + 4); | |
724 | ||
725 | /* now program the OMAP registers */ | |
726 | writel(lower_32_bits(pci_addr), pcie->base + omap_offset); | |
727 | writel(upper_32_bits(pci_addr), pcie->base + omap_offset + 4); | |
728 | ||
729 | dev_info(dev, "ob window [%d]: offset 0x%x axi %pap pci %pap\n", | |
730 | window_idx, oarr_offset, &axi_addr, &pci_addr); | |
731 | dev_info(dev, "oarr lo 0x%x oarr hi 0x%x\n", | |
732 | readl(pcie->base + oarr_offset), | |
733 | readl(pcie->base + oarr_offset + 4)); | |
734 | dev_info(dev, "omap lo 0x%x omap hi 0x%x\n", | |
735 | readl(pcie->base + omap_offset), | |
736 | readl(pcie->base + omap_offset + 4)); | |
737 | ||
738 | return 0; | |
739 | } | |
740 | ||
e99a187b RJ |
741 | /** |
742 | * Some iProc SoCs require the SW to configure the outbound address mapping | |
743 | * | |
744 | * Outbound address translation: | |
745 | * | |
746 | * iproc_pcie_address = axi_address - axi_offset | |
747 | * OARR = iproc_pcie_address | |
748 | * OMAP = pci_addr | |
749 | * | |
750 | * axi_addr -> iproc_pcie_address -> OARR -> OMAP -> pci_address | |
751 | */ | |
752 | static int iproc_pcie_setup_ob(struct iproc_pcie *pcie, u64 axi_addr, | |
753 | u64 pci_addr, resource_size_t size) | |
754 | { | |
755 | struct iproc_pcie_ob *ob = &pcie->ob; | |
786aeccb | 756 | struct device *dev = pcie->dev; |
4213e15c | 757 | int ret = -EINVAL, window_idx, size_idx; |
e99a187b RJ |
758 | |
759 | if (axi_addr < ob->axi_offset) { | |
786aeccb | 760 | dev_err(dev, "axi address %pap less than offset %pap\n", |
e99a187b RJ |
761 | &axi_addr, &ob->axi_offset); |
762 | return -EINVAL; | |
763 | } | |
764 | ||
765 | /* | |
766 | * Translate the AXI address to the internal address used by the iProc | |
767 | * PCIe core before programming the OARR | |
768 | */ | |
769 | axi_addr -= ob->axi_offset; | |
770 | ||
4213e15c RJ |
771 | /* iterate through all OARR/OMAP mapping windows */ |
772 | for (window_idx = ob->nr_windows - 1; window_idx >= 0; window_idx--) { | |
773 | const struct iproc_pcie_ob_map *ob_map = | |
774 | &pcie->ob_map[window_idx]; | |
775 | ||
776 | /* | |
777 | * If current outbound window is already in use, move on to the | |
778 | * next one. | |
779 | */ | |
780 | if (iproc_pcie_ob_is_valid(pcie, window_idx)) | |
781 | continue; | |
782 | ||
783 | /* | |
784 | * Iterate through all supported window sizes within the | |
785 | * OARR/OMAP pair to find a match. Go through the window sizes | |
786 | * in a descending order. | |
787 | */ | |
788 | for (size_idx = ob_map->nr_sizes - 1; size_idx >= 0; | |
789 | size_idx--) { | |
790 | resource_size_t window_size = | |
791 | ob_map->window_sizes[size_idx] * SZ_1M; | |
792 | ||
793 | if (size < window_size) | |
794 | continue; | |
795 | ||
796 | if (!IS_ALIGNED(axi_addr, window_size) || | |
797 | !IS_ALIGNED(pci_addr, window_size)) { | |
798 | dev_err(dev, | |
799 | "axi %pap or pci %pap not aligned\n", | |
800 | &axi_addr, &pci_addr); | |
801 | return -EINVAL; | |
802 | } | |
803 | ||
804 | /* | |
805 | * Match found! Program both OARR and OMAP and mark | |
806 | * them as a valid entry. | |
807 | */ | |
808 | ret = iproc_pcie_ob_write(pcie, window_idx, size_idx, | |
809 | axi_addr, pci_addr); | |
810 | if (ret) | |
811 | goto err_ob; | |
812 | ||
813 | size -= window_size; | |
814 | if (size == 0) | |
815 | return 0; | |
816 | ||
817 | /* | |
818 | * If we are here, we are done with the current window, | |
819 | * but not yet finished all mappings. Need to move on | |
820 | * to the next window. | |
821 | */ | |
822 | axi_addr += window_size; | |
823 | pci_addr += window_size; | |
e99a187b | 824 | break; |
4213e15c | 825 | } |
e99a187b RJ |
826 | } |
827 | ||
4213e15c RJ |
828 | err_ob: |
829 | dev_err(dev, "unable to configure outbound mapping\n"); | |
830 | dev_err(dev, | |
831 | "axi %pap, axi offset %pap, pci %pap, res size %pap\n", | |
832 | &axi_addr, &ob->axi_offset, &pci_addr, &size); | |
833 | ||
834 | return ret; | |
e99a187b RJ |
835 | } |
836 | ||
837 | static int iproc_pcie_map_ranges(struct iproc_pcie *pcie, | |
838 | struct list_head *resources) | |
839 | { | |
786aeccb | 840 | struct device *dev = pcie->dev; |
e99a187b RJ |
841 | struct resource_entry *window; |
842 | int ret; | |
843 | ||
844 | resource_list_for_each_entry(window, resources) { | |
845 | struct resource *res = window->res; | |
846 | u64 res_type = resource_type(res); | |
847 | ||
848 | switch (res_type) { | |
849 | case IORESOURCE_IO: | |
850 | case IORESOURCE_BUS: | |
851 | break; | |
852 | case IORESOURCE_MEM: | |
853 | ret = iproc_pcie_setup_ob(pcie, res->start, | |
854 | res->start - window->offset, | |
855 | resource_size(res)); | |
856 | if (ret) | |
857 | return ret; | |
858 | break; | |
859 | default: | |
786aeccb | 860 | dev_err(dev, "invalid resource %pR\n", res); |
e99a187b RJ |
861 | return -EINVAL; |
862 | } | |
863 | } | |
864 | ||
865 | return 0; | |
866 | } | |
867 | ||
dd9d4e74 RJ |
868 | static inline bool iproc_pcie_ib_is_in_use(struct iproc_pcie *pcie, |
869 | int region_idx) | |
870 | { | |
871 | const struct iproc_pcie_ib_map *ib_map = &pcie->ib_map[region_idx]; | |
872 | u32 val; | |
873 | ||
874 | val = iproc_pcie_read_reg(pcie, MAP_REG(IPROC_PCIE_IARR0, region_idx)); | |
875 | ||
876 | return !!(val & (BIT(ib_map->nr_sizes) - 1)); | |
877 | } | |
878 | ||
879 | static inline bool iproc_pcie_ib_check_type(const struct iproc_pcie_ib_map *ib_map, | |
880 | enum iproc_pcie_ib_map_type type) | |
881 | { | |
882 | return !!(ib_map->type == type); | |
883 | } | |
884 | ||
885 | static int iproc_pcie_ib_write(struct iproc_pcie *pcie, int region_idx, | |
886 | int size_idx, int nr_windows, u64 axi_addr, | |
887 | u64 pci_addr, resource_size_t size) | |
888 | { | |
889 | struct device *dev = pcie->dev; | |
890 | const struct iproc_pcie_ib_map *ib_map = &pcie->ib_map[region_idx]; | |
891 | u16 iarr_offset, imap_offset; | |
892 | u32 val; | |
893 | int window_idx; | |
894 | ||
895 | iarr_offset = iproc_pcie_reg_offset(pcie, | |
896 | MAP_REG(IPROC_PCIE_IARR0, region_idx)); | |
897 | imap_offset = iproc_pcie_reg_offset(pcie, | |
898 | MAP_REG(IPROC_PCIE_IMAP0, region_idx)); | |
899 | if (iproc_pcie_reg_is_invalid(iarr_offset) || | |
900 | iproc_pcie_reg_is_invalid(imap_offset)) | |
901 | return -EINVAL; | |
902 | ||
903 | dev_info(dev, "ib region [%d]: offset 0x%x axi %pap pci %pap\n", | |
904 | region_idx, iarr_offset, &axi_addr, &pci_addr); | |
905 | ||
906 | /* | |
907 | * Program the IARR registers. The upper 32-bit IARR register is | |
908 | * always right after the lower 32-bit IARR register. | |
909 | */ | |
910 | writel(lower_32_bits(pci_addr) | BIT(size_idx), | |
911 | pcie->base + iarr_offset); | |
912 | writel(upper_32_bits(pci_addr), pcie->base + iarr_offset + 4); | |
913 | ||
914 | dev_info(dev, "iarr lo 0x%x iarr hi 0x%x\n", | |
915 | readl(pcie->base + iarr_offset), | |
916 | readl(pcie->base + iarr_offset + 4)); | |
917 | ||
918 | /* | |
919 | * Now program the IMAP registers. Each IARR region may have one or | |
920 | * more IMAP windows. | |
921 | */ | |
922 | size >>= ilog2(nr_windows); | |
923 | for (window_idx = 0; window_idx < nr_windows; window_idx++) { | |
924 | val = readl(pcie->base + imap_offset); | |
925 | val |= lower_32_bits(axi_addr) | IMAP_VALID; | |
926 | writel(val, pcie->base + imap_offset); | |
927 | writel(upper_32_bits(axi_addr), | |
928 | pcie->base + imap_offset + ib_map->imap_addr_offset); | |
929 | ||
930 | dev_info(dev, "imap window [%d] lo 0x%x hi 0x%x\n", | |
931 | window_idx, readl(pcie->base + imap_offset), | |
932 | readl(pcie->base + imap_offset + | |
933 | ib_map->imap_addr_offset)); | |
934 | ||
935 | imap_offset += ib_map->imap_window_offset; | |
936 | axi_addr += size; | |
937 | } | |
938 | ||
939 | return 0; | |
940 | } | |
941 | ||
942 | static int iproc_pcie_setup_ib(struct iproc_pcie *pcie, | |
943 | struct of_pci_range *range, | |
944 | enum iproc_pcie_ib_map_type type) | |
945 | { | |
946 | struct device *dev = pcie->dev; | |
947 | struct iproc_pcie_ib *ib = &pcie->ib; | |
948 | int ret; | |
949 | unsigned int region_idx, size_idx; | |
950 | u64 axi_addr = range->cpu_addr, pci_addr = range->pci_addr; | |
951 | resource_size_t size = range->size; | |
952 | ||
953 | /* iterate through all IARR mapping regions */ | |
954 | for (region_idx = 0; region_idx < ib->nr_regions; region_idx++) { | |
955 | const struct iproc_pcie_ib_map *ib_map = | |
956 | &pcie->ib_map[region_idx]; | |
957 | ||
958 | /* | |
959 | * If current inbound region is already in use or not a | |
960 | * compatible type, move on to the next. | |
961 | */ | |
962 | if (iproc_pcie_ib_is_in_use(pcie, region_idx) || | |
963 | !iproc_pcie_ib_check_type(ib_map, type)) | |
964 | continue; | |
965 | ||
966 | /* iterate through all supported region sizes to find a match */ | |
967 | for (size_idx = 0; size_idx < ib_map->nr_sizes; size_idx++) { | |
968 | resource_size_t region_size = | |
969 | ib_map->region_sizes[size_idx] * ib_map->size_unit; | |
970 | ||
971 | if (size != region_size) | |
972 | continue; | |
973 | ||
974 | if (!IS_ALIGNED(axi_addr, region_size) || | |
975 | !IS_ALIGNED(pci_addr, region_size)) { | |
976 | dev_err(dev, | |
977 | "axi %pap or pci %pap not aligned\n", | |
978 | &axi_addr, &pci_addr); | |
979 | return -EINVAL; | |
980 | } | |
981 | ||
982 | /* Match found! Program IARR and all IMAP windows. */ | |
983 | ret = iproc_pcie_ib_write(pcie, region_idx, size_idx, | |
984 | ib_map->nr_windows, axi_addr, | |
985 | pci_addr, size); | |
986 | if (ret) | |
987 | goto err_ib; | |
988 | else | |
989 | return 0; | |
990 | ||
991 | } | |
992 | } | |
993 | ret = -EINVAL; | |
994 | ||
995 | err_ib: | |
996 | dev_err(dev, "unable to configure inbound mapping\n"); | |
997 | dev_err(dev, "axi %pap, pci %pap, res size %pap\n", | |
998 | &axi_addr, &pci_addr, &size); | |
999 | ||
1000 | return ret; | |
1001 | } | |
1002 | ||
1003 | static int pci_dma_range_parser_init(struct of_pci_range_parser *parser, | |
1004 | struct device_node *node) | |
1005 | { | |
1006 | const int na = 3, ns = 2; | |
1007 | int rlen; | |
1008 | ||
1009 | parser->node = node; | |
1010 | parser->pna = of_n_addr_cells(node); | |
1011 | parser->np = parser->pna + na + ns; | |
1012 | ||
1013 | parser->range = of_get_property(node, "dma-ranges", &rlen); | |
1014 | if (!parser->range) | |
1015 | return -ENOENT; | |
1016 | ||
1017 | parser->end = parser->range + rlen / sizeof(__be32); | |
1018 | return 0; | |
1019 | } | |
1020 | ||
1021 | static int iproc_pcie_map_dma_ranges(struct iproc_pcie *pcie) | |
1022 | { | |
1023 | struct of_pci_range range; | |
1024 | struct of_pci_range_parser parser; | |
1025 | int ret; | |
1026 | ||
1027 | /* Get the dma-ranges from DT */ | |
1028 | ret = pci_dma_range_parser_init(&parser, pcie->dev->of_node); | |
1029 | if (ret) | |
1030 | return ret; | |
1031 | ||
1032 | for_each_of_pci_range(&parser, &range) { | |
1033 | /* Each range entry corresponds to an inbound mapping region */ | |
1034 | ret = iproc_pcie_setup_ib(pcie, &range, IPROC_PCIE_IB_MAP_MEM); | |
1035 | if (ret) | |
1036 | return ret; | |
1037 | } | |
1038 | ||
1039 | return 0; | |
1040 | } | |
1041 | ||
787b3c4f RJ |
1042 | static int iproce_pcie_get_msi(struct iproc_pcie *pcie, |
1043 | struct device_node *msi_node, | |
1044 | u64 *msi_addr) | |
1045 | { | |
1046 | struct device *dev = pcie->dev; | |
1047 | int ret; | |
1048 | struct resource res; | |
1049 | ||
1050 | /* | |
1051 | * Check if 'msi-map' points to ARM GICv3 ITS, which is the only | |
1052 | * supported external MSI controller that requires steering. | |
1053 | */ | |
1054 | if (!of_device_is_compatible(msi_node, "arm,gic-v3-its")) { | |
1055 | dev_err(dev, "unable to find compatible MSI controller\n"); | |
1056 | return -ENODEV; | |
1057 | } | |
1058 | ||
1059 | /* derive GITS_TRANSLATER address from GICv3 */ | |
1060 | ret = of_address_to_resource(msi_node, 0, &res); | |
1061 | if (ret < 0) { | |
1062 | dev_err(dev, "unable to obtain MSI controller resources\n"); | |
1063 | return ret; | |
1064 | } | |
1065 | ||
1066 | *msi_addr = res.start + GITS_TRANSLATER; | |
1067 | return 0; | |
1068 | } | |
1069 | ||
c7c44527 RJ |
1070 | static int iproc_pcie_paxb_v2_msi_steer(struct iproc_pcie *pcie, u64 msi_addr) |
1071 | { | |
1072 | int ret; | |
1073 | struct of_pci_range range; | |
1074 | ||
1075 | memset(&range, 0, sizeof(range)); | |
1076 | range.size = SZ_32K; | |
feacdb4a | 1077 | range.pci_addr = range.cpu_addr = msi_addr & ~(range.size - 1); |
c7c44527 RJ |
1078 | |
1079 | ret = iproc_pcie_setup_ib(pcie, &range, IPROC_PCIE_IB_MAP_IO); | |
1080 | return ret; | |
1081 | } | |
1082 | ||
787b3c4f RJ |
1083 | static void iproc_pcie_paxc_v2_msi_steer(struct iproc_pcie *pcie, u64 msi_addr) |
1084 | { | |
1085 | u32 val; | |
1086 | ||
1087 | /* | |
1088 | * Program bits [43:13] of address of GITS_TRANSLATER register into | |
1089 | * bits [30:0] of the MSI base address register. In fact, in all iProc | |
1090 | * based SoCs, all I/O register bases are well below the 32-bit | |
1091 | * boundary, so we can safely assume bits [43:32] are always zeros. | |
1092 | */ | |
1093 | iproc_pcie_write_reg(pcie, IPROC_PCIE_MSI_BASE_ADDR, | |
1094 | (u32)(msi_addr >> 13)); | |
1095 | ||
1096 | /* use a default 8K window size */ | |
1097 | iproc_pcie_write_reg(pcie, IPROC_PCIE_MSI_WINDOW_SIZE, 0); | |
1098 | ||
1099 | /* steering MSI to GICv3 ITS */ | |
1100 | val = iproc_pcie_read_reg(pcie, IPROC_PCIE_MSI_GIC_MODE); | |
1101 | val |= GIC_V3_CFG; | |
1102 | iproc_pcie_write_reg(pcie, IPROC_PCIE_MSI_GIC_MODE, val); | |
1103 | ||
1104 | /* | |
1105 | * Program bits [43:2] of address of GITS_TRANSLATER register into the | |
1106 | * iProc MSI address registers. | |
1107 | */ | |
1108 | msi_addr >>= 2; | |
1109 | iproc_pcie_write_reg(pcie, IPROC_PCIE_MSI_ADDR_HI, | |
1110 | upper_32_bits(msi_addr)); | |
1111 | iproc_pcie_write_reg(pcie, IPROC_PCIE_MSI_ADDR_LO, | |
1112 | lower_32_bits(msi_addr)); | |
1113 | ||
1114 | /* enable MSI */ | |
1115 | val = iproc_pcie_read_reg(pcie, IPROC_PCIE_MSI_EN_CFG); | |
1116 | val |= MSI_ENABLE_CFG; | |
1117 | iproc_pcie_write_reg(pcie, IPROC_PCIE_MSI_EN_CFG, val); | |
1118 | } | |
1119 | ||
1120 | static int iproc_pcie_msi_steer(struct iproc_pcie *pcie, | |
1121 | struct device_node *msi_node) | |
1122 | { | |
1123 | struct device *dev = pcie->dev; | |
1124 | int ret; | |
1125 | u64 msi_addr; | |
1126 | ||
1127 | ret = iproce_pcie_get_msi(pcie, msi_node, &msi_addr); | |
1128 | if (ret < 0) { | |
1129 | dev_err(dev, "msi steering failed\n"); | |
1130 | return ret; | |
1131 | } | |
1132 | ||
1133 | switch (pcie->type) { | |
c7c44527 RJ |
1134 | case IPROC_PCIE_PAXB_V2: |
1135 | ret = iproc_pcie_paxb_v2_msi_steer(pcie, msi_addr); | |
1136 | if (ret) | |
1137 | return ret; | |
1138 | break; | |
787b3c4f RJ |
1139 | case IPROC_PCIE_PAXC_V2: |
1140 | iproc_pcie_paxc_v2_msi_steer(pcie, msi_addr); | |
1141 | break; | |
1142 | default: | |
1143 | return -EINVAL; | |
1144 | } | |
1145 | ||
1146 | return 0; | |
1147 | } | |
1148 | ||
3bc2b234 RJ |
1149 | static int iproc_pcie_msi_enable(struct iproc_pcie *pcie) |
1150 | { | |
1151 | struct device_node *msi_node; | |
787b3c4f RJ |
1152 | int ret; |
1153 | ||
1154 | /* | |
1155 | * Either the "msi-parent" or the "msi-map" phandle needs to exist | |
1156 | * for us to obtain the MSI node. | |
1157 | */ | |
3bc2b234 RJ |
1158 | |
1159 | msi_node = of_parse_phandle(pcie->dev->of_node, "msi-parent", 0); | |
787b3c4f RJ |
1160 | if (!msi_node) { |
1161 | const __be32 *msi_map = NULL; | |
1162 | int len; | |
1163 | u32 phandle; | |
1164 | ||
1165 | msi_map = of_get_property(pcie->dev->of_node, "msi-map", &len); | |
1166 | if (!msi_map) | |
1167 | return -ENODEV; | |
1168 | ||
1169 | phandle = be32_to_cpup(msi_map + 1); | |
1170 | msi_node = of_find_node_by_phandle(phandle); | |
1171 | if (!msi_node) | |
1172 | return -ENODEV; | |
1173 | } | |
1174 | ||
1175 | /* | |
1176 | * Certain revisions of the iProc PCIe controller require additional | |
1177 | * configurations to steer the MSI writes towards an external MSI | |
1178 | * controller. | |
1179 | */ | |
1180 | if (pcie->need_msi_steer) { | |
1181 | ret = iproc_pcie_msi_steer(pcie, msi_node); | |
1182 | if (ret) | |
1183 | return ret; | |
1184 | } | |
3bc2b234 RJ |
1185 | |
1186 | /* | |
1187 | * If another MSI controller is being used, the call below should fail | |
1188 | * but that is okay | |
1189 | */ | |
1190 | return iproc_msi_init(pcie, msi_node); | |
1191 | } | |
1192 | ||
1193 | static void iproc_pcie_msi_disable(struct iproc_pcie *pcie) | |
1194 | { | |
1195 | iproc_msi_exit(pcie); | |
1196 | } | |
1197 | ||
06324ede RJ |
1198 | static int iproc_pcie_rev_init(struct iproc_pcie *pcie) |
1199 | { | |
1200 | struct device *dev = pcie->dev; | |
1201 | unsigned int reg_idx; | |
1202 | const u16 *regs; | |
1203 | ||
1204 | switch (pcie->type) { | |
404349c5 RJ |
1205 | case IPROC_PCIE_PAXB_BCMA: |
1206 | regs = iproc_pcie_reg_paxb_bcma; | |
1207 | break; | |
06324ede RJ |
1208 | case IPROC_PCIE_PAXB: |
1209 | regs = iproc_pcie_reg_paxb; | |
538928fd | 1210 | pcie->has_apb_err_disable = true; |
4213e15c RJ |
1211 | if (pcie->need_ob_cfg) { |
1212 | pcie->ob_map = paxb_ob_map; | |
1213 | pcie->ob.nr_windows = ARRAY_SIZE(paxb_ob_map); | |
1214 | } | |
06324ede | 1215 | break; |
c7c44527 RJ |
1216 | case IPROC_PCIE_PAXB_V2: |
1217 | regs = iproc_pcie_reg_paxb_v2; | |
1218 | pcie->has_apb_err_disable = true; | |
1219 | if (pcie->need_ob_cfg) { | |
1220 | pcie->ob_map = paxb_v2_ob_map; | |
1221 | pcie->ob.nr_windows = ARRAY_SIZE(paxb_v2_ob_map); | |
1222 | } | |
1223 | pcie->ib.nr_regions = ARRAY_SIZE(paxb_v2_ib_map); | |
1224 | pcie->ib_map = paxb_v2_ib_map; | |
1225 | pcie->need_msi_steer = true; | |
1226 | break; | |
06324ede RJ |
1227 | case IPROC_PCIE_PAXC: |
1228 | regs = iproc_pcie_reg_paxc; | |
1229 | pcie->ep_is_internal = true; | |
1230 | break; | |
787b3c4f RJ |
1231 | case IPROC_PCIE_PAXC_V2: |
1232 | regs = iproc_pcie_reg_paxc_v2; | |
1233 | pcie->ep_is_internal = true; | |
1234 | pcie->need_msi_steer = true; | |
1235 | break; | |
06324ede RJ |
1236 | default: |
1237 | dev_err(dev, "incompatible iProc PCIe interface\n"); | |
1238 | return -EINVAL; | |
1239 | } | |
1240 | ||
1241 | pcie->reg_offsets = devm_kcalloc(dev, IPROC_PCIE_MAX_NUM_REG, | |
1242 | sizeof(*pcie->reg_offsets), | |
1243 | GFP_KERNEL); | |
1244 | if (!pcie->reg_offsets) | |
1245 | return -ENOMEM; | |
1246 | ||
1247 | /* go through the register table and populate all valid registers */ | |
787b3c4f RJ |
1248 | pcie->reg_offsets[0] = (pcie->type == IPROC_PCIE_PAXC_V2) ? |
1249 | IPROC_PCIE_REG_INVALID : regs[0]; | |
06324ede RJ |
1250 | for (reg_idx = 1; reg_idx < IPROC_PCIE_MAX_NUM_REG; reg_idx++) |
1251 | pcie->reg_offsets[reg_idx] = regs[reg_idx] ? | |
1252 | regs[reg_idx] : IPROC_PCIE_REG_INVALID; | |
1253 | ||
1254 | return 0; | |
1255 | } | |
1256 | ||
18c4342a | 1257 | int iproc_pcie_setup(struct iproc_pcie *pcie, struct list_head *res) |
1fb37a81 | 1258 | { |
786aeccb | 1259 | struct device *dev; |
1fb37a81 | 1260 | int ret; |
8d9bfe37 | 1261 | void *sysdata; |
52774076 LP |
1262 | struct pci_bus *child; |
1263 | struct pci_host_bridge *host = pci_host_bridge_from_priv(pcie); | |
1fb37a81 | 1264 | |
786aeccb | 1265 | dev = pcie->dev; |
06324ede RJ |
1266 | |
1267 | ret = iproc_pcie_rev_init(pcie); | |
1268 | if (ret) { | |
1269 | dev_err(dev, "unable to initialize controller parameters\n"); | |
1270 | return ret; | |
1271 | } | |
1272 | ||
786aeccb | 1273 | ret = devm_request_pci_bus_resources(dev, res); |
c3245a56 BH |
1274 | if (ret) |
1275 | return ret; | |
1276 | ||
93972d18 ME |
1277 | ret = phy_init(pcie->phy); |
1278 | if (ret) { | |
786aeccb | 1279 | dev_err(dev, "unable to initialize PCIe PHY\n"); |
93972d18 ME |
1280 | return ret; |
1281 | } | |
1fb37a81 | 1282 | |
93972d18 ME |
1283 | ret = phy_power_on(pcie->phy); |
1284 | if (ret) { | |
786aeccb | 1285 | dev_err(dev, "unable to power on PCIe PHY\n"); |
93972d18 | 1286 | goto err_exit_phy; |
1fb37a81 RJ |
1287 | } |
1288 | ||
1289 | iproc_pcie_reset(pcie); | |
1290 | ||
e99a187b RJ |
1291 | if (pcie->need_ob_cfg) { |
1292 | ret = iproc_pcie_map_ranges(pcie, res); | |
1293 | if (ret) { | |
786aeccb | 1294 | dev_err(dev, "map failed\n"); |
e99a187b RJ |
1295 | goto err_power_off_phy; |
1296 | } | |
1297 | } | |
1298 | ||
dd9d4e74 RJ |
1299 | ret = iproc_pcie_map_dma_ranges(pcie); |
1300 | if (ret && ret != -ENOENT) | |
1301 | goto err_power_off_phy; | |
1302 | ||
8d9bfe37 | 1303 | #ifdef CONFIG_ARM |
1fb37a81 | 1304 | pcie->sysdata.private_data = pcie; |
8d9bfe37 RJ |
1305 | sysdata = &pcie->sysdata; |
1306 | #else | |
1307 | sysdata = pcie; | |
1308 | #endif | |
1fb37a81 | 1309 | |
022adcfc | 1310 | ret = iproc_pcie_check_link(pcie); |
1fb37a81 | 1311 | if (ret) { |
786aeccb | 1312 | dev_err(dev, "no PCIe EP device detected\n"); |
52774076 | 1313 | goto err_power_off_phy; |
1fb37a81 RJ |
1314 | } |
1315 | ||
1316 | iproc_pcie_enable(pcie); | |
1317 | ||
3bc2b234 RJ |
1318 | if (IS_ENABLED(CONFIG_PCI_MSI)) |
1319 | if (iproc_pcie_msi_enable(pcie)) | |
786aeccb | 1320 | dev_info(dev, "not using iProc MSI\n"); |
3bc2b234 | 1321 | |
52774076 LP |
1322 | list_splice_init(res, &host->windows); |
1323 | host->busnr = 0; | |
1324 | host->dev.parent = dev; | |
1325 | host->ops = &iproc_pcie_ops; | |
1326 | host->sysdata = sysdata; | |
1327 | ||
1328 | ret = pci_scan_root_bus_bridge(host); | |
1329 | if (ret < 0) { | |
1330 | dev_err(dev, "failed to scan host: %d\n", ret); | |
1331 | goto err_power_off_phy; | |
1332 | } | |
ffbd7968 AG |
1333 | |
1334 | if (pcie->map_irq) | |
1335 | pci_fixup_irqs(pci_common_swizzle, pcie->map_irq); | |
1336 | ||
52774076 LP |
1337 | pci_assign_unassigned_bus_resources(host->bus); |
1338 | ||
1339 | pcie->root_bus = host->bus; | |
1340 | ||
1341 | list_for_each_entry(child, &host->bus->children, node) | |
4d4836ab JM |
1342 | pcie_bus_configure_settings(child); |
1343 | ||
52774076 | 1344 | pci_bus_add_devices(host->bus); |
1fb37a81 RJ |
1345 | |
1346 | return 0; | |
1347 | ||
1fb37a81 | 1348 | err_power_off_phy: |
93972d18 | 1349 | phy_power_off(pcie->phy); |
1fb37a81 | 1350 | err_exit_phy: |
93972d18 | 1351 | phy_exit(pcie->phy); |
1fb37a81 RJ |
1352 | return ret; |
1353 | } | |
1354 | EXPORT_SYMBOL(iproc_pcie_setup); | |
1355 | ||
1356 | int iproc_pcie_remove(struct iproc_pcie *pcie) | |
1357 | { | |
1358 | pci_stop_root_bus(pcie->root_bus); | |
1359 | pci_remove_root_bus(pcie->root_bus); | |
1360 | ||
3bc2b234 RJ |
1361 | iproc_pcie_msi_disable(pcie); |
1362 | ||
93972d18 ME |
1363 | phy_power_off(pcie->phy); |
1364 | phy_exit(pcie->phy); | |
1fb37a81 RJ |
1365 | |
1366 | return 0; | |
1367 | } | |
1368 | EXPORT_SYMBOL(iproc_pcie_remove); | |
1369 | ||
1370 | MODULE_AUTHOR("Ray Jui <rjui@broadcom.com>"); | |
1371 | MODULE_DESCRIPTION("Broadcom iPROC PCIe common driver"); | |
1372 | MODULE_LICENSE("GPL v2"); |