.. | .. |
---|
| 1 | +// SPDX-License-Identifier: GPL-2.0-only |
---|
1 | 2 | /* |
---|
2 | 3 | * sdhci-brcmstb.c Support for SDHCI on Broadcom BRCMSTB SoC's |
---|
3 | 4 | * |
---|
4 | 5 | * Copyright (C) 2015 Broadcom Corporation |
---|
5 | | - * |
---|
6 | | - * This program is free software; you can redistribute it and/or modify |
---|
7 | | - * it under the terms of the GNU General Public License version 2 as |
---|
8 | | - * published by the Free Software Foundation. |
---|
9 | | - * |
---|
10 | | - * This program is distributed in the hope that it will be useful, |
---|
11 | | - * but WITHOUT ANY WARRANTY; without even the implied warranty of |
---|
12 | | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
---|
13 | | - * GNU General Public License for more details. |
---|
14 | | - * |
---|
15 | 6 | */ |
---|
16 | 7 | |
---|
17 | 8 | #include <linux/io.h> |
---|
18 | 9 | #include <linux/mmc/host.h> |
---|
19 | 10 | #include <linux/module.h> |
---|
20 | 11 | #include <linux/of.h> |
---|
| 12 | +#include <linux/bitops.h> |
---|
| 13 | +#include <linux/delay.h> |
---|
21 | 14 | |
---|
| 15 | +#include "sdhci-cqhci.h" |
---|
22 | 16 | #include "sdhci-pltfm.h" |
---|
| 17 | +#include "cqhci.h" |
---|
23 | 18 | |
---|
24 | | -static const struct sdhci_ops sdhci_brcmstb_ops = { |
---|
| 19 | +#define SDHCI_VENDOR 0x78 |
---|
| 20 | +#define SDHCI_VENDOR_ENHANCED_STRB 0x1 |
---|
| 21 | +#define SDHCI_VENDOR_GATE_SDCLK_EN 0x2 |
---|
| 22 | + |
---|
| 23 | +#define BRCMSTB_MATCH_FLAGS_NO_64BIT BIT(0) |
---|
| 24 | +#define BRCMSTB_MATCH_FLAGS_BROKEN_TIMEOUT BIT(1) |
---|
| 25 | +#define BRCMSTB_MATCH_FLAGS_HAS_CLOCK_GATE BIT(2) |
---|
| 26 | + |
---|
| 27 | +#define BRCMSTB_PRIV_FLAGS_HAS_CQE BIT(0) |
---|
| 28 | +#define BRCMSTB_PRIV_FLAGS_GATE_CLOCK BIT(1) |
---|
| 29 | + |
---|
| 30 | +#define SDHCI_ARASAN_CQE_BASE_ADDR 0x200 |
---|
| 31 | + |
---|
| 32 | +struct sdhci_brcmstb_priv { |
---|
| 33 | + void __iomem *cfg_regs; |
---|
| 34 | + unsigned int flags; |
---|
| 35 | +}; |
---|
| 36 | + |
---|
| 37 | +struct brcmstb_match_priv { |
---|
| 38 | + void (*hs400es)(struct mmc_host *mmc, struct mmc_ios *ios); |
---|
| 39 | + struct sdhci_ops *ops; |
---|
| 40 | + const unsigned int flags; |
---|
| 41 | +}; |
---|
| 42 | + |
---|
| 43 | +static inline void enable_clock_gating(struct sdhci_host *host) |
---|
| 44 | +{ |
---|
| 45 | + u32 reg; |
---|
| 46 | + |
---|
| 47 | + reg = sdhci_readl(host, SDHCI_VENDOR); |
---|
| 48 | + reg |= SDHCI_VENDOR_GATE_SDCLK_EN; |
---|
| 49 | + sdhci_writel(host, reg, SDHCI_VENDOR); |
---|
| 50 | +} |
---|
| 51 | + |
---|
| 52 | +void brcmstb_reset(struct sdhci_host *host, u8 mask) |
---|
| 53 | +{ |
---|
| 54 | + struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host); |
---|
| 55 | + struct sdhci_brcmstb_priv *priv = sdhci_pltfm_priv(pltfm_host); |
---|
| 56 | + |
---|
| 57 | + sdhci_and_cqhci_reset(host, mask); |
---|
| 58 | + |
---|
| 59 | + /* Reset will clear this, so re-enable it */ |
---|
| 60 | + if (priv->flags & BRCMSTB_PRIV_FLAGS_GATE_CLOCK) |
---|
| 61 | + enable_clock_gating(host); |
---|
| 62 | +} |
---|
| 63 | + |
---|
| 64 | +static void sdhci_brcmstb_hs400es(struct mmc_host *mmc, struct mmc_ios *ios) |
---|
| 65 | +{ |
---|
| 66 | + struct sdhci_host *host = mmc_priv(mmc); |
---|
| 67 | + |
---|
| 68 | + u32 reg; |
---|
| 69 | + |
---|
| 70 | + dev_dbg(mmc_dev(mmc), "%s(): Setting HS400-Enhanced-Strobe mode\n", |
---|
| 71 | + __func__); |
---|
| 72 | + reg = readl(host->ioaddr + SDHCI_VENDOR); |
---|
| 73 | + if (ios->enhanced_strobe) |
---|
| 74 | + reg |= SDHCI_VENDOR_ENHANCED_STRB; |
---|
| 75 | + else |
---|
| 76 | + reg &= ~SDHCI_VENDOR_ENHANCED_STRB; |
---|
| 77 | + writel(reg, host->ioaddr + SDHCI_VENDOR); |
---|
| 78 | +} |
---|
| 79 | + |
---|
| 80 | +static void sdhci_brcmstb_set_clock(struct sdhci_host *host, unsigned int clock) |
---|
| 81 | +{ |
---|
| 82 | + u16 clk; |
---|
| 83 | + |
---|
| 84 | + host->mmc->actual_clock = 0; |
---|
| 85 | + |
---|
| 86 | + clk = sdhci_calc_clk(host, clock, &host->mmc->actual_clock); |
---|
| 87 | + sdhci_writew(host, clk, SDHCI_CLOCK_CONTROL); |
---|
| 88 | + |
---|
| 89 | + if (clock == 0) |
---|
| 90 | + return; |
---|
| 91 | + |
---|
| 92 | + sdhci_enable_clk(host, clk); |
---|
| 93 | +} |
---|
| 94 | + |
---|
| 95 | +static void sdhci_brcmstb_set_uhs_signaling(struct sdhci_host *host, |
---|
| 96 | + unsigned int timing) |
---|
| 97 | +{ |
---|
| 98 | + u16 ctrl_2; |
---|
| 99 | + |
---|
| 100 | + dev_dbg(mmc_dev(host->mmc), "%s: Setting UHS signaling for %d timing\n", |
---|
| 101 | + __func__, timing); |
---|
| 102 | + ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); |
---|
| 103 | + /* Select Bus Speed Mode for host */ |
---|
| 104 | + ctrl_2 &= ~SDHCI_CTRL_UHS_MASK; |
---|
| 105 | + if ((timing == MMC_TIMING_MMC_HS200) || |
---|
| 106 | + (timing == MMC_TIMING_UHS_SDR104)) |
---|
| 107 | + ctrl_2 |= SDHCI_CTRL_UHS_SDR104; |
---|
| 108 | + else if (timing == MMC_TIMING_UHS_SDR12) |
---|
| 109 | + ctrl_2 |= SDHCI_CTRL_UHS_SDR12; |
---|
| 110 | + else if (timing == MMC_TIMING_SD_HS || |
---|
| 111 | + timing == MMC_TIMING_MMC_HS || |
---|
| 112 | + timing == MMC_TIMING_UHS_SDR25) |
---|
| 113 | + ctrl_2 |= SDHCI_CTRL_UHS_SDR25; |
---|
| 114 | + else if (timing == MMC_TIMING_UHS_SDR50) |
---|
| 115 | + ctrl_2 |= SDHCI_CTRL_UHS_SDR50; |
---|
| 116 | + else if ((timing == MMC_TIMING_UHS_DDR50) || |
---|
| 117 | + (timing == MMC_TIMING_MMC_DDR52)) |
---|
| 118 | + ctrl_2 |= SDHCI_CTRL_UHS_DDR50; |
---|
| 119 | + else if (timing == MMC_TIMING_MMC_HS400) |
---|
| 120 | + ctrl_2 |= SDHCI_CTRL_HS400; /* Non-standard */ |
---|
| 121 | + sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); |
---|
| 122 | +} |
---|
| 123 | + |
---|
| 124 | +static void sdhci_brcmstb_dumpregs(struct mmc_host *mmc) |
---|
| 125 | +{ |
---|
| 126 | + sdhci_dumpregs(mmc_priv(mmc)); |
---|
| 127 | +} |
---|
| 128 | + |
---|
| 129 | +static void sdhci_brcmstb_cqe_enable(struct mmc_host *mmc) |
---|
| 130 | +{ |
---|
| 131 | + struct sdhci_host *host = mmc_priv(mmc); |
---|
| 132 | + u32 reg; |
---|
| 133 | + |
---|
| 134 | + reg = sdhci_readl(host, SDHCI_PRESENT_STATE); |
---|
| 135 | + while (reg & SDHCI_DATA_AVAILABLE) { |
---|
| 136 | + sdhci_readl(host, SDHCI_BUFFER); |
---|
| 137 | + reg = sdhci_readl(host, SDHCI_PRESENT_STATE); |
---|
| 138 | + } |
---|
| 139 | + |
---|
| 140 | + sdhci_cqe_enable(mmc); |
---|
| 141 | +} |
---|
| 142 | + |
---|
| 143 | +static const struct cqhci_host_ops sdhci_brcmstb_cqhci_ops = { |
---|
| 144 | + .enable = sdhci_brcmstb_cqe_enable, |
---|
| 145 | + .disable = sdhci_cqe_disable, |
---|
| 146 | + .dumpregs = sdhci_brcmstb_dumpregs, |
---|
| 147 | +}; |
---|
| 148 | + |
---|
| 149 | +static struct sdhci_ops sdhci_brcmstb_ops = { |
---|
25 | 150 | .set_clock = sdhci_set_clock, |
---|
26 | 151 | .set_bus_width = sdhci_set_bus_width, |
---|
27 | 152 | .reset = sdhci_reset, |
---|
28 | 153 | .set_uhs_signaling = sdhci_set_uhs_signaling, |
---|
29 | 154 | }; |
---|
30 | 155 | |
---|
31 | | -static const struct sdhci_pltfm_data sdhci_brcmstb_pdata = { |
---|
| 156 | +static struct sdhci_ops sdhci_brcmstb_ops_7216 = { |
---|
| 157 | + .set_clock = sdhci_brcmstb_set_clock, |
---|
| 158 | + .set_bus_width = sdhci_set_bus_width, |
---|
| 159 | + .reset = brcmstb_reset, |
---|
| 160 | + .set_uhs_signaling = sdhci_brcmstb_set_uhs_signaling, |
---|
| 161 | +}; |
---|
| 162 | + |
---|
| 163 | +static struct brcmstb_match_priv match_priv_7425 = { |
---|
| 164 | + .flags = BRCMSTB_MATCH_FLAGS_NO_64BIT | |
---|
| 165 | + BRCMSTB_MATCH_FLAGS_BROKEN_TIMEOUT, |
---|
32 | 166 | .ops = &sdhci_brcmstb_ops, |
---|
33 | 167 | }; |
---|
34 | 168 | |
---|
| 169 | +static struct brcmstb_match_priv match_priv_7445 = { |
---|
| 170 | + .flags = BRCMSTB_MATCH_FLAGS_BROKEN_TIMEOUT, |
---|
| 171 | + .ops = &sdhci_brcmstb_ops, |
---|
| 172 | +}; |
---|
| 173 | + |
---|
| 174 | +static const struct brcmstb_match_priv match_priv_7216 = { |
---|
| 175 | + .flags = BRCMSTB_MATCH_FLAGS_HAS_CLOCK_GATE, |
---|
| 176 | + .hs400es = sdhci_brcmstb_hs400es, |
---|
| 177 | + .ops = &sdhci_brcmstb_ops_7216, |
---|
| 178 | +}; |
---|
| 179 | + |
---|
| 180 | +static const struct of_device_id sdhci_brcm_of_match[] = { |
---|
| 181 | + { .compatible = "brcm,bcm7425-sdhci", .data = &match_priv_7425 }, |
---|
| 182 | + { .compatible = "brcm,bcm7445-sdhci", .data = &match_priv_7445 }, |
---|
| 183 | + { .compatible = "brcm,bcm7216-sdhci", .data = &match_priv_7216 }, |
---|
| 184 | + {}, |
---|
| 185 | +}; |
---|
| 186 | + |
---|
| 187 | +static u32 sdhci_brcmstb_cqhci_irq(struct sdhci_host *host, u32 intmask) |
---|
| 188 | +{ |
---|
| 189 | + int cmd_error = 0; |
---|
| 190 | + int data_error = 0; |
---|
| 191 | + |
---|
| 192 | + if (!sdhci_cqe_irq(host, intmask, &cmd_error, &data_error)) |
---|
| 193 | + return intmask; |
---|
| 194 | + |
---|
| 195 | + cqhci_irq(host->mmc, intmask, cmd_error, data_error); |
---|
| 196 | + |
---|
| 197 | + return 0; |
---|
| 198 | +} |
---|
| 199 | + |
---|
| 200 | +static int sdhci_brcmstb_add_host(struct sdhci_host *host, |
---|
| 201 | + struct sdhci_brcmstb_priv *priv) |
---|
| 202 | +{ |
---|
| 203 | + struct cqhci_host *cq_host; |
---|
| 204 | + bool dma64; |
---|
| 205 | + int ret; |
---|
| 206 | + |
---|
| 207 | + if ((priv->flags & BRCMSTB_PRIV_FLAGS_HAS_CQE) == 0) |
---|
| 208 | + return sdhci_add_host(host); |
---|
| 209 | + |
---|
| 210 | + dev_dbg(mmc_dev(host->mmc), "CQE is enabled\n"); |
---|
| 211 | + host->mmc->caps2 |= MMC_CAP2_CQE | MMC_CAP2_CQE_DCMD; |
---|
| 212 | + ret = sdhci_setup_host(host); |
---|
| 213 | + if (ret) |
---|
| 214 | + return ret; |
---|
| 215 | + |
---|
| 216 | + cq_host = devm_kzalloc(mmc_dev(host->mmc), |
---|
| 217 | + sizeof(*cq_host), GFP_KERNEL); |
---|
| 218 | + if (!cq_host) { |
---|
| 219 | + ret = -ENOMEM; |
---|
| 220 | + goto cleanup; |
---|
| 221 | + } |
---|
| 222 | + |
---|
| 223 | + cq_host->mmio = host->ioaddr + SDHCI_ARASAN_CQE_BASE_ADDR; |
---|
| 224 | + cq_host->ops = &sdhci_brcmstb_cqhci_ops; |
---|
| 225 | + |
---|
| 226 | + dma64 = host->flags & SDHCI_USE_64_BIT_DMA; |
---|
| 227 | + if (dma64) { |
---|
| 228 | + dev_dbg(mmc_dev(host->mmc), "Using 64 bit DMA\n"); |
---|
| 229 | + cq_host->caps |= CQHCI_TASK_DESC_SZ_128; |
---|
| 230 | + } |
---|
| 231 | + |
---|
| 232 | + ret = cqhci_init(cq_host, host->mmc, dma64); |
---|
| 233 | + if (ret) |
---|
| 234 | + goto cleanup; |
---|
| 235 | + |
---|
| 236 | + ret = __sdhci_add_host(host); |
---|
| 237 | + if (ret) |
---|
| 238 | + goto cleanup; |
---|
| 239 | + |
---|
| 240 | + return 0; |
---|
| 241 | + |
---|
| 242 | +cleanup: |
---|
| 243 | + sdhci_cleanup_host(host); |
---|
| 244 | + return ret; |
---|
| 245 | +} |
---|
| 246 | + |
---|
35 | 247 | static int sdhci_brcmstb_probe(struct platform_device *pdev) |
---|
36 | 248 | { |
---|
37 | | - struct sdhci_host *host; |
---|
| 249 | + const struct brcmstb_match_priv *match_priv; |
---|
| 250 | + struct sdhci_pltfm_data brcmstb_pdata; |
---|
38 | 251 | struct sdhci_pltfm_host *pltfm_host; |
---|
| 252 | + const struct of_device_id *match; |
---|
| 253 | + struct sdhci_brcmstb_priv *priv; |
---|
| 254 | + struct sdhci_host *host; |
---|
| 255 | + struct resource *iomem; |
---|
39 | 256 | struct clk *clk; |
---|
40 | 257 | int res; |
---|
41 | 258 | |
---|
42 | | - clk = devm_clk_get(&pdev->dev, NULL); |
---|
43 | | - if (IS_ERR(clk)) { |
---|
44 | | - dev_err(&pdev->dev, "Clock not found in Device Tree\n"); |
---|
45 | | - clk = NULL; |
---|
46 | | - } |
---|
| 259 | + match = of_match_node(sdhci_brcm_of_match, pdev->dev.of_node); |
---|
| 260 | + match_priv = match->data; |
---|
| 261 | + |
---|
| 262 | + dev_dbg(&pdev->dev, "Probe found match for %s\n", match->compatible); |
---|
| 263 | + |
---|
| 264 | + clk = devm_clk_get_optional(&pdev->dev, NULL); |
---|
| 265 | + if (IS_ERR(clk)) |
---|
| 266 | + return dev_err_probe(&pdev->dev, PTR_ERR(clk), |
---|
| 267 | + "Failed to get clock from Device Tree\n"); |
---|
| 268 | + |
---|
47 | 269 | res = clk_prepare_enable(clk); |
---|
48 | 270 | if (res) |
---|
49 | 271 | return res; |
---|
50 | 272 | |
---|
51 | | - host = sdhci_pltfm_init(pdev, &sdhci_brcmstb_pdata, 0); |
---|
| 273 | + memset(&brcmstb_pdata, 0, sizeof(brcmstb_pdata)); |
---|
| 274 | + brcmstb_pdata.ops = match_priv->ops; |
---|
| 275 | + host = sdhci_pltfm_init(pdev, &brcmstb_pdata, |
---|
| 276 | + sizeof(struct sdhci_brcmstb_priv)); |
---|
52 | 277 | if (IS_ERR(host)) { |
---|
53 | 278 | res = PTR_ERR(host); |
---|
54 | 279 | goto err_clk; |
---|
| 280 | + } |
---|
| 281 | + |
---|
| 282 | + pltfm_host = sdhci_priv(host); |
---|
| 283 | + priv = sdhci_pltfm_priv(pltfm_host); |
---|
| 284 | + if (device_property_read_bool(&pdev->dev, "supports-cqe")) { |
---|
| 285 | + priv->flags |= BRCMSTB_PRIV_FLAGS_HAS_CQE; |
---|
| 286 | + match_priv->ops->irq = sdhci_brcmstb_cqhci_irq; |
---|
| 287 | + } |
---|
| 288 | + |
---|
| 289 | + /* Map in the non-standard CFG registers */ |
---|
| 290 | + iomem = platform_get_resource(pdev, IORESOURCE_MEM, 1); |
---|
| 291 | + priv->cfg_regs = devm_ioremap_resource(&pdev->dev, iomem); |
---|
| 292 | + if (IS_ERR(priv->cfg_regs)) { |
---|
| 293 | + res = PTR_ERR(priv->cfg_regs); |
---|
| 294 | + goto err; |
---|
55 | 295 | } |
---|
56 | 296 | |
---|
57 | 297 | sdhci_get_of_property(pdev); |
---|
.. | .. |
---|
60 | 300 | goto err; |
---|
61 | 301 | |
---|
62 | 302 | /* |
---|
| 303 | + * Automatic clock gating does not work for SD cards that may |
---|
| 304 | + * voltage switch so only enable it for non-removable devices. |
---|
| 305 | + */ |
---|
| 306 | + if ((match_priv->flags & BRCMSTB_MATCH_FLAGS_HAS_CLOCK_GATE) && |
---|
| 307 | + (host->mmc->caps & MMC_CAP_NONREMOVABLE)) |
---|
| 308 | + priv->flags |= BRCMSTB_PRIV_FLAGS_GATE_CLOCK; |
---|
| 309 | + |
---|
| 310 | + /* |
---|
| 311 | + * If the chip has enhanced strobe and it's enabled, add |
---|
| 312 | + * callback |
---|
| 313 | + */ |
---|
| 314 | + if (match_priv->hs400es && |
---|
| 315 | + (host->mmc->caps2 & MMC_CAP2_HS400_ES)) |
---|
| 316 | + host->mmc_host_ops.hs400_enhanced_strobe = match_priv->hs400es; |
---|
| 317 | + |
---|
| 318 | + /* |
---|
63 | 319 | * Supply the existing CAPS, but clear the UHS modes. This |
---|
64 | 320 | * will allow these modes to be specified by device tree |
---|
65 | 321 | * properties through mmc_of_parse(). |
---|
66 | 322 | */ |
---|
67 | 323 | host->caps = sdhci_readl(host, SDHCI_CAPABILITIES); |
---|
68 | | - if (of_device_is_compatible(pdev->dev.of_node, "brcm,bcm7425-sdhci")) |
---|
| 324 | + if (match_priv->flags & BRCMSTB_MATCH_FLAGS_NO_64BIT) |
---|
69 | 325 | host->caps &= ~SDHCI_CAN_64BIT; |
---|
70 | 326 | host->caps1 = sdhci_readl(host, SDHCI_CAPABILITIES_1); |
---|
71 | 327 | host->caps1 &= ~(SDHCI_SUPPORT_SDR50 | SDHCI_SUPPORT_SDR104 | |
---|
72 | | - SDHCI_SUPPORT_DDR50); |
---|
73 | | - host->quirks |= SDHCI_QUIRK_MISSING_CAPS | |
---|
74 | | - SDHCI_QUIRK_BROKEN_TIMEOUT_VAL; |
---|
| 328 | + SDHCI_SUPPORT_DDR50); |
---|
| 329 | + host->quirks |= SDHCI_QUIRK_MISSING_CAPS; |
---|
75 | 330 | |
---|
76 | | - res = sdhci_add_host(host); |
---|
| 331 | + if (match_priv->flags & BRCMSTB_MATCH_FLAGS_BROKEN_TIMEOUT) |
---|
| 332 | + host->quirks |= SDHCI_QUIRK_BROKEN_TIMEOUT_VAL; |
---|
| 333 | + |
---|
| 334 | + res = sdhci_brcmstb_add_host(host, priv); |
---|
77 | 335 | if (res) |
---|
78 | 336 | goto err; |
---|
79 | 337 | |
---|
80 | | - pltfm_host = sdhci_priv(host); |
---|
81 | 338 | pltfm_host->clk = clk; |
---|
82 | 339 | return res; |
---|
83 | 340 | |
---|
.. | .. |
---|
88 | 345 | return res; |
---|
89 | 346 | } |
---|
90 | 347 | |
---|
91 | | -static const struct of_device_id sdhci_brcm_of_match[] = { |
---|
92 | | - { .compatible = "brcm,bcm7425-sdhci" }, |
---|
93 | | - { .compatible = "brcm,bcm7445-sdhci" }, |
---|
94 | | - {}, |
---|
95 | | -}; |
---|
| 348 | +static void sdhci_brcmstb_shutdown(struct platform_device *pdev) |
---|
| 349 | +{ |
---|
| 350 | + sdhci_pltfm_suspend(&pdev->dev); |
---|
| 351 | +} |
---|
| 352 | + |
---|
96 | 353 | MODULE_DEVICE_TABLE(of, sdhci_brcm_of_match); |
---|
97 | 354 | |
---|
98 | 355 | static struct platform_driver sdhci_brcmstb_driver = { |
---|
99 | 356 | .driver = { |
---|
100 | 357 | .name = "sdhci-brcmstb", |
---|
| 358 | + .probe_type = PROBE_PREFER_ASYNCHRONOUS, |
---|
101 | 359 | .pm = &sdhci_pltfm_pmops, |
---|
102 | 360 | .of_match_table = of_match_ptr(sdhci_brcm_of_match), |
---|
103 | 361 | }, |
---|
104 | 362 | .probe = sdhci_brcmstb_probe, |
---|
105 | 363 | .remove = sdhci_pltfm_unregister, |
---|
| 364 | + .shutdown = sdhci_brcmstb_shutdown, |
---|
106 | 365 | }; |
---|
107 | 366 | |
---|
108 | 367 | module_platform_driver(sdhci_brcmstb_driver); |
---|