fsl_esdhc.c 27.4 KB
Newer Older
1
/*
2
 * Copyright 2007, 2010-2011 Freescale Semiconductor, Inc
3 4 5 6 7 8
 * Andy Fleming
 *
 * Based vaguely on the pxa mmc code:
 * (C) Copyright 2003
 * Kyle Harris, Nexus Technologies, Inc. kharris@nexus-tech.net
 *
9
 * SPDX-License-Identifier:	GPL-2.0+
10 11 12 13 14
 */

#include <config.h>
#include <common.h>
#include <command.h>
15
#include <errno.h>
16
#include <hwconfig.h>
17 18
#include <mmc.h>
#include <part.h>
19
#include <power/regulator.h>
20 21
#include <malloc.h>
#include <fsl_esdhc.h>
22
#include <fdt_support.h>
23
#include <asm/io.h>
P
Peng Fan 已提交
24 25
#include <dm.h>
#include <asm-generic/gpio.h>
26 27 28

DECLARE_GLOBAL_DATA_PTR;

29 30 31 32 33 34 35
#define SDHCI_IRQ_EN_BITS		(IRQSTATEN_CC | IRQSTATEN_TC | \
				IRQSTATEN_CINT | \
				IRQSTATEN_CTOE | IRQSTATEN_CCE | IRQSTATEN_CEBE | \
				IRQSTATEN_CIE | IRQSTATEN_DTOE | IRQSTATEN_DCE | \
				IRQSTATEN_DEBE | IRQSTATEN_BRR | IRQSTATEN_BWR | \
				IRQSTATEN_DINT)

36
struct fsl_esdhc {
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
	uint    dsaddr;		/* SDMA system address register */
	uint    blkattr;	/* Block attributes register */
	uint    cmdarg;		/* Command argument register */
	uint    xfertyp;	/* Transfer type register */
	uint    cmdrsp0;	/* Command response 0 register */
	uint    cmdrsp1;	/* Command response 1 register */
	uint    cmdrsp2;	/* Command response 2 register */
	uint    cmdrsp3;	/* Command response 3 register */
	uint    datport;	/* Buffer data port register */
	uint    prsstat;	/* Present state register */
	uint    proctl;		/* Protocol control register */
	uint    sysctl;		/* System Control Register */
	uint    irqstat;	/* Interrupt status register */
	uint    irqstaten;	/* Interrupt status enable register */
	uint    irqsigen;	/* Interrupt signal enable register */
	uint    autoc12err;	/* Auto CMD error status register */
	uint    hostcapblt;	/* Host controller capabilities register */
	uint    wml;		/* Watermark level register */
	uint    mixctrl;	/* For USDHC */
	char    reserved1[4];	/* reserved */
	uint    fevt;		/* Force event register */
	uint    admaes;		/* ADMA error status register */
	uint    adsaddr;	/* ADMA system address register */
60 61 62 63 64 65 66 67 68
	char    reserved2[4];
	uint    dllctrl;
	uint    dllstat;
	uint    clktunectrlstatus;
	char    reserved3[84];
	uint    vendorspec;
	uint    mmcboot;
	uint    vendorspec2;
	char	reserved4[48];
69 70
	uint    hostver;	/* Host controller version register */
	char    reserved5[4];	/* reserved */
71
	uint    dmaerraddr;	/* DMA error address register */
72
	char    reserved6[4];	/* reserved */
73 74
	uint    dmaerrattr;	/* DMA error attribute register */
	char    reserved7[4];	/* reserved */
75
	uint    hostcapblt2;	/* Host controller capabilities register 2 */
76
	char    reserved8[8];	/* reserved */
77
	uint    tcr;		/* Tuning control register */
78
	char    reserved9[28];	/* reserved */
79
	uint    sddirctl;	/* SD direction control register */
80
	char    reserved10[712];/* reserved */
81
	uint    scr;		/* eSDHC control register */
82 83
};

P
Peng Fan 已提交
84 85 86 87 88 89 90 91 92 93 94
/**
 * struct fsl_esdhc_priv
 *
 * @esdhc_regs: registers of the sdhc controller
 * @sdhc_clk: Current clk of the sdhc controller
 * @bus_width: bus width, 1bit, 4bit or 8bit
 * @cfg: mmc config
 * @mmc: mmc
 * Following is used when Driver Model is enabled for MMC
 * @dev: pointer for the device
 * @non_removable: 0: removable; 1: non-removable
P
Peng Fan 已提交
95
 * @wp_enable: 1: enable checking wp; 0: no check
96
 * @vs18_enable: 1: use 1.8V voltage; 0: use 3.3V
P
Peng Fan 已提交
97
 * @cd_gpio: gpio for card detection
P
Peng Fan 已提交
98
 * @wp_gpio: gpio for write protection
P
Peng Fan 已提交
99 100 101 102 103 104 105 106 107
 */
struct fsl_esdhc_priv {
	struct fsl_esdhc *esdhc_regs;
	unsigned int sdhc_clk;
	unsigned int bus_width;
	struct mmc_config cfg;
	struct mmc *mmc;
	struct udevice *dev;
	int non_removable;
P
Peng Fan 已提交
108
	int wp_enable;
109
	int vs18_enable;
110
#ifdef CONFIG_DM_GPIO
P
Peng Fan 已提交
111
	struct gpio_desc cd_gpio;
P
Peng Fan 已提交
112
	struct gpio_desc wp_gpio;
113
#endif
P
Peng Fan 已提交
114 115
};

