]>
Commit | Line | Data |
---|---|---|
5b7ec7fb NA |
1 | // SPDX-License-Identifier: GPL-2.0+ |
2 | ||
3 | #include <clk.h> | |
4 | #include <dm.h> | |
5 | #include <generic-phy.h> | |
6 | #include <pci.h> | |
7 | #include <u-boot/crc.h> | |
8 | #include <power-domain.h> | |
9 | #include <reset.h> | |
10 | #include <syscon.h> | |
11 | #include <malloc.h> | |
12 | #include <power/regulator.h> | |
13 | #include <asm/global_data.h> | |
14 | #include <asm/io.h> | |
15 | #include <asm-generic/gpio.h> | |
16 | #include <dm/device_compat.h> | |
17 | #include <linux/iopoll.h> | |
18 | #include <linux/delay.h> | |
19 | #include <linux/log2.h> | |
20 | #include <linux/bitfield.h> | |
21 | ||
22 | #include "pcie_dw_common.h" | |
23 | ||
24 | DECLARE_GLOBAL_DATA_PTR; | |
25 | ||
26 | struct qcom_pcie; | |
27 | ||
28 | struct qcom_pcie_ops { | |
29 | int (*config_sid)(struct qcom_pcie *priv); | |
30 | }; | |
31 | ||
32 | #define NUM_SUPPLIES 2 | |
33 | ||
34 | struct qcom_pcie { | |
35 | /* Must be first member of the struct */ | |
36 | struct pcie_dw dw; | |
37 | void *parf; | |
38 | struct phy phy; | |
39 | struct reset_ctl_bulk rsts; | |
40 | struct clk_bulk clks; | |
41 | struct gpio_desc rst_gpio; | |
42 | struct qcom_pcie_ops *ops; | |
43 | struct udevice *vregs[NUM_SUPPLIES]; | |
44 | }; | |
45 | ||
46 | /* PARF registers */ | |
47 | #define PARF_SYS_CTRL 0x00 | |
48 | #define PARF_PM_CTRL 0x20 | |
49 | #define PARF_PCS_DEEMPH 0x34 | |
50 | #define PARF_PCS_SWING 0x38 | |
51 | #define PARF_PHY_CTRL 0x40 | |
52 | #define PARF_PHY_REFCLK 0x4c | |
53 | #define PARF_CONFIG_BITS 0x50 | |
54 | #define PARF_DBI_BASE_ADDR 0x168 | |
55 | #define PARF_MHI_CLOCK_RESET_CTRL 0x174 | |
56 | #define PARF_AXI_MSTR_WR_ADDR_HALT 0x178 | |
57 | #define PARF_AXI_MSTR_WR_ADDR_HALT_V2 0x1a8 | |
58 | #define PARF_Q2A_FLUSH 0x1ac | |
59 | #define PARF_LTSSM 0x1b0 | |
60 | #define PARF_SID_OFFSET 0x234 | |
61 | #define PARF_BDF_TRANSLATE_CFG 0x24c | |
62 | #define PARF_SLV_ADDR_SPACE_SIZE 0x358 | |
63 | #define PARF_DEVICE_TYPE 0x1000 | |
64 | #define PARF_BDF_TO_SID_TABLE_N 0x2000 | |
65 | ||
66 | /* ELBI registers */ | |
67 | #define ELBI_SYS_CTRL 0x04 | |
68 | ||
69 | /* DBI registers */ | |
70 | #define AXI_MSTR_RESP_COMP_CTRL0 0x818 | |
71 | #define AXI_MSTR_RESP_COMP_CTRL1 0x81c | |
72 | #define MISC_CONTROL_1_REG 0x8bc | |
73 | ||
74 | /* MHI registers */ | |
75 | #define PARF_DEBUG_CNT_PM_LINKST_IN_L2 0xc04 | |
76 | #define PARF_DEBUG_CNT_PM_LINKST_IN_L1 0xc0c | |
77 | #define PARF_DEBUG_CNT_PM_LINKST_IN_L0S 0xc10 | |
78 | #define PARF_DEBUG_CNT_AUX_CLK_IN_L1SUB_L1 0xc84 | |
79 | #define PARF_DEBUG_CNT_AUX_CLK_IN_L1SUB_L2 0xc88 | |
80 | ||
81 | /* PARF_SYS_CTRL register fields */ | |
82 | #define MAC_PHY_POWERDOWN_IN_P2_D_MUX_EN BIT(29) | |
83 | #define MST_WAKEUP_EN BIT(13) | |
84 | #define SLV_WAKEUP_EN BIT(12) | |
85 | #define MSTR_ACLK_CGC_DIS BIT(10) | |
86 | #define SLV_ACLK_CGC_DIS BIT(9) | |
87 | #define CORE_CLK_CGC_DIS BIT(6) | |
88 | #define AUX_PWR_DET BIT(4) | |
89 | #define L23_CLK_RMV_DIS BIT(2) | |
90 | #define L1_CLK_RMV_DIS BIT(1) | |
91 | ||
92 | /* PARF_PM_CTRL register fields */ | |
93 | #define REQ_NOT_ENTR_L1 BIT(5) | |
94 | ||
95 | /* PARF_PCS_DEEMPH register fields */ | |
96 | #define PCS_DEEMPH_TX_DEEMPH_GEN1(x) FIELD_PREP(GENMASK(21, 16), x) | |
97 | #define PCS_DEEMPH_TX_DEEMPH_GEN2_3_5DB(x) FIELD_PREP(GENMASK(13, 8), x) | |
98 | #define PCS_DEEMPH_TX_DEEMPH_GEN2_6DB(x) FIELD_PREP(GENMASK(5, 0), x) | |
99 | ||
100 | /* PARF_PCS_SWING register fields */ | |
101 | #define PCS_SWING_TX_SWING_FULL(x) FIELD_PREP(GENMASK(14, 8), x) | |
102 | #define PCS_SWING_TX_SWING_LOW(x) FIELD_PREP(GENMASK(6, 0), x) | |
103 | ||
104 | /* PARF_PHY_CTRL register fields */ | |
105 | #define PHY_CTRL_PHY_TX0_TERM_OFFSET_MASK GENMASK(20, 16) | |
106 | #define PHY_CTRL_PHY_TX0_TERM_OFFSET(x) FIELD_PREP(PHY_CTRL_PHY_TX0_TERM_OFFSET_MASK, x) | |
107 | #define PHY_TEST_PWR_DOWN BIT(0) | |
108 | ||
109 | /* PARF_PHY_REFCLK register fields */ | |
110 | #define PHY_REFCLK_SSP_EN BIT(16) | |
111 | #define PHY_REFCLK_USE_PAD BIT(12) | |
112 | ||
113 | /* PARF_CONFIG_BITS register fields */ | |
114 | #define PHY_RX0_EQ(x) FIELD_PREP(GENMASK(26, 24), x) | |
115 | ||
116 | /* PARF_SLV_ADDR_SPACE_SIZE register value */ | |
117 | #define SLV_ADDR_SPACE_SZ 0x10000000 | |
118 | ||
119 | /* PARF_MHI_CLOCK_RESET_CTRL register fields */ | |
120 | #define AHB_CLK_EN BIT(0) | |
121 | #define MSTR_AXI_CLK_EN BIT(1) | |
122 | #define BYPASS BIT(4) | |
123 | ||
124 | /* PARF_AXI_MSTR_WR_ADDR_HALT register fields */ | |
125 | #define EN BIT(31) | |
126 | ||
127 | /* PARF_LTSSM register fields */ | |
128 | #define LTSSM_EN BIT(8) | |
129 | ||
130 | /* PARF_DEVICE_TYPE register fields */ | |
131 | #define DEVICE_TYPE_RC 0x4 | |
132 | ||
133 | /* ELBI_SYS_CTRL register fields */ | |
134 | #define ELBI_SYS_CTRL_LT_ENABLE BIT(0) | |
135 | ||
136 | /* AXI_MSTR_RESP_COMP_CTRL0 register fields */ | |
137 | #define CFG_REMOTE_RD_REQ_BRIDGE_SIZE_2K 0x4 | |
138 | #define CFG_REMOTE_RD_REQ_BRIDGE_SIZE_4K 0x5 | |
139 | ||
140 | /* AXI_MSTR_RESP_COMP_CTRL1 register fields */ | |
141 | #define CFG_BRIDGE_SB_INIT BIT(0) | |
142 | ||
143 | /* MISC_CONTROL_1_REG register fields */ | |
144 | #define DBI_RO_WR_EN 1 | |
145 | ||
146 | /* PCI_EXP_SLTCAP register fields */ | |
147 | #define PCIE_CAP_SLOT_POWER_LIMIT_VAL FIELD_PREP(PCI_EXP_SLTCAP_SPLV, 250) | |
148 | #define PCIE_CAP_SLOT_POWER_LIMIT_SCALE FIELD_PREP(PCI_EXP_SLTCAP_SPLS, 1) | |
149 | #define PCIE_CAP_SLOT_VAL (PCI_EXP_SLTCAP_ABP | \ | |
150 | PCI_EXP_SLTCAP_PCP | \ | |
151 | PCI_EXP_SLTCAP_MRLSP | \ | |
152 | PCI_EXP_SLTCAP_AIP | \ | |
153 | PCI_EXP_SLTCAP_PIP | \ | |
154 | PCI_EXP_SLTCAP_HPS | \ | |
155 | PCI_EXP_SLTCAP_HPC | \ | |
156 | PCI_EXP_SLTCAP_EIP | \ | |
157 | PCIE_CAP_SLOT_POWER_LIMIT_VAL | \ | |
158 | PCIE_CAP_SLOT_POWER_LIMIT_SCALE) | |
159 | ||
160 | #define PERST_DELAY_US 1000 | |
161 | ||
162 | #define LINK_WAIT_MAX_RETRIES 10 | |
163 | #define LINK_WAIT_USLEEP 100000 | |
164 | ||
165 | #define QCOM_PCIE_CRC8_POLYNOMIAL (BIT(2) | BIT(1) | BIT(0)) | |
166 | ||
167 | #define CRC8_TABLE_SIZE 256 | |
168 | ||
169 | static bool qcom_pcie_wait_link_up(struct qcom_pcie *priv) | |
170 | { | |
171 | u8 offset = pcie_dw_find_capability(&priv->dw, PCI_CAP_ID_EXP); | |
172 | unsigned int cnt = 0; | |
173 | u16 val; | |
174 | ||
175 | do { | |
176 | val = readw(priv->dw.dbi_base + offset + PCI_EXP_LNKSTA); | |
177 | ||
178 | if ((val & PCI_EXP_LNKSTA_DLLLA)) | |
179 | return true; | |
180 | cnt++; | |
181 | ||
182 | udelay(LINK_WAIT_USLEEP); | |
183 | } while (cnt < LINK_WAIT_MAX_RETRIES); | |
184 | ||
185 | return false; | |
186 | } | |
187 | ||
188 | static void qcom_pcie_clear_aspm_l0s(struct qcom_pcie *priv) | |
189 | { | |
190 | u8 offset = pcie_dw_find_capability(&priv->dw, PCI_CAP_ID_EXP); | |
191 | u32 val; | |
192 | ||
193 | dw_pcie_dbi_write_enable(&priv->dw, true); | |
194 | ||
195 | val = readl(priv->dw.dbi_base + offset + PCI_EXP_LNKCAP); | |
196 | val &= ~PCI_EXP_LNKCAP_ASPM_L0S; | |
197 | writel(val, priv->dw.dbi_base + offset + PCI_EXP_LNKCAP); | |
198 | ||
199 | dw_pcie_dbi_write_enable(&priv->dw, false); | |
200 | } | |
201 | ||
202 | static void qcom_pcie_clear_hpc(struct qcom_pcie *priv) | |
203 | { | |
204 | u8 offset = pcie_dw_find_capability(&priv->dw, PCI_CAP_ID_EXP); | |
205 | u32 val; | |
206 | ||
207 | dw_pcie_dbi_write_enable(&priv->dw, true); | |
208 | ||
209 | val = readl(priv->dw.dbi_base + offset + PCI_EXP_SLTCAP); | |
210 | val &= ~PCI_EXP_SLTCAP_HPC; | |
211 | writel(val, priv->dw.dbi_base + offset + PCI_EXP_SLTCAP); | |
212 | ||
213 | dw_pcie_dbi_write_enable(&priv->dw, false); | |
214 | } | |
215 | ||
216 | static void qcom_pcie_set_lanes(struct qcom_pcie *priv, unsigned int lanes) | |
217 | { | |
218 | u8 offset = pcie_dw_find_capability(&priv->dw, PCI_CAP_ID_EXP); | |
219 | u32 val; | |
220 | ||
221 | val = readl(priv->dw.dbi_base + offset + PCI_EXP_LNKCAP); | |
222 | val &= ~PCI_EXP_LNKCAP_MLW; | |
223 | val |= FIELD_PREP(PCI_EXP_LNKCAP_MLW, lanes); | |
224 | writel(val, priv->dw.dbi_base + offset + PCI_EXP_LNKCAP); | |
225 | } | |
226 | ||
227 | static int qcom_pcie_config_sid_1_9_0(struct qcom_pcie *priv) | |
228 | { | |
229 | /* iommu map structure */ | |
230 | struct { | |
231 | u32 bdf; | |
232 | u32 phandle; | |
233 | u32 smmu_sid; | |
234 | u32 smmu_sid_len; | |
235 | } *map; | |
236 | void *bdf_to_sid_base = priv->parf + PARF_BDF_TO_SID_TABLE_N; | |
237 | int i, nr_map, size = 0; | |
238 | u32 smmu_sid_base; | |
239 | ||
240 | dev_read_prop(priv->dw.dev, "iommu-map", &size); | |
241 | if (!size) | |
242 | return 0; | |
243 | ||
244 | map = malloc(size); | |
245 | if (!map) | |
246 | return -ENOMEM; | |
247 | ||
248 | dev_read_u32_array(priv->dw.dev, "iommu-map", (u32 *)map, size / sizeof(u32)); | |
249 | ||
250 | nr_map = size / (sizeof(*map)); | |
251 | ||
252 | /* Registers need to be zero out first */ | |
253 | memset_io(bdf_to_sid_base, 0, CRC8_TABLE_SIZE * sizeof(u32)); | |
254 | ||
255 | /* Extract the SMMU SID base from the first entry of iommu-map */ | |
256 | smmu_sid_base = map[0].smmu_sid; | |
257 | ||
258 | /* Look for an available entry to hold the mapping */ | |
259 | for (i = 0; i < nr_map; i++) { | |
260 | __be16 bdf_be = cpu_to_be16(map[i].bdf); | |
261 | u32 val; | |
262 | u8 hash; | |
263 | ||
264 | hash = crc8(QCOM_PCIE_CRC8_POLYNOMIAL, (u8 *)&bdf_be, sizeof(bdf_be)); | |
265 | ||
266 | val = readl(bdf_to_sid_base + hash * sizeof(u32)); | |
267 | ||
268 | /* If the register is already populated, look for next available entry */ | |
269 | while (val) { | |
270 | u8 current_hash = hash++; | |
271 | u8 next_mask = 0xff; | |
272 | ||
273 | /* If NEXT field is NULL then update it with next hash */ | |
274 | if (!(val & next_mask)) { | |
275 | val |= (u32)hash; | |
276 | writel(val, bdf_to_sid_base + current_hash * sizeof(u32)); | |
277 | } | |
278 | ||
279 | val = readl(bdf_to_sid_base + hash * sizeof(u32)); | |
280 | } | |
281 | ||
282 | /* BDF [31:16] | SID [15:8] | NEXT [7:0] */ | |
283 | val = map[i].bdf << 16 | (map[i].smmu_sid - smmu_sid_base) << 8 | 0; | |
284 | writel(val, bdf_to_sid_base + hash * sizeof(u32)); | |
285 | } | |
286 | ||
287 | free(map); | |
288 | ||
289 | return 0; | |
290 | } | |
291 | ||
292 | static void qcom_pcie_configure(struct qcom_pcie *priv) | |
293 | { | |
294 | u32 val; | |
295 | ||
296 | dw_pcie_dbi_write_enable(&priv->dw, true); | |
297 | ||
298 | val = readl(priv->dw.dbi_base + PCIE_PORT_LINK_CONTROL); | |
299 | val &= ~PORT_LINK_FAST_LINK_MODE; | |
300 | val |= PORT_LINK_DLL_LINK_EN; | |
301 | val &= ~PORT_LINK_MODE_MASK; | |
302 | val |= PORT_LINK_MODE_2_LANES; | |
303 | writel(val, priv->dw.dbi_base + PCIE_PORT_LINK_CONTROL); | |
304 | ||
305 | val = readl(priv->dw.dbi_base + PCIE_LINK_WIDTH_SPEED_CONTROL); | |
306 | val &= ~PORT_LOGIC_LINK_WIDTH_MASK; | |
307 | val |= PORT_LOGIC_LINK_WIDTH_2_LANES; | |
308 | writel(val, priv->dw.dbi_base + PCIE_LINK_WIDTH_SPEED_CONTROL); | |
309 | ||
310 | qcom_pcie_set_lanes(priv, 2); | |
311 | ||
312 | dw_pcie_dbi_write_enable(&priv->dw, false); | |
313 | } | |
314 | ||
315 | static int qcom_pcie_init_port(struct udevice *dev) | |
316 | { | |
317 | struct qcom_pcie *priv = dev_get_priv(dev); | |
318 | int vreg, ret; | |
319 | u32 val; | |
320 | ||
321 | dm_gpio_set_value(&priv->rst_gpio, 1); | |
322 | udelay(PERST_DELAY_US); | |
323 | ||
324 | ret = generic_phy_init(&priv->phy); | |
325 | if (ret) { | |
326 | dev_err(dev, "failed to init phy (%d)\n", ret); | |
327 | return ret; | |
328 | } | |
329 | ||
330 | udelay(PERST_DELAY_US); | |
331 | ||
332 | for (vreg = 0; vreg < NUM_SUPPLIES; ++vreg) { | |
333 | ret = regulator_set_enable(priv->vregs[vreg], true); | |
334 | if (ret && ret != -ENOSYS) | |
335 | dev_warn(dev, "failed to enable regulator %d (%d)\n", vreg, ret); | |
336 | } | |
337 | ||
338 | ret = clk_enable_bulk(&priv->clks); | |
339 | if (ret) { | |
340 | dev_err(dev, "failed to enable clocks (%d)\n", ret); | |
341 | goto err_power_off_phy; | |
342 | } | |
343 | ||
344 | ret = reset_assert_bulk(&priv->rsts); | |
345 | if (ret) { | |
346 | dev_err(dev, "failed to assert resets (%d)\n", ret); | |
347 | goto err_disable_clks; | |
348 | } | |
349 | ||
350 | udelay(PERST_DELAY_US); | |
351 | ||
352 | ret = reset_deassert_bulk(&priv->rsts); | |
353 | if (ret) { | |
354 | dev_err(dev, "failed to deassert resets (%d)\n", ret); | |
355 | goto err_power_off_phy; | |
356 | } | |
357 | ||
358 | udelay(PERST_DELAY_US); | |
359 | ||
360 | /* configure PCIe to RC mode */ | |
361 | writel(DEVICE_TYPE_RC, priv->parf + PARF_DEVICE_TYPE); | |
362 | ||
363 | /* enable PCIe clocks and resets */ | |
364 | val = readl(priv->parf + PARF_PHY_CTRL); | |
365 | val &= ~PHY_TEST_PWR_DOWN; | |
366 | writel(val, priv->parf + PARF_PHY_CTRL); | |
367 | ||
368 | /* change DBI base address */ | |
369 | writel(0, priv->parf + PARF_DBI_BASE_ADDR); | |
370 | ||
371 | /* MAC PHY_POWERDOWN MUX DISABLE */ | |
372 | val = readl(priv->parf + PARF_SYS_CTRL); | |
373 | val &= ~MAC_PHY_POWERDOWN_IN_P2_D_MUX_EN; | |
374 | writel(val, priv->parf + PARF_SYS_CTRL); | |
375 | ||
376 | val = readl(priv->parf + PARF_MHI_CLOCK_RESET_CTRL); | |
377 | val |= BYPASS; | |
378 | writel(val, priv->parf + PARF_MHI_CLOCK_RESET_CTRL); | |
379 | ||
380 | /* Enable L1 and L1SS */ | |
381 | val = readl(priv->parf + PARF_PM_CTRL); | |
382 | val &= ~REQ_NOT_ENTR_L1; | |
383 | writel(val, priv->parf + PARF_PM_CTRL); | |
384 | ||
385 | val = readl(priv->parf + PARF_AXI_MSTR_WR_ADDR_HALT_V2); | |
386 | val |= EN; | |
387 | writel(val, priv->parf + PARF_AXI_MSTR_WR_ADDR_HALT_V2); | |
388 | ||
389 | ret = generic_phy_power_on(&priv->phy); | |
390 | if (ret) { | |
391 | dev_err(dev, "failed to power on phy (%d)\n", ret); | |
392 | goto err_exit_phy; | |
393 | } | |
394 | ||
395 | qcom_pcie_clear_aspm_l0s(priv); | |
396 | qcom_pcie_clear_hpc(priv); | |
397 | ||
398 | mdelay(100); | |
399 | dm_gpio_set_value(&priv->rst_gpio, 0); | |
400 | udelay(PERST_DELAY_US); | |
401 | ||
402 | if (priv->ops && priv->ops->config_sid) { | |
403 | ret = priv->ops->config_sid(priv); | |
404 | if (ret) | |
405 | goto err_deassert_bulk; | |
406 | } | |
407 | ||
408 | qcom_pcie_configure(priv); | |
409 | ||
410 | pcie_dw_setup_host(&priv->dw); | |
411 | ||
412 | /* enable link training */ | |
413 | val = readl(priv->parf + PARF_LTSSM); | |
414 | val |= LTSSM_EN; | |
415 | writel(val, priv->parf + PARF_LTSSM); | |
416 | ||
417 | return 0; | |
418 | err_deassert_bulk: | |
419 | reset_assert_bulk(&priv->rsts); | |
420 | err_disable_clks: | |
421 | clk_disable_bulk(&priv->clks); | |
422 | err_power_off_phy: | |
423 | generic_phy_power_off(&priv->phy); | |
424 | err_exit_phy: | |
425 | generic_phy_exit(&priv->phy); | |
426 | ||
427 | return ret; | |
428 | } | |
429 | ||
430 | static const char *qcom_pcie_vregs[NUM_SUPPLIES] = { | |
431 | "vdda-supply", | |
432 | "vddpe-3v3-supply", | |
433 | }; | |
434 | ||
435 | static int qcom_pcie_parse_dt(struct udevice *dev) | |
436 | { | |
437 | struct qcom_pcie *priv = dev_get_priv(dev); | |
438 | int vreg, ret; | |
439 | ||
440 | priv->dw.dbi_base = dev_read_addr_name_ptr(dev, "dbi"); | |
441 | if (!priv->dw.dbi_base) | |
442 | return -EINVAL; | |
443 | ||
444 | dev_dbg(dev, "DBI address is 0x%p\n", priv->dw.dbi_base); | |
445 | ||
446 | priv->dw.atu_base = dev_read_addr_name_ptr(dev, "atu"); | |
447 | if (!priv->dw.atu_base) | |
448 | return -EINVAL; | |
449 | ||
450 | dev_dbg(dev, "ATU address is 0x%p\n", priv->dw.atu_base); | |
451 | ||
452 | priv->parf = dev_read_addr_name_ptr(dev, "parf"); | |
453 | if (!priv->parf) | |
454 | return -EINVAL; | |
455 | ||
456 | dev_dbg(dev, "PARF address is 0x%p\n", priv->parf); | |
457 | ||
458 | ret = gpio_request_by_name(dev, "perst-gpios", 0, | |
459 | &priv->rst_gpio, GPIOD_IS_OUT); | |
460 | if (ret) { | |
461 | dev_err(dev, "failed to find reset-gpios property\n"); | |
462 | return ret; | |
463 | } | |
464 | ||
465 | ret = reset_get_bulk(dev, &priv->rsts); | |
466 | if (ret) { | |
467 | dev_err(dev, "failed to get resets (%d)\n", ret); | |
468 | return ret; | |
469 | } | |
470 | ||
471 | ret = clk_get_bulk(dev, &priv->clks); | |
472 | if (ret) { | |
473 | dev_err(dev, "failed to get clocks (%d)\n", ret); | |
474 | return ret; | |
475 | } | |
476 | ||
477 | ret = generic_phy_get_by_index(dev, 0, &priv->phy); | |
478 | if (ret) { | |
479 | dev_err(dev, "failed to get pcie phy (%d)\n", ret); | |
480 | return ret; | |
481 | } | |
482 | ||
483 | for (vreg = 0; vreg < NUM_SUPPLIES; ++vreg) { | |
484 | ret = device_get_supply_regulator(dev, qcom_pcie_vregs[vreg], &priv->vregs[vreg]); | |
485 | if (ret) | |
486 | dev_warn(dev, "failed to get regulator %d (%d)\n", vreg, ret); | |
487 | } | |
488 | ||
489 | return 0; | |
490 | } | |
491 | ||
492 | /** | |
493 | * qcom_pcie_probe() - Probe the PCIe bus for active link | |
494 | * | |
495 | * @dev: A pointer to the device being operated on | |
496 | * | |
497 | * Probe for an active link on the PCIe bus and configure the controller | |
498 | * to enable this port. | |
499 | * | |
500 | * Return: 0 on success, else -ENODEV | |
501 | */ | |
502 | static int qcom_pcie_probe(struct udevice *dev) | |
503 | { | |
504 | struct qcom_pcie *priv = dev_get_priv(dev); | |
505 | struct udevice *ctlr = pci_get_controller(dev); | |
506 | struct pci_controller *hose = dev_get_uclass_priv(ctlr); | |
507 | int ret = 0; | |
508 | ||
509 | priv->dw.first_busno = dev_seq(dev); | |
510 | priv->dw.dev = dev; | |
511 | ||
512 | ret = qcom_pcie_parse_dt(dev); | |
513 | if (ret) | |
514 | return ret; | |
515 | ||
516 | ret = qcom_pcie_init_port(dev); | |
517 | if (ret) { | |
518 | dm_gpio_free(dev, &priv->rst_gpio); | |
519 | return ret; | |
520 | } | |
521 | ||
522 | if (qcom_pcie_wait_link_up(priv)) | |
523 | printf("PCIE-%d: Link up (Gen%d-x%d, Bus%d)\n", | |
524 | dev_seq(dev), pcie_dw_get_link_speed(&priv->dw), | |
525 | pcie_dw_get_link_width(&priv->dw), | |
526 | hose->first_busno); | |
527 | else | |
528 | printf("PCIE-%d: Link up timeout\n", dev_seq(dev)); | |
529 | ||
530 | return pcie_dw_prog_outbound_atu_unroll(&priv->dw, | |
531 | PCIE_ATU_REGION_INDEX0, | |
532 | PCIE_ATU_TYPE_MEM, | |
533 | priv->dw.mem.phys_start, | |
534 | priv->dw.mem.bus_start, | |
535 | priv->dw.mem.size); | |
536 | } | |
537 | ||
538 | static const struct dm_pci_ops qcom_pcie_ops = { | |
539 | .read_config = pcie_dw_read_config, | |
540 | .write_config = pcie_dw_write_config, | |
541 | }; | |
542 | ||
543 | static const struct qcom_pcie_ops ops_1_9_0 = { | |
544 | .config_sid = qcom_pcie_config_sid_1_9_0, | |
545 | }; | |
546 | ||
547 | static const struct udevice_id qcom_pcie_ids[] = { | |
548 | { .compatible = "qcom,pcie-sa8540p", .data = (ulong)&ops_1_9_0 }, | |
549 | { .compatible = "qcom,pcie-sc7280", .data = (ulong)&ops_1_9_0 }, | |
550 | { .compatible = "qcom,pcie-sc8180x", .data = (ulong)&ops_1_9_0 }, | |
551 | { .compatible = "qcom,pcie-sc8280xp", .data = (ulong)&ops_1_9_0 }, | |
552 | { .compatible = "qcom,pcie-sdm845" }, | |
553 | { .compatible = "qcom,pcie-sdx55", .data = (ulong)&ops_1_9_0 }, | |
554 | { .compatible = "qcom,pcie-sm8150", .data = (ulong)&ops_1_9_0 }, | |
555 | { .compatible = "qcom,pcie-sm8250", .data = (ulong)&ops_1_9_0 }, | |
556 | { .compatible = "qcom,pcie-sm8350", .data = (ulong)&ops_1_9_0 }, | |
557 | { .compatible = "qcom,pcie-sm8450-pcie0", .data = (ulong)&ops_1_9_0 }, | |
558 | { .compatible = "qcom,pcie-sm8450-pcie1", .data = (ulong)&ops_1_9_0 }, | |
559 | { .compatible = "qcom,pcie-sm8550", .data = (ulong)&ops_1_9_0 }, | |
560 | { .compatible = "qcom,pcie-x1e80100", .data = (ulong)&ops_1_9_0 }, | |
561 | { } | |
562 | }; | |
563 | ||
564 | U_BOOT_DRIVER(qcom_dw_pcie) = { | |
565 | .name = "pcie_dw_qcom", | |
566 | .id = UCLASS_PCI, | |
567 | .of_match = qcom_pcie_ids, | |
568 | .ops = &qcom_pcie_ops, | |
569 | .probe = qcom_pcie_probe, | |
570 | .priv_auto = sizeof(struct qcom_pcie), | |
571 | }; |