blob: 0b56d1405bee54044f671e10bd3e313584cbff9a [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Ian Campbelle24ea552014-05-05 14:42:31 +01002/*
3 * (C) Copyright 2007-2011
4 * Allwinner Technology Co., Ltd. <www.allwinnertech.com>
5 * Aaron <leafy.myeh@allwinnertech.com>
6 *
7 * MMC driver for allwinner sunxi platform.
Andre Przywaraba16b532022-07-13 17:21:44 +01008 *
9 * This driver is used by the (ARM) SPL with the legacy MMC interface, and
10 * by U-Boot proper using the full DM interface. The actual hardware access
11 * code is common, and comes first in this file.
12 * The legacy MMC interface implementation comes next, followed by the
13 * proper DM_MMC implementation at the end.
Ian Campbelle24ea552014-05-05 14:42:31 +010014 */
15
Simon Glassdd279182017-07-04 13:31:27 -060016#include <dm.h>
Hans de Goede90641f82015-04-22 17:03:17 +020017#include <errno.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -060018#include <log.h>
Ian Campbelle24ea552014-05-05 14:42:31 +010019#include <malloc.h>
20#include <mmc.h>
Andre Przywarac57572e2019-01-29 15:54:13 +000021#include <clk.h>
22#include <reset.h>
Samuel Holland42508462021-09-11 16:50:47 -050023#include <asm/gpio.h>
Ian Campbelle24ea552014-05-05 14:42:31 +010024#include <asm/io.h>
25#include <asm/arch/clock.h>
26#include <asm/arch/cpu.h>
Samuel Holland43b573d2023-10-31 00:22:35 -050027#if !CONFIG_IS_ENABLED(DM_MMC)
Ian Campbelle24ea552014-05-05 14:42:31 +010028#include <asm/arch/mmc.h>
Samuel Holland43b573d2023-10-31 00:22:35 -050029#endif
Simon Glassc05ed002020-05-10 11:40:11 -060030#include <linux/delay.h>
Andre Przywara207ed0a2022-09-06 10:36:38 +010031#include <sunxi_gpio.h>
Ian Campbelle24ea552014-05-05 14:42:31 +010032
Samuel Holland43b573d2023-10-31 00:22:35 -050033#include "sunxi_mmc.h"
34
Andre Przywaraf85c0912021-05-05 09:57:47 +010035#ifndef CCM_MMC_CTRL_MODE_SEL_NEW
36#define CCM_MMC_CTRL_MODE_SEL_NEW 0
37#endif
38
Simon Glassdd279182017-07-04 13:31:27 -060039struct sunxi_mmc_plat {
40 struct mmc_config cfg;
41 struct mmc mmc;
42};
43
Simon Glasse3c794e2017-07-04 13:31:23 -060044struct sunxi_mmc_priv {
Ian Campbelle24ea552014-05-05 14:42:31 +010045 unsigned mmc_no;
46 uint32_t *mclkreg;
Ian Campbelle24ea552014-05-05 14:42:31 +010047 unsigned fatal_err;
Simon Glassdd279182017-07-04 13:31:27 -060048 struct gpio_desc cd_gpio; /* Change Detect GPIO */
Ian Campbelle24ea552014-05-05 14:42:31 +010049 struct sunxi_mmc *reg;
50 struct mmc_config cfg;
51};
52
Andre Przywarab5dd39c2021-05-05 10:06:24 +010053/*
54 * All A64 and later MMC controllers feature auto-calibration. This would
55 * normally be detected via the compatible string, but we need something
56 * which works in the SPL as well.
57 */
58static bool sunxi_mmc_can_calibrate(void)
59{
60 return IS_ENABLED(CONFIG_MACH_SUN50I) ||
61 IS_ENABLED(CONFIG_MACH_SUN50I_H5) ||
62 IS_ENABLED(CONFIG_SUN50I_GEN_H6) ||
Andre Przywara4a9e89a2022-10-05 17:54:19 +010063 IS_ENABLED(CONFIG_SUNXI_GEN_NCAT2) ||
Andre Przywarab5dd39c2021-05-05 10:06:24 +010064 IS_ENABLED(CONFIG_MACH_SUN8I_R40);
65}
66
Simon Glass3f5af122017-07-04 13:31:24 -060067static int mmc_set_mod_clk(struct sunxi_mmc_priv *priv, unsigned int hz)
Hans de Goedefc3a8322014-12-07 20:55:10 +010068{
69 unsigned int pll, pll_hz, div, n, oclk_dly, sclk_dly;
Andre Przywaraf85c0912021-05-05 09:57:47 +010070 bool new_mode = IS_ENABLED(CONFIG_MMC_SUNXI_HAS_NEW_MODE);
Maxime Ripardde9b1772017-08-23 12:03:41 +020071 u32 val = 0;
72
Vasily Khoruzhick0e21a2f2018-11-09 20:41:46 -080073 /* A83T support new mode only on eMMC */
74 if (IS_ENABLED(CONFIG_MACH_SUN8I_A83T) && priv->mmc_no != 2)
75 new_mode = false;
Maxime Ripardde9b1772017-08-23 12:03:41 +020076
Hans de Goedefc3a8322014-12-07 20:55:10 +010077 if (hz <= 24000000) {
78 pll = CCM_MMC_CTRL_OSCM24;
79 pll_hz = 24000000;
80 } else {
Hans de Goededaf22632015-01-14 19:05:03 +010081#ifdef CONFIG_MACH_SUN9I
82 pll = CCM_MMC_CTRL_PLL_PERIPH0;
83 pll_hz = clock_get_pll4_periph0();
84#else
Andre Przywara937ee312021-05-05 09:57:47 +010085 /*
86 * SoCs since the A64 (H5, H6, H616) actually use the doubled
87 * rate of PLL6/PERIPH0 as an input clock, but compensate for
88 * that with a fixed post-divider of 2 in the mod clock.
89 * This cancels each other out, so for simplicity we just
90 * pretend it's always PLL6 without a post divider here.
91 */
Hans de Goedefc3a8322014-12-07 20:55:10 +010092 pll = CCM_MMC_CTRL_PLL6;
93 pll_hz = clock_get_pll6();
Hans de Goededaf22632015-01-14 19:05:03 +010094#endif
Hans de Goedefc3a8322014-12-07 20:55:10 +010095 }
96
97 div = pll_hz / hz;
98 if (pll_hz % hz)
99 div++;
100
101 n = 0;
102 while (div > 16) {
103 n++;
104 div = (div + 1) / 2;
105 }
106
107 if (n > 3) {
Simon Glass3f5af122017-07-04 13:31:24 -0600108 printf("mmc %u error cannot set clock to %u\n", priv->mmc_no,
109 hz);
Hans de Goedefc3a8322014-12-07 20:55:10 +0100110 return -1;
111 }
112
113 /* determine delays */
114 if (hz <= 400000) {
115 oclk_dly = 0;
Hans de Goedebe909742015-09-23 16:13:10 +0200116 sclk_dly = 0;
Hans de Goedefc3a8322014-12-07 20:55:10 +0100117 } else if (hz <= 25000000) {
118 oclk_dly = 0;
119 sclk_dly = 5;
Hans de Goedefc3a8322014-12-07 20:55:10 +0100120 } else {
Andre Przywaraf4826fb2020-12-18 22:02:11 +0000121 if (IS_ENABLED(CONFIG_MACH_SUN9I)) {
122 if (hz <= 52000000)
123 oclk_dly = 5;
124 else
125 oclk_dly = 2;
126 } else {
127 if (hz <= 52000000)
128 oclk_dly = 3;
129 else
130 oclk_dly = 1;
131 }
Hans de Goedefc3a8322014-12-07 20:55:10 +0100132 sclk_dly = 4;
133 }
134
Maxime Ripardde9b1772017-08-23 12:03:41 +0200135 if (new_mode) {
Andre Przywaraf85c0912021-05-05 09:57:47 +0100136 val |= CCM_MMC_CTRL_MODE_SEL_NEW;
Chen-Yu Tsai8a647fc2017-08-31 21:57:48 +0800137 setbits_le32(&priv->reg->ntsr, SUNXI_MMC_NTSR_MODE_SEL_NEW);
Andre Przywarab5dd39c2021-05-05 10:06:24 +0100138 }
139
140 if (!sunxi_mmc_can_calibrate()) {
Vasily Khoruzhick20940ef2018-11-05 20:24:28 -0800141 /*
142 * Use hardcoded delay values if controller doesn't support
143 * calibration
144 */
Maxime Ripardde9b1772017-08-23 12:03:41 +0200145 val = CCM_MMC_CTRL_OCLK_DLY(oclk_dly) |
146 CCM_MMC_CTRL_SCLK_DLY(sclk_dly);
147 }
148
149 writel(CCM_MMC_CTRL_ENABLE| pll | CCM_MMC_CTRL_N(n) |
150 CCM_MMC_CTRL_M(div) | val, priv->mclkreg);
Hans de Goedefc3a8322014-12-07 20:55:10 +0100151
152 debug("mmc %u set mod-clk req %u parent %u n %u m %u rate %u\n",
Simon Glass3f5af122017-07-04 13:31:24 -0600153 priv->mmc_no, hz, pll_hz, 1u << n, div, pll_hz / (1u << n) / div);
Hans de Goedefc3a8322014-12-07 20:55:10 +0100154
155 return 0;
156}
157
Simon Glass034e2262017-07-04 13:31:25 -0600158static int mmc_update_clk(struct sunxi_mmc_priv *priv)
Ian Campbelle24ea552014-05-05 14:42:31 +0100159{
Ian Campbelle24ea552014-05-05 14:42:31 +0100160 unsigned int cmd;
161 unsigned timeout_msecs = 2000;
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100162 unsigned long start = get_timer(0);
Ian Campbelle24ea552014-05-05 14:42:31 +0100163
164 cmd = SUNXI_MMC_CMD_START |
165 SUNXI_MMC_CMD_UPCLK_ONLY |
166 SUNXI_MMC_CMD_WAIT_PRE_OVER;
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100167
Simon Glass3f5af122017-07-04 13:31:24 -0600168 writel(cmd, &priv->reg->cmd);
169 while (readl(&priv->reg->cmd) & SUNXI_MMC_CMD_START) {
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100170 if (get_timer(start) > timeout_msecs)
Ian Campbelle24ea552014-05-05 14:42:31 +0100171 return -1;
Ian Campbelle24ea552014-05-05 14:42:31 +0100172 }
173
174 /* clock update sets various irq status bits, clear these */
Simon Glass3f5af122017-07-04 13:31:24 -0600175 writel(readl(&priv->reg->rint), &priv->reg->rint);
Ian Campbelle24ea552014-05-05 14:42:31 +0100176
177 return 0;
178}
179
Simon Glass034e2262017-07-04 13:31:25 -0600180static int mmc_config_clock(struct sunxi_mmc_priv *priv, struct mmc *mmc)
Ian Campbelle24ea552014-05-05 14:42:31 +0100181{
Simon Glass3f5af122017-07-04 13:31:24 -0600182 unsigned rval = readl(&priv->reg->clkcr);
Ian Campbelle24ea552014-05-05 14:42:31 +0100183
184 /* Disable Clock */
185 rval &= ~SUNXI_MMC_CLK_ENABLE;
Simon Glass3f5af122017-07-04 13:31:24 -0600186 writel(rval, &priv->reg->clkcr);
Simon Glass034e2262017-07-04 13:31:25 -0600187 if (mmc_update_clk(priv))
Ian Campbelle24ea552014-05-05 14:42:31 +0100188 return -1;
189
Hans de Goedefc3a8322014-12-07 20:55:10 +0100190 /* Set mod_clk to new rate */
Simon Glass3f5af122017-07-04 13:31:24 -0600191 if (mmc_set_mod_clk(priv, mmc->clock))
Ian Campbelle24ea552014-05-05 14:42:31 +0100192 return -1;
Hans de Goedefc3a8322014-12-07 20:55:10 +0100193
194 /* Clear internal divider */
195 rval &= ~SUNXI_MMC_CLK_DIVIDER_MASK;
Simon Glass3f5af122017-07-04 13:31:24 -0600196 writel(rval, &priv->reg->clkcr);
Hans de Goedefc3a8322014-12-07 20:55:10 +0100197
Andre Przywara4a9e89a2022-10-05 17:54:19 +0100198#if defined(CONFIG_SUNXI_GEN_SUN6I) || defined(CONFIG_SUN50I_GEN_H6) || defined(CONFIG_SUNXI_GEN_NCAT2)
Vasily Khoruzhick20940ef2018-11-05 20:24:28 -0800199 /* A64 supports calibration of delays on MMC controller and we
200 * have to set delay of zero before starting calibration.
201 * Allwinner BSP driver sets a delay only in the case of
202 * using HS400 which is not supported by mainline U-Boot or
203 * Linux at the moment
204 */
Andre Przywarab5dd39c2021-05-05 10:06:24 +0100205 if (sunxi_mmc_can_calibrate())
206 writel(SUNXI_MMC_CAL_DL_SW_EN, &priv->reg->samp_dl);
Vasily Khoruzhick20940ef2018-11-05 20:24:28 -0800207#endif
208
Ian Campbelle24ea552014-05-05 14:42:31 +0100209 /* Re-enable Clock */
210 rval |= SUNXI_MMC_CLK_ENABLE;
Simon Glass3f5af122017-07-04 13:31:24 -0600211 writel(rval, &priv->reg->clkcr);
Simon Glass034e2262017-07-04 13:31:25 -0600212 if (mmc_update_clk(priv))
Ian Campbelle24ea552014-05-05 14:42:31 +0100213 return -1;
214
215 return 0;
216}
217
Simon Glass034e2262017-07-04 13:31:25 -0600218static int sunxi_mmc_set_ios_common(struct sunxi_mmc_priv *priv,
219 struct mmc *mmc)
Ian Campbelle24ea552014-05-05 14:42:31 +0100220{
Hans de Goedefc3a8322014-12-07 20:55:10 +0100221 debug("set ios: bus_width: %x, clock: %d\n",
222 mmc->bus_width, mmc->clock);
Ian Campbelle24ea552014-05-05 14:42:31 +0100223
224 /* Change clock first */
Simon Glass034e2262017-07-04 13:31:25 -0600225 if (mmc->clock && mmc_config_clock(priv, mmc) != 0) {
Simon Glass3f5af122017-07-04 13:31:24 -0600226 priv->fatal_err = 1;
Jaehoon Chung07b0b9c2016-12-30 15:30:16 +0900227 return -EINVAL;
Ian Campbelle24ea552014-05-05 14:42:31 +0100228 }
229
230 /* Change bus width */
231 if (mmc->bus_width == 8)
Simon Glass3f5af122017-07-04 13:31:24 -0600232 writel(0x2, &priv->reg->width);
Ian Campbelle24ea552014-05-05 14:42:31 +0100233 else if (mmc->bus_width == 4)
Simon Glass3f5af122017-07-04 13:31:24 -0600234 writel(0x1, &priv->reg->width);
Ian Campbelle24ea552014-05-05 14:42:31 +0100235 else
Simon Glass3f5af122017-07-04 13:31:24 -0600236 writel(0x0, &priv->reg->width);
Jaehoon Chung07b0b9c2016-12-30 15:30:16 +0900237
238 return 0;
Ian Campbelle24ea552014-05-05 14:42:31 +0100239}
240
Simon Glass034e2262017-07-04 13:31:25 -0600241static int mmc_trans_data_by_cpu(struct sunxi_mmc_priv *priv, struct mmc *mmc,
242 struct mmc_data *data)
Ian Campbelle24ea552014-05-05 14:42:31 +0100243{
Ian Campbelle24ea552014-05-05 14:42:31 +0100244 const int reading = !!(data->flags & MMC_DATA_READ);
245 const uint32_t status_bit = reading ? SUNXI_MMC_STATUS_FIFO_EMPTY :
246 SUNXI_MMC_STATUS_FIFO_FULL;
247 unsigned i;
Ian Campbelle24ea552014-05-05 14:42:31 +0100248 unsigned *buff = (unsigned int *)(reading ? data->dest : data->src);
Andre Przywara9faae542021-05-05 11:33:40 +0100249 unsigned word_cnt = (data->blocksize * data->blocks) >> 2;
250 unsigned timeout_msecs = word_cnt >> 6;
251 uint32_t status;
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100252 unsigned long start;
253
254 if (timeout_msecs < 2000)
255 timeout_msecs = 2000;
Ian Campbelle24ea552014-05-05 14:42:31 +0100256
Hans de Goedeb6ae6762014-06-09 11:36:55 +0200257 /* Always read / write data through the CPU */
Simon Glass3f5af122017-07-04 13:31:24 -0600258 setbits_le32(&priv->reg->gctrl, SUNXI_MMC_GCTRL_ACCESS_BY_AHB);
Hans de Goedeb6ae6762014-06-09 11:36:55 +0200259
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100260 start = get_timer(0);
261
Andre Przywara9faae542021-05-05 11:33:40 +0100262 for (i = 0; i < word_cnt;) {
263 unsigned int in_fifo;
264
265 while ((status = readl(&priv->reg->status)) & status_bit) {
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100266 if (get_timer(start) > timeout_msecs)
Ian Campbelle24ea552014-05-05 14:42:31 +0100267 return -1;
Ian Campbelle24ea552014-05-05 14:42:31 +0100268 }
269
Andre Przywara9faae542021-05-05 11:33:40 +0100270 /*
271 * For writing we do not easily know the FIFO size, so have
272 * to check the FIFO status after every word written.
273 * TODO: For optimisation we could work out a minimum FIFO
274 * size across all SoCs, and use that together with the current
275 * fill level to write chunks of words.
276 */
277 if (!reading) {
278 writel(buff[i++], &priv->reg->fifo);
279 continue;
280 }
281
282 /*
283 * The status register holds the current FIFO level, so we
284 * can be sure to collect as many words from the FIFO
285 * register without checking the status register after every
286 * read. That saves half of the costly MMIO reads, effectively
287 * doubling the read performance.
Andre Przywara0b508ca2021-09-03 16:49:16 +0100288 * Some SoCs (A20) report a level of 0 if the FIFO is
289 * completely full (value masked out?). Use a safe minimal
290 * FIFO size in this case.
Andre Przywara9faae542021-05-05 11:33:40 +0100291 */
Andre Przywara0b508ca2021-09-03 16:49:16 +0100292 in_fifo = SUNXI_MMC_STATUS_FIFO_LEVEL(status);
293 if (in_fifo == 0 && (status & SUNXI_MMC_STATUS_FIFO_FULL))
294 in_fifo = 32;
295 for (; in_fifo > 0; in_fifo--)
Andre Przywara9faae542021-05-05 11:33:40 +0100296 buff[i++] = readl_relaxed(&priv->reg->fifo);
297 dmb();
Ian Campbelle24ea552014-05-05 14:42:31 +0100298 }
299
300 return 0;
301}
302
Simon Glass034e2262017-07-04 13:31:25 -0600303static int mmc_rint_wait(struct sunxi_mmc_priv *priv, struct mmc *mmc,
304 uint timeout_msecs, uint done_bit, const char *what)
Ian Campbelle24ea552014-05-05 14:42:31 +0100305{
Ian Campbelle24ea552014-05-05 14:42:31 +0100306 unsigned int status;
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100307 unsigned long start = get_timer(0);
Ian Campbelle24ea552014-05-05 14:42:31 +0100308
309 do {
Simon Glass3f5af122017-07-04 13:31:24 -0600310 status = readl(&priv->reg->rint);
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100311 if ((get_timer(start) > timeout_msecs) ||
Ian Campbelle24ea552014-05-05 14:42:31 +0100312 (status & SUNXI_MMC_RINT_INTERRUPT_ERROR_BIT)) {
313 debug("%s timeout %x\n", what,
314 status & SUNXI_MMC_RINT_INTERRUPT_ERROR_BIT);
Jaehoon Chung915ffa52016-07-19 16:33:36 +0900315 return -ETIMEDOUT;
Ian Campbelle24ea552014-05-05 14:42:31 +0100316 }
Ian Campbelle24ea552014-05-05 14:42:31 +0100317 } while (!(status & done_bit));
318
319 return 0;
320}
321
Simon Glass034e2262017-07-04 13:31:25 -0600322static int sunxi_mmc_send_cmd_common(struct sunxi_mmc_priv *priv,
323 struct mmc *mmc, struct mmc_cmd *cmd,
324 struct mmc_data *data)
Ian Campbelle24ea552014-05-05 14:42:31 +0100325{
Ian Campbelle24ea552014-05-05 14:42:31 +0100326 unsigned int cmdval = SUNXI_MMC_CMD_START;
327 unsigned int timeout_msecs;
328 int error = 0;
329 unsigned int status = 0;
Ian Campbelle24ea552014-05-05 14:42:31 +0100330 unsigned int bytecnt = 0;
331
Simon Glass3f5af122017-07-04 13:31:24 -0600332 if (priv->fatal_err)
Ian Campbelle24ea552014-05-05 14:42:31 +0100333 return -1;
334 if (cmd->resp_type & MMC_RSP_BUSY)
335 debug("mmc cmd %d check rsp busy\n", cmd->cmdidx);
336 if (cmd->cmdidx == 12)
337 return 0;
338
339 if (!cmd->cmdidx)
340 cmdval |= SUNXI_MMC_CMD_SEND_INIT_SEQ;
341 if (cmd->resp_type & MMC_RSP_PRESENT)
342 cmdval |= SUNXI_MMC_CMD_RESP_EXPIRE;
343 if (cmd->resp_type & MMC_RSP_136)
344 cmdval |= SUNXI_MMC_CMD_LONG_RESPONSE;
345 if (cmd->resp_type & MMC_RSP_CRC)
346 cmdval |= SUNXI_MMC_CMD_CHK_RESPONSE_CRC;
347
348 if (data) {
Alexander Graf0ea5a042016-03-29 17:29:09 +0200349 if ((u32)(long)data->dest & 0x3) {
Ian Campbelle24ea552014-05-05 14:42:31 +0100350 error = -1;
351 goto out;
352 }
353
354 cmdval |= SUNXI_MMC_CMD_DATA_EXPIRE|SUNXI_MMC_CMD_WAIT_PRE_OVER;
355 if (data->flags & MMC_DATA_WRITE)
356 cmdval |= SUNXI_MMC_CMD_WRITE;
357 if (data->blocks > 1)
358 cmdval |= SUNXI_MMC_CMD_AUTO_STOP;
Simon Glass3f5af122017-07-04 13:31:24 -0600359 writel(data->blocksize, &priv->reg->blksz);
360 writel(data->blocks * data->blocksize, &priv->reg->bytecnt);
Ian Campbelle24ea552014-05-05 14:42:31 +0100361 }
362
Simon Glass3f5af122017-07-04 13:31:24 -0600363 debug("mmc %d, cmd %d(0x%08x), arg 0x%08x\n", priv->mmc_no,
Ian Campbelle24ea552014-05-05 14:42:31 +0100364 cmd->cmdidx, cmdval | cmd->cmdidx, cmd->cmdarg);
Simon Glass3f5af122017-07-04 13:31:24 -0600365 writel(cmd->cmdarg, &priv->reg->arg);
Ian Campbelle24ea552014-05-05 14:42:31 +0100366
367 if (!data)
Simon Glass3f5af122017-07-04 13:31:24 -0600368 writel(cmdval | cmd->cmdidx, &priv->reg->cmd);
Ian Campbelle24ea552014-05-05 14:42:31 +0100369
370 /*
371 * transfer data and check status
372 * STATREG[2] : FIFO empty
373 * STATREG[3] : FIFO full
374 */
375 if (data) {
376 int ret = 0;
377
378 bytecnt = data->blocksize * data->blocks;
379 debug("trans data %d bytes\n", bytecnt);
Simon Glass3f5af122017-07-04 13:31:24 -0600380 writel(cmdval | cmd->cmdidx, &priv->reg->cmd);
Simon Glass034e2262017-07-04 13:31:25 -0600381 ret = mmc_trans_data_by_cpu(priv, mmc, data);
Ian Campbelle24ea552014-05-05 14:42:31 +0100382 if (ret) {
Simon Glass3f5af122017-07-04 13:31:24 -0600383 error = readl(&priv->reg->rint) &
Ian Campbelle24ea552014-05-05 14:42:31 +0100384 SUNXI_MMC_RINT_INTERRUPT_ERROR_BIT;
Jaehoon Chung915ffa52016-07-19 16:33:36 +0900385 error = -ETIMEDOUT;
Ian Campbelle24ea552014-05-05 14:42:31 +0100386 goto out;
387 }
388 }
389
Simon Glass034e2262017-07-04 13:31:25 -0600390 error = mmc_rint_wait(priv, mmc, 1000, SUNXI_MMC_RINT_COMMAND_DONE,
391 "cmd");
Ian Campbelle24ea552014-05-05 14:42:31 +0100392 if (error)
393 goto out;
394
395 if (data) {
Hans de Goedeb6ae6762014-06-09 11:36:55 +0200396 timeout_msecs = 120;
Ian Campbelle24ea552014-05-05 14:42:31 +0100397 debug("cacl timeout %x msec\n", timeout_msecs);
Simon Glass034e2262017-07-04 13:31:25 -0600398 error = mmc_rint_wait(priv, mmc, timeout_msecs,
Ian Campbelle24ea552014-05-05 14:42:31 +0100399 data->blocks > 1 ?
400 SUNXI_MMC_RINT_AUTO_COMMAND_DONE :
401 SUNXI_MMC_RINT_DATA_OVER,
402 "data");
403 if (error)
404 goto out;
405 }
406
407 if (cmd->resp_type & MMC_RSP_BUSY) {
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100408 unsigned long start = get_timer(0);
Ian Campbelle24ea552014-05-05 14:42:31 +0100409 timeout_msecs = 2000;
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100410
Ian Campbelle24ea552014-05-05 14:42:31 +0100411 do {
Simon Glass3f5af122017-07-04 13:31:24 -0600412 status = readl(&priv->reg->status);
Philipp Tomsich5ff8e542018-03-21 12:18:58 +0100413 if (get_timer(start) > timeout_msecs) {
Ian Campbelle24ea552014-05-05 14:42:31 +0100414 debug("busy timeout\n");
Jaehoon Chung915ffa52016-07-19 16:33:36 +0900415 error = -ETIMEDOUT;
Ian Campbelle24ea552014-05-05 14:42:31 +0100416 goto out;
417 }
Ian Campbelle24ea552014-05-05 14:42:31 +0100418 } while (status & SUNXI_MMC_STATUS_CARD_DATA_BUSY);
419 }
420
421 if (cmd->resp_type & MMC_RSP_136) {
Simon Glass3f5af122017-07-04 13:31:24 -0600422 cmd->response[0] = readl(&priv->reg->resp3);
423 cmd->response[1] = readl(&priv->reg->resp2);
424 cmd->response[2] = readl(&priv->reg->resp1);
425 cmd->response[3] = readl(&priv->reg->resp0);
Ian Campbelle24ea552014-05-05 14:42:31 +0100426 debug("mmc resp 0x%08x 0x%08x 0x%08x 0x%08x\n",
427 cmd->response[3], cmd->response[2],
428 cmd->response[1], cmd->response[0]);
429 } else {
Simon Glass3f5af122017-07-04 13:31:24 -0600430 cmd->response[0] = readl(&priv->reg->resp0);
Ian Campbelle24ea552014-05-05 14:42:31 +0100431 debug("mmc resp 0x%08x\n", cmd->response[0]);
432 }
433out:
Ian Campbelle24ea552014-05-05 14:42:31 +0100434 if (error < 0) {
Simon Glass3f5af122017-07-04 13:31:24 -0600435 writel(SUNXI_MMC_GCTRL_RESET, &priv->reg->gctrl);
Simon Glass034e2262017-07-04 13:31:25 -0600436 mmc_update_clk(priv);
Ian Campbelle24ea552014-05-05 14:42:31 +0100437 }
Simon Glass3f5af122017-07-04 13:31:24 -0600438 writel(0xffffffff, &priv->reg->rint);
439 writel(readl(&priv->reg->gctrl) | SUNXI_MMC_GCTRL_FIFO_RESET,
440 &priv->reg->gctrl);
Ian Campbelle24ea552014-05-05 14:42:31 +0100441
442 return error;
443}
444
Andre Przywaraba16b532022-07-13 17:21:44 +0100445/* non-DM code here is used by the (ARM) SPL only */
446
Simon Glassdd279182017-07-04 13:31:27 -0600447#if !CONFIG_IS_ENABLED(DM_MMC)
Andre Przywaraba16b532022-07-13 17:21:44 +0100448/* support 4 mmc hosts */
449struct sunxi_mmc_priv mmc_host[4];
450
451static int mmc_resource_init(int sdc_no)
452{
453 struct sunxi_mmc_priv *priv = &mmc_host[sdc_no];
454 struct sunxi_ccm_reg *ccm = (struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
455
456 debug("init mmc %d resource\n", sdc_no);
457
458 switch (sdc_no) {
459 case 0:
460 priv->reg = (struct sunxi_mmc *)SUNXI_MMC0_BASE;
461 priv->mclkreg = &ccm->sd0_clk_cfg;
462 break;
463 case 1:
464 priv->reg = (struct sunxi_mmc *)SUNXI_MMC1_BASE;
465 priv->mclkreg = &ccm->sd1_clk_cfg;
466 break;
467#ifdef SUNXI_MMC2_BASE
468 case 2:
469 priv->reg = (struct sunxi_mmc *)SUNXI_MMC2_BASE;
470 priv->mclkreg = &ccm->sd2_clk_cfg;
471 break;
472#endif
473#ifdef SUNXI_MMC3_BASE
474 case 3:
475 priv->reg = (struct sunxi_mmc *)SUNXI_MMC3_BASE;
476 priv->mclkreg = &ccm->sd3_clk_cfg;
477 break;
478#endif
479 default:
480 printf("Wrong mmc number %d\n", sdc_no);
481 return -1;
482 }
483 priv->mmc_no = sdc_no;
484
485 return 0;
486}
487
488static int sunxi_mmc_core_init(struct mmc *mmc)
489{
490 struct sunxi_mmc_priv *priv = mmc->priv;
491
492 /* Reset controller */
493 writel(SUNXI_MMC_GCTRL_RESET, &priv->reg->gctrl);
494 udelay(1000);
495
496 return 0;
497}
498
Simon Glass034e2262017-07-04 13:31:25 -0600499static int sunxi_mmc_set_ios_legacy(struct mmc *mmc)
500{
501 struct sunxi_mmc_priv *priv = mmc->priv;
502
503 return sunxi_mmc_set_ios_common(priv, mmc);
504}
505
506static int sunxi_mmc_send_cmd_legacy(struct mmc *mmc, struct mmc_cmd *cmd,
507 struct mmc_data *data)
508{
509 struct sunxi_mmc_priv *priv = mmc->priv;
510
511 return sunxi_mmc_send_cmd_common(priv, mmc, cmd, data);
512}
513
Andre Przywara5db81f12022-07-13 17:21:43 +0100514/* .getcd is not needed by the SPL */
Ian Campbelle24ea552014-05-05 14:42:31 +0100515static const struct mmc_ops sunxi_mmc_ops = {
Simon Glass034e2262017-07-04 13:31:25 -0600516 .send_cmd = sunxi_mmc_send_cmd_legacy,
517 .set_ios = sunxi_mmc_set_ios_legacy,
Siarhei Siamashka5abdb152015-02-01 00:42:14 +0200518 .init = sunxi_mmc_core_init,
Ian Campbelle24ea552014-05-05 14:42:31 +0100519};
520
Hans de Goedee79c7c82014-10-02 21:13:54 +0200521struct mmc *sunxi_mmc_init(int sdc_no)
Ian Campbelle24ea552014-05-05 14:42:31 +0100522{
Simon Glassec73d962017-07-04 13:31:26 -0600523 struct sunxi_ccm_reg *ccm = (struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
Simon Glass034e2262017-07-04 13:31:25 -0600524 struct sunxi_mmc_priv *priv = &mmc_host[sdc_no];
525 struct mmc_config *cfg = &priv->cfg;
Simon Glassec73d962017-07-04 13:31:26 -0600526 int ret;
Ian Campbelle24ea552014-05-05 14:42:31 +0100527
Simon Glass034e2262017-07-04 13:31:25 -0600528 memset(priv, '\0', sizeof(struct sunxi_mmc_priv));
Ian Campbelle24ea552014-05-05 14:42:31 +0100529
530 cfg->name = "SUNXI SD/MMC";
531 cfg->ops = &sunxi_mmc_ops;
532
533 cfg->voltages = MMC_VDD_32_33 | MMC_VDD_33_34;
534 cfg->host_caps = MMC_MODE_4BIT;
Andre Przywaraf4826fb2020-12-18 22:02:11 +0000535
536 if ((IS_ENABLED(CONFIG_MACH_SUN50I) || IS_ENABLED(CONFIG_MACH_SUN8I) ||
537 IS_ENABLED(CONFIG_SUN50I_GEN_H6)) && (sdc_no == 2))
Siarhei Siamashkad96ebc42016-03-29 17:29:10 +0200538 cfg->host_caps = MMC_MODE_8BIT;
Andre Przywaraf4826fb2020-12-18 22:02:11 +0000539
Rob Herring5a203972015-03-23 17:56:59 -0500540 cfg->host_caps |= MMC_MODE_HS_52MHz | MMC_MODE_HS;
Ian Campbelle24ea552014-05-05 14:42:31 +0100541 cfg->b_max = CONFIG_SYS_MMC_MAX_BLK_COUNT;
542
543 cfg->f_min = 400000;
544 cfg->f_max = 52000000;
545
Hans de Goede967325f2014-10-31 16:55:02 +0100546 if (mmc_resource_init(sdc_no) != 0)
547 return NULL;
548
Simon Glassec73d962017-07-04 13:31:26 -0600549 /* config ahb clock */
550 debug("init mmc %d clock and io\n", sdc_no);
Andre Przywara4a9e89a2022-10-05 17:54:19 +0100551#if !defined(CONFIG_SUN50I_GEN_H6) && !defined(CONFIG_SUNXI_GEN_NCAT2)
Simon Glassec73d962017-07-04 13:31:26 -0600552 setbits_le32(&ccm->ahb_gate0, 1 << AHB_GATE_OFFSET_MMC(sdc_no));
553
554#ifdef CONFIG_SUNXI_GEN_SUN6I
555 /* unassert reset */
556 setbits_le32(&ccm->ahb_reset0_cfg, 1 << AHB_RESET_OFFSET_MMC(sdc_no));
557#endif
558#if defined(CONFIG_MACH_SUN9I)
559 /* sun9i has a mmc-common module, also set the gate and reset there */
560 writel(SUNXI_MMC_COMMON_CLK_GATE | SUNXI_MMC_COMMON_RESET,
561 SUNXI_MMC_COMMON_BASE + 4 * sdc_no);
562#endif
Jernej Skrabecaaebb902021-01-11 21:11:35 +0100563#else /* CONFIG_SUN50I_GEN_H6 */
Icenowy Zheng42956f12018-07-21 16:20:29 +0800564 setbits_le32(&ccm->sd_gate_reset, 1 << sdc_no);
565 /* unassert reset */
566 setbits_le32(&ccm->sd_gate_reset, 1 << (RESET_SHIFT + sdc_no));
567#endif
Simon Glassec73d962017-07-04 13:31:26 -0600568 ret = mmc_set_mod_clk(priv, 24000000);
569 if (ret)
570 return NULL;
Ian Campbelle24ea552014-05-05 14:42:31 +0100571
Maxime Ripardead36972017-08-23 13:41:33 +0200572 return mmc_create(cfg, priv);
Ian Campbelle24ea552014-05-05 14:42:31 +0100573}
Andre Przywaraba16b532022-07-13 17:21:44 +0100574
575#else /* CONFIG_DM_MMC code below, as used by U-Boot proper */
Simon Glassdd279182017-07-04 13:31:27 -0600576
577static int sunxi_mmc_set_ios(struct udevice *dev)
578{
Simon Glassc69cda22020-12-03 16:55:20 -0700579 struct sunxi_mmc_plat *plat = dev_get_plat(dev);
Simon Glassdd279182017-07-04 13:31:27 -0600580 struct sunxi_mmc_priv *priv = dev_get_priv(dev);
581
582 return sunxi_mmc_set_ios_common(priv, &plat->mmc);
583}
584
585static int sunxi_mmc_send_cmd(struct udevice *dev, struct mmc_cmd *cmd,
586 struct mmc_data *data)
587{
Simon Glassc69cda22020-12-03 16:55:20 -0700588 struct sunxi_mmc_plat *plat = dev_get_plat(dev);
Simon Glassdd279182017-07-04 13:31:27 -0600589 struct sunxi_mmc_priv *priv = dev_get_priv(dev);
590
591 return sunxi_mmc_send_cmd_common(priv, &plat->mmc, cmd, data);
592}
593
594static int sunxi_mmc_getcd(struct udevice *dev)
595{
Andre Przywaraac62dad2021-04-21 09:33:04 +0100596 struct mmc *mmc = mmc_get_mmc_dev(dev);
Simon Glassdd279182017-07-04 13:31:27 -0600597 struct sunxi_mmc_priv *priv = dev_get_priv(dev);
598
Andre Przywaraac62dad2021-04-21 09:33:04 +0100599 /* If polling, assume that the card is always present. */
600 if ((mmc->cfg->host_caps & MMC_CAP_NONREMOVABLE) ||
601 (mmc->cfg->host_caps & MMC_CAP_NEEDS_POLL))
602 return 1;
603
Heinrich Schuchardt8be4e612018-02-01 23:39:19 +0100604 if (dm_gpio_is_valid(&priv->cd_gpio)) {
605 int cd_state = dm_gpio_get_value(&priv->cd_gpio);
Simon Glassdd279182017-07-04 13:31:27 -0600606
Andre Przywaraac62dad2021-04-21 09:33:04 +0100607 if (mmc->cfg->host_caps & MMC_CAP_CD_ACTIVE_HIGH)
608 return !cd_state;
609 else
610 return cd_state;
Heinrich Schuchardt8be4e612018-02-01 23:39:19 +0100611 }
Simon Glassdd279182017-07-04 13:31:27 -0600612 return 1;
613}
614
615static const struct dm_mmc_ops sunxi_mmc_ops = {
616 .send_cmd = sunxi_mmc_send_cmd,
617 .set_ios = sunxi_mmc_set_ios,
618 .get_cd = sunxi_mmc_getcd,
619};
620
Andre Przywara0237b302021-01-11 21:11:44 +0100621static unsigned get_mclk_offset(void)
622{
623 if (IS_ENABLED(CONFIG_MACH_SUN9I_A80))
624 return 0x410;
625
Andre Przywara4a9e89a2022-10-05 17:54:19 +0100626 if (IS_ENABLED(CONFIG_SUN50I_GEN_H6) || IS_ENABLED(CONFIG_SUNXI_GEN_NCAT2))
Andre Przywara0237b302021-01-11 21:11:44 +0100627 return 0x830;
628
629 return 0x88;
630};
631
Simon Glassdd279182017-07-04 13:31:27 -0600632static int sunxi_mmc_probe(struct udevice *dev)
633{
634 struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
Simon Glassc69cda22020-12-03 16:55:20 -0700635 struct sunxi_mmc_plat *plat = dev_get_plat(dev);
Simon Glassdd279182017-07-04 13:31:27 -0600636 struct sunxi_mmc_priv *priv = dev_get_priv(dev);
Andre Przywarac57572e2019-01-29 15:54:13 +0000637 struct reset_ctl_bulk reset_bulk;
638 struct clk gate_clk;
Simon Glassdd279182017-07-04 13:31:27 -0600639 struct mmc_config *cfg = &plat->cfg;
640 struct ofnode_phandle_args args;
Andre Przywarac57572e2019-01-29 15:54:13 +0000641 u32 *ccu_reg;
Andre Przywaraac62dad2021-04-21 09:33:04 +0100642 int ret;
Simon Glassdd279182017-07-04 13:31:27 -0600643
644 cfg->name = dev->name;
Simon Glassdd279182017-07-04 13:31:27 -0600645
646 cfg->voltages = MMC_VDD_32_33 | MMC_VDD_33_34;
Andre Przywaraac62dad2021-04-21 09:33:04 +0100647 cfg->host_caps = MMC_MODE_HS_52MHz | MMC_MODE_HS;
Simon Glassdd279182017-07-04 13:31:27 -0600648 cfg->b_max = CONFIG_SYS_MMC_MAX_BLK_COUNT;
649
650 cfg->f_min = 400000;
651 cfg->f_max = 52000000;
652
Andre Przywaraac62dad2021-04-21 09:33:04 +0100653 ret = mmc_of_parse(dev, cfg);
654 if (ret)
655 return ret;
656
Andre Przywaraca496ba2021-04-29 09:31:58 +0100657 priv->reg = dev_read_addr_ptr(dev);
Simon Glassdd279182017-07-04 13:31:27 -0600658
659 /* We don't have a sunxi clock driver so find the clock address here */
660 ret = dev_read_phandle_with_args(dev, "clocks", "#clock-cells", 0,
661 1, &args);
662 if (ret)
663 return ret;
Andre Przywaraca496ba2021-04-29 09:31:58 +0100664 ccu_reg = (u32 *)(uintptr_t)ofnode_get_addr(args.node);
Simon Glassdd279182017-07-04 13:31:27 -0600665
Jagan Tekie8f37f42019-01-09 16:58:39 +0530666 priv->mmc_no = ((uintptr_t)priv->reg - SUNXI_MMC0_BASE) / 0x1000;
Andre Przywara0237b302021-01-11 21:11:44 +0100667 priv->mclkreg = (void *)ccu_reg + get_mclk_offset() + priv->mmc_no * 4;
Andre Przywarac57572e2019-01-29 15:54:13 +0000668
669 ret = clk_get_by_name(dev, "ahb", &gate_clk);
670 if (!ret)
671 clk_enable(&gate_clk);
672
673 ret = reset_get_bulk(dev, &reset_bulk);
674 if (!ret)
675 reset_deassert_bulk(&reset_bulk);
Simon Glassdd279182017-07-04 13:31:27 -0600676
677 ret = mmc_set_mod_clk(priv, 24000000);
678 if (ret)
679 return ret;
680
681 /* This GPIO is optional */
Samuel Hollandfb6f6702021-10-20 23:52:57 -0500682 gpio_request_by_name(dev, "cd-gpios", 0, &priv->cd_gpio,
683 GPIOD_IS_IN | GPIOD_PULL_UP);
Simon Glassdd279182017-07-04 13:31:27 -0600684
685 upriv->mmc = &plat->mmc;
686
687 /* Reset controller */
688 writel(SUNXI_MMC_GCTRL_RESET, &priv->reg->gctrl);
689 udelay(1000);
690
691 return 0;
692}
693
694static int sunxi_mmc_bind(struct udevice *dev)
695{
Simon Glassc69cda22020-12-03 16:55:20 -0700696 struct sunxi_mmc_plat *plat = dev_get_plat(dev);
Simon Glassdd279182017-07-04 13:31:27 -0600697
698 return mmc_bind(dev, &plat->mmc, &plat->cfg);
699}
700
701static const struct udevice_id sunxi_mmc_ids[] = {
Andre Przywara0237b302021-01-11 21:11:44 +0100702 { .compatible = "allwinner,sun4i-a10-mmc" },
703 { .compatible = "allwinner,sun5i-a13-mmc" },
704 { .compatible = "allwinner,sun7i-a20-mmc" },
705 { .compatible = "allwinner,sun8i-a83t-emmc" },
706 { .compatible = "allwinner,sun9i-a80-mmc" },
Samuel Hollandd379bcb2023-10-31 00:22:34 -0500707 { .compatible = "allwinner,sun20i-d1-mmc" },
Andre Przywara0237b302021-01-11 21:11:44 +0100708 { .compatible = "allwinner,sun50i-a64-mmc" },
709 { .compatible = "allwinner,sun50i-a64-emmc" },
710 { .compatible = "allwinner,sun50i-h6-mmc" },
711 { .compatible = "allwinner,sun50i-h6-emmc" },
712 { .compatible = "allwinner,sun50i-a100-mmc" },
713 { .compatible = "allwinner,sun50i-a100-emmc" },
Jagan Tekie8f37f42019-01-09 16:58:39 +0530714 { /* sentinel */ }
Simon Glassdd279182017-07-04 13:31:27 -0600715};
716
717U_BOOT_DRIVER(sunxi_mmc_drv) = {
718 .name = "sunxi_mmc",
719 .id = UCLASS_MMC,
720 .of_match = sunxi_mmc_ids,
721 .bind = sunxi_mmc_bind,
722 .probe = sunxi_mmc_probe,
723 .ops = &sunxi_mmc_ops,
Simon Glasscaa4daa2020-12-03 16:55:18 -0700724 .plat_auto = sizeof(struct sunxi_mmc_plat),
Simon Glass41575d82020-12-03 16:55:17 -0700725 .priv_auto = sizeof(struct sunxi_mmc_priv),
Simon Glassdd279182017-07-04 13:31:27 -0600726};
727#endif