xref: /aosp_15_r20/external/arm-trusted-firmware/drivers/imx/usdhc/imx_usdhc.c (revision 54fd6939e177f8ff529b10183254802c76df6d08)
1*54fd6939SJiyong Park /*
2*54fd6939SJiyong Park  * Copyright (c) 2018, ARM Limited and Contributors. All rights reserved.
3*54fd6939SJiyong Park  *
4*54fd6939SJiyong Park  * SPDX-License-Identifier: BSD-3-Clause
5*54fd6939SJiyong Park  */
6*54fd6939SJiyong Park 
7*54fd6939SJiyong Park #include <assert.h>
8*54fd6939SJiyong Park #include <errno.h>
9*54fd6939SJiyong Park #include <string.h>
10*54fd6939SJiyong Park 
11*54fd6939SJiyong Park #include <arch.h>
12*54fd6939SJiyong Park #include <arch_helpers.h>
13*54fd6939SJiyong Park #include <common/debug.h>
14*54fd6939SJiyong Park #include <drivers/delay_timer.h>
15*54fd6939SJiyong Park #include <drivers/mmc.h>
16*54fd6939SJiyong Park #include <lib/mmio.h>
17*54fd6939SJiyong Park 
18*54fd6939SJiyong Park #include <imx_usdhc.h>
19*54fd6939SJiyong Park 
20*54fd6939SJiyong Park static void imx_usdhc_initialize(void);
21*54fd6939SJiyong Park static int imx_usdhc_send_cmd(struct mmc_cmd *cmd);
22*54fd6939SJiyong Park static int imx_usdhc_set_ios(unsigned int clk, unsigned int width);
23*54fd6939SJiyong Park static int imx_usdhc_prepare(int lba, uintptr_t buf, size_t size);
24*54fd6939SJiyong Park static int imx_usdhc_read(int lba, uintptr_t buf, size_t size);
25*54fd6939SJiyong Park static int imx_usdhc_write(int lba, uintptr_t buf, size_t size);
26*54fd6939SJiyong Park 
27*54fd6939SJiyong Park static const struct mmc_ops imx_usdhc_ops = {
28*54fd6939SJiyong Park 	.init		= imx_usdhc_initialize,
29*54fd6939SJiyong Park 	.send_cmd	= imx_usdhc_send_cmd,
30*54fd6939SJiyong Park 	.set_ios	= imx_usdhc_set_ios,
31*54fd6939SJiyong Park 	.prepare	= imx_usdhc_prepare,
32*54fd6939SJiyong Park 	.read		= imx_usdhc_read,
33*54fd6939SJiyong Park 	.write		= imx_usdhc_write,
34*54fd6939SJiyong Park };
35*54fd6939SJiyong Park 
36*54fd6939SJiyong Park static imx_usdhc_params_t imx_usdhc_params;
37*54fd6939SJiyong Park 
38*54fd6939SJiyong Park #define IMX7_MMC_SRC_CLK_RATE (200 * 1000 * 1000)
imx_usdhc_set_clk(int clk)39*54fd6939SJiyong Park static void imx_usdhc_set_clk(int clk)
40*54fd6939SJiyong Park {
41*54fd6939SJiyong Park 	int div = 1;
42*54fd6939SJiyong Park 	int pre_div = 1;
43*54fd6939SJiyong Park 	unsigned int sdhc_clk = IMX7_MMC_SRC_CLK_RATE;
44*54fd6939SJiyong Park 	uintptr_t reg_base = imx_usdhc_params.reg_base;
45*54fd6939SJiyong Park 
46*54fd6939SJiyong Park 	assert(clk > 0);
47*54fd6939SJiyong Park 
48*54fd6939SJiyong Park 	while (sdhc_clk / (16 * pre_div) > clk && pre_div < 256)
49*54fd6939SJiyong Park 		pre_div *= 2;
50*54fd6939SJiyong Park 
51*54fd6939SJiyong Park 	while (sdhc_clk / div > clk && div < 16)
52*54fd6939SJiyong Park 		div++;
53*54fd6939SJiyong Park 
54*54fd6939SJiyong Park 	pre_div >>= 1;
55*54fd6939SJiyong Park 	div -= 1;
56*54fd6939SJiyong Park 	clk = (pre_div << 8) | (div << 4);
57*54fd6939SJiyong Park 
58*54fd6939SJiyong Park 	mmio_clrbits32(reg_base + VENDSPEC, VENDSPEC_CARD_CLKEN);
59*54fd6939SJiyong Park 	mmio_clrsetbits32(reg_base + SYSCTRL, SYSCTRL_CLOCK_MASK, clk);
60*54fd6939SJiyong Park 	udelay(10000);
61*54fd6939SJiyong Park 
62*54fd6939SJiyong Park 	mmio_setbits32(reg_base + VENDSPEC, VENDSPEC_PER_CLKEN | VENDSPEC_CARD_CLKEN);
63*54fd6939SJiyong Park }
64*54fd6939SJiyong Park 
imx_usdhc_initialize(void)65*54fd6939SJiyong Park static void imx_usdhc_initialize(void)
66*54fd6939SJiyong Park {
67*54fd6939SJiyong Park 	unsigned int timeout = 10000;
68*54fd6939SJiyong Park 	uintptr_t reg_base = imx_usdhc_params.reg_base;
69*54fd6939SJiyong Park 
70*54fd6939SJiyong Park 	assert((imx_usdhc_params.reg_base & MMC_BLOCK_MASK) == 0);
71*54fd6939SJiyong Park 
72*54fd6939SJiyong Park 	/* reset the controller */
73*54fd6939SJiyong Park 	mmio_setbits32(reg_base + SYSCTRL, SYSCTRL_RSTA);
74*54fd6939SJiyong Park 
75*54fd6939SJiyong Park 	/* wait for reset done */
76*54fd6939SJiyong Park 	while ((mmio_read_32(reg_base + SYSCTRL) & SYSCTRL_RSTA)) {
77*54fd6939SJiyong Park 		if (!timeout)
78*54fd6939SJiyong Park 			ERROR("IMX MMC reset timeout.\n");
79*54fd6939SJiyong Park 		timeout--;
80*54fd6939SJiyong Park 	}
81*54fd6939SJiyong Park 
82*54fd6939SJiyong Park 	mmio_write_32(reg_base + MMCBOOT, 0);
83*54fd6939SJiyong Park 	mmio_write_32(reg_base + MIXCTRL, 0);
84*54fd6939SJiyong Park 	mmio_write_32(reg_base + CLKTUNECTRLSTS, 0);
85*54fd6939SJiyong Park 
86*54fd6939SJiyong Park 	mmio_write_32(reg_base + VENDSPEC, VENDSPEC_INIT);
87*54fd6939SJiyong Park 	mmio_write_32(reg_base + DLLCTRL, 0);
88*54fd6939SJiyong Park 	mmio_setbits32(reg_base + VENDSPEC, VENDSPEC_IPG_CLKEN | VENDSPEC_PER_CLKEN);
89*54fd6939SJiyong Park 
90*54fd6939SJiyong Park 	/* Set the initial boot clock rate */
91*54fd6939SJiyong Park 	imx_usdhc_set_clk(MMC_BOOT_CLK_RATE);
92*54fd6939SJiyong Park 	udelay(100);
93*54fd6939SJiyong Park 
94*54fd6939SJiyong Park 	/* Clear read/write ready status */
95*54fd6939SJiyong Park 	mmio_clrbits32(reg_base + INTSTATEN, INTSTATEN_BRR | INTSTATEN_BWR);
96*54fd6939SJiyong Park 
97*54fd6939SJiyong Park 	/* configure as little endian */
98*54fd6939SJiyong Park 	mmio_write_32(reg_base + PROTCTRL, PROTCTRL_LE);
99*54fd6939SJiyong Park 
100*54fd6939SJiyong Park 	/* Set timeout to the maximum value */
101*54fd6939SJiyong Park 	mmio_clrsetbits32(reg_base + SYSCTRL, SYSCTRL_TIMEOUT_MASK,
102*54fd6939SJiyong Park 			  SYSCTRL_TIMEOUT(15));
103*54fd6939SJiyong Park 
104*54fd6939SJiyong Park 	/* set wartermark level as 16 for safe for MMC */
105*54fd6939SJiyong Park 	mmio_clrsetbits32(reg_base + WATERMARKLEV, WMKLV_MASK, 16 | (16 << 16));
106*54fd6939SJiyong Park }
107*54fd6939SJiyong Park 
108*54fd6939SJiyong Park #define FSL_CMD_RETRIES	1000
109*54fd6939SJiyong Park 
imx_usdhc_send_cmd(struct mmc_cmd * cmd)110*54fd6939SJiyong Park static int imx_usdhc_send_cmd(struct mmc_cmd *cmd)
111*54fd6939SJiyong Park {
112*54fd6939SJiyong Park 	uintptr_t reg_base = imx_usdhc_params.reg_base;
113*54fd6939SJiyong Park 	unsigned int xfertype = 0, mixctl = 0, multiple = 0, data = 0, err = 0;
114*54fd6939SJiyong Park 	unsigned int state, flags = INTSTATEN_CC | INTSTATEN_CTOE;
115*54fd6939SJiyong Park 	unsigned int cmd_retries = 0;
116*54fd6939SJiyong Park 
117*54fd6939SJiyong Park 	assert(cmd);
118*54fd6939SJiyong Park 
119*54fd6939SJiyong Park 	/* clear all irq status */
120*54fd6939SJiyong Park 	mmio_write_32(reg_base + INTSTAT, 0xffffffff);
121*54fd6939SJiyong Park 
122*54fd6939SJiyong Park 	/* Wait for the bus to be idle */
123*54fd6939SJiyong Park 	do {
124*54fd6939SJiyong Park 		state = mmio_read_32(reg_base + PSTATE);
125*54fd6939SJiyong Park 	} while (state & (PSTATE_CDIHB | PSTATE_CIHB));
126*54fd6939SJiyong Park 
127*54fd6939SJiyong Park 	while (mmio_read_32(reg_base + PSTATE) & PSTATE_DLA)
128*54fd6939SJiyong Park 		;
129*54fd6939SJiyong Park 
130*54fd6939SJiyong Park 	mmio_write_32(reg_base + INTSIGEN, 0);
131*54fd6939SJiyong Park 	udelay(1000);
132*54fd6939SJiyong Park 
133*54fd6939SJiyong Park 	switch (cmd->cmd_idx) {
134*54fd6939SJiyong Park 	case MMC_CMD(12):
135*54fd6939SJiyong Park 		xfertype |= XFERTYPE_CMDTYP_ABORT;
136*54fd6939SJiyong Park 		break;
137*54fd6939SJiyong Park 	case MMC_CMD(18):
138*54fd6939SJiyong Park 		multiple = 1;
139*54fd6939SJiyong Park 		/* fall thru for read op */
140*54fd6939SJiyong Park 	case MMC_CMD(17):
141*54fd6939SJiyong Park 	case MMC_CMD(8):
142*54fd6939SJiyong Park 		mixctl |= MIXCTRL_DTDSEL;
143*54fd6939SJiyong Park 		data = 1;
144*54fd6939SJiyong Park 		break;
145*54fd6939SJiyong Park 	case MMC_CMD(25):
146*54fd6939SJiyong Park 		multiple = 1;
147*54fd6939SJiyong Park 		/* fall thru for data op flag */
148*54fd6939SJiyong Park 	case MMC_CMD(24):
149*54fd6939SJiyong Park 		data = 1;
150*54fd6939SJiyong Park 		break;
151*54fd6939SJiyong Park 	default:
152*54fd6939SJiyong Park 		break;
153*54fd6939SJiyong Park 	}
154*54fd6939SJiyong Park 
155*54fd6939SJiyong Park 	if (multiple) {
156*54fd6939SJiyong Park 		mixctl |= MIXCTRL_MSBSEL;
157*54fd6939SJiyong Park 		mixctl |= MIXCTRL_BCEN;
158*54fd6939SJiyong Park 	}
159*54fd6939SJiyong Park 
160*54fd6939SJiyong Park 	if (data) {
161*54fd6939SJiyong Park 		xfertype |= XFERTYPE_DPSEL;
162*54fd6939SJiyong Park 		mixctl |= MIXCTRL_DMAEN;
163*54fd6939SJiyong Park 	}
164*54fd6939SJiyong Park 
165*54fd6939SJiyong Park 	if (cmd->resp_type & MMC_RSP_48 && cmd->resp_type != MMC_RESPONSE_R2)
166*54fd6939SJiyong Park 		xfertype |= XFERTYPE_RSPTYP_48;
167*54fd6939SJiyong Park 	else if (cmd->resp_type & MMC_RSP_136)
168*54fd6939SJiyong Park 		xfertype |= XFERTYPE_RSPTYP_136;
169*54fd6939SJiyong Park 	else if (cmd->resp_type & MMC_RSP_BUSY)
170*54fd6939SJiyong Park 		xfertype |= XFERTYPE_RSPTYP_48_BUSY;
171*54fd6939SJiyong Park 
172*54fd6939SJiyong Park 	if (cmd->resp_type & MMC_RSP_CMD_IDX)
173*54fd6939SJiyong Park 		xfertype |= XFERTYPE_CICEN;
174*54fd6939SJiyong Park 
175*54fd6939SJiyong Park 	if (cmd->resp_type & MMC_RSP_CRC)
176*54fd6939SJiyong Park 		xfertype |= XFERTYPE_CCCEN;
177*54fd6939SJiyong Park 
178*54fd6939SJiyong Park 	xfertype |= XFERTYPE_CMD(cmd->cmd_idx);
179*54fd6939SJiyong Park 
180*54fd6939SJiyong Park 	/* Send the command */
181*54fd6939SJiyong Park 	mmio_write_32(reg_base + CMDARG, cmd->cmd_arg);
182*54fd6939SJiyong Park 	mmio_clrsetbits32(reg_base + MIXCTRL, MIXCTRL_DATMASK, mixctl);
183*54fd6939SJiyong Park 	mmio_write_32(reg_base + XFERTYPE, xfertype);
184*54fd6939SJiyong Park 
185*54fd6939SJiyong Park 	/* Wait for the command done */
186*54fd6939SJiyong Park 	do {
187*54fd6939SJiyong Park 		state = mmio_read_32(reg_base + INTSTAT);
188*54fd6939SJiyong Park 		if (cmd_retries)
189*54fd6939SJiyong Park 			udelay(1);
190*54fd6939SJiyong Park 	} while ((!(state & flags)) && ++cmd_retries < FSL_CMD_RETRIES);
191*54fd6939SJiyong Park 
192*54fd6939SJiyong Park 	if ((state & (INTSTATEN_CTOE | CMD_ERR)) || cmd_retries == FSL_CMD_RETRIES) {
193*54fd6939SJiyong Park 		if (cmd_retries == FSL_CMD_RETRIES)
194*54fd6939SJiyong Park 			err = -ETIMEDOUT;
195*54fd6939SJiyong Park 		else
196*54fd6939SJiyong Park 			err = -EIO;
197*54fd6939SJiyong Park 		ERROR("imx_usdhc mmc cmd %d state 0x%x errno=%d\n",
198*54fd6939SJiyong Park 		      cmd->cmd_idx, state, err);
199*54fd6939SJiyong Park 		goto out;
200*54fd6939SJiyong Park 	}
201*54fd6939SJiyong Park 
202*54fd6939SJiyong Park 	/* Copy the response to the response buffer */
203*54fd6939SJiyong Park 	if (cmd->resp_type & MMC_RSP_136) {
204*54fd6939SJiyong Park 		unsigned int cmdrsp3, cmdrsp2, cmdrsp1, cmdrsp0;
205*54fd6939SJiyong Park 
206*54fd6939SJiyong Park 		cmdrsp3 = mmio_read_32(reg_base + CMDRSP3);
207*54fd6939SJiyong Park 		cmdrsp2 = mmio_read_32(reg_base + CMDRSP2);
208*54fd6939SJiyong Park 		cmdrsp1 = mmio_read_32(reg_base + CMDRSP1);
209*54fd6939SJiyong Park 		cmdrsp0 = mmio_read_32(reg_base + CMDRSP0);
210*54fd6939SJiyong Park 		cmd->resp_data[3] = (cmdrsp3 << 8) | (cmdrsp2 >> 24);
211*54fd6939SJiyong Park 		cmd->resp_data[2] = (cmdrsp2 << 8) | (cmdrsp1 >> 24);
212*54fd6939SJiyong Park 		cmd->resp_data[1] = (cmdrsp1 << 8) | (cmdrsp0 >> 24);
213*54fd6939SJiyong Park 		cmd->resp_data[0] = (cmdrsp0 << 8);
214*54fd6939SJiyong Park 	} else {
215*54fd6939SJiyong Park 		cmd->resp_data[0] = mmio_read_32(reg_base + CMDRSP0);
216*54fd6939SJiyong Park 	}
217*54fd6939SJiyong Park 
218*54fd6939SJiyong Park 	/* Wait until all of the blocks are transferred */
219*54fd6939SJiyong Park 	if (data) {
220*54fd6939SJiyong Park 		flags = DATA_COMPLETE;
221*54fd6939SJiyong Park 		do {
222*54fd6939SJiyong Park 			state = mmio_read_32(reg_base + INTSTAT);
223*54fd6939SJiyong Park 
224*54fd6939SJiyong Park 			if (state & (INTSTATEN_DTOE | DATA_ERR)) {
225*54fd6939SJiyong Park 				err = -EIO;
226*54fd6939SJiyong Park 				ERROR("imx_usdhc mmc data state 0x%x\n", state);
227*54fd6939SJiyong Park 				goto out;
228*54fd6939SJiyong Park 			}
229*54fd6939SJiyong Park 		} while ((state & flags) != flags);
230*54fd6939SJiyong Park 	}
231*54fd6939SJiyong Park 
232*54fd6939SJiyong Park out:
233*54fd6939SJiyong Park 	/* Reset CMD and DATA on error */
234*54fd6939SJiyong Park 	if (err) {
235*54fd6939SJiyong Park 		mmio_setbits32(reg_base + SYSCTRL, SYSCTRL_RSTC);
236*54fd6939SJiyong Park 		while (mmio_read_32(reg_base + SYSCTRL) & SYSCTRL_RSTC)
237*54fd6939SJiyong Park 			;
238*54fd6939SJiyong Park 
239*54fd6939SJiyong Park 		if (data) {
240*54fd6939SJiyong Park 			mmio_setbits32(reg_base + SYSCTRL, SYSCTRL_RSTD);
241*54fd6939SJiyong Park 			while (mmio_read_32(reg_base + SYSCTRL) & SYSCTRL_RSTD)
242*54fd6939SJiyong Park 				;
243*54fd6939SJiyong Park 		}
244*54fd6939SJiyong Park 	}
245*54fd6939SJiyong Park 
246*54fd6939SJiyong Park 	/* clear all irq status */
247*54fd6939SJiyong Park 	mmio_write_32(reg_base + INTSTAT, 0xffffffff);
248*54fd6939SJiyong Park 
249*54fd6939SJiyong Park 	return err;
250*54fd6939SJiyong Park }
251*54fd6939SJiyong Park 
imx_usdhc_set_ios(unsigned int clk,unsigned int width)252*54fd6939SJiyong Park static int imx_usdhc_set_ios(unsigned int clk, unsigned int width)
253*54fd6939SJiyong Park {
254*54fd6939SJiyong Park 	uintptr_t reg_base = imx_usdhc_params.reg_base;
255*54fd6939SJiyong Park 
256*54fd6939SJiyong Park 	imx_usdhc_set_clk(clk);
257*54fd6939SJiyong Park 
258*54fd6939SJiyong Park 	if (width == MMC_BUS_WIDTH_4)
259*54fd6939SJiyong Park 		mmio_clrsetbits32(reg_base + PROTCTRL, PROTCTRL_WIDTH_MASK,
260*54fd6939SJiyong Park 				  PROTCTRL_WIDTH_4);
261*54fd6939SJiyong Park 	else if (width == MMC_BUS_WIDTH_8)
262*54fd6939SJiyong Park 		mmio_clrsetbits32(reg_base + PROTCTRL, PROTCTRL_WIDTH_MASK,
263*54fd6939SJiyong Park 				  PROTCTRL_WIDTH_8);
264*54fd6939SJiyong Park 
265*54fd6939SJiyong Park 	return 0;
266*54fd6939SJiyong Park }
267*54fd6939SJiyong Park 
imx_usdhc_prepare(int lba,uintptr_t buf,size_t size)268*54fd6939SJiyong Park static int imx_usdhc_prepare(int lba, uintptr_t buf, size_t size)
269*54fd6939SJiyong Park {
270*54fd6939SJiyong Park 	uintptr_t reg_base = imx_usdhc_params.reg_base;
271*54fd6939SJiyong Park 
272*54fd6939SJiyong Park 	mmio_write_32(reg_base + DSADDR, buf);
273*54fd6939SJiyong Park 	mmio_write_32(reg_base + BLKATT,
274*54fd6939SJiyong Park 		      (size / MMC_BLOCK_SIZE) << 16 | MMC_BLOCK_SIZE);
275*54fd6939SJiyong Park 
276*54fd6939SJiyong Park 	return 0;
277*54fd6939SJiyong Park }
278*54fd6939SJiyong Park 
imx_usdhc_read(int lba,uintptr_t buf,size_t size)279*54fd6939SJiyong Park static int imx_usdhc_read(int lba, uintptr_t buf, size_t size)
280*54fd6939SJiyong Park {
281*54fd6939SJiyong Park 	return 0;
282*54fd6939SJiyong Park }
283*54fd6939SJiyong Park 
imx_usdhc_write(int lba,uintptr_t buf,size_t size)284*54fd6939SJiyong Park static int imx_usdhc_write(int lba, uintptr_t buf, size_t size)
285*54fd6939SJiyong Park {
286*54fd6939SJiyong Park 	return 0;
287*54fd6939SJiyong Park }
288*54fd6939SJiyong Park 
imx_usdhc_init(imx_usdhc_params_t * params,struct mmc_device_info * mmc_dev_info)289*54fd6939SJiyong Park void imx_usdhc_init(imx_usdhc_params_t *params,
290*54fd6939SJiyong Park 		    struct mmc_device_info *mmc_dev_info)
291*54fd6939SJiyong Park {
292*54fd6939SJiyong Park 	assert((params != 0) &&
293*54fd6939SJiyong Park 	       ((params->reg_base & MMC_BLOCK_MASK) == 0) &&
294*54fd6939SJiyong Park 	       (params->clk_rate > 0) &&
295*54fd6939SJiyong Park 	       ((params->bus_width == MMC_BUS_WIDTH_1) ||
296*54fd6939SJiyong Park 		(params->bus_width == MMC_BUS_WIDTH_4) ||
297*54fd6939SJiyong Park 		(params->bus_width == MMC_BUS_WIDTH_8)));
298*54fd6939SJiyong Park 
299*54fd6939SJiyong Park 	memcpy(&imx_usdhc_params, params, sizeof(imx_usdhc_params_t));
300*54fd6939SJiyong Park 	mmc_init(&imx_usdhc_ops, params->clk_rate, params->bus_width,
301*54fd6939SJiyong Park 		 params->flags, mmc_dev_info);
302*54fd6939SJiyong Park }
303