]>
Commit | Line | Data |
---|---|---|
2874c5fd | 1 | // SPDX-License-Identifier: GPL-2.0-or-later |
b8c86fc5 PO |
2 | /* linux/drivers/mmc/host/sdhci-pci.c - SDHCI on PCI bus interface |
3 | * | |
4 | * Copyright (C) 2005-2008 Pierre Ossman, All Rights Reserved. | |
5 | * | |
b8c86fc5 PO |
6 | * Thanks to the following companies for their support: |
7 | * | |
8 | * - JMicron (hardware and technical support) | |
9 | */ | |
10 | ||
5305ec6a | 11 | #include <linux/bitfield.h> |
a72016a4 | 12 | #include <linux/string.h> |
b8c86fc5 PO |
13 | #include <linux/delay.h> |
14 | #include <linux/highmem.h> | |
88b47679 | 15 | #include <linux/module.h> |
b8c86fc5 PO |
16 | #include <linux/pci.h> |
17 | #include <linux/dma-mapping.h> | |
5a0e3ad6 | 18 | #include <linux/slab.h> |
ccc92c23 | 19 | #include <linux/device.h> |
b8c86fc5 | 20 | #include <linux/mmc/host.h> |
e1bfad6d | 21 | #include <linux/mmc/mmc.h> |
b177bc91 AP |
22 | #include <linux/scatterlist.h> |
23 | #include <linux/io.h> | |
7a869f00 | 24 | #include <linux/iopoll.h> |
0f201655 | 25 | #include <linux/gpio.h> |
66fd8ad5 | 26 | #include <linux/pm_runtime.h> |
ff59c520 | 27 | #include <linux/mmc/slot-gpio.h> |
52c506f0 | 28 | #include <linux/mmc/sdhci-pci-data.h> |
3f23df72 | 29 | #include <linux/acpi.h> |
bedf9fc0 | 30 | #include <linux/dmi.h> |
b8c86fc5 | 31 | |
0a49a619 AH |
32 | #ifdef CONFIG_X86 |
33 | #include <asm/iosf_mbi.h> | |
34 | #endif | |
35 | ||
8ee82bda AH |
36 | #include "cqhci.h" |
37 | ||
b8c86fc5 | 38 | #include "sdhci.h" |
522624f9 | 39 | #include "sdhci-pci.h" |
22606405 | 40 | |
fee686b7 | 41 | static void sdhci_pci_hw_reset(struct sdhci_host *host); |
fee686b7 | 42 | |
30cf2803 | 43 | #ifdef CONFIG_PM_SLEEP |
5c3c6126 AH |
44 | static int sdhci_pci_init_wakeup(struct sdhci_pci_chip *chip) |
45 | { | |
46 | mmc_pm_flag_t pm_flags = 0; | |
d56ee1ff | 47 | bool cap_cd_wake = false; |
5c3c6126 AH |
48 | int i; |
49 | ||
50 | for (i = 0; i < chip->num_slots; i++) { | |
51 | struct sdhci_pci_slot *slot = chip->slots[i]; | |
52 | ||
d56ee1ff | 53 | if (slot) { |
5c3c6126 | 54 | pm_flags |= slot->host->mmc->pm_flags; |
d56ee1ff AH |
55 | if (slot->host->mmc->caps & MMC_CAP_CD_WAKE) |
56 | cap_cd_wake = true; | |
57 | } | |
5c3c6126 AH |
58 | } |
59 | ||
d56ee1ff AH |
60 | if ((pm_flags & MMC_PM_KEEP_POWER) && (pm_flags & MMC_PM_WAKE_SDIO_IRQ)) |
61 | return device_wakeup_enable(&chip->pdev->dev); | |
62 | else if (!cap_cd_wake) | |
63 | return device_wakeup_disable(&chip->pdev->dev); | |
64 | ||
65 | return 0; | |
5c3c6126 AH |
66 | } |
67 | ||
68 | static int sdhci_pci_suspend_host(struct sdhci_pci_chip *chip) | |
30cf2803 AH |
69 | { |
70 | int i, ret; | |
71 | ||
5c3c6126 AH |
72 | sdhci_pci_init_wakeup(chip); |
73 | ||
30cf2803 AH |
74 | for (i = 0; i < chip->num_slots; i++) { |
75 | struct sdhci_pci_slot *slot = chip->slots[i]; | |
76 | struct sdhci_host *host; | |
77 | ||
78 | if (!slot) | |
79 | continue; | |
80 | ||
81 | host = slot->host; | |
82 | ||
83 | if (chip->pm_retune && host->tuning_mode != SDHCI_TUNING_MODE_3) | |
84 | mmc_retune_needed(host->mmc); | |
85 | ||
86 | ret = sdhci_suspend_host(host); | |
87 | if (ret) | |
88 | goto err_pci_suspend; | |
d56ee1ff AH |
89 | |
90 | if (device_may_wakeup(&chip->pdev->dev)) | |
91 | mmc_gpio_set_cd_wake(host->mmc, true); | |
30cf2803 AH |
92 | } |
93 | ||
94 | return 0; | |
95 | ||
96 | err_pci_suspend: | |
97 | while (--i >= 0) | |
98 | sdhci_resume_host(chip->slots[i]->host); | |
99 | return ret; | |
100 | } | |
101 | ||
30cf2803 AH |
102 | int sdhci_pci_resume_host(struct sdhci_pci_chip *chip) |
103 | { | |
104 | struct sdhci_pci_slot *slot; | |
105 | int i, ret; | |
106 | ||
107 | for (i = 0; i < chip->num_slots; i++) { | |
108 | slot = chip->slots[i]; | |
109 | if (!slot) | |
110 | continue; | |
111 | ||
112 | ret = sdhci_resume_host(slot->host); | |
113 | if (ret) | |
114 | return ret; | |
d56ee1ff AH |
115 | |
116 | mmc_gpio_set_cd_wake(slot->host->mmc, false); | |
30cf2803 AH |
117 | } |
118 | ||
119 | return 0; | |
120 | } | |
8ee82bda AH |
121 | |
122 | static int sdhci_cqhci_suspend(struct sdhci_pci_chip *chip) | |
123 | { | |
124 | int ret; | |
125 | ||
126 | ret = cqhci_suspend(chip->slots[0]->host->mmc); | |
127 | if (ret) | |
128 | return ret; | |
129 | ||
130 | return sdhci_pci_suspend_host(chip); | |
131 | } | |
132 | ||
133 | static int sdhci_cqhci_resume(struct sdhci_pci_chip *chip) | |
134 | { | |
135 | int ret; | |
136 | ||
137 | ret = sdhci_pci_resume_host(chip); | |
138 | if (ret) | |
139 | return ret; | |
140 | ||
141 | return cqhci_resume(chip->slots[0]->host->mmc); | |
142 | } | |
30cf2803 AH |
143 | #endif |
144 | ||
966d696a AH |
145 | #ifdef CONFIG_PM |
146 | static int sdhci_pci_runtime_suspend_host(struct sdhci_pci_chip *chip) | |
147 | { | |
148 | struct sdhci_pci_slot *slot; | |
149 | struct sdhci_host *host; | |
150 | int i, ret; | |
151 | ||
152 | for (i = 0; i < chip->num_slots; i++) { | |
153 | slot = chip->slots[i]; | |
154 | if (!slot) | |
155 | continue; | |
156 | ||
157 | host = slot->host; | |
158 | ||
159 | ret = sdhci_runtime_suspend_host(host); | |
160 | if (ret) | |
161 | goto err_pci_runtime_suspend; | |
162 | ||
163 | if (chip->rpm_retune && | |
164 | host->tuning_mode != SDHCI_TUNING_MODE_3) | |
165 | mmc_retune_needed(host->mmc); | |
166 | } | |
167 | ||
168 | return 0; | |
169 | ||
170 | err_pci_runtime_suspend: | |
171 | while (--i >= 0) | |
c6303c5d | 172 | sdhci_runtime_resume_host(chip->slots[i]->host, 0); |
966d696a AH |
173 | return ret; |
174 | } | |
175 | ||
176 | static int sdhci_pci_runtime_resume_host(struct sdhci_pci_chip *chip) | |
177 | { | |
178 | struct sdhci_pci_slot *slot; | |
179 | int i, ret; | |
180 | ||
181 | for (i = 0; i < chip->num_slots; i++) { | |
182 | slot = chip->slots[i]; | |
183 | if (!slot) | |
184 | continue; | |
185 | ||
c6303c5d | 186 | ret = sdhci_runtime_resume_host(slot->host, 0); |
966d696a AH |
187 | if (ret) |
188 | return ret; | |
189 | } | |
190 | ||
191 | return 0; | |
192 | } | |
8ee82bda AH |
193 | |
194 | static int sdhci_cqhci_runtime_suspend(struct sdhci_pci_chip *chip) | |
195 | { | |
196 | int ret; | |
197 | ||
198 | ret = cqhci_suspend(chip->slots[0]->host->mmc); | |
199 | if (ret) | |
200 | return ret; | |
201 | ||
202 | return sdhci_pci_runtime_suspend_host(chip); | |
203 | } | |
204 | ||
205 | static int sdhci_cqhci_runtime_resume(struct sdhci_pci_chip *chip) | |
206 | { | |
207 | int ret; | |
208 | ||
209 | ret = sdhci_pci_runtime_resume_host(chip); | |
210 | if (ret) | |
211 | return ret; | |
212 | ||
213 | return cqhci_resume(chip->slots[0]->host->mmc); | |
214 | } | |
966d696a AH |
215 | #endif |
216 | ||
8ee82bda AH |
217 | static u32 sdhci_cqhci_irq(struct sdhci_host *host, u32 intmask) |
218 | { | |
219 | int cmd_error = 0; | |
220 | int data_error = 0; | |
221 | ||
222 | if (!sdhci_cqe_irq(host, intmask, &cmd_error, &data_error)) | |
223 | return intmask; | |
224 | ||
225 | cqhci_irq(host->mmc, intmask, cmd_error, data_error); | |
226 | ||
227 | return 0; | |
228 | } | |
229 | ||
230 | static void sdhci_pci_dumpregs(struct mmc_host *mmc) | |
231 | { | |
232 | sdhci_dumpregs(mmc_priv(mmc)); | |
233 | } | |
234 | ||
22606405 PO |
235 | /*****************************************************************************\ |
236 | * * | |
237 | * Hardware specific quirk handling * | |
238 | * * | |
239 | \*****************************************************************************/ | |
240 | ||
241 | static int ricoh_probe(struct sdhci_pci_chip *chip) | |
242 | { | |
c99436fb CB |
243 | if (chip->pdev->subsystem_vendor == PCI_VENDOR_ID_SAMSUNG || |
244 | chip->pdev->subsystem_vendor == PCI_VENDOR_ID_SONY) | |
22606405 | 245 | chip->quirks |= SDHCI_QUIRK_NO_CARD_NO_RESET; |
ccc92c23 ML |
246 | return 0; |
247 | } | |
248 | ||
249 | static int ricoh_mmc_probe_slot(struct sdhci_pci_slot *slot) | |
250 | { | |
251 | slot->host->caps = | |
252 | ((0x21 << SDHCI_TIMEOUT_CLK_SHIFT) | |
253 | & SDHCI_TIMEOUT_CLK_MASK) | | |
22606405 | 254 | |
ccc92c23 ML |
255 | ((0x21 << SDHCI_CLOCK_BASE_SHIFT) |
256 | & SDHCI_CLOCK_BASE_MASK) | | |
257 | ||
258 | SDHCI_TIMEOUT_CLK_UNIT | | |
259 | SDHCI_CAN_VDD_330 | | |
1a1f1f04 | 260 | SDHCI_CAN_DO_HISPD | |
ccc92c23 ML |
261 | SDHCI_CAN_DO_SDMA; |
262 | return 0; | |
263 | } | |
264 | ||
b7813f0f | 265 | #ifdef CONFIG_PM_SLEEP |
ccc92c23 ML |
266 | static int ricoh_mmc_resume(struct sdhci_pci_chip *chip) |
267 | { | |
268 | /* Apply a delay to allow controller to settle */ | |
269 | /* Otherwise it becomes confused if card state changed | |
270 | during suspend */ | |
271 | msleep(500); | |
30cf2803 | 272 | return sdhci_pci_resume_host(chip); |
22606405 | 273 | } |
b7813f0f | 274 | #endif |
22606405 PO |
275 | |
276 | static const struct sdhci_pci_fixes sdhci_ricoh = { | |
277 | .probe = ricoh_probe, | |
84938294 VK |
278 | .quirks = SDHCI_QUIRK_32BIT_DMA_ADDR | |
279 | SDHCI_QUIRK_FORCE_DMA | | |
280 | SDHCI_QUIRK_CLOCK_BEFORE_RESET, | |
22606405 PO |
281 | }; |
282 | ||
ccc92c23 ML |
283 | static const struct sdhci_pci_fixes sdhci_ricoh_mmc = { |
284 | .probe_slot = ricoh_mmc_probe_slot, | |
b7813f0f | 285 | #ifdef CONFIG_PM_SLEEP |
ccc92c23 | 286 | .resume = ricoh_mmc_resume, |
b7813f0f | 287 | #endif |
ccc92c23 ML |
288 | .quirks = SDHCI_QUIRK_32BIT_DMA_ADDR | |
289 | SDHCI_QUIRK_CLOCK_BEFORE_RESET | | |
290 | SDHCI_QUIRK_NO_CARD_NO_RESET | | |
291 | SDHCI_QUIRK_MISSING_CAPS | |
292 | }; | |
293 | ||
22606405 PO |
294 | static const struct sdhci_pci_fixes sdhci_ene_712 = { |
295 | .quirks = SDHCI_QUIRK_SINGLE_POWER_WRITE | | |
296 | SDHCI_QUIRK_BROKEN_DMA, | |
297 | }; | |
298 | ||
299 | static const struct sdhci_pci_fixes sdhci_ene_714 = { | |
300 | .quirks = SDHCI_QUIRK_SINGLE_POWER_WRITE | | |
301 | SDHCI_QUIRK_RESET_CMD_DATA_ON_IOS | | |
302 | SDHCI_QUIRK_BROKEN_DMA, | |
303 | }; | |
304 | ||
305 | static const struct sdhci_pci_fixes sdhci_cafe = { | |
306 | .quirks = SDHCI_QUIRK_NO_SIMULT_VDD_AND_POWER | | |
a0874897 | 307 | SDHCI_QUIRK_NO_BUSY_IRQ | |
55fc05b7 | 308 | SDHCI_QUIRK_BROKEN_CARD_DETECTION | |
ee53ab5d | 309 | SDHCI_QUIRK_BROKEN_TIMEOUT_VAL, |
22606405 PO |
310 | }; |
311 | ||
43e968ce DB |
312 | static const struct sdhci_pci_fixes sdhci_intel_qrk = { |
313 | .quirks = SDHCI_QUIRK_NO_HISPD_BIT, | |
314 | }; | |
315 | ||
68077b02 ML |
316 | static int mrst_hc_probe_slot(struct sdhci_pci_slot *slot) |
317 | { | |
318 | slot->host->mmc->caps |= MMC_CAP_8_BIT_DATA; | |
319 | return 0; | |
320 | } | |
321 | ||
f9ee3eab AC |
322 | /* |
323 | * ADMA operation is disabled for Moorestown platform due to | |
324 | * hardware bugs. | |
325 | */ | |
35ac6f08 | 326 | static int mrst_hc_probe(struct sdhci_pci_chip *chip) |
f9ee3eab AC |
327 | { |
328 | /* | |
35ac6f08 JP |
329 | * slots number is fixed here for MRST as SDIO3/5 are never used and |
330 | * have hardware bugs. | |
f9ee3eab AC |
331 | */ |
332 | chip->num_slots = 1; | |
333 | return 0; | |
334 | } | |
335 | ||
296e0b03 AS |
336 | static int pch_hc_probe_slot(struct sdhci_pci_slot *slot) |
337 | { | |
338 | slot->host->mmc->caps |= MMC_CAP_8_BIT_DATA; | |
339 | return 0; | |
340 | } | |
341 | ||
162d6f98 | 342 | #ifdef CONFIG_PM |
66fd8ad5 | 343 | |
c5e027a4 | 344 | static irqreturn_t sdhci_pci_sd_cd(int irq, void *dev_id) |
66fd8ad5 AH |
345 | { |
346 | struct sdhci_pci_slot *slot = dev_id; | |
347 | struct sdhci_host *host = slot->host; | |
348 | ||
349 | mmc_detect_change(host->mmc, msecs_to_jiffies(200)); | |
350 | return IRQ_HANDLED; | |
351 | } | |
352 | ||
c5e027a4 | 353 | static void sdhci_pci_add_own_cd(struct sdhci_pci_slot *slot) |
66fd8ad5 | 354 | { |
c5e027a4 | 355 | int err, irq, gpio = slot->cd_gpio; |
66fd8ad5 AH |
356 | |
357 | slot->cd_gpio = -EINVAL; | |
358 | slot->cd_irq = -EINVAL; | |
359 | ||
c5e027a4 AH |
360 | if (!gpio_is_valid(gpio)) |
361 | return; | |
362 | ||
c10bc372 | 363 | err = devm_gpio_request(&slot->chip->pdev->dev, gpio, "sd_cd"); |
66fd8ad5 AH |
364 | if (err < 0) |
365 | goto out; | |
366 | ||
367 | err = gpio_direction_input(gpio); | |
368 | if (err < 0) | |
369 | goto out_free; | |
370 | ||
371 | irq = gpio_to_irq(gpio); | |
372 | if (irq < 0) | |
373 | goto out_free; | |
374 | ||
c5e027a4 | 375 | err = request_irq(irq, sdhci_pci_sd_cd, IRQF_TRIGGER_RISING | |
66fd8ad5 AH |
376 | IRQF_TRIGGER_FALLING, "sd_cd", slot); |
377 | if (err) | |
378 | goto out_free; | |
379 | ||
380 | slot->cd_gpio = gpio; | |
381 | slot->cd_irq = irq; | |
66fd8ad5 | 382 | |
c5e027a4 | 383 | return; |
66fd8ad5 AH |
384 | |
385 | out_free: | |
c10bc372 | 386 | devm_gpio_free(&slot->chip->pdev->dev, gpio); |
66fd8ad5 AH |
387 | out: |
388 | dev_warn(&slot->chip->pdev->dev, "failed to setup card detect wake up\n"); | |
66fd8ad5 AH |
389 | } |
390 | ||
c5e027a4 | 391 | static void sdhci_pci_remove_own_cd(struct sdhci_pci_slot *slot) |
66fd8ad5 AH |
392 | { |
393 | if (slot->cd_irq >= 0) | |
394 | free_irq(slot->cd_irq, slot); | |
66fd8ad5 AH |
395 | } |
396 | ||
397 | #else | |
398 | ||
c5e027a4 AH |
399 | static inline void sdhci_pci_add_own_cd(struct sdhci_pci_slot *slot) |
400 | { | |
401 | } | |
402 | ||
403 | static inline void sdhci_pci_remove_own_cd(struct sdhci_pci_slot *slot) | |
404 | { | |
405 | } | |
66fd8ad5 AH |
406 | |
407 | #endif | |
408 | ||
0d013bcf AH |
409 | static int mfd_emmc_probe_slot(struct sdhci_pci_slot *slot) |
410 | { | |
66fd8ad5 | 411 | slot->host->mmc->caps |= MMC_CAP_8_BIT_DATA | MMC_CAP_NONREMOVABLE; |
d2a47176 | 412 | slot->host->mmc->caps2 |= MMC_CAP2_BOOTPART_NOACC; |
0d013bcf AH |
413 | return 0; |
414 | } | |
415 | ||
93933508 AH |
416 | static int mfd_sdio_probe_slot(struct sdhci_pci_slot *slot) |
417 | { | |
012e4671 | 418 | slot->host->mmc->caps |= MMC_CAP_POWER_OFF_CARD | MMC_CAP_NONREMOVABLE; |
93933508 AH |
419 | return 0; |
420 | } | |
421 | ||
f9ee3eab AC |
422 | static const struct sdhci_pci_fixes sdhci_intel_mrst_hc0 = { |
423 | .quirks = SDHCI_QUIRK_BROKEN_ADMA | SDHCI_QUIRK_NO_HISPD_BIT, | |
68077b02 | 424 | .probe_slot = mrst_hc_probe_slot, |
f9ee3eab AC |
425 | }; |
426 | ||
35ac6f08 | 427 | static const struct sdhci_pci_fixes sdhci_intel_mrst_hc1_hc2 = { |
f9ee3eab | 428 | .quirks = SDHCI_QUIRK_BROKEN_ADMA | SDHCI_QUIRK_NO_HISPD_BIT, |
35ac6f08 | 429 | .probe = mrst_hc_probe, |
f9ee3eab AC |
430 | }; |
431 | ||
29229052 XS |
432 | static const struct sdhci_pci_fixes sdhci_intel_mfd_sd = { |
433 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, | |
c43fd774 | 434 | .allow_runtime_pm = true, |
77a0122e | 435 | .own_cd_for_runtime_pm = true, |
29229052 XS |
436 | }; |
437 | ||
0d013bcf AH |
438 | static const struct sdhci_pci_fixes sdhci_intel_mfd_sdio = { |
439 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, | |
f3c55a7b | 440 | .quirks2 = SDHCI_QUIRK2_HOST_OFF_CARD_ON, |
c43fd774 | 441 | .allow_runtime_pm = true, |
93933508 | 442 | .probe_slot = mfd_sdio_probe_slot, |
0d013bcf AH |
443 | }; |
444 | ||
445 | static const struct sdhci_pci_fixes sdhci_intel_mfd_emmc = { | |
29229052 | 446 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, |
c43fd774 | 447 | .allow_runtime_pm = true, |
0d013bcf | 448 | .probe_slot = mfd_emmc_probe_slot, |
29229052 XS |
449 | }; |
450 | ||
296e0b03 AS |
451 | static const struct sdhci_pci_fixes sdhci_intel_pch_sdio = { |
452 | .quirks = SDHCI_QUIRK_BROKEN_ADMA, | |
453 | .probe_slot = pch_hc_probe_slot, | |
454 | }; | |
455 | ||
0a49a619 AH |
456 | #ifdef CONFIG_X86 |
457 | ||
458 | #define BYT_IOSF_SCCEP 0x63 | |
459 | #define BYT_IOSF_OCP_NETCTRL0 0x1078 | |
460 | #define BYT_IOSF_OCP_TIMEOUT_BASE GENMASK(10, 8) | |
461 | ||
462 | static void byt_ocp_setting(struct pci_dev *pdev) | |
463 | { | |
464 | u32 val = 0; | |
465 | ||
466 | if (pdev->device != PCI_DEVICE_ID_INTEL_BYT_EMMC && | |
467 | pdev->device != PCI_DEVICE_ID_INTEL_BYT_SDIO && | |
468 | pdev->device != PCI_DEVICE_ID_INTEL_BYT_SD && | |
469 | pdev->device != PCI_DEVICE_ID_INTEL_BYT_EMMC2) | |
470 | return; | |
471 | ||
472 | if (iosf_mbi_read(BYT_IOSF_SCCEP, MBI_CR_READ, BYT_IOSF_OCP_NETCTRL0, | |
473 | &val)) { | |
474 | dev_err(&pdev->dev, "%s read error\n", __func__); | |
475 | return; | |
476 | } | |
477 | ||
478 | if (!(val & BYT_IOSF_OCP_TIMEOUT_BASE)) | |
479 | return; | |
480 | ||
481 | val &= ~BYT_IOSF_OCP_TIMEOUT_BASE; | |
482 | ||
483 | if (iosf_mbi_write(BYT_IOSF_SCCEP, MBI_CR_WRITE, BYT_IOSF_OCP_NETCTRL0, | |
484 | val)) { | |
485 | dev_err(&pdev->dev, "%s write error\n", __func__); | |
486 | return; | |
487 | } | |
488 | ||
489 | dev_dbg(&pdev->dev, "%s completed\n", __func__); | |
490 | } | |
491 | ||
492 | #else | |
493 | ||
494 | static inline void byt_ocp_setting(struct pci_dev *pdev) | |
495 | { | |
496 | } | |
497 | ||
498 | #endif | |
499 | ||
c959a6b0 AH |
500 | enum { |
501 | INTEL_DSM_FNS = 0, | |
6ae03368 | 502 | INTEL_DSM_V18_SWITCH = 3, |
be17355a | 503 | INTEL_DSM_V33_SWITCH = 4, |
51ced59c | 504 | INTEL_DSM_DRV_STRENGTH = 9, |
c959a6b0 AH |
505 | INTEL_DSM_D3_RETUNE = 10, |
506 | }; | |
507 | ||
508 | struct intel_host { | |
509 | u32 dsm_fns; | |
51ced59c | 510 | int drv_strength; |
c959a6b0 | 511 | bool d3_retune; |
5305ec6a AH |
512 | bool rpm_retune_ok; |
513 | u32 glk_rx_ctrl1; | |
514 | u32 glk_tun_val; | |
c959a6b0 AH |
515 | }; |
516 | ||
c37f69ff | 517 | static const guid_t intel_dsm_guid = |
94116f81 AS |
518 | GUID_INIT(0xF6C13EA5, 0x65CD, 0x461F, |
519 | 0xAB, 0x7A, 0x29, 0xF7, 0xE8, 0xD5, 0xBD, 0x61); | |
c959a6b0 AH |
520 | |
521 | static int __intel_dsm(struct intel_host *intel_host, struct device *dev, | |
522 | unsigned int fn, u32 *result) | |
523 | { | |
524 | union acpi_object *obj; | |
525 | int err = 0; | |
a72016a4 | 526 | size_t len; |
c959a6b0 | 527 | |
94116f81 | 528 | obj = acpi_evaluate_dsm(ACPI_HANDLE(dev), &intel_dsm_guid, 0, fn, NULL); |
c959a6b0 AH |
529 | if (!obj) |
530 | return -EOPNOTSUPP; | |
531 | ||
532 | if (obj->type != ACPI_TYPE_BUFFER || obj->buffer.length < 1) { | |
533 | err = -EINVAL; | |
534 | goto out; | |
535 | } | |
536 | ||
a72016a4 AH |
537 | len = min_t(size_t, obj->buffer.length, 4); |
538 | ||
539 | *result = 0; | |
540 | memcpy(result, obj->buffer.pointer, len); | |
c959a6b0 AH |
541 | out: |
542 | ACPI_FREE(obj); | |
543 | ||
544 | return err; | |
545 | } | |
546 | ||
547 | static int intel_dsm(struct intel_host *intel_host, struct device *dev, | |
548 | unsigned int fn, u32 *result) | |
549 | { | |
550 | if (fn > 31 || !(intel_host->dsm_fns & (1 << fn))) | |
551 | return -EOPNOTSUPP; | |
552 | ||
553 | return __intel_dsm(intel_host, dev, fn, result); | |
554 | } | |
555 | ||
556 | static void intel_dsm_init(struct intel_host *intel_host, struct device *dev, | |
557 | struct mmc_host *mmc) | |
558 | { | |
559 | int err; | |
560 | u32 val; | |
561 | ||
eb701ce1 AH |
562 | intel_host->d3_retune = true; |
563 | ||
c959a6b0 AH |
564 | err = __intel_dsm(intel_host, dev, INTEL_DSM_FNS, &intel_host->dsm_fns); |
565 | if (err) { | |
566 | pr_debug("%s: DSM not supported, error %d\n", | |
567 | mmc_hostname(mmc), err); | |
568 | return; | |
569 | } | |
570 | ||
571 | pr_debug("%s: DSM function mask %#x\n", | |
572 | mmc_hostname(mmc), intel_host->dsm_fns); | |
573 | ||
51ced59c AH |
574 | err = intel_dsm(intel_host, dev, INTEL_DSM_DRV_STRENGTH, &val); |
575 | intel_host->drv_strength = err ? 0 : val; | |
576 | ||
c959a6b0 AH |
577 | err = intel_dsm(intel_host, dev, INTEL_DSM_D3_RETUNE, &val); |
578 | intel_host->d3_retune = err ? true : !!val; | |
579 | } | |
580 | ||
c9faff6c AH |
581 | static void sdhci_pci_int_hw_reset(struct sdhci_host *host) |
582 | { | |
583 | u8 reg; | |
584 | ||
585 | reg = sdhci_readb(host, SDHCI_POWER_CONTROL); | |
586 | reg |= 0x10; | |
587 | sdhci_writeb(host, reg, SDHCI_POWER_CONTROL); | |
588 | /* For eMMC, minimum is 1us but give it 9us for good measure */ | |
589 | udelay(9); | |
590 | reg &= ~0x10; | |
591 | sdhci_writeb(host, reg, SDHCI_POWER_CONTROL); | |
592 | /* For eMMC, minimum is 200us but give it 300us for good measure */ | |
593 | usleep_range(300, 1000); | |
594 | } | |
595 | ||
51ced59c AH |
596 | static int intel_select_drive_strength(struct mmc_card *card, |
597 | unsigned int max_dtr, int host_drv, | |
598 | int card_drv, int *drv_type) | |
e1bfad6d | 599 | { |
51ced59c AH |
600 | struct sdhci_host *host = mmc_priv(card->host); |
601 | struct sdhci_pci_slot *slot = sdhci_priv(host); | |
602 | struct intel_host *intel_host = sdhci_pci_priv(slot); | |
e1bfad6d | 603 | |
1a8eb6b3 AH |
604 | if (!(mmc_driver_type_mask(intel_host->drv_strength) & card_drv)) |
605 | return 0; | |
606 | ||
51ced59c | 607 | return intel_host->drv_strength; |
e1bfad6d AH |
608 | } |
609 | ||
163cbe31 AH |
610 | static int bxt_get_cd(struct mmc_host *mmc) |
611 | { | |
612 | int gpio_cd = mmc_gpio_get_cd(mmc); | |
613 | struct sdhci_host *host = mmc_priv(mmc); | |
614 | unsigned long flags; | |
615 | int ret = 0; | |
616 | ||
617 | if (!gpio_cd) | |
618 | return 0; | |
619 | ||
163cbe31 AH |
620 | spin_lock_irqsave(&host->lock, flags); |
621 | ||
622 | if (host->flags & SDHCI_DEVICE_DEAD) | |
623 | goto out; | |
624 | ||
625 | ret = !!(sdhci_readl(host, SDHCI_PRESENT_STATE) & SDHCI_CARD_PRESENT); | |
626 | out: | |
627 | spin_unlock_irqrestore(&host->lock, flags); | |
628 | ||
163cbe31 AH |
629 | return ret; |
630 | } | |
631 | ||
48d685a2 AH |
632 | #define SDHCI_INTEL_PWR_TIMEOUT_CNT 20 |
633 | #define SDHCI_INTEL_PWR_TIMEOUT_UDELAY 100 | |
634 | ||
635 | static void sdhci_intel_set_power(struct sdhci_host *host, unsigned char mode, | |
636 | unsigned short vdd) | |
637 | { | |
638 | int cntr; | |
639 | u8 reg; | |
640 | ||
641 | sdhci_set_power(host, mode, vdd); | |
642 | ||
643 | if (mode == MMC_POWER_OFF) | |
644 | return; | |
645 | ||
646 | /* | |
647 | * Bus power might not enable after D3 -> D0 transition due to the | |
648 | * present state not yet having propagated. Retry for up to 2ms. | |
649 | */ | |
650 | for (cntr = 0; cntr < SDHCI_INTEL_PWR_TIMEOUT_CNT; cntr++) { | |
651 | reg = sdhci_readb(host, SDHCI_POWER_CONTROL); | |
652 | if (reg & SDHCI_POWER_ON) | |
653 | break; | |
654 | udelay(SDHCI_INTEL_PWR_TIMEOUT_UDELAY); | |
655 | reg |= SDHCI_POWER_ON; | |
656 | sdhci_writeb(host, reg, SDHCI_POWER_CONTROL); | |
657 | } | |
658 | } | |
659 | ||
bc55dcd8 AH |
660 | #define INTEL_HS400_ES_REG 0x78 |
661 | #define INTEL_HS400_ES_BIT BIT(0) | |
662 | ||
663 | static void intel_hs400_enhanced_strobe(struct mmc_host *mmc, | |
664 | struct mmc_ios *ios) | |
665 | { | |
666 | struct sdhci_host *host = mmc_priv(mmc); | |
667 | u32 val; | |
668 | ||
669 | val = sdhci_readl(host, INTEL_HS400_ES_REG); | |
670 | if (ios->enhanced_strobe) | |
671 | val |= INTEL_HS400_ES_BIT; | |
672 | else | |
673 | val &= ~INTEL_HS400_ES_BIT; | |
674 | sdhci_writel(host, val, INTEL_HS400_ES_REG); | |
675 | } | |
676 | ||
be17355a AH |
677 | static int intel_start_signal_voltage_switch(struct mmc_host *mmc, |
678 | struct mmc_ios *ios) | |
6ae03368 | 679 | { |
be17355a AH |
680 | struct device *dev = mmc_dev(mmc); |
681 | struct sdhci_host *host = mmc_priv(mmc); | |
6ae03368 AH |
682 | struct sdhci_pci_slot *slot = sdhci_priv(host); |
683 | struct intel_host *intel_host = sdhci_pci_priv(slot); | |
be17355a | 684 | unsigned int fn; |
6ae03368 AH |
685 | u32 result = 0; |
686 | int err; | |
687 | ||
be17355a AH |
688 | err = sdhci_start_signal_voltage_switch(mmc, ios); |
689 | if (err) | |
690 | return err; | |
691 | ||
692 | switch (ios->signal_voltage) { | |
693 | case MMC_SIGNAL_VOLTAGE_330: | |
694 | fn = INTEL_DSM_V33_SWITCH; | |
695 | break; | |
696 | case MMC_SIGNAL_VOLTAGE_180: | |
697 | fn = INTEL_DSM_V18_SWITCH; | |
698 | break; | |
699 | default: | |
700 | return 0; | |
701 | } | |
702 | ||
703 | err = intel_dsm(intel_host, dev, fn, &result); | |
704 | pr_debug("%s: %s DSM fn %u error %d result %u\n", | |
705 | mmc_hostname(mmc), __func__, fn, err, result); | |
706 | ||
707 | return 0; | |
6ae03368 AH |
708 | } |
709 | ||
48d685a2 AH |
710 | static const struct sdhci_ops sdhci_intel_byt_ops = { |
711 | .set_clock = sdhci_set_clock, | |
712 | .set_power = sdhci_intel_set_power, | |
713 | .enable_dma = sdhci_pci_enable_dma, | |
adc16398 | 714 | .set_bus_width = sdhci_set_bus_width, |
48d685a2 AH |
715 | .reset = sdhci_reset, |
716 | .set_uhs_signaling = sdhci_set_uhs_signaling, | |
717 | .hw_reset = sdhci_pci_hw_reset, | |
718 | }; | |
719 | ||
8ee82bda AH |
720 | static const struct sdhci_ops sdhci_intel_glk_ops = { |
721 | .set_clock = sdhci_set_clock, | |
722 | .set_power = sdhci_intel_set_power, | |
723 | .enable_dma = sdhci_pci_enable_dma, | |
724 | .set_bus_width = sdhci_set_bus_width, | |
725 | .reset = sdhci_reset, | |
726 | .set_uhs_signaling = sdhci_set_uhs_signaling, | |
727 | .hw_reset = sdhci_pci_hw_reset, | |
8ee82bda AH |
728 | .irq = sdhci_cqhci_irq, |
729 | }; | |
730 | ||
c959a6b0 AH |
731 | static void byt_read_dsm(struct sdhci_pci_slot *slot) |
732 | { | |
733 | struct intel_host *intel_host = sdhci_pci_priv(slot); | |
734 | struct device *dev = &slot->chip->pdev->dev; | |
735 | struct mmc_host *mmc = slot->host->mmc; | |
736 | ||
737 | intel_dsm_init(intel_host, dev, mmc); | |
738 | slot->chip->rpm_retune = intel_host->d3_retune; | |
739 | } | |
740 | ||
f8870ae6 AH |
741 | static int intel_execute_tuning(struct mmc_host *mmc, u32 opcode) |
742 | { | |
743 | int err = sdhci_execute_tuning(mmc, opcode); | |
744 | struct sdhci_host *host = mmc_priv(mmc); | |
745 | ||
746 | if (err) | |
747 | return err; | |
748 | ||
749 | /* | |
750 | * Tuning can leave the IP in an active state (Buffer Read Enable bit | |
751 | * set) which prevents the entry to low power states (i.e. S0i3). Data | |
752 | * reset will clear it. | |
753 | */ | |
754 | sdhci_reset(host, SDHCI_RESET_DATA); | |
755 | ||
756 | return 0; | |
757 | } | |
758 | ||
759 | static void byt_probe_slot(struct sdhci_pci_slot *slot) | |
728ef3d1 | 760 | { |
f8870ae6 | 761 | struct mmc_host_ops *ops = &slot->host->mmc_host_ops; |
809090e8 AH |
762 | struct device *dev = &slot->chip->pdev->dev; |
763 | struct mmc_host *mmc = slot->host->mmc; | |
f8870ae6 | 764 | |
c959a6b0 | 765 | byt_read_dsm(slot); |
f8870ae6 | 766 | |
0a49a619 AH |
767 | byt_ocp_setting(slot->chip->pdev); |
768 | ||
f8870ae6 | 769 | ops->execute_tuning = intel_execute_tuning; |
be17355a | 770 | ops->start_signal_voltage_switch = intel_start_signal_voltage_switch; |
809090e8 AH |
771 | |
772 | device_property_read_u32(dev, "max-frequency", &mmc->f_max); | |
f8870ae6 AH |
773 | } |
774 | ||
775 | static int byt_emmc_probe_slot(struct sdhci_pci_slot *slot) | |
776 | { | |
777 | byt_probe_slot(slot); | |
c9faff6c | 778 | slot->host->mmc->caps |= MMC_CAP_8_BIT_DATA | MMC_CAP_NONREMOVABLE | |
6aab23a8 | 779 | MMC_CAP_HW_RESET | MMC_CAP_1_8V_DDR | |
32828857 | 780 | MMC_CAP_CMD_DURING_TFR | |
6aab23a8 | 781 | MMC_CAP_WAIT_WHILE_BUSY; |
c9faff6c | 782 | slot->hw_reset = sdhci_pci_int_hw_reset; |
a06586b6 AH |
783 | if (slot->chip->pdev->device == PCI_DEVICE_ID_INTEL_BSW_EMMC) |
784 | slot->host->timeout_clk = 1000; /* 1000 kHz i.e. 1 MHz */ | |
51ced59c AH |
785 | slot->host->mmc_host_ops.select_drive_strength = |
786 | intel_select_drive_strength; | |
728ef3d1 AH |
787 | return 0; |
788 | } | |
789 | ||
bedf9fc0 AH |
790 | static bool glk_broken_cqhci(struct sdhci_pci_slot *slot) |
791 | { | |
792 | return slot->chip->pdev->device == PCI_DEVICE_ID_INTEL_GLK_EMMC && | |
793 | dmi_match(DMI_BIOS_VENDOR, "LENOVO"); | |
794 | } | |
795 | ||
bc55dcd8 AH |
796 | static int glk_emmc_probe_slot(struct sdhci_pci_slot *slot) |
797 | { | |
798 | int ret = byt_emmc_probe_slot(slot); | |
799 | ||
bedf9fc0 AH |
800 | if (!glk_broken_cqhci(slot)) |
801 | slot->host->mmc->caps2 |= MMC_CAP2_CQE; | |
8ee82bda | 802 | |
bc55dcd8 AH |
803 | if (slot->chip->pdev->device != PCI_DEVICE_ID_INTEL_GLK_EMMC) { |
804 | slot->host->mmc->caps2 |= MMC_CAP2_HS400_ES, | |
805 | slot->host->mmc_host_ops.hs400_enhanced_strobe = | |
806 | intel_hs400_enhanced_strobe; | |
8ee82bda | 807 | slot->host->mmc->caps2 |= MMC_CAP2_CQE_DCMD; |
bc55dcd8 AH |
808 | } |
809 | ||
810 | return ret; | |
811 | } | |
812 | ||
8ee82bda | 813 | static const struct cqhci_host_ops glk_cqhci_ops = { |
7b7d57fd | 814 | .enable = sdhci_cqe_enable, |
8ee82bda AH |
815 | .disable = sdhci_cqe_disable, |
816 | .dumpregs = sdhci_pci_dumpregs, | |
817 | }; | |
818 | ||
819 | static int glk_emmc_add_host(struct sdhci_pci_slot *slot) | |
820 | { | |
821 | struct device *dev = &slot->chip->pdev->dev; | |
822 | struct sdhci_host *host = slot->host; | |
823 | struct cqhci_host *cq_host; | |
824 | bool dma64; | |
825 | int ret; | |
826 | ||
827 | ret = sdhci_setup_host(host); | |
828 | if (ret) | |
829 | return ret; | |
830 | ||
831 | cq_host = devm_kzalloc(dev, sizeof(*cq_host), GFP_KERNEL); | |
832 | if (!cq_host) { | |
833 | ret = -ENOMEM; | |
834 | goto cleanup; | |
835 | } | |
836 | ||
837 | cq_host->mmio = host->ioaddr + 0x200; | |
838 | cq_host->quirks |= CQHCI_QUIRK_SHORT_TXFR_DESC_SZ; | |
839 | cq_host->ops = &glk_cqhci_ops; | |
840 | ||
841 | dma64 = host->flags & SDHCI_USE_64_BIT_DMA; | |
842 | if (dma64) | |
843 | cq_host->caps |= CQHCI_TASK_DESC_SZ_128; | |
844 | ||
845 | ret = cqhci_init(cq_host, host->mmc, dma64); | |
846 | if (ret) | |
847 | goto cleanup; | |
848 | ||
849 | ret = __sdhci_add_host(host); | |
850 | if (ret) | |
851 | goto cleanup; | |
852 | ||
853 | return 0; | |
854 | ||
855 | cleanup: | |
856 | sdhci_cleanup_host(host); | |
857 | return ret; | |
858 | } | |
859 | ||
5305ec6a AH |
860 | #ifdef CONFIG_PM |
861 | #define GLK_RX_CTRL1 0x834 | |
862 | #define GLK_TUN_VAL 0x840 | |
863 | #define GLK_PATH_PLL GENMASK(13, 8) | |
864 | #define GLK_DLY GENMASK(6, 0) | |
865 | /* Workaround firmware failing to restore the tuning value */ | |
866 | static void glk_rpm_retune_wa(struct sdhci_pci_chip *chip, bool susp) | |
867 | { | |
868 | struct sdhci_pci_slot *slot = chip->slots[0]; | |
869 | struct intel_host *intel_host = sdhci_pci_priv(slot); | |
870 | struct sdhci_host *host = slot->host; | |
871 | u32 glk_rx_ctrl1; | |
872 | u32 glk_tun_val; | |
873 | u32 dly; | |
874 | ||
875 | if (intel_host->rpm_retune_ok || !mmc_can_retune(host->mmc)) | |
876 | return; | |
877 | ||
878 | glk_rx_ctrl1 = sdhci_readl(host, GLK_RX_CTRL1); | |
879 | glk_tun_val = sdhci_readl(host, GLK_TUN_VAL); | |
880 | ||
881 | if (susp) { | |
882 | intel_host->glk_rx_ctrl1 = glk_rx_ctrl1; | |
883 | intel_host->glk_tun_val = glk_tun_val; | |
884 | return; | |
885 | } | |
886 | ||
887 | if (!intel_host->glk_tun_val) | |
888 | return; | |
889 | ||
890 | if (glk_rx_ctrl1 != intel_host->glk_rx_ctrl1) { | |
891 | intel_host->rpm_retune_ok = true; | |
892 | return; | |
893 | } | |
894 | ||
895 | dly = FIELD_PREP(GLK_DLY, FIELD_GET(GLK_PATH_PLL, glk_rx_ctrl1) + | |
896 | (intel_host->glk_tun_val << 1)); | |
897 | if (dly == FIELD_GET(GLK_DLY, glk_rx_ctrl1)) | |
898 | return; | |
899 | ||
900 | glk_rx_ctrl1 = (glk_rx_ctrl1 & ~GLK_DLY) | dly; | |
901 | sdhci_writel(host, glk_rx_ctrl1, GLK_RX_CTRL1); | |
902 | ||
903 | intel_host->rpm_retune_ok = true; | |
904 | chip->rpm_retune = true; | |
905 | mmc_retune_needed(host->mmc); | |
906 | pr_info("%s: Requiring re-tune after rpm resume", mmc_hostname(host->mmc)); | |
907 | } | |
908 | ||
909 | static void glk_rpm_retune_chk(struct sdhci_pci_chip *chip, bool susp) | |
910 | { | |
911 | if (chip->pdev->device == PCI_DEVICE_ID_INTEL_GLK_EMMC && | |
912 | !chip->rpm_retune) | |
913 | glk_rpm_retune_wa(chip, susp); | |
914 | } | |
915 | ||
916 | static int glk_runtime_suspend(struct sdhci_pci_chip *chip) | |
917 | { | |
918 | glk_rpm_retune_chk(chip, true); | |
919 | ||
920 | return sdhci_cqhci_runtime_suspend(chip); | |
921 | } | |
922 | ||
923 | static int glk_runtime_resume(struct sdhci_pci_chip *chip) | |
924 | { | |
925 | glk_rpm_retune_chk(chip, false); | |
926 | ||
927 | return sdhci_cqhci_runtime_resume(chip); | |
928 | } | |
929 | #endif | |
930 | ||
3f23df72 ZB |
931 | #ifdef CONFIG_ACPI |
932 | static int ni_set_max_freq(struct sdhci_pci_slot *slot) | |
933 | { | |
934 | acpi_status status; | |
935 | unsigned long long max_freq; | |
936 | ||
937 | status = acpi_evaluate_integer(ACPI_HANDLE(&slot->chip->pdev->dev), | |
938 | "MXFQ", NULL, &max_freq); | |
939 | if (ACPI_FAILURE(status)) { | |
940 | dev_err(&slot->chip->pdev->dev, | |
941 | "MXFQ not found in acpi table\n"); | |
942 | return -EINVAL; | |
943 | } | |
944 | ||
945 | slot->host->mmc->f_max = max_freq * 1000000; | |
946 | ||
947 | return 0; | |
948 | } | |
949 | #else | |
950 | static inline int ni_set_max_freq(struct sdhci_pci_slot *slot) | |
951 | { | |
952 | return 0; | |
953 | } | |
954 | #endif | |
955 | ||
42b06496 ZB |
956 | static int ni_byt_sdio_probe_slot(struct sdhci_pci_slot *slot) |
957 | { | |
3f23df72 ZB |
958 | int err; |
959 | ||
f8870ae6 | 960 | byt_probe_slot(slot); |
c959a6b0 | 961 | |
3f23df72 ZB |
962 | err = ni_set_max_freq(slot); |
963 | if (err) | |
964 | return err; | |
965 | ||
42b06496 ZB |
966 | slot->host->mmc->caps |= MMC_CAP_POWER_OFF_CARD | MMC_CAP_NONREMOVABLE | |
967 | MMC_CAP_WAIT_WHILE_BUSY; | |
968 | return 0; | |
969 | } | |
970 | ||
728ef3d1 AH |
971 | static int byt_sdio_probe_slot(struct sdhci_pci_slot *slot) |
972 | { | |
f8870ae6 | 973 | byt_probe_slot(slot); |
6aab23a8 | 974 | slot->host->mmc->caps |= MMC_CAP_POWER_OFF_CARD | MMC_CAP_NONREMOVABLE | |
6aab23a8 | 975 | MMC_CAP_WAIT_WHILE_BUSY; |
728ef3d1 AH |
976 | return 0; |
977 | } | |
978 | ||
ff59c520 AH |
979 | static int byt_sd_probe_slot(struct sdhci_pci_slot *slot) |
980 | { | |
f8870ae6 | 981 | byt_probe_slot(slot); |
c2c49a2e | 982 | slot->host->mmc->caps |= MMC_CAP_WAIT_WHILE_BUSY | |
6cf4156c | 983 | MMC_CAP_AGGRESSIVE_PM | MMC_CAP_CD_WAKE; |
ff59c520 AH |
984 | slot->cd_idx = 0; |
985 | slot->cd_override_level = true; | |
163cbe31 | 986 | if (slot->chip->pdev->device == PCI_DEVICE_ID_INTEL_BXT_SD || |
01d6b2a4 | 987 | slot->chip->pdev->device == PCI_DEVICE_ID_INTEL_BXTM_SD || |
2d1956d0 | 988 | slot->chip->pdev->device == PCI_DEVICE_ID_INTEL_APL_SD || |
c2c49a2e | 989 | slot->chip->pdev->device == PCI_DEVICE_ID_INTEL_GLK_SD) |
163cbe31 AH |
990 | slot->host->mmc_host_ops.get_cd = bxt_get_cd; |
991 | ||
bb26b841 KR |
992 | if (slot->chip->pdev->subsystem_vendor == PCI_VENDOR_ID_NI && |
993 | slot->chip->pdev->subsystem_device == PCI_SUBDEVICE_ID_NI_78E3) | |
994 | slot->host->mmc->caps2 |= MMC_CAP2_AVOID_3_3V; | |
995 | ||
ff59c520 AH |
996 | return 0; |
997 | } | |
998 | ||
0a49a619 AH |
999 | #ifdef CONFIG_PM_SLEEP |
1000 | ||
1001 | static int byt_resume(struct sdhci_pci_chip *chip) | |
1002 | { | |
1003 | byt_ocp_setting(chip->pdev); | |
1004 | ||
1005 | return sdhci_pci_resume_host(chip); | |
1006 | } | |
1007 | ||
1008 | #endif | |
1009 | ||
1010 | #ifdef CONFIG_PM | |
1011 | ||
1012 | static int byt_runtime_resume(struct sdhci_pci_chip *chip) | |
1013 | { | |
1014 | byt_ocp_setting(chip->pdev); | |
1015 | ||
1016 | return sdhci_pci_runtime_resume_host(chip); | |
1017 | } | |
1018 | ||
1019 | #endif | |
1020 | ||
728ef3d1 | 1021 | static const struct sdhci_pci_fixes sdhci_intel_byt_emmc = { |
0a49a619 AH |
1022 | #ifdef CONFIG_PM_SLEEP |
1023 | .resume = byt_resume, | |
1024 | #endif | |
1025 | #ifdef CONFIG_PM | |
1026 | .runtime_resume = byt_runtime_resume, | |
1027 | #endif | |
728ef3d1 AH |
1028 | .allow_runtime_pm = true, |
1029 | .probe_slot = byt_emmc_probe_slot, | |
aeae6ad3 AH |
1030 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC | |
1031 | SDHCI_QUIRK_NO_LED, | |
e58e4a0d | 1032 | .quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN | |
b69587e2 | 1033 | SDHCI_QUIRK2_CAPS_BIT63_FOR_HS400 | |
e58e4a0d | 1034 | SDHCI_QUIRK2_STOP_WITH_TC, |
fee686b7 | 1035 | .ops = &sdhci_intel_byt_ops, |
c959a6b0 | 1036 | .priv_size = sizeof(struct intel_host), |
728ef3d1 AH |
1037 | }; |
1038 | ||
bc55dcd8 AH |
1039 | static const struct sdhci_pci_fixes sdhci_intel_glk_emmc = { |
1040 | .allow_runtime_pm = true, | |
1041 | .probe_slot = glk_emmc_probe_slot, | |
8ee82bda AH |
1042 | .add_host = glk_emmc_add_host, |
1043 | #ifdef CONFIG_PM_SLEEP | |
1044 | .suspend = sdhci_cqhci_suspend, | |
1045 | .resume = sdhci_cqhci_resume, | |
1046 | #endif | |
1047 | #ifdef CONFIG_PM | |
5305ec6a AH |
1048 | .runtime_suspend = glk_runtime_suspend, |
1049 | .runtime_resume = glk_runtime_resume, | |
8ee82bda | 1050 | #endif |
aeae6ad3 AH |
1051 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC | |
1052 | SDHCI_QUIRK_NO_LED, | |
bc55dcd8 AH |
1053 | .quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN | |
1054 | SDHCI_QUIRK2_CAPS_BIT63_FOR_HS400 | | |
1055 | SDHCI_QUIRK2_STOP_WITH_TC, | |
8ee82bda | 1056 | .ops = &sdhci_intel_glk_ops, |
bc55dcd8 AH |
1057 | .priv_size = sizeof(struct intel_host), |
1058 | }; | |
1059 | ||
42b06496 | 1060 | static const struct sdhci_pci_fixes sdhci_ni_byt_sdio = { |
0a49a619 AH |
1061 | #ifdef CONFIG_PM_SLEEP |
1062 | .resume = byt_resume, | |
1063 | #endif | |
1064 | #ifdef CONFIG_PM | |
1065 | .runtime_resume = byt_runtime_resume, | |
1066 | #endif | |
aeae6ad3 AH |
1067 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC | |
1068 | SDHCI_QUIRK_NO_LED, | |
42b06496 ZB |
1069 | .quirks2 = SDHCI_QUIRK2_HOST_OFF_CARD_ON | |
1070 | SDHCI_QUIRK2_PRESET_VALUE_BROKEN, | |
1071 | .allow_runtime_pm = true, | |
1072 | .probe_slot = ni_byt_sdio_probe_slot, | |
1073 | .ops = &sdhci_intel_byt_ops, | |
c959a6b0 | 1074 | .priv_size = sizeof(struct intel_host), |
42b06496 ZB |
1075 | }; |
1076 | ||
728ef3d1 | 1077 | static const struct sdhci_pci_fixes sdhci_intel_byt_sdio = { |
0a49a619 AH |
1078 | #ifdef CONFIG_PM_SLEEP |
1079 | .resume = byt_resume, | |
1080 | #endif | |
1081 | #ifdef CONFIG_PM | |
1082 | .runtime_resume = byt_runtime_resume, | |
1083 | #endif | |
aeae6ad3 AH |
1084 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC | |
1085 | SDHCI_QUIRK_NO_LED, | |
b7574bad GY |
1086 | .quirks2 = SDHCI_QUIRK2_HOST_OFF_CARD_ON | |
1087 | SDHCI_QUIRK2_PRESET_VALUE_BROKEN, | |
728ef3d1 AH |
1088 | .allow_runtime_pm = true, |
1089 | .probe_slot = byt_sdio_probe_slot, | |
fee686b7 | 1090 | .ops = &sdhci_intel_byt_ops, |
c959a6b0 | 1091 | .priv_size = sizeof(struct intel_host), |
728ef3d1 AH |
1092 | }; |
1093 | ||
1094 | static const struct sdhci_pci_fixes sdhci_intel_byt_sd = { | |
0a49a619 AH |
1095 | #ifdef CONFIG_PM_SLEEP |
1096 | .resume = byt_resume, | |
1097 | #endif | |
1098 | #ifdef CONFIG_PM | |
1099 | .runtime_resume = byt_runtime_resume, | |
1100 | #endif | |
aeae6ad3 AH |
1101 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC | |
1102 | SDHCI_QUIRK_NO_LED, | |
b7574bad | 1103 | .quirks2 = SDHCI_QUIRK2_CARD_ON_NEEDS_BUS_ON | |
e58e4a0d AH |
1104 | SDHCI_QUIRK2_PRESET_VALUE_BROKEN | |
1105 | SDHCI_QUIRK2_STOP_WITH_TC, | |
7396e318 | 1106 | .allow_runtime_pm = true, |
77a0122e | 1107 | .own_cd_for_runtime_pm = true, |
ff59c520 | 1108 | .probe_slot = byt_sd_probe_slot, |
fee686b7 | 1109 | .ops = &sdhci_intel_byt_ops, |
c959a6b0 | 1110 | .priv_size = sizeof(struct intel_host), |
728ef3d1 AH |
1111 | }; |
1112 | ||
8776a165 | 1113 | /* Define Host controllers for Intel Merrifield platform */ |
1f64cec2 AS |
1114 | #define INTEL_MRFLD_EMMC_0 0 |
1115 | #define INTEL_MRFLD_EMMC_1 1 | |
4674b6c8 | 1116 | #define INTEL_MRFLD_SD 2 |
d5565577 | 1117 | #define INTEL_MRFLD_SDIO 3 |
8776a165 | 1118 | |
0e39220e AS |
1119 | #ifdef CONFIG_ACPI |
1120 | static void intel_mrfld_mmc_fix_up_power_slot(struct sdhci_pci_slot *slot) | |
1121 | { | |
1122 | struct acpi_device *device, *child; | |
1123 | ||
1124 | device = ACPI_COMPANION(&slot->chip->pdev->dev); | |
1125 | if (!device) | |
1126 | return; | |
1127 | ||
1128 | acpi_device_fix_up_power(device); | |
1129 | list_for_each_entry(child, &device->children, node) | |
1130 | if (child->status.present && child->status.enabled) | |
1131 | acpi_device_fix_up_power(child); | |
1132 | } | |
1133 | #else | |
1134 | static inline void intel_mrfld_mmc_fix_up_power_slot(struct sdhci_pci_slot *slot) {} | |
1135 | #endif | |
1136 | ||
1f64cec2 | 1137 | static int intel_mrfld_mmc_probe_slot(struct sdhci_pci_slot *slot) |
8776a165 | 1138 | { |
2e57bbe2 AS |
1139 | unsigned int func = PCI_FUNC(slot->chip->pdev->devfn); |
1140 | ||
1141 | switch (func) { | |
1142 | case INTEL_MRFLD_EMMC_0: | |
1143 | case INTEL_MRFLD_EMMC_1: | |
1144 | slot->host->mmc->caps |= MMC_CAP_NONREMOVABLE | | |
1145 | MMC_CAP_8_BIT_DATA | | |
1146 | MMC_CAP_1_8V_DDR; | |
1147 | break; | |
4674b6c8 AS |
1148 | case INTEL_MRFLD_SD: |
1149 | slot->host->quirks2 |= SDHCI_QUIRK2_NO_1_8_V; | |
1150 | break; | |
d5565577 | 1151 | case INTEL_MRFLD_SDIO: |
2a609abe AS |
1152 | /* Advertise 2.0v for compatibility with the SDIO card's OCR */ |
1153 | slot->host->ocr_mask = MMC_VDD_20_21 | MMC_VDD_165_195; | |
d5565577 AS |
1154 | slot->host->mmc->caps |= MMC_CAP_NONREMOVABLE | |
1155 | MMC_CAP_POWER_OFF_CARD; | |
1156 | break; | |
2e57bbe2 | 1157 | default: |
8776a165 | 1158 | return -ENODEV; |
2e57bbe2 | 1159 | } |
0e39220e AS |
1160 | |
1161 | intel_mrfld_mmc_fix_up_power_slot(slot); | |
8776a165 DC |
1162 | return 0; |
1163 | } | |
1164 | ||
1f64cec2 | 1165 | static const struct sdhci_pci_fixes sdhci_intel_mrfld_mmc = { |
8776a165 | 1166 | .quirks = SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC, |
b7574bad GY |
1167 | .quirks2 = SDHCI_QUIRK2_BROKEN_HS200 | |
1168 | SDHCI_QUIRK2_PRESET_VALUE_BROKEN, | |
f1b55a55 | 1169 | .allow_runtime_pm = true, |
1f64cec2 | 1170 | .probe_slot = intel_mrfld_mmc_probe_slot, |
8776a165 DC |
1171 | }; |
1172 | ||
45211e21 PO |
1173 | static int jmicron_pmos(struct sdhci_pci_chip *chip, int on) |
1174 | { | |
1175 | u8 scratch; | |
1176 | int ret; | |
1177 | ||
1178 | ret = pci_read_config_byte(chip->pdev, 0xAE, &scratch); | |
1179 | if (ret) | |
1180 | return ret; | |
1181 | ||
1182 | /* | |
1183 | * Turn PMOS on [bit 0], set over current detection to 2.4 V | |
1184 | * [bit 1:2] and enable over current debouncing [bit 6]. | |
1185 | */ | |
1186 | if (on) | |
1187 | scratch |= 0x47; | |
1188 | else | |
1189 | scratch &= ~0x47; | |
1190 | ||
7582041f | 1191 | return pci_write_config_byte(chip->pdev, 0xAE, scratch); |
45211e21 PO |
1192 | } |
1193 | ||
1194 | static int jmicron_probe(struct sdhci_pci_chip *chip) | |
1195 | { | |
1196 | int ret; | |
8f230f45 | 1197 | u16 mmcdev = 0; |
45211e21 | 1198 | |
93fc48c7 PO |
1199 | if (chip->pdev->revision == 0) { |
1200 | chip->quirks |= SDHCI_QUIRK_32BIT_DMA_ADDR | | |
1201 | SDHCI_QUIRK_32BIT_DMA_SIZE | | |
2134a922 | 1202 | SDHCI_QUIRK_32BIT_ADMA_SIZE | |
4a3cba32 | 1203 | SDHCI_QUIRK_RESET_AFTER_REQUEST | |
86a6a874 | 1204 | SDHCI_QUIRK_BROKEN_SMALL_PIO; |
93fc48c7 PO |
1205 | } |
1206 | ||
4489428a PO |
1207 | /* |
1208 | * JMicron chips can have two interfaces to the same hardware | |
1209 | * in order to work around limitations in Microsoft's driver. | |
1210 | * We need to make sure we only bind to one of them. | |
1211 | * | |
1212 | * This code assumes two things: | |
1213 | * | |
1214 | * 1. The PCI code adds subfunctions in order. | |
1215 | * | |
1216 | * 2. The MMC interface has a lower subfunction number | |
1217 | * than the SD interface. | |
1218 | */ | |
8f230f45 TI |
1219 | if (chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB38X_SD) |
1220 | mmcdev = PCI_DEVICE_ID_JMICRON_JMB38X_MMC; | |
1221 | else if (chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_SD) | |
1222 | mmcdev = PCI_DEVICE_ID_JMICRON_JMB388_ESD; | |
1223 | ||
1224 | if (mmcdev) { | |
4489428a PO |
1225 | struct pci_dev *sd_dev; |
1226 | ||
1227 | sd_dev = NULL; | |
1228 | while ((sd_dev = pci_get_device(PCI_VENDOR_ID_JMICRON, | |
8f230f45 | 1229 | mmcdev, sd_dev)) != NULL) { |
4489428a PO |
1230 | if ((PCI_SLOT(chip->pdev->devfn) == |
1231 | PCI_SLOT(sd_dev->devfn)) && | |
1232 | (chip->pdev->bus == sd_dev->bus)) | |
1233 | break; | |
1234 | } | |
1235 | ||
1236 | if (sd_dev) { | |
1237 | pci_dev_put(sd_dev); | |
1238 | dev_info(&chip->pdev->dev, "Refusing to bind to " | |
1239 | "secondary interface.\n"); | |
1240 | return -ENODEV; | |
1241 | } | |
1242 | } | |
1243 | ||
45211e21 PO |
1244 | /* |
1245 | * JMicron chips need a bit of a nudge to enable the power | |
1246 | * output pins. | |
1247 | */ | |
1248 | ret = jmicron_pmos(chip, 1); | |
1249 | if (ret) { | |
1250 | dev_err(&chip->pdev->dev, "Failure enabling card power\n"); | |
1251 | return ret; | |
1252 | } | |
1253 | ||
82b0e23a TI |
1254 | /* quirk for unsable RO-detection on JM388 chips */ |
1255 | if (chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_SD || | |
1256 | chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_ESD) | |
1257 | chip->quirks |= SDHCI_QUIRK_UNSTABLE_RO_DETECT; | |
1258 | ||
45211e21 PO |
1259 | return 0; |
1260 | } | |
1261 | ||
4489428a PO |
1262 | static void jmicron_enable_mmc(struct sdhci_host *host, int on) |
1263 | { | |
1264 | u8 scratch; | |
1265 | ||
1266 | scratch = readb(host->ioaddr + 0xC0); | |
1267 | ||
1268 | if (on) | |
1269 | scratch |= 0x01; | |
1270 | else | |
1271 | scratch &= ~0x01; | |
1272 | ||
1273 | writeb(scratch, host->ioaddr + 0xC0); | |
1274 | } | |
1275 | ||
1276 | static int jmicron_probe_slot(struct sdhci_pci_slot *slot) | |
1277 | { | |
2134a922 PO |
1278 | if (slot->chip->pdev->revision == 0) { |
1279 | u16 version; | |
1280 | ||
1281 | version = readl(slot->host->ioaddr + SDHCI_HOST_VERSION); | |
1282 | version = (version & SDHCI_VENDOR_VER_MASK) >> | |
1283 | SDHCI_VENDOR_VER_SHIFT; | |
1284 | ||
1285 | /* | |
1286 | * Older versions of the chip have lots of nasty glitches | |
1287 | * in the ADMA engine. It's best just to avoid it | |
1288 | * completely. | |
1289 | */ | |
1290 | if (version < 0xAC) | |
1291 | slot->host->quirks |= SDHCI_QUIRK_BROKEN_ADMA; | |
1292 | } | |
1293 | ||
8f230f45 TI |
1294 | /* JM388 MMC doesn't support 1.8V while SD supports it */ |
1295 | if (slot->chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_ESD) { | |
1296 | slot->host->ocr_avail_sd = MMC_VDD_32_33 | MMC_VDD_33_34 | | |
1297 | MMC_VDD_29_30 | MMC_VDD_30_31 | | |
1298 | MMC_VDD_165_195; /* allow 1.8V */ | |
1299 | slot->host->ocr_avail_mmc = MMC_VDD_32_33 | MMC_VDD_33_34 | | |
1300 | MMC_VDD_29_30 | MMC_VDD_30_31; /* no 1.8V for MMC */ | |
1301 | } | |
1302 | ||
4489428a PO |
1303 | /* |
1304 | * The secondary interface requires a bit set to get the | |
1305 | * interrupts. | |
1306 | */ | |
8f230f45 TI |
1307 | if (slot->chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB38X_MMC || |
1308 | slot->chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_ESD) | |
4489428a PO |
1309 | jmicron_enable_mmc(slot->host, 1); |
1310 | ||
d75c1084 TI |
1311 | slot->host->mmc->caps |= MMC_CAP_BUS_WIDTH_TEST; |
1312 | ||
4489428a PO |
1313 | return 0; |
1314 | } | |
1315 | ||
1e72859e | 1316 | static void jmicron_remove_slot(struct sdhci_pci_slot *slot, int dead) |
4489428a | 1317 | { |
1e72859e PO |
1318 | if (dead) |
1319 | return; | |
1320 | ||
8f230f45 TI |
1321 | if (slot->chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB38X_MMC || |
1322 | slot->chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_ESD) | |
4489428a PO |
1323 | jmicron_enable_mmc(slot->host, 0); |
1324 | } | |
1325 | ||
b7813f0f | 1326 | #ifdef CONFIG_PM_SLEEP |
29495aa0 | 1327 | static int jmicron_suspend(struct sdhci_pci_chip *chip) |
4489428a | 1328 | { |
30cf2803 AH |
1329 | int i, ret; |
1330 | ||
5c3c6126 | 1331 | ret = sdhci_pci_suspend_host(chip); |
30cf2803 AH |
1332 | if (ret) |
1333 | return ret; | |
4489428a | 1334 | |
8f230f45 TI |
1335 | if (chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB38X_MMC || |
1336 | chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_ESD) { | |
b177bc91 | 1337 | for (i = 0; i < chip->num_slots; i++) |
4489428a PO |
1338 | jmicron_enable_mmc(chip->slots[i]->host, 0); |
1339 | } | |
1340 | ||
1341 | return 0; | |
1342 | } | |
1343 | ||
45211e21 PO |
1344 | static int jmicron_resume(struct sdhci_pci_chip *chip) |
1345 | { | |
4489428a PO |
1346 | int ret, i; |
1347 | ||
8f230f45 TI |
1348 | if (chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB38X_MMC || |
1349 | chip->pdev->device == PCI_DEVICE_ID_JMICRON_JMB388_ESD) { | |
b177bc91 | 1350 | for (i = 0; i < chip->num_slots; i++) |
4489428a PO |
1351 | jmicron_enable_mmc(chip->slots[i]->host, 1); |
1352 | } | |
45211e21 PO |
1353 | |
1354 | ret = jmicron_pmos(chip, 1); | |
1355 | if (ret) { | |
1356 | dev_err(&chip->pdev->dev, "Failure enabling card power\n"); | |
1357 | return ret; | |
1358 | } | |
1359 | ||
30cf2803 | 1360 | return sdhci_pci_resume_host(chip); |
45211e21 | 1361 | } |
b7813f0f | 1362 | #endif |
45211e21 | 1363 | |
22606405 | 1364 | static const struct sdhci_pci_fixes sdhci_jmicron = { |
45211e21 PO |
1365 | .probe = jmicron_probe, |
1366 | ||
4489428a PO |
1367 | .probe_slot = jmicron_probe_slot, |
1368 | .remove_slot = jmicron_remove_slot, | |
1369 | ||
b7813f0f | 1370 | #ifdef CONFIG_PM_SLEEP |
4489428a | 1371 | .suspend = jmicron_suspend, |
45211e21 | 1372 | .resume = jmicron_resume, |
b7813f0f | 1373 | #endif |
22606405 PO |
1374 | }; |
1375 | ||
a7a6186c NP |
1376 | /* SysKonnect CardBus2SDIO extra registers */ |
1377 | #define SYSKT_CTRL 0x200 | |
1378 | #define SYSKT_RDFIFO_STAT 0x204 | |
1379 | #define SYSKT_WRFIFO_STAT 0x208 | |
1380 | #define SYSKT_POWER_DATA 0x20c | |
1381 | #define SYSKT_POWER_330 0xef | |
1382 | #define SYSKT_POWER_300 0xf8 | |
1383 | #define SYSKT_POWER_184 0xcc | |
1384 | #define SYSKT_POWER_CMD 0x20d | |
1385 | #define SYSKT_POWER_START (1 << 7) | |
1386 | #define SYSKT_POWER_STATUS 0x20e | |
1387 | #define SYSKT_POWER_STATUS_OK (1 << 0) | |
1388 | #define SYSKT_BOARD_REV 0x210 | |
1389 | #define SYSKT_CHIP_REV 0x211 | |
1390 | #define SYSKT_CONF_DATA 0x212 | |
1391 | #define SYSKT_CONF_DATA_1V8 (1 << 2) | |
1392 | #define SYSKT_CONF_DATA_2V5 (1 << 1) | |
1393 | #define SYSKT_CONF_DATA_3V3 (1 << 0) | |
1394 | ||
1395 | static int syskt_probe(struct sdhci_pci_chip *chip) | |
1396 | { | |
1397 | if ((chip->pdev->class & 0x0000FF) == PCI_SDHCI_IFVENDOR) { | |
1398 | chip->pdev->class &= ~0x0000FF; | |
1399 | chip->pdev->class |= PCI_SDHCI_IFDMA; | |
1400 | } | |
1401 | return 0; | |
1402 | } | |
1403 | ||
1404 | static int syskt_probe_slot(struct sdhci_pci_slot *slot) | |
1405 | { | |
1406 | int tm, ps; | |
1407 | ||
1408 | u8 board_rev = readb(slot->host->ioaddr + SYSKT_BOARD_REV); | |
1409 | u8 chip_rev = readb(slot->host->ioaddr + SYSKT_CHIP_REV); | |
1410 | dev_info(&slot->chip->pdev->dev, "SysKonnect CardBus2SDIO, " | |
1411 | "board rev %d.%d, chip rev %d.%d\n", | |
1412 | board_rev >> 4, board_rev & 0xf, | |
1413 | chip_rev >> 4, chip_rev & 0xf); | |
1414 | if (chip_rev >= 0x20) | |
1415 | slot->host->quirks |= SDHCI_QUIRK_FORCE_DMA; | |
1416 | ||
1417 | writeb(SYSKT_POWER_330, slot->host->ioaddr + SYSKT_POWER_DATA); | |
1418 | writeb(SYSKT_POWER_START, slot->host->ioaddr + SYSKT_POWER_CMD); | |
1419 | udelay(50); | |
1420 | tm = 10; /* Wait max 1 ms */ | |
1421 | do { | |
1422 | ps = readw(slot->host->ioaddr + SYSKT_POWER_STATUS); | |
1423 | if (ps & SYSKT_POWER_STATUS_OK) | |
1424 | break; | |
1425 | udelay(100); | |
1426 | } while (--tm); | |
1427 | if (!tm) { | |
1428 | dev_err(&slot->chip->pdev->dev, | |
1429 | "power regulator never stabilized"); | |
1430 | writeb(0, slot->host->ioaddr + SYSKT_POWER_CMD); | |
1431 | return -ENODEV; | |
1432 | } | |
1433 | ||
1434 | return 0; | |
1435 | } | |
1436 | ||
1437 | static const struct sdhci_pci_fixes sdhci_syskt = { | |
1438 | .quirks = SDHCI_QUIRK_NO_SIMULT_VDD_AND_POWER, | |
1439 | .probe = syskt_probe, | |
1440 | .probe_slot = syskt_probe_slot, | |
1441 | }; | |
1442 | ||
557b0697 HW |
1443 | static int via_probe(struct sdhci_pci_chip *chip) |
1444 | { | |
1445 | if (chip->pdev->revision == 0x10) | |
1446 | chip->quirks |= SDHCI_QUIRK_DELAY_AFTER_POWER; | |
1447 | ||
1448 | return 0; | |
1449 | } | |
1450 | ||
1451 | static const struct sdhci_pci_fixes sdhci_via = { | |
1452 | .probe = via_probe, | |
1453 | }; | |
1454 | ||
9107ebbf MC |
1455 | static int rtsx_probe_slot(struct sdhci_pci_slot *slot) |
1456 | { | |
1457 | slot->host->mmc->caps2 |= MMC_CAP2_HS200; | |
1458 | return 0; | |
1459 | } | |
1460 | ||
1461 | static const struct sdhci_pci_fixes sdhci_rtsx = { | |
1462 | .quirks2 = SDHCI_QUIRK2_PRESET_VALUE_BROKEN | | |
e30b978f | 1463 | SDHCI_QUIRK2_BROKEN_64_BIT_DMA | |
9107ebbf MC |
1464 | SDHCI_QUIRK2_BROKEN_DDR50, |
1465 | .probe_slot = rtsx_probe_slot, | |
1466 | }; | |
1467 | ||
b5e97d6e VW |
1468 | /*AMD chipset generation*/ |
1469 | enum amd_chipset_gen { | |
1470 | AMD_CHIPSET_BEFORE_ML, | |
1471 | AMD_CHIPSET_CZ, | |
1472 | AMD_CHIPSET_NL, | |
1473 | AMD_CHIPSET_UNKNOWN, | |
1474 | }; | |
1475 | ||
c31165d7 SS |
1476 | /* AMD registers */ |
1477 | #define AMD_SD_AUTO_PATTERN 0xB8 | |
1478 | #define AMD_MSLEEP_DURATION 4 | |
1479 | #define AMD_SD_MISC_CONTROL 0xD0 | |
1480 | #define AMD_MAX_TUNE_VALUE 0x0B | |
1481 | #define AMD_AUTO_TUNE_SEL 0x10800 | |
1482 | #define AMD_FIFO_PTR 0x30 | |
1483 | #define AMD_BIT_MASK 0x1F | |
1484 | ||
1485 | static void amd_tuning_reset(struct sdhci_host *host) | |
1486 | { | |
1487 | unsigned int val; | |
1488 | ||
1489 | val = sdhci_readw(host, SDHCI_HOST_CONTROL2); | |
1490 | val |= SDHCI_CTRL_PRESET_VAL_ENABLE | SDHCI_CTRL_EXEC_TUNING; | |
1491 | sdhci_writew(host, val, SDHCI_HOST_CONTROL2); | |
1492 | ||
1493 | val = sdhci_readw(host, SDHCI_HOST_CONTROL2); | |
1494 | val &= ~SDHCI_CTRL_EXEC_TUNING; | |
1495 | sdhci_writew(host, val, SDHCI_HOST_CONTROL2); | |
1496 | } | |
1497 | ||
1498 | static void amd_config_tuning_phase(struct pci_dev *pdev, u8 phase) | |
1499 | { | |
1500 | unsigned int val; | |
1501 | ||
1502 | pci_read_config_dword(pdev, AMD_SD_AUTO_PATTERN, &val); | |
1503 | val &= ~AMD_BIT_MASK; | |
1504 | val |= (AMD_AUTO_TUNE_SEL | (phase << 1)); | |
1505 | pci_write_config_dword(pdev, AMD_SD_AUTO_PATTERN, val); | |
1506 | } | |
1507 | ||
1508 | static void amd_enable_manual_tuning(struct pci_dev *pdev) | |
1509 | { | |
1510 | unsigned int val; | |
1511 | ||
1512 | pci_read_config_dword(pdev, AMD_SD_MISC_CONTROL, &val); | |
1513 | val |= AMD_FIFO_PTR; | |
1514 | pci_write_config_dword(pdev, AMD_SD_MISC_CONTROL, val); | |
1515 | } | |
1516 | ||
300ad899 | 1517 | static int amd_execute_tuning_hs200(struct sdhci_host *host, u32 opcode) |
c31165d7 SS |
1518 | { |
1519 | struct sdhci_pci_slot *slot = sdhci_priv(host); | |
1520 | struct pci_dev *pdev = slot->chip->pdev; | |
1521 | u8 valid_win = 0; | |
1522 | u8 valid_win_max = 0; | |
1523 | u8 valid_win_end = 0; | |
1524 | u8 ctrl, tune_around; | |
1525 | ||
1526 | amd_tuning_reset(host); | |
1527 | ||
1528 | for (tune_around = 0; tune_around < 12; tune_around++) { | |
1529 | amd_config_tuning_phase(pdev, tune_around); | |
1530 | ||
1531 | if (mmc_send_tuning(host->mmc, opcode, NULL)) { | |
1532 | valid_win = 0; | |
1533 | msleep(AMD_MSLEEP_DURATION); | |
1534 | ctrl = SDHCI_RESET_CMD | SDHCI_RESET_DATA; | |
1535 | sdhci_writeb(host, ctrl, SDHCI_SOFTWARE_RESET); | |
1536 | } else if (++valid_win > valid_win_max) { | |
1537 | valid_win_max = valid_win; | |
1538 | valid_win_end = tune_around; | |
1539 | } | |
1540 | } | |
1541 | ||
1542 | if (!valid_win_max) { | |
1543 | dev_err(&pdev->dev, "no tuning point found\n"); | |
1544 | return -EIO; | |
1545 | } | |
1546 | ||
1547 | amd_config_tuning_phase(pdev, valid_win_end - valid_win_max / 2); | |
1548 | ||
1549 | amd_enable_manual_tuning(pdev); | |
1550 | ||
1551 | host->mmc->retune_period = 0; | |
1552 | ||
1553 | return 0; | |
1554 | } | |
1555 | ||
300ad899 DK |
1556 | static int amd_execute_tuning(struct mmc_host *mmc, u32 opcode) |
1557 | { | |
1558 | struct sdhci_host *host = mmc_priv(mmc); | |
1559 | ||
1560 | /* AMD requires custom HS200 tuning */ | |
1561 | if (host->timing == MMC_TIMING_MMC_HS200) | |
1562 | return amd_execute_tuning_hs200(host, opcode); | |
1563 | ||
1564 | /* Otherwise perform standard SDHCI tuning */ | |
1565 | return sdhci_execute_tuning(mmc, opcode); | |
1566 | } | |
1567 | ||
1568 | static int amd_probe_slot(struct sdhci_pci_slot *slot) | |
1569 | { | |
1570 | struct mmc_host_ops *ops = &slot->host->mmc_host_ops; | |
1571 | ||
1572 | ops->execute_tuning = amd_execute_tuning; | |
1573 | ||
1574 | return 0; | |
1575 | } | |
1576 | ||
d44f88da VW |
1577 | static int amd_probe(struct sdhci_pci_chip *chip) |
1578 | { | |
1579 | struct pci_dev *smbus_dev; | |
b5e97d6e | 1580 | enum amd_chipset_gen gen; |
d44f88da VW |
1581 | |
1582 | smbus_dev = pci_get_device(PCI_VENDOR_ID_AMD, | |
1583 | PCI_DEVICE_ID_AMD_HUDSON2_SMBUS, NULL); | |
b5e97d6e VW |
1584 | if (smbus_dev) { |
1585 | gen = AMD_CHIPSET_BEFORE_ML; | |
1586 | } else { | |
1587 | smbus_dev = pci_get_device(PCI_VENDOR_ID_AMD, | |
1588 | PCI_DEVICE_ID_AMD_KERNCZ_SMBUS, NULL); | |
1589 | if (smbus_dev) { | |
1590 | if (smbus_dev->revision < 0x51) | |
1591 | gen = AMD_CHIPSET_CZ; | |
1592 | else | |
1593 | gen = AMD_CHIPSET_NL; | |
1594 | } else { | |
1595 | gen = AMD_CHIPSET_UNKNOWN; | |
1596 | } | |
1597 | } | |
d44f88da | 1598 | |
c31165d7 | 1599 | if (gen == AMD_CHIPSET_BEFORE_ML || gen == AMD_CHIPSET_CZ) |
d44f88da VW |
1600 | chip->quirks2 |= SDHCI_QUIRK2_CLEAR_TRANSFERMODE_REG_BEFORE_CMD; |
1601 | ||
1602 | return 0; | |
1603 | } | |
1604 | ||
7a869f00 RR |
1605 | static u32 sdhci_read_present_state(struct sdhci_host *host) |
1606 | { | |
1607 | return sdhci_readl(host, SDHCI_PRESENT_STATE); | |
1608 | } | |
1609 | ||
38413ce3 | 1610 | static void amd_sdhci_reset(struct sdhci_host *host, u8 mask) |
7a869f00 RR |
1611 | { |
1612 | struct sdhci_pci_slot *slot = sdhci_priv(host); | |
1613 | struct pci_dev *pdev = slot->chip->pdev; | |
1614 | u32 present_state; | |
1615 | ||
1616 | /* | |
1617 | * SDHC 0x7906 requires a hard reset to clear all internal state. | |
1618 | * Otherwise it can get into a bad state where the DATA lines are always | |
1619 | * read as zeros. | |
1620 | */ | |
1621 | if (pdev->device == 0x7906 && (mask & SDHCI_RESET_ALL)) { | |
1622 | pci_clear_master(pdev); | |
1623 | ||
1624 | pci_save_state(pdev); | |
1625 | ||
1626 | pci_set_power_state(pdev, PCI_D3cold); | |
1627 | pr_debug("%s: power_state=%u\n", mmc_hostname(host->mmc), | |
1628 | pdev->current_state); | |
1629 | pci_set_power_state(pdev, PCI_D0); | |
1630 | ||
1631 | pci_restore_state(pdev); | |
1632 | ||
1633 | /* | |
1634 | * SDHCI_RESET_ALL says the card detect logic should not be | |
1635 | * reset, but since we need to reset the entire controller | |
1636 | * we should wait until the card detect logic has stabilized. | |
1637 | * | |
1638 | * This normally takes about 40ms. | |
1639 | */ | |
1640 | readx_poll_timeout( | |
1641 | sdhci_read_present_state, | |
1642 | host, | |
1643 | present_state, | |
1644 | present_state & SDHCI_CD_STABLE, | |
1645 | 10000, | |
1646 | 100000 | |
1647 | ); | |
1648 | } | |
1649 | ||
1650 | return sdhci_reset(host, mask); | |
1651 | } | |
1652 | ||
c31165d7 SS |
1653 | static const struct sdhci_ops amd_sdhci_pci_ops = { |
1654 | .set_clock = sdhci_set_clock, | |
1655 | .enable_dma = sdhci_pci_enable_dma, | |
adc16398 | 1656 | .set_bus_width = sdhci_set_bus_width, |
7a869f00 | 1657 | .reset = amd_sdhci_reset, |
c31165d7 | 1658 | .set_uhs_signaling = sdhci_set_uhs_signaling, |
c31165d7 SS |
1659 | }; |
1660 | ||
d44f88da VW |
1661 | static const struct sdhci_pci_fixes sdhci_amd = { |
1662 | .probe = amd_probe, | |
c31165d7 | 1663 | .ops = &amd_sdhci_pci_ops, |
300ad899 | 1664 | .probe_slot = amd_probe_slot, |
d44f88da VW |
1665 | }; |
1666 | ||
9647f84d | 1667 | static const struct pci_device_id pci_ids[] = { |
c949c907 MK |
1668 | SDHCI_PCI_DEVICE(RICOH, R5C822, ricoh), |
1669 | SDHCI_PCI_DEVICE(RICOH, R5C843, ricoh_mmc), | |
1670 | SDHCI_PCI_DEVICE(RICOH, R5CE822, ricoh_mmc), | |
1671 | SDHCI_PCI_DEVICE(RICOH, R5CE823, ricoh_mmc), | |
1672 | SDHCI_PCI_DEVICE(ENE, CB712_SD, ene_712), | |
1673 | SDHCI_PCI_DEVICE(ENE, CB712_SD_2, ene_712), | |
1674 | SDHCI_PCI_DEVICE(ENE, CB714_SD, ene_714), | |
1675 | SDHCI_PCI_DEVICE(ENE, CB714_SD_2, ene_714), | |
1676 | SDHCI_PCI_DEVICE(MARVELL, 88ALP01_SD, cafe), | |
1677 | SDHCI_PCI_DEVICE(JMICRON, JMB38X_SD, jmicron), | |
1678 | SDHCI_PCI_DEVICE(JMICRON, JMB38X_MMC, jmicron), | |
1679 | SDHCI_PCI_DEVICE(JMICRON, JMB388_SD, jmicron), | |
1680 | SDHCI_PCI_DEVICE(JMICRON, JMB388_ESD, jmicron), | |
1681 | SDHCI_PCI_DEVICE(SYSKONNECT, 8000, syskt), | |
1682 | SDHCI_PCI_DEVICE(VIA, 95D0, via), | |
1683 | SDHCI_PCI_DEVICE(REALTEK, 5250, rtsx), | |
1684 | SDHCI_PCI_DEVICE(INTEL, QRK_SD, intel_qrk), | |
1685 | SDHCI_PCI_DEVICE(INTEL, MRST_SD0, intel_mrst_hc0), | |
1686 | SDHCI_PCI_DEVICE(INTEL, MRST_SD1, intel_mrst_hc1_hc2), | |
1687 | SDHCI_PCI_DEVICE(INTEL, MRST_SD2, intel_mrst_hc1_hc2), | |
1688 | SDHCI_PCI_DEVICE(INTEL, MFD_SD, intel_mfd_sd), | |
1689 | SDHCI_PCI_DEVICE(INTEL, MFD_SDIO1, intel_mfd_sdio), | |
1690 | SDHCI_PCI_DEVICE(INTEL, MFD_SDIO2, intel_mfd_sdio), | |
1691 | SDHCI_PCI_DEVICE(INTEL, MFD_EMMC0, intel_mfd_emmc), | |
1692 | SDHCI_PCI_DEVICE(INTEL, MFD_EMMC1, intel_mfd_emmc), | |
1693 | SDHCI_PCI_DEVICE(INTEL, PCH_SDIO0, intel_pch_sdio), | |
1694 | SDHCI_PCI_DEVICE(INTEL, PCH_SDIO1, intel_pch_sdio), | |
1695 | SDHCI_PCI_DEVICE(INTEL, BYT_EMMC, intel_byt_emmc), | |
1696 | SDHCI_PCI_SUBDEVICE(INTEL, BYT_SDIO, NI, 7884, ni_byt_sdio), | |
1697 | SDHCI_PCI_DEVICE(INTEL, BYT_SDIO, intel_byt_sdio), | |
1698 | SDHCI_PCI_DEVICE(INTEL, BYT_SD, intel_byt_sd), | |
1699 | SDHCI_PCI_DEVICE(INTEL, BYT_EMMC2, intel_byt_emmc), | |
1700 | SDHCI_PCI_DEVICE(INTEL, BSW_EMMC, intel_byt_emmc), | |
1701 | SDHCI_PCI_DEVICE(INTEL, BSW_SDIO, intel_byt_sdio), | |
1702 | SDHCI_PCI_DEVICE(INTEL, BSW_SD, intel_byt_sd), | |
1703 | SDHCI_PCI_DEVICE(INTEL, CLV_SDIO0, intel_mfd_sd), | |
1704 | SDHCI_PCI_DEVICE(INTEL, CLV_SDIO1, intel_mfd_sdio), | |
1705 | SDHCI_PCI_DEVICE(INTEL, CLV_SDIO2, intel_mfd_sdio), | |
1706 | SDHCI_PCI_DEVICE(INTEL, CLV_EMMC0, intel_mfd_emmc), | |
1707 | SDHCI_PCI_DEVICE(INTEL, CLV_EMMC1, intel_mfd_emmc), | |
1708 | SDHCI_PCI_DEVICE(INTEL, MRFLD_MMC, intel_mrfld_mmc), | |
1709 | SDHCI_PCI_DEVICE(INTEL, SPT_EMMC, intel_byt_emmc), | |
1710 | SDHCI_PCI_DEVICE(INTEL, SPT_SDIO, intel_byt_sdio), | |
1711 | SDHCI_PCI_DEVICE(INTEL, SPT_SD, intel_byt_sd), | |
1712 | SDHCI_PCI_DEVICE(INTEL, DNV_EMMC, intel_byt_emmc), | |
cdaba732 | 1713 | SDHCI_PCI_DEVICE(INTEL, CDF_EMMC, intel_glk_emmc), |
c949c907 MK |
1714 | SDHCI_PCI_DEVICE(INTEL, BXT_EMMC, intel_byt_emmc), |
1715 | SDHCI_PCI_DEVICE(INTEL, BXT_SDIO, intel_byt_sdio), | |
1716 | SDHCI_PCI_DEVICE(INTEL, BXT_SD, intel_byt_sd), | |
1717 | SDHCI_PCI_DEVICE(INTEL, BXTM_EMMC, intel_byt_emmc), | |
1718 | SDHCI_PCI_DEVICE(INTEL, BXTM_SDIO, intel_byt_sdio), | |
1719 | SDHCI_PCI_DEVICE(INTEL, BXTM_SD, intel_byt_sd), | |
1720 | SDHCI_PCI_DEVICE(INTEL, APL_EMMC, intel_byt_emmc), | |
1721 | SDHCI_PCI_DEVICE(INTEL, APL_SDIO, intel_byt_sdio), | |
1722 | SDHCI_PCI_DEVICE(INTEL, APL_SD, intel_byt_sd), | |
bc55dcd8 | 1723 | SDHCI_PCI_DEVICE(INTEL, GLK_EMMC, intel_glk_emmc), |
c949c907 MK |
1724 | SDHCI_PCI_DEVICE(INTEL, GLK_SDIO, intel_byt_sdio), |
1725 | SDHCI_PCI_DEVICE(INTEL, GLK_SD, intel_byt_sd), | |
bc55dcd8 AH |
1726 | SDHCI_PCI_DEVICE(INTEL, CNP_EMMC, intel_glk_emmc), |
1727 | SDHCI_PCI_DEVICE(INTEL, CNP_SD, intel_byt_sd), | |
1728 | SDHCI_PCI_DEVICE(INTEL, CNPH_SD, intel_byt_sd), | |
5637ffad AH |
1729 | SDHCI_PCI_DEVICE(INTEL, ICP_EMMC, intel_glk_emmc), |
1730 | SDHCI_PCI_DEVICE(INTEL, ICP_SD, intel_byt_sd), | |
cb3a7d4a AH |
1731 | SDHCI_PCI_DEVICE(INTEL, EHL_EMMC, intel_glk_emmc), |
1732 | SDHCI_PCI_DEVICE(INTEL, EHL_SD, intel_byt_sd), | |
765c5967 AH |
1733 | SDHCI_PCI_DEVICE(INTEL, CML_EMMC, intel_glk_emmc), |
1734 | SDHCI_PCI_DEVICE(INTEL, CML_SD, intel_byt_sd), | |
8f05eee6 | 1735 | SDHCI_PCI_DEVICE(INTEL, CMLH_SD, intel_byt_sd), |
315e3bd7 AH |
1736 | SDHCI_PCI_DEVICE(INTEL, JSL_EMMC, intel_glk_emmc), |
1737 | SDHCI_PCI_DEVICE(INTEL, JSL_SD, intel_byt_sd), | |
c949c907 MK |
1738 | SDHCI_PCI_DEVICE(O2, 8120, o2), |
1739 | SDHCI_PCI_DEVICE(O2, 8220, o2), | |
1740 | SDHCI_PCI_DEVICE(O2, 8221, o2), | |
1741 | SDHCI_PCI_DEVICE(O2, 8320, o2), | |
1742 | SDHCI_PCI_DEVICE(O2, 8321, o2), | |
1743 | SDHCI_PCI_DEVICE(O2, FUJIN2, o2), | |
1744 | SDHCI_PCI_DEVICE(O2, SDS0, o2), | |
1745 | SDHCI_PCI_DEVICE(O2, SDS1, o2), | |
1746 | SDHCI_PCI_DEVICE(O2, SEABIRD0, o2), | |
1747 | SDHCI_PCI_DEVICE(O2, SEABIRD1, o2), | |
d72d72cd | 1748 | SDHCI_PCI_DEVICE(ARASAN, PHY_EMMC, arasan), |
152f8204 | 1749 | SDHCI_PCI_DEVICE(SYNOPSYS, DWC_MSHC, snps), |
e51df6ce BC |
1750 | SDHCI_PCI_DEVICE(GLI, 9750, gl9750), |
1751 | SDHCI_PCI_DEVICE(GLI, 9755, gl9755), | |
c949c907 MK |
1752 | SDHCI_PCI_DEVICE_CLASS(AMD, SYSTEM_SDHCI, PCI_CLASS_MASK, amd), |
1753 | /* Generic SD host controller */ | |
1754 | {PCI_DEVICE_CLASS(SYSTEM_SDHCI, PCI_CLASS_MASK)}, | |
b8c86fc5 PO |
1755 | { /* end: all zeroes */ }, |
1756 | }; | |
1757 | ||
1758 | MODULE_DEVICE_TABLE(pci, pci_ids); | |
1759 | ||
b8c86fc5 PO |
1760 | /*****************************************************************************\ |
1761 | * * | |
1762 | * SDHCI core callbacks * | |
1763 | * * | |
1764 | \*****************************************************************************/ | |
1765 | ||
d72d72cd | 1766 | int sdhci_pci_enable_dma(struct sdhci_host *host) |
b8c86fc5 PO |
1767 | { |
1768 | struct sdhci_pci_slot *slot; | |
1769 | struct pci_dev *pdev; | |
b8c86fc5 PO |
1770 | |
1771 | slot = sdhci_priv(host); | |
1772 | pdev = slot->chip->pdev; | |
1773 | ||
1774 | if (((pdev->class & 0xFFFF00) == (PCI_CLASS_SYSTEM_SDHCI << 8)) && | |
1775 | ((pdev->class & 0x0000FF) != PCI_SDHCI_IFDMA) && | |
a13abc7b | 1776 | (host->flags & SDHCI_USE_SDMA)) { |
b8c86fc5 PO |
1777 | dev_warn(&pdev->dev, "Will use DMA mode even though HW " |
1778 | "doesn't fully claim to support it.\n"); | |
1779 | } | |
1780 | ||
b8c86fc5 PO |
1781 | pci_set_master(pdev); |
1782 | ||
1783 | return 0; | |
1784 | } | |
1785 | ||
c9faff6c | 1786 | static void sdhci_pci_gpio_hw_reset(struct sdhci_host *host) |
0f201655 AH |
1787 | { |
1788 | struct sdhci_pci_slot *slot = sdhci_priv(host); | |
1789 | int rst_n_gpio = slot->rst_n_gpio; | |
1790 | ||
1791 | if (!gpio_is_valid(rst_n_gpio)) | |
1792 | return; | |
1793 | gpio_set_value_cansleep(rst_n_gpio, 0); | |
1794 | /* For eMMC, minimum is 1us but give it 10us for good measure */ | |
1795 | udelay(10); | |
1796 | gpio_set_value_cansleep(rst_n_gpio, 1); | |
1797 | /* For eMMC, minimum is 200us but give it 300us for good measure */ | |
1798 | usleep_range(300, 1000); | |
1799 | } | |
1800 | ||
c9faff6c AH |
1801 | static void sdhci_pci_hw_reset(struct sdhci_host *host) |
1802 | { | |
1803 | struct sdhci_pci_slot *slot = sdhci_priv(host); | |
1804 | ||
1805 | if (slot->hw_reset) | |
1806 | slot->hw_reset(host); | |
1807 | } | |
1808 | ||
c915568d | 1809 | static const struct sdhci_ops sdhci_pci_ops = { |
1771059c | 1810 | .set_clock = sdhci_set_clock, |
b8c86fc5 | 1811 | .enable_dma = sdhci_pci_enable_dma, |
adc16398 | 1812 | .set_bus_width = sdhci_set_bus_width, |
03231f9b | 1813 | .reset = sdhci_reset, |
96d7b78c | 1814 | .set_uhs_signaling = sdhci_set_uhs_signaling, |
0f201655 | 1815 | .hw_reset = sdhci_pci_hw_reset, |
b8c86fc5 PO |
1816 | }; |
1817 | ||
1818 | /*****************************************************************************\ | |
1819 | * * | |
1820 | * Suspend/resume * | |
1821 | * * | |
1822 | \*****************************************************************************/ | |
1823 | ||
f9900f15 | 1824 | #ifdef CONFIG_PM_SLEEP |
29495aa0 | 1825 | static int sdhci_pci_suspend(struct device *dev) |
b8c86fc5 | 1826 | { |
90b51e3c | 1827 | struct sdhci_pci_chip *chip = dev_get_drvdata(dev); |
b8c86fc5 | 1828 | |
b8c86fc5 PO |
1829 | if (!chip) |
1830 | return 0; | |
1831 | ||
30cf2803 AH |
1832 | if (chip->fixes && chip->fixes->suspend) |
1833 | return chip->fixes->suspend(chip); | |
b8c86fc5 | 1834 | |
30cf2803 | 1835 | return sdhci_pci_suspend_host(chip); |
b8c86fc5 PO |
1836 | } |
1837 | ||
29495aa0 | 1838 | static int sdhci_pci_resume(struct device *dev) |
b8c86fc5 | 1839 | { |
90b51e3c | 1840 | struct sdhci_pci_chip *chip = dev_get_drvdata(dev); |
b8c86fc5 | 1841 | |
b8c86fc5 PO |
1842 | if (!chip) |
1843 | return 0; | |
1844 | ||
30cf2803 AH |
1845 | if (chip->fixes && chip->fixes->resume) |
1846 | return chip->fixes->resume(chip); | |
b8c86fc5 | 1847 | |
30cf2803 | 1848 | return sdhci_pci_resume_host(chip); |
b8c86fc5 | 1849 | } |
f9900f15 | 1850 | #endif |
b8c86fc5 | 1851 | |
f9900f15 | 1852 | #ifdef CONFIG_PM |
66fd8ad5 AH |
1853 | static int sdhci_pci_runtime_suspend(struct device *dev) |
1854 | { | |
90b51e3c | 1855 | struct sdhci_pci_chip *chip = dev_get_drvdata(dev); |
66fd8ad5 | 1856 | |
66fd8ad5 AH |
1857 | if (!chip) |
1858 | return 0; | |
1859 | ||
966d696a AH |
1860 | if (chip->fixes && chip->fixes->runtime_suspend) |
1861 | return chip->fixes->runtime_suspend(chip); | |
66fd8ad5 | 1862 | |
966d696a | 1863 | return sdhci_pci_runtime_suspend_host(chip); |
66fd8ad5 AH |
1864 | } |
1865 | ||
1866 | static int sdhci_pci_runtime_resume(struct device *dev) | |
1867 | { | |
90b51e3c | 1868 | struct sdhci_pci_chip *chip = dev_get_drvdata(dev); |
66fd8ad5 | 1869 | |
66fd8ad5 AH |
1870 | if (!chip) |
1871 | return 0; | |
1872 | ||
966d696a AH |
1873 | if (chip->fixes && chip->fixes->runtime_resume) |
1874 | return chip->fixes->runtime_resume(chip); | |
66fd8ad5 | 1875 | |
966d696a | 1876 | return sdhci_pci_runtime_resume_host(chip); |
66fd8ad5 | 1877 | } |
f9900f15 | 1878 | #endif |
66fd8ad5 AH |
1879 | |
1880 | static const struct dev_pm_ops sdhci_pci_pm_ops = { | |
f9900f15 | 1881 | SET_SYSTEM_SLEEP_PM_OPS(sdhci_pci_suspend, sdhci_pci_resume) |
f3a92b1a | 1882 | SET_RUNTIME_PM_OPS(sdhci_pci_runtime_suspend, |
106276bb | 1883 | sdhci_pci_runtime_resume, NULL) |
66fd8ad5 AH |
1884 | }; |
1885 | ||
b8c86fc5 PO |
1886 | /*****************************************************************************\ |
1887 | * * | |
1888 | * Device probing/removal * | |
1889 | * * | |
1890 | \*****************************************************************************/ | |
1891 | ||
c3be1efd | 1892 | static struct sdhci_pci_slot *sdhci_pci_probe_slot( |
52c506f0 AH |
1893 | struct pci_dev *pdev, struct sdhci_pci_chip *chip, int first_bar, |
1894 | int slotno) | |
b8c86fc5 PO |
1895 | { |
1896 | struct sdhci_pci_slot *slot; | |
1897 | struct sdhci_host *host; | |
52c506f0 | 1898 | int ret, bar = first_bar + slotno; |
ac9f67b5 | 1899 | size_t priv_size = chip->fixes ? chip->fixes->priv_size : 0; |
b8c86fc5 PO |
1900 | |
1901 | if (!(pci_resource_flags(pdev, bar) & IORESOURCE_MEM)) { | |
1902 | dev_err(&pdev->dev, "BAR %d is not iomem. Aborting.\n", bar); | |
1903 | return ERR_PTR(-ENODEV); | |
1904 | } | |
1905 | ||
90b3e6c5 | 1906 | if (pci_resource_len(pdev, bar) < 0x100) { |
b8c86fc5 PO |
1907 | dev_err(&pdev->dev, "Invalid iomem size. You may " |
1908 | "experience problems.\n"); | |
1909 | } | |
1910 | ||
1911 | if ((pdev->class & 0x0000FF) == PCI_SDHCI_IFVENDOR) { | |
1912 | dev_err(&pdev->dev, "Vendor specific interface. Aborting.\n"); | |
1913 | return ERR_PTR(-ENODEV); | |
1914 | } | |
1915 | ||
1916 | if ((pdev->class & 0x0000FF) > PCI_SDHCI_IFVENDOR) { | |
1917 | dev_err(&pdev->dev, "Unknown interface. Aborting.\n"); | |
1918 | return ERR_PTR(-ENODEV); | |
1919 | } | |
1920 | ||
ac9f67b5 | 1921 | host = sdhci_alloc_host(&pdev->dev, sizeof(*slot) + priv_size); |
b8c86fc5 | 1922 | if (IS_ERR(host)) { |
c60a32cd | 1923 | dev_err(&pdev->dev, "cannot allocate host\n"); |
dc0fd7b5 | 1924 | return ERR_CAST(host); |
b8c86fc5 PO |
1925 | } |
1926 | ||
1927 | slot = sdhci_priv(host); | |
1928 | ||
1929 | slot->chip = chip; | |
1930 | slot->host = host; | |
0f201655 | 1931 | slot->rst_n_gpio = -EINVAL; |
c5e027a4 | 1932 | slot->cd_gpio = -EINVAL; |
ff59c520 | 1933 | slot->cd_idx = -1; |
b8c86fc5 | 1934 | |
52c506f0 AH |
1935 | /* Retrieve platform data if there is any */ |
1936 | if (*sdhci_pci_get_data) | |
1937 | slot->data = sdhci_pci_get_data(pdev, slotno); | |
1938 | ||
1939 | if (slot->data) { | |
1940 | if (slot->data->setup) { | |
1941 | ret = slot->data->setup(slot->data); | |
1942 | if (ret) { | |
1943 | dev_err(&pdev->dev, "platform setup failed\n"); | |
1944 | goto free; | |
1945 | } | |
1946 | } | |
c5e027a4 AH |
1947 | slot->rst_n_gpio = slot->data->rst_n_gpio; |
1948 | slot->cd_gpio = slot->data->cd_gpio; | |
52c506f0 AH |
1949 | } |
1950 | ||
b8c86fc5 | 1951 | host->hw_name = "PCI"; |
6bc09063 AH |
1952 | host->ops = chip->fixes && chip->fixes->ops ? |
1953 | chip->fixes->ops : | |
1954 | &sdhci_pci_ops; | |
b8c86fc5 | 1955 | host->quirks = chip->quirks; |
f3c55a7b | 1956 | host->quirks2 = chip->quirks2; |
b8c86fc5 PO |
1957 | |
1958 | host->irq = pdev->irq; | |
1959 | ||
c10bc372 | 1960 | ret = pcim_iomap_regions(pdev, BIT(bar), mmc_hostname(host->mmc)); |
b8c86fc5 PO |
1961 | if (ret) { |
1962 | dev_err(&pdev->dev, "cannot request region\n"); | |
52c506f0 | 1963 | goto cleanup; |
b8c86fc5 PO |
1964 | } |
1965 | ||
c10bc372 | 1966 | host->ioaddr = pcim_iomap_table(pdev)[bar]; |
b8c86fc5 | 1967 | |
4489428a PO |
1968 | if (chip->fixes && chip->fixes->probe_slot) { |
1969 | ret = chip->fixes->probe_slot(slot); | |
1970 | if (ret) | |
c10bc372 | 1971 | goto cleanup; |
4489428a PO |
1972 | } |
1973 | ||
c5e027a4 | 1974 | if (gpio_is_valid(slot->rst_n_gpio)) { |
c10bc372 | 1975 | if (!devm_gpio_request(&pdev->dev, slot->rst_n_gpio, "eMMC_reset")) { |
c5e027a4 AH |
1976 | gpio_direction_output(slot->rst_n_gpio, 1); |
1977 | slot->host->mmc->caps |= MMC_CAP_HW_RESET; | |
c9faff6c | 1978 | slot->hw_reset = sdhci_pci_gpio_hw_reset; |
c5e027a4 AH |
1979 | } else { |
1980 | dev_warn(&pdev->dev, "failed to request rst_n_gpio\n"); | |
1981 | slot->rst_n_gpio = -EINVAL; | |
1982 | } | |
1983 | } | |
1984 | ||
e92cc35d | 1985 | host->mmc->pm_caps = MMC_PM_KEEP_POWER; |
eed222ac | 1986 | host->mmc->slotno = slotno; |
a08b17be | 1987 | host->mmc->caps2 |= MMC_CAP2_NO_PRESCAN_POWERUP; |
2f4cbb3d | 1988 | |
e92cc35d AH |
1989 | if (device_can_wakeup(&pdev->dev)) |
1990 | host->mmc->pm_caps |= MMC_PM_WAKE_SDIO_IRQ; | |
1991 | ||
d56ee1ff AH |
1992 | if (host->mmc->caps & MMC_CAP_CD_WAKE) |
1993 | device_init_wakeup(&pdev->dev, true); | |
1994 | ||
8f743d03 | 1995 | if (slot->cd_idx >= 0) { |
cdcefe6b | 1996 | ret = mmc_gpiod_request_cd(host->mmc, "cd", slot->cd_idx, |
d0052ad9 | 1997 | slot->cd_override_level, 0); |
cdcefe6b RJ |
1998 | if (ret && ret != -EPROBE_DEFER) |
1999 | ret = mmc_gpiod_request_cd(host->mmc, NULL, | |
2000 | slot->cd_idx, | |
2001 | slot->cd_override_level, | |
d0052ad9 | 2002 | 0); |
8f743d03 DB |
2003 | if (ret == -EPROBE_DEFER) |
2004 | goto remove; | |
2005 | ||
2006 | if (ret) { | |
2007 | dev_warn(&pdev->dev, "failed to setup card detect gpio\n"); | |
2008 | slot->cd_idx = -1; | |
2009 | } | |
ff59c520 AH |
2010 | } |
2011 | ||
61c951de AH |
2012 | if (chip->fixes && chip->fixes->add_host) |
2013 | ret = chip->fixes->add_host(slot); | |
2014 | else | |
2015 | ret = sdhci_add_host(host); | |
b8c86fc5 | 2016 | if (ret) |
4489428a | 2017 | goto remove; |
b8c86fc5 | 2018 | |
c5e027a4 AH |
2019 | sdhci_pci_add_own_cd(slot); |
2020 | ||
77a0122e AH |
2021 | /* |
2022 | * Check if the chip needs a separate GPIO for card detect to wake up | |
2023 | * from runtime suspend. If it is not there, don't allow runtime PM. | |
2024 | * Note sdhci_pci_add_own_cd() sets slot->cd_gpio to -EINVAL on failure. | |
2025 | */ | |
945be38c | 2026 | if (chip->fixes && chip->fixes->own_cd_for_runtime_pm && |
ff59c520 | 2027 | !gpio_is_valid(slot->cd_gpio) && slot->cd_idx < 0) |
77a0122e AH |
2028 | chip->allow_runtime_pm = false; |
2029 | ||
b8c86fc5 PO |
2030 | return slot; |
2031 | ||
4489428a PO |
2032 | remove: |
2033 | if (chip->fixes && chip->fixes->remove_slot) | |
1e72859e | 2034 | chip->fixes->remove_slot(slot, 0); |
4489428a | 2035 | |
52c506f0 AH |
2036 | cleanup: |
2037 | if (slot->data && slot->data->cleanup) | |
2038 | slot->data->cleanup(slot->data); | |
2039 | ||
c60a32cd | 2040 | free: |
b8c86fc5 PO |
2041 | sdhci_free_host(host); |
2042 | ||
2043 | return ERR_PTR(ret); | |
2044 | } | |
2045 | ||
2046 | static void sdhci_pci_remove_slot(struct sdhci_pci_slot *slot) | |
2047 | { | |
1e72859e PO |
2048 | int dead; |
2049 | u32 scratch; | |
2050 | ||
c5e027a4 AH |
2051 | sdhci_pci_remove_own_cd(slot); |
2052 | ||
1e72859e PO |
2053 | dead = 0; |
2054 | scratch = readl(slot->host->ioaddr + SDHCI_INT_STATUS); | |
2055 | if (scratch == (u32)-1) | |
2056 | dead = 1; | |
2057 | ||
2058 | sdhci_remove_host(slot->host, dead); | |
4489428a PO |
2059 | |
2060 | if (slot->chip->fixes && slot->chip->fixes->remove_slot) | |
1e72859e | 2061 | slot->chip->fixes->remove_slot(slot, dead); |
4489428a | 2062 | |
52c506f0 AH |
2063 | if (slot->data && slot->data->cleanup) |
2064 | slot->data->cleanup(slot->data); | |
2065 | ||
b8c86fc5 PO |
2066 | sdhci_free_host(slot->host); |
2067 | } | |
2068 | ||
c3be1efd | 2069 | static void sdhci_pci_runtime_pm_allow(struct device *dev) |
66fd8ad5 | 2070 | { |
00884b61 | 2071 | pm_suspend_ignore_children(dev, 1); |
66fd8ad5 AH |
2072 | pm_runtime_set_autosuspend_delay(dev, 50); |
2073 | pm_runtime_use_autosuspend(dev); | |
00884b61 AH |
2074 | pm_runtime_allow(dev); |
2075 | /* Stay active until mmc core scans for a card */ | |
2076 | pm_runtime_put_noidle(dev); | |
66fd8ad5 AH |
2077 | } |
2078 | ||
6e0ee714 | 2079 | static void sdhci_pci_runtime_pm_forbid(struct device *dev) |
66fd8ad5 AH |
2080 | { |
2081 | pm_runtime_forbid(dev); | |
2082 | pm_runtime_get_noresume(dev); | |
2083 | } | |
2084 | ||
c3be1efd | 2085 | static int sdhci_pci_probe(struct pci_dev *pdev, |
b8c86fc5 PO |
2086 | const struct pci_device_id *ent) |
2087 | { | |
2088 | struct sdhci_pci_chip *chip; | |
2089 | struct sdhci_pci_slot *slot; | |
2090 | ||
cf5e23e1 | 2091 | u8 slots, first_bar; |
b8c86fc5 PO |
2092 | int ret, i; |
2093 | ||
2094 | BUG_ON(pdev == NULL); | |
2095 | BUG_ON(ent == NULL); | |
2096 | ||
b8c86fc5 | 2097 | dev_info(&pdev->dev, "SDHCI controller found [%04x:%04x] (rev %x)\n", |
cf5e23e1 | 2098 | (int)pdev->vendor, (int)pdev->device, (int)pdev->revision); |
b8c86fc5 PO |
2099 | |
2100 | ret = pci_read_config_byte(pdev, PCI_SLOT_INFO, &slots); | |
2101 | if (ret) | |
2102 | return ret; | |
2103 | ||
2104 | slots = PCI_SLOT_INFO_SLOTS(slots) + 1; | |
2105 | dev_dbg(&pdev->dev, "found %d slot(s)\n", slots); | |
b8c86fc5 PO |
2106 | |
2107 | BUG_ON(slots > MAX_SLOTS); | |
2108 | ||
2109 | ret = pci_read_config_byte(pdev, PCI_SLOT_INFO, &first_bar); | |
2110 | if (ret) | |
2111 | return ret; | |
2112 | ||
2113 | first_bar &= PCI_SLOT_INFO_FIRST_BAR_MASK; | |
2114 | ||
2115 | if (first_bar > 5) { | |
2116 | dev_err(&pdev->dev, "Invalid first BAR. Aborting.\n"); | |
2117 | return -ENODEV; | |
2118 | } | |
2119 | ||
52ac7acf | 2120 | ret = pcim_enable_device(pdev); |
b8c86fc5 PO |
2121 | if (ret) |
2122 | return ret; | |
2123 | ||
52ac7acf AS |
2124 | chip = devm_kzalloc(&pdev->dev, sizeof(*chip), GFP_KERNEL); |
2125 | if (!chip) | |
2126 | return -ENOMEM; | |
b8c86fc5 PO |
2127 | |
2128 | chip->pdev = pdev; | |
b177bc91 | 2129 | chip->fixes = (const struct sdhci_pci_fixes *)ent->driver_data; |
c43fd774 | 2130 | if (chip->fixes) { |
22606405 | 2131 | chip->quirks = chip->fixes->quirks; |
f3c55a7b | 2132 | chip->quirks2 = chip->fixes->quirks2; |
c43fd774 AH |
2133 | chip->allow_runtime_pm = chip->fixes->allow_runtime_pm; |
2134 | } | |
b8c86fc5 | 2135 | chip->num_slots = slots; |
d38dcad4 AH |
2136 | chip->pm_retune = true; |
2137 | chip->rpm_retune = true; | |
b8c86fc5 PO |
2138 | |
2139 | pci_set_drvdata(pdev, chip); | |
2140 | ||
22606405 PO |
2141 | if (chip->fixes && chip->fixes->probe) { |
2142 | ret = chip->fixes->probe(chip); | |
2143 | if (ret) | |
52ac7acf | 2144 | return ret; |
22606405 PO |
2145 | } |
2146 | ||
225d85fe AC |
2147 | slots = chip->num_slots; /* Quirk may have changed this */ |
2148 | ||
b177bc91 | 2149 | for (i = 0; i < slots; i++) { |
52c506f0 | 2150 | slot = sdhci_pci_probe_slot(pdev, chip, first_bar, i); |
b8c86fc5 | 2151 | if (IS_ERR(slot)) { |
b177bc91 | 2152 | for (i--; i >= 0; i--) |
b8c86fc5 | 2153 | sdhci_pci_remove_slot(chip->slots[i]); |
52ac7acf | 2154 | return PTR_ERR(slot); |
b8c86fc5 PO |
2155 | } |
2156 | ||
2157 | chip->slots[i] = slot; | |
2158 | } | |
2159 | ||
c43fd774 AH |
2160 | if (chip->allow_runtime_pm) |
2161 | sdhci_pci_runtime_pm_allow(&pdev->dev); | |
66fd8ad5 | 2162 | |
b8c86fc5 | 2163 | return 0; |
b8c86fc5 PO |
2164 | } |
2165 | ||
6e0ee714 | 2166 | static void sdhci_pci_remove(struct pci_dev *pdev) |
b8c86fc5 PO |
2167 | { |
2168 | int i; | |
52ac7acf | 2169 | struct sdhci_pci_chip *chip = pci_get_drvdata(pdev); |
c43fd774 | 2170 | |
52ac7acf AS |
2171 | if (chip->allow_runtime_pm) |
2172 | sdhci_pci_runtime_pm_forbid(&pdev->dev); | |
b8c86fc5 | 2173 | |
52ac7acf AS |
2174 | for (i = 0; i < chip->num_slots; i++) |
2175 | sdhci_pci_remove_slot(chip->slots[i]); | |
b8c86fc5 PO |
2176 | } |
2177 | ||
2178 | static struct pci_driver sdhci_driver = { | |
b177bc91 | 2179 | .name = "sdhci-pci", |
b8c86fc5 | 2180 | .id_table = pci_ids, |
b177bc91 | 2181 | .probe = sdhci_pci_probe, |
0433c143 | 2182 | .remove = sdhci_pci_remove, |
66fd8ad5 AH |
2183 | .driver = { |
2184 | .pm = &sdhci_pci_pm_ops | |
2185 | }, | |
b8c86fc5 PO |
2186 | }; |
2187 | ||
acc69646 | 2188 | module_pci_driver(sdhci_driver); |
b8c86fc5 | 2189 | |
32710e8f | 2190 | MODULE_AUTHOR("Pierre Ossman <pierre@ossman.eu>"); |
b8c86fc5 PO |
2191 | MODULE_DESCRIPTION("Secure Digital Host Controller Interface PCI driver"); |
2192 | MODULE_LICENSE("GPL"); |