116
/* Return the XFERTYP flags for a given command and data packet */
117
static uint esdhc_xfertyp(struct mmc_cmd *cmd, struct mmc_data *data)
118 119 120 121
{
	uint xfertyp = 0;

	if (data) {
122 123 124 125
		xfertyp |= XFERTYP_DPSEL;
#ifndef CONFIG_SYS_FSL_ESDHC_USE_PIO
		xfertyp |= XFERTYP_DMAEN;
#endif
126 127 128
		if (data->blocks > 1) {
			xfertyp |= XFERTYP_MSBSEL;
			xfertyp |= XFERTYP_BCEN;
129 130 131
#ifdef CONFIG_SYS_FSL_ERRATUM_ESDHC111
			xfertyp |= XFERTYP_AC12EN;
#endif
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
		}

		if (data->flags & MMC_DATA_READ)
			xfertyp |= XFERTYP_DTDSEL;
	}

	if (cmd->resp_type & MMC_RSP_CRC)
		xfertyp |= XFERTYP_CCCEN;
	if (cmd->resp_type & MMC_RSP_OPCODE)
		xfertyp |= XFERTYP_CICEN;
	if (cmd->resp_type & MMC_RSP_136)
		xfertyp |= XFERTYP_RSPTYP_136;
	else if (cmd->resp_type & MMC_RSP_BUSY)
		xfertyp |= XFERTYP_RSPTYP_48_BUSY;
	else if (cmd->resp_type & MMC_RSP_PRESENT)
		xfertyp |= XFERTYP_RSPTYP_48;

149 150
	if (cmd->cmdidx == MMC_CMD_STOP_TRANSMISSION)
		xfertyp |= XFERTYP_CMDTYP_ABORT;
151

152 153 154
	return XFERTYP_CMD(cmd->cmdidx) | xfertyp;
}

155 156 157 158
#ifdef CONFIG_SYS_FSL_ESDHC_USE_PIO
/*
 * PIO Read/Write Mode reduce the performace as DMA is not used in this mode.
 */
159 160
static void esdhc_pio_read_write(struct fsl_esdhc_priv *priv,
				 struct mmc_data *data)
161
{
P
Peng Fan 已提交
162
	struct fsl_esdhc *regs = priv->esdhc_regs;
163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180
	uint blocks;
	char *buffer;
	uint databuf;
	uint size;
	uint irqstat;
	uint timeout;

	if (data->flags & MMC_DATA_READ) {
		blocks = data->blocks;
		buffer = data->dest;
		while (blocks) {
			timeout = PIO_TIMEOUT;
			size = data->blocksize;
			irqstat = esdhc_read32(&regs->irqstat);
			while (!(esdhc_read32(&regs->prsstat) & PRSSTAT_BREN)
				&& --timeout);
			if (timeout <= 0) {
				printf("\nData Read Failed in PIO Mode.");
181
				return;
182 183 184 185 186 187 188 189 190 191 192 193 194
			}
			while (size && (!(irqstat & IRQSTAT_TC))) {
				udelay(100); /* Wait before last byte transfer complete */
				irqstat = esdhc_read32(&regs->irqstat);
				databuf = in_le32(&regs->datport);
				*((uint *)buffer) = databuf;
				buffer += 4;
				size -= 4;
			}
			blocks--;
		}
	} else {
		blocks = data->blocks;
195
		buffer = (char *)data->src;
196 197 198 199 200 201 202 203
		while (blocks) {
			timeout = PIO_TIMEOUT;
			size = data->blocksize;
			irqstat = esdhc_read32(&regs->irqstat);
			while (!(esdhc_read32(&regs->prsstat) & PRSSTAT_BWEN)
				&& --timeout);
			if (timeout <= 0) {
				printf("\nData Write Failed in PIO Mode.");
204
				return;
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
			}
			while (size && (!(irqstat & IRQSTAT_TC))) {
				udelay(100); /* Wait before last byte transfer complete */
				databuf = *((uint *)buffer);
				buffer += 4;
				size -= 4;
				irqstat = esdhc_read32(&regs->irqstat);
				out_le32(&regs->datport, databuf);
			}
			blocks--;
		}
	}
}
#endif

220 221
static int esdhc_setup_data(struct fsl_esdhc_priv *priv, struct mmc *mmc,
			    struct mmc_data *data)
222 223
{
	int timeout;
P
Peng Fan 已提交
224
	struct fsl_esdhc *regs = priv->esdhc_regs;
225
#if defined(CONFIG_FSL_LAYERSCAPE) || defined(CONFIG_S32V234)
226 227
	dma_addr_t addr;
#endif
228
	uint wml_value;
229 230 231 232

	wml_value = data->blocksize/4;

	if (data->flags & MMC_DATA_READ) {
233 234
		if (wml_value > WML_RD_WML_MAX)
			wml_value = WML_RD_WML_MAX_VAL;
235

236
		esdhc_clrsetbits32(&regs->wml, WML_RD_WML_MASK, wml_value);
237
#ifndef CONFIG_SYS_FSL_ESDHC_USE_PIO
238
#if defined(CONFIG_FSL_LAYERSCAPE) || defined(CONFIG_S32V234)
239 240 241 242 243 244
		addr = virt_to_phys((void *)(data->dest));
		if (upper_32_bits(addr))
			printf("Error found for upper 32 bits\n");
		else
			esdhc_write32(&regs->dsaddr, lower_32_bits(addr));
#else
245
		esdhc_write32(&regs->dsaddr, (u32)data->dest);
246
#endif
247
#endif
248
	} else {
249
#ifndef CONFIG_SYS_FSL_ESDHC_USE_PIO
250 251 252
		flush_dcache_range((ulong)data->src,
				   (ulong)data->src+data->blocks
					 *data->blocksize);
253
#endif
254 255
		if (wml_value > WML_WR_WML_MAX)
			wml_value = WML_WR_WML_MAX_VAL;
P
Peng Fan 已提交
256 257 258 259
		if (priv->wp_enable) {
			if ((esdhc_read32(&regs->prsstat) &
			    PRSSTAT_WPSPL) == 0) {
				printf("\nThe SD card is locked. Can not write to a locked card.\n\n");
260
				return -ETIMEDOUT;
P
Peng Fan 已提交
261
			}
262
		}
263 264 265

		esdhc_clrsetbits32(&regs->wml, WML_WR_WML_MASK,
					wml_value << 16);
266
#ifndef CONFIG_SYS_FSL_ESDHC_USE_PIO
267
#if defined(CONFIG_FSL_LAYERSCAPE) || defined(CONFIG_S32V234)
268 269 270 271 272 273
		addr = virt_to_phys((void *)(data->src));
		if (upper_32_bits(addr))
			printf("Error found for upper 32 bits\n");
		else
			esdhc_write32(&regs->dsaddr, lower_32_bits(addr));
#else
274
		esdhc_write32(&regs->dsaddr, (u32)data->src);
275
#endif
276
#endif
277 278
	}

279
	esdhc_write32(&regs->blkattr, data->blocks << 16 | data->blocksize);
280 281

	/* Calculate the timeout period for data transactions */
282 283 284 285 286
	/*
	 * 1)Timeout period = (2^(timeout+13)) SD Clock cycles
	 * 2)Timeout period should be minimum 0.250sec as per SD Card spec
	 *  So, Number of SD Clock cycles for 0.25sec should be minimum
	 *		(SD Clock/sec * 0.25 sec) SD Clock cycles
287
	 *		= (mmc->clock * 1/4) SD Clock cycles
288
	 * As 1) >=  2)
289
	 * => (2^(timeout+13)) >= mmc->clock * 1/4
290
	 * Taking log2 both the sides
291
	 * => timeout + 13 >= log2(mmc->clock/4)
292
	 * Rounding up to next power of 2
293 294
	 * => timeout + 13 = log2(mmc->clock/4) + 1
	 * => timeout + 13 = fls(mmc->clock/4)
295 296 297 298 299 300 301
	 *
	 * However, the MMC spec "It is strongly recommended for hosts to
	 * implement more than 500ms timeout value even if the card
	 * indicates the 250ms maximum busy length."  Even the previous
	 * value of 300ms is known to be insufficient for some cards.
	 * So, we use
	 * => timeout + 13 = fls(mmc->clock/2)
302
	 */
303
	timeout = fls(mmc->clock/2);
304 305 306 307 308 309 310 311
	timeout -= 13;

	if (timeout > 14)
		timeout = 14;

	if (timeout < 0)
		timeout = 0;

312 313 314 315 316
#ifdef CONFIG_SYS_FSL_ERRATUM_ESDHC_A001
	if ((timeout == 4) || (timeout == 8) || (timeout == 12))
		timeout++;
#endif

317 318 319
#ifdef ESDHCI_QUIRK_BROKEN_TIMEOUT_VALUE
	timeout = 0xE;
#endif
320
	esdhc_clrsetbits32(&regs->sysctl, SYSCTL_TIMEOUT_MASK, timeout << 16);
321 322 323 324

	return 0;
}

325 326 327
static void check_and_invalidate_dcache_range
	(struct mmc_cmd *cmd,
	 struct mmc_data *data) {
328
	unsigned start = 0;
329
	unsigned end = 0;
330 331
	unsigned size = roundup(ARCH_DMA_MINALIGN,
				data->blocks*data->blocksize);
332
#if defined(CONFIG_FSL_LAYERSCAPE) || defined(CONFIG_S32V234)
333 334 335 336 337 338 339
	dma_addr_t addr;

	addr = virt_to_phys((void *)(data->dest));
	if (upper_32_bits(addr))
		printf("Error found for upper 32 bits\n");
	else
		start = lower_32_bits(addr);
340 341
#else
	start = (unsigned)data->dest;
342
#endif
343
	end = start + size;
344 345
	invalidate_dcache_range(start, end);
}
346

347 348 349 350
/*
 * Sends a command out on the bus.  Takes the mmc pointer,
 * a command pointer, and an optional data pointer.
 */
351 352
static int esdhc_send_cmd_common(struct fsl_esdhc_priv *priv, struct mmc *mmc,
				 struct mmc_cmd *cmd, struct mmc_data *data)
353
{
354
	int	err = 0;
355 356
	uint	xfertyp;
	uint	irqstat;
P
Peng Fan 已提交
357
	struct fsl_esdhc *regs = priv->esdhc_regs;
358

359 360 361 362 363
#ifdef CONFIG_SYS_FSL_ERRATUM_ESDHC111
	if (cmd->cmdidx == MMC_CMD_STOP_TRANSMISSION)
		return 0;
#endif

364
	esdhc_write32(&regs->irqstat, -1);
365 366 367 368

	sync();

	/* Wait for the bus to be idle */
369 370 371
	while ((esdhc_read32(&regs->prsstat) & PRSSTAT_CICHB) ||
			(esdhc_read32(&regs->prsstat) & PRSSTAT_CIDHB))
		;
372

373 374
	while (esdhc_read32(&regs->prsstat) & PRSSTAT_DLA)
		;
375 376 377 378 379 380 381 382 383 384

	/* Wait at least 8 SD clock cycles before the next command */
	/*
	 * Note: This is way more than 8 cycles, but 1ms seems to
	 * resolve timing issues with some cards
	 */
	udelay(1000);

	/* Set up for a data transfer if we have one */
	if (data) {
385
		err = esdhc_setup_data(priv, mmc, data);
386 387
		if(err)
			return err;
388 389 390

		if (data->flags & MMC_DATA_READ)
			check_and_invalidate_dcache_range(cmd, data);
391 392 393 394 395
	}

	/* Figure out the transfer arguments */
	xfertyp = esdhc_xfertyp(cmd, data);

396 397 398
	/* Mask all irqs */
	esdhc_write32(&regs->irqsigen, 0);

399
	/* Send the command */
400
	esdhc_write32(&regs->cmdarg, cmd->cmdarg);
401 402
#if defined(CONFIG_FSL_USDHC)
	esdhc_write32(&regs->mixctrl,
403 404
	(esdhc_read32(&regs->mixctrl) & 0xFFFFFF80) | (xfertyp & 0x7F)
			| (mmc->ddr_mode ? XFERTYP_DDREN : 0));
405 406
	esdhc_write32(&regs->xfertyp, xfertyp & 0xFFFF0000);
#else
407
	esdhc_write32(&regs->xfertyp, xfertyp);
408
#endif
409

410
	/* Wait for the command to complete */
411
	while (!(esdhc_read32(&regs->irqstat) & (IRQSTAT_CC | IRQSTAT_CTOE)))
412
		;
413

414
	irqstat = esdhc_read32(&regs->irqstat);
415

416
	if (irqstat & CMD_ERR) {
417
		err = -ECOMM;
418
		goto out;
419 420
	}

421
	if (irqstat & IRQSTAT_CTOE) {
422
		err = -ETIMEDOUT;
423 424
		goto out;
	}
425

426 427 428 429 430 431 432 433 434
	/* Switch voltage to 1.8V if CMD11 succeeded */
	if (cmd->cmdidx == SD_CMD_SWITCH_UHS18V) {
		esdhc_setbits32(&regs->vendorspec, ESDHC_VENDORSPEC_VSELECT);

		printf("Run CMD11 1.8V switch\n");
		/* Sleep for 5 ms - max time for card to switch to 1.8V */
		udelay(5000);
	}

435 436
	/* Workaround for ESDHC errata ENGcm03648 */
	if (!data && (cmd->resp_type & MMC_RSP_BUSY)) {
437
		int timeout = 6000;
438

439
		/* Poll on DATA0 line for cmd with busy signal for 600 ms */
440 441 442 443 444 445 446 447
		while (timeout > 0 && !(esdhc_read32(&regs->prsstat) &
					PRSSTAT_DAT0)) {
			udelay(100);
			timeout--;
		}

		if (timeout <= 0) {
			printf("Timeout waiting for DAT0 to go high!\n");
448
			err = -ETIMEDOUT;
449
			goto out;
450 451 452
		}
	}

453 454 455 456
	/* Copy the response to the response buffer */
	if (cmd->resp_type & MMC_RSP_136) {
		u32 cmdrsp3, cmdrsp2, cmdrsp1, cmdrsp0;

457 458 459 460
		cmdrsp3 = esdhc_read32(&regs->cmdrsp3);
		cmdrsp2 = esdhc_read32(&regs->cmdrsp2);
		cmdrsp1 = esdhc_read32(&regs->cmdrsp1);
		cmdrsp0 = esdhc_read32(&regs->cmdrsp0);
R
Rabin Vincent 已提交
461 462 463 464
		cmd->response[0] = (cmdrsp3 << 8) | (cmdrsp2 >> 24);
		cmd->response[1] = (cmdrsp2 << 8) | (cmdrsp1 >> 24);
		cmd->response[2] = (cmdrsp1 << 8) | (cmdrsp0 >> 24);
		cmd->response[3] = (cmdrsp0 << 8);
465
	} else
466
		cmd->response[0] = esdhc_read32(&regs->cmdrsp0);
467 468 469

	/* Wait until all of the blocks are transferred */
	if (data) {
470
#ifdef CONFIG_SYS_FSL_ESDHC_USE_PIO
471
		esdhc_pio_read_write(priv, data);
472
#else
473
		do {
474
			irqstat = esdhc_read32(&regs->irqstat);
475

476
			if (irqstat & IRQSTAT_DTOE) {
477
				err = -ETIMEDOUT;
478 479
				goto out;
			}
480

481
			if (irqstat & DATA_ERR) {
482
				err = -ECOMM;
483 484
				goto out;
			}
485
		} while ((irqstat & DATA_COMPLETE) != DATA_COMPLETE);
486

487 488 489 490 491
		/*
		 * Need invalidate the dcache here again to avoid any
		 * cache-fill during the DMA operations such as the
		 * speculative pre-fetching etc.
		 */
492 493
		if (data->flags & MMC_DATA_READ)
			check_and_invalidate_dcache_range(cmd, data);
494
#endif
495 496
	}

497 498 499 500 501 502 503 504 505 506 507 508 509 510 511
out:
	/* Reset CMD and DATA portions on error */
	if (err) {
		esdhc_write32(&regs->sysctl, esdhc_read32(&regs->sysctl) |
			      SYSCTL_RSTC);
		while (esdhc_read32(&regs->sysctl) & SYSCTL_RSTC)
			;

		if (data) {
			esdhc_write32(&regs->sysctl,
				      esdhc_read32(&regs->sysctl) |
				      SYSCTL_RSTD);
			while ((esdhc_read32(&regs->sysctl) & SYSCTL_RSTD))
				;
		}
512 513 514 515

		/* If this was CMD11, then notify that power cycle is needed */
		if (cmd->cmdidx == SD_CMD_SWITCH_UHS18V)
			printf("CMD11 to switch to 1.8V mode failed, card requires power cycle.\n");
516 517
	}

518
	esdhc_write32(&regs->irqstat, -1);
519

520
	return err;
521 522
}

523
static void set_sysctl(struct fsl_esdhc_priv *priv, struct mmc *mmc, uint clock)
524
{
525 526 527 528 529 530 531
	int div = 1;
#ifdef ARCH_MXC
	int pre_div = 1;
#else
	int pre_div = 2;
#endif
	int ddr_pre_div = mmc->ddr_mode ? 2 : 1;
P
Peng Fan 已提交
532 533
	struct fsl_esdhc *regs = priv->esdhc_regs;
	int sdhc_clk = priv->sdhc_clk;
534 535
	uint clk;

536 537
	if (clock < mmc->cfg->f_min)
		clock = mmc->cfg->f_min;
538

539 540
	while (sdhc_clk / (16 * pre_div * ddr_pre_div) > clock && pre_div < 256)
		pre_div *= 2;
541

542 543
	while (sdhc_clk / (div * pre_div * ddr_pre_div) > clock && div < 16)
		div++;
544

545
	pre_div >>= 1;
546 547 548 549
	div -= 1;

	clk = (pre_div << 8) | (div << 4);

550
#ifdef CONFIG_FSL_USDHC
551
	esdhc_clrbits32(&regs->vendorspec, VENDORSPEC_CKEN);
552
#else
553
	esdhc_clrbits32(&regs->sysctl, SYSCTL_CKEN);
554
#endif
555 556

	esdhc_clrsetbits32(&regs->sysctl, SYSCTL_CLOCK_MASK, clk);
557 558 559

	udelay(10000);

560
#ifdef CONFIG_FSL_USDHC
561
	esdhc_setbits32(&regs->vendorspec, VENDORSPEC_PEREN | VENDORSPEC_CKEN);
562 563 564
#else
	esdhc_setbits32(&regs->sysctl, SYSCTL_PEREN | SYSCTL_CKEN);
#endif
565

566 567
}

568
#ifdef CONFIG_FSL_ESDHC_USE_PERIPHERAL_CLK
569
static void esdhc_clock_control(struct fsl_esdhc_priv *priv, bool enable)
570
{
P
Peng Fan 已提交
571
	struct fsl_esdhc *regs = priv->esdhc_regs;
572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596
	u32 value;
	u32 time_out;

	value = esdhc_read32(&regs->sysctl);

	if (enable)
		value |= SYSCTL_CKEN;
	else
		value &= ~SYSCTL_CKEN;

	esdhc_write32(&regs->sysctl, value);

	time_out = 20;
	value = PRSSTAT_SDSTB;
	while (!(esdhc_read32(&regs->prsstat) & value)) {
		if (time_out == 0) {
			printf("fsl_esdhc: Internal clock never stabilised.\n");
			break;
		}
		time_out--;
		mdelay(1);
	}
}
#endif

597
static int esdhc_set_ios_common(struct fsl_esdhc_priv *priv, struct mmc *mmc)
598
{
P
Peng Fan 已提交
599
	struct fsl_esdhc *regs = priv->esdhc_regs;
600

601 602
#ifdef CONFIG_FSL_ESDHC_USE_PERIPHERAL_CLK
	/* Select to use peripheral clock */
603
	esdhc_clock_control(priv, false);
604
	esdhc_setbits32(&regs->scr, ESDHCCTL_PCS);
605
	esdhc_clock_control(priv, true);
606
#endif
607
	/* Set the clock speed */
608
	set_sysctl(priv, mmc, mmc->clock);
609 610

	/* Set the bus width */
611
	esdhc_clrbits32(&regs->proctl, PROCTL_DTW_4 | PROCTL_DTW_8);
612 613

	if (mmc->bus_width == 4)
614
		esdhc_setbits32(&regs->proctl, PROCTL_DTW_4);
615
	else if (mmc->bus_width == 8)
616 617
		esdhc_setbits32(&regs->proctl, PROCTL_DTW_8);

618
	return 0;
619 620
}

621
static int esdhc_init_common(struct fsl_esdhc_priv *priv, struct mmc *mmc)
622
{
P
Peng Fan 已提交
623
	struct fsl_esdhc *regs = priv->esdhc_regs;
624 625
	int timeout = 1000;

626
	/* Reset the entire host controller */
627
	esdhc_setbits32(&regs->sysctl, SYSCTL_RSTA);
628 629 630 631

	/* Wait until the controller is available */
	while ((esdhc_read32(&regs->sysctl) & SYSCTL_RSTA) && --timeout)
		udelay(1000);
632

633 634 635 636 637 638 639 640 641 642 643 644 645 646
#if defined(CONFIG_FSL_USDHC)
	/* RSTA doesn't reset MMC_BOOT register, so manually reset it */
	esdhc_write32(&regs->mmcboot, 0x0);
	/* Reset MIX_CTRL and CLK_TUNE_CTRL_STATUS regs to 0 */
	esdhc_write32(&regs->mixctrl, 0x0);
	esdhc_write32(&regs->clktunectrlstatus, 0x0);

	/* Put VEND_SPEC to default value */
	esdhc_write32(&regs->vendorspec, VENDORSPEC_INIT);

	/* Disable DLL_CTRL delay line */
	esdhc_write32(&regs->dllctrl, 0x0);
#endif

647
#ifndef ARCH_MXC
648
	/* Enable cache snooping */
649 650
	esdhc_write32(&regs->scr, 0x00000040);
#endif
651

652
#ifndef CONFIG_FSL_USDHC
653
	esdhc_setbits32(&regs->sysctl, SYSCTL_HCKEN | SYSCTL_IPGEN);
654 655
#else
	esdhc_setbits32(&regs->vendorspec, VENDORSPEC_HCKEN | VENDORSPEC_IPGEN);
656
#endif
657 658

	/* Set the initial clock speed */
659
	mmc_set_clock(mmc, 400000);
660 661

	/* Disable the BRR and BWR bits in IRQSTAT */
662
	esdhc_clrbits32(&regs->irqstaten, IRQSTATEN_BRR | IRQSTATEN_BWR);
663 664

	/* Put the PROCTL reg back to the default */
665
	esdhc_write32(&regs->proctl, PROCTL_INIT);
666

667 668
	/* Set timout to the maximum value */
	esdhc_clrsetbits32(&regs->sysctl, SYSCTL_TIMEOUT_MASK, 14 << 16);
669

670 671 672
	if (priv->vs18_enable)
		esdhc_setbits32(&regs->vendorspec, ESDHC_VENDORSPEC_VSELECT);

673 674
	return 0;
}
675

676
static int esdhc_getcd_common(struct fsl_esdhc_priv *priv)
677
{
P
Peng Fan 已提交
678
	struct fsl_esdhc *regs = priv->esdhc_regs;
679 680
	int timeout = 1000;

681 682 683 684
#ifdef CONFIG_ESDHC_DETECT_QUIRK
	if (CONFIG_ESDHC_DETECT_QUIRK)
		return 1;
#endif
P
Peng Fan 已提交
685 686 687 688

#ifdef CONFIG_DM_MMC
	if (priv->non_removable)
		return 1;
689
#ifdef CONFIG_DM_GPIO
P
Peng Fan 已提交
690 691
	if (dm_gpio_is_valid(&priv->cd_gpio))
		return dm_gpio_get_value(&priv->cd_gpio);
692
#endif
P
Peng Fan 已提交
693 694
#endif

695 696
	while (!(esdhc_read32(&regs->prsstat) & PRSSTAT_CINS) && --timeout)
		udelay(1000);
697

698
	return timeout > 0;
699 700
}

701
static int esdhc_reset(struct fsl_esdhc *regs)
702
{
703
	ulong start;
704 705

	/* reset the controller */
706
	esdhc_setbits32(&regs->sysctl, SYSCTL_RSTA);
707 708

	/* hardware clears the bit when it is done */
709 710 711 712 713 714 715 716 717
	start = get_timer(0);
	while ((esdhc_read32(&regs->sysctl) & SYSCTL_RSTA)) {
		if (get_timer(start) > 100) {
			printf("MMC/SD: Reset never completed.\n");
			return -ETIMEDOUT;
		}
	}

	return 0;
718 719
}

720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748
static int esdhc_getcd(struct mmc *mmc)
{
	struct fsl_esdhc_priv *priv = mmc->priv;

	return esdhc_getcd_common(priv);
}

static int esdhc_init(struct mmc *mmc)
{
	struct fsl_esdhc_priv *priv = mmc->priv;

	return esdhc_init_common(priv, mmc);
}

static int esdhc_send_cmd(struct mmc *mmc, struct mmc_cmd *cmd,
			  struct mmc_data *data)
{
	struct fsl_esdhc_priv *priv = mmc->priv;

	return esdhc_send_cmd_common(priv, mmc, cmd, data);
}

static int esdhc_set_ios(struct mmc *mmc)
{
	struct fsl_esdhc_priv *priv = mmc->priv;

	return esdhc_set_ios_common(priv, mmc);
}

749
static const struct mmc_ops esdhc_ops = {
750 751
	.getcd		= esdhc_getcd,
	.init		= esdhc_init,
752 753 754 755
	.send_cmd	= esdhc_send_cmd,
	.set_ios	= esdhc_set_ios,
};

P
Peng Fan 已提交
756
static int fsl_esdhc_init(struct fsl_esdhc_priv *priv)
757
{
758
	struct fsl_esdhc *regs;
759
	struct mmc *mmc;
760
	u32 caps, voltage_caps;
761
	int ret;
762

P
Peng Fan 已提交
763 764
	if (!priv)
		return -EINVAL;
765

P
Peng Fan 已提交
766
	regs = priv->esdhc_regs;
767

768
	/* First reset the eSDHC controller */
769 770 771
	ret = esdhc_reset(regs);
	if (ret)
		return ret;
772

773
#ifndef CONFIG_FSL_USDHC
774 775
	esdhc_setbits32(&regs->sysctl, SYSCTL_PEREN | SYSCTL_HCKEN
				| SYSCTL_IPGEN | SYSCTL_CKEN);
776 777 778
#else
	esdhc_setbits32(&regs->vendorspec, VENDORSPEC_PEREN |
			VENDORSPEC_HCKEN | VENDORSPEC_IPGEN | VENDORSPEC_CKEN);
779
#endif
780

781 782 783
	if (priv->vs18_enable)
		esdhc_setbits32(&regs->vendorspec, ESDHC_VENDORSPEC_VSELECT);

784
	writel(SDHCI_IRQ_EN_BITS, &regs->irqstaten);
P
Peng Fan 已提交
785
	memset(&priv->cfg, 0, sizeof(priv->cfg));
786

787
	voltage_caps = 0;
788
	caps = esdhc_read32(&regs->hostcapblt);
789 790 791 792 793

#ifdef CONFIG_SYS_FSL_ERRATUM_ESDHC135
	caps = caps & ~(ESDHC_HOSTCAPBLT_SRS |
			ESDHC_HOSTCAPBLT_VS18 | ESDHC_HOSTCAPBLT_VS30);
#endif
794 795 796 797 798 799

/* T4240 host controller capabilities register should have VS33 bit */
#ifdef CONFIG_SYS_FSL_MMC_HAS_CAPBLT_VS33
	caps = caps | ESDHC_HOSTCAPBLT_VS33;
#endif

800
	if (caps & ESDHC_HOSTCAPBLT_VS18)
801
		voltage_caps |= MMC_VDD_165_195;
802
	if (caps & ESDHC_HOSTCAPBLT_VS30)
803
		voltage_caps |= MMC_VDD_29_30 | MMC_VDD_30_31;
804
	if (caps & ESDHC_HOSTCAPBLT_VS33)
805 806
		voltage_caps |= MMC_VDD_32_33 | MMC_VDD_33_34;

P
Peng Fan 已提交
807 808
	priv->cfg.name = "FSL_SDHC";
	priv->cfg.ops = &esdhc_ops;
809
#ifdef CONFIG_SYS_SD_VOLTAGE
P
Peng Fan 已提交
810
	priv->cfg.voltages = CONFIG_SYS_SD_VOLTAGE;
811
#else
P
Peng Fan 已提交
812
	priv->cfg.voltages = MMC_VDD_32_33 | MMC_VDD_33_34;
813
#endif
P
Peng Fan 已提交
814
	if ((priv->cfg.voltages & voltage_caps) == 0) {
815 816 817
		printf("voltage not supported by controller\n");
		return -1;
	}
818

P
Peng Fan 已提交
819 820 821 822 823 824
	if (priv->bus_width == 8)
		priv->cfg.host_caps = MMC_MODE_4BIT | MMC_MODE_8BIT;
	else if (priv->bus_width == 4)
		priv->cfg.host_caps = MMC_MODE_4BIT;

	priv->cfg.host_caps = MMC_MODE_4BIT | MMC_MODE_8BIT;
825
#ifdef CONFIG_SYS_FSL_ESDHC_HAS_DDR_MODE
P
Peng Fan 已提交
826
	priv->cfg.host_caps |= MMC_MODE_DDR_52MHz;
827
#endif
828

P
Peng Fan 已提交
829 830 831 832 833
	if (priv->bus_width > 0) {
		if (priv->bus_width < 8)
			priv->cfg.host_caps &= ~MMC_MODE_8BIT;
		if (priv->bus_width < 4)
			priv->cfg.host_caps &= ~MMC_MODE_4BIT;
834 835
	}

836
	if (caps & ESDHC_HOSTCAPBLT_HSS)
P
Peng Fan 已提交
837
		priv->cfg.host_caps |= MMC_MODE_HS_52MHz | MMC_MODE_HS;
838

839 840
#ifdef CONFIG_ESDHC_DETECT_8_BIT_QUIRK
	if (CONFIG_ESDHC_DETECT_8_BIT_QUIRK)
P
Peng Fan 已提交
841
		priv->cfg.host_caps &= ~MMC_MODE_8BIT;
842 843
#endif

P
Peng Fan 已提交
844 845
	priv->cfg.f_min = 400000;
	priv->cfg.f_max = min(priv->sdhc_clk, (u32)52000000);
846

P
Peng Fan 已提交
847
	priv->cfg.b_max = CONFIG_SYS_MMC_MAX_BLK_COUNT;
848

P
Peng Fan 已提交
849
	mmc = mmc_create(&priv->cfg, priv);
850 851
	if (mmc == NULL)
		return -1;
852

P
Peng Fan 已提交
853 854 855 856 857
	priv->mmc = mmc;

	return 0;
}

858 859 860 861 862 863 864 865 866 867 868
#ifndef CONFIG_DM_MMC
static int fsl_esdhc_cfg_to_priv(struct fsl_esdhc_cfg *cfg,
				 struct fsl_esdhc_priv *priv)
{
	if (!cfg || !priv)
		return -EINVAL;

	priv->esdhc_regs = (struct fsl_esdhc *)(unsigned long)(cfg->esdhc_base);
	priv->bus_width = cfg->max_bus_width;
	priv->sdhc_clk = cfg->sdhc_clk;
	priv->wp_enable  = cfg->wp_enable;
869
	priv->vs18_enable  = cfg->vs18_enable;
870 871 872 873

	return 0;
};

P
Peng Fan 已提交
874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899
int fsl_esdhc_initialize(bd_t *bis, struct fsl_esdhc_cfg *cfg)
{
	struct fsl_esdhc_priv *priv;
	int ret;

	if (!cfg)
		return -EINVAL;

	priv = calloc(sizeof(struct fsl_esdhc_priv), 1);
	if (!priv)
		return -ENOMEM;

	ret = fsl_esdhc_cfg_to_priv(cfg, priv);
	if (ret) {
		debug("%s xlate failure\n", __func__);
		free(priv);
		return ret;
	}

	ret = fsl_esdhc_init(priv);
	if (ret) {
		debug("%s init failure\n", __func__);
		free(priv);
		return ret;
	}

900 901 902 903 904
	return 0;
}

int fsl_esdhc_mmc_init(bd_t *bis)
{
905 906
	struct fsl_esdhc_cfg *cfg;

F
Fabio Estevam 已提交
907
	cfg = calloc(sizeof(struct fsl_esdhc_cfg), 1);
908
	cfg->esdhc_base = CONFIG_SYS_FSL_ESDHC_ADDR;
909
	cfg->sdhc_clk = gd->arch.sdhc_clk;
910
	return fsl_esdhc_initialize(bis, cfg);
911
}
912
#endif
913

914 915 916 917 918 919 920 921 922 923 924
#ifdef CONFIG_FSL_ESDHC_ADAPTER_IDENT
void mmc_adapter_card_type_ident(void)
{
	u8 card_id;
	u8 value;

	card_id = QIXIS_READ(present) & QIXIS_SDID_MASK;
	gd->arch.sdhc_adapter = card_id;

	switch (card_id) {
	case QIXIS_ESDHC_ADAPTER_TYPE_EMMC45:
925 926 927
		value = QIXIS_READ(brdcfg[5]);
		value |= (QIXIS_DAT4 | QIXIS_DAT5_6_7);
		QIXIS_WRITE(brdcfg[5], value);
928 929
		break;
	case QIXIS_ESDHC_ADAPTER_TYPE_SDMMC_LEGACY:
930 931 932
		value = QIXIS_READ(pwr_ctl[1]);
		value |= QIXIS_EVDD_BY_SDHC_VS;
		QIXIS_WRITE(pwr_ctl[1], value);
933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952
		break;
	case QIXIS_ESDHC_ADAPTER_TYPE_EMMC44:
		value = QIXIS_READ(brdcfg[5]);
		value |= (QIXIS_SDCLKIN | QIXIS_SDCLKOUT);
		QIXIS_WRITE(brdcfg[5], value);
		break;
	case QIXIS_ESDHC_ADAPTER_TYPE_RSV:
		break;
	case QIXIS_ESDHC_ADAPTER_TYPE_MMC:
		break;
	case QIXIS_ESDHC_ADAPTER_TYPE_SD:
		break;
	case QIXIS_ESDHC_NO_ADAPTER:
		break;
	default:
		break;
	}
}
#endif

953
#ifdef CONFIG_OF_LIBFDT
954
__weak int esdhc_status_fixup(void *blob, const char *compat)
955
{
956
#ifdef CONFIG_FSL_ESDHC_PIN_MUX
957
	if (!hwconfig("esdhc")) {
958
		do_fixup_by_compat(blob, compat, "status", "disabled",
959 960
				sizeof("disabled"), 1);
		return 1;
961
	}
962
#endif
963 964 965 966 967 968 969 970 971
	return 0;
}

void fdt_fixup_esdhc(void *blob, bd_t *bd)
{
	const char *compat = "fsl,esdhc";

	if (esdhc_status_fixup(blob, compat))
		return;
972

973 974 975 976
#ifdef CONFIG_FSL_ESDHC_USE_PERIPHERAL_CLK
	do_fixup_by_compat_u32(blob, compat, "peripheral-frequency",
			       gd->arch.sdhc_clk, 1);
#else
977
	do_fixup_by_compat_u32(blob, compat, "clock-frequency",
978
			       gd->arch.sdhc_clk, 1);
979
#endif
980 981 982 983
#ifdef CONFIG_FSL_ESDHC_ADAPTER_IDENT
	do_fixup_by_compat_u32(blob, compat, "adapter-type",
			       (u32)(gd->arch.sdhc_adapter), 1);
#endif
984
}
985
#endif
P
Peng Fan 已提交
986 987 988

#ifdef CONFIG_DM_MMC
#include <asm/arch/clock.h>
P
Peng Fan 已提交
989 990 991 992
__weak void init_clk_usdhc(u32 index)
{
}

P
Peng Fan 已提交
993 994 995 996 997
static int fsl_esdhc_probe(struct udevice *dev)
{
	struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
	struct fsl_esdhc_priv *priv = dev_get_priv(dev);
	const void *fdt = gd->fdt_blob;
998
	int node = dev_of_offset(dev);
999
#ifdef CONFIG_DM_REGULATOR
1000
	struct udevice *vqmmc_dev;
1001
#endif
P
Peng Fan 已提交
1002 1003 1004 1005
	fdt_addr_t addr;
	unsigned int val;
	int ret;

S
Simon Glass 已提交
1006
	addr = devfdt_get_addr(dev);
P
Peng Fan 已提交
1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024
	if (addr == FDT_ADDR_T_NONE)
		return -EINVAL;

	priv->esdhc_regs = (struct fsl_esdhc *)addr;
	priv->dev = dev;

	val = fdtdec_get_int(fdt, node, "bus-width", -1);
	if (val == 8)
		priv->bus_width = 8;
	else if (val == 4)
		priv->bus_width = 4;
	else
		priv->bus_width = 1;

	if (fdt_get_property(fdt, node, "non-removable", NULL)) {
		priv->non_removable = 1;
	 } else {
		priv->non_removable = 0;
1025
#ifdef CONFIG_DM_GPIO
S
Simon Glass 已提交
1026 1027
		gpio_request_by_name_nodev(offset_to_ofnode(node), "cd-gpios",
					   0, &priv->cd_gpio, GPIOD_IS_IN);
1028
#endif
P
Peng Fan 已提交
1029 1030
	}

P
Peng Fan 已提交
1031 1032
	priv->wp_enable = 1;

1033
#ifdef CONFIG_DM_GPIO
S
Simon Glass 已提交
1034
	ret = gpio_request_by_name_nodev(offset_to_ofnode(node), "wp-gpios", 0,
P
Peng Fan 已提交
1035 1036 1037
					 &priv->wp_gpio, GPIOD_IS_IN);
	if (ret)
		priv->wp_enable = 0;
1038
#endif
1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061

	priv->vs18_enable = 0;

#ifdef CONFIG_DM_REGULATOR
	/*
	 * If emmc I/O has a fixed voltage at 1.8V, this must be provided,
	 * otherwise, emmc will work abnormally.
	 */
	ret = device_get_supply_regulator(dev, "vqmmc-supply", &vqmmc_dev);
	if (ret) {
		dev_dbg(dev, "no vqmmc-supply\n");
	} else {
		ret = regulator_set_enable(vqmmc_dev, true);
		if (ret) {
			dev_err(dev, "fail to enable vqmmc-supply\n");
			return ret;
		}

		if (regulator_get_value(vqmmc_dev) == 1800000)
			priv->vs18_enable = 1;
	}
#endif

P
Peng Fan 已提交
1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080
	/*
	 * TODO:
	 * Because lack of clk driver, if SDHC clk is not enabled,
	 * need to enable it first before this driver is invoked.
	 *
	 * we use MXC_ESDHC_CLK to get clk freq.
	 * If one would like to make this function work,
	 * the aliases should be provided in dts as this:
	 *
	 *  aliases {
	 *	mmc0 = &usdhc1;
	 *	mmc1 = &usdhc2;
	 *	mmc2 = &usdhc3;
	 *	mmc3 = &usdhc4;
	 *	};
	 * Then if your board only supports mmc2 and mmc3, but we can
	 * correctly get the seq as 2 and 3, then let mxc_get_clock
	 * work as expected.
	 */
P
Peng Fan 已提交
1081 1082 1083

	init_clk_usdhc(dev->seq);

P
Peng Fan 已提交
1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096
	priv->sdhc_clk = mxc_get_clock(MXC_ESDHC_CLK + dev->seq);
	if (priv->sdhc_clk <= 0) {
		dev_err(dev, "Unable to get clk for %s\n", dev->name);
		return -EINVAL;
	}

	ret = fsl_esdhc_init(priv);
	if (ret) {
		dev_err(dev, "fsl_esdhc_init failure\n");
		return ret;
	}

	upriv->mmc = priv->mmc;
P
Peng Fan 已提交
1097
	priv->mmc->dev = dev;
P
Peng Fan 已提交
1098 1099 1100 1101 1102 1103 1104 1105 1106 1107

	return 0;
}

static const struct udevice_id fsl_esdhc_ids[] = {
	{ .compatible = "fsl,imx6ul-usdhc", },
	{ .compatible = "fsl,imx6sx-usdhc", },
	{ .compatible = "fsl,imx6sl-usdhc", },
	{ .compatible = "fsl,imx6q-usdhc", },
	{ .compatible = "fsl,imx7d-usdhc", },
P
Peng Fan 已提交
1108
	{ .compatible = "fsl,imx7ulp-usdhc", },
1109
	{ .compatible = "fsl,esdhc", },
P
Peng Fan 已提交
1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120
	{ /* sentinel */ }
};

U_BOOT_DRIVER(fsl_esdhc) = {
	.name	= "fsl-esdhc-mmc",
	.id	= UCLASS_MMC,
	.of_match = fsl_esdhc_ids,
	.probe	= fsl_esdhc_probe,
	.priv_auto_alloc_size = sizeof(struct fsl_esdhc_priv),
};
#endif