18b659130SJun Nie /*
28b659130SJun Nie * Copyright (c) 2018, ARM Limited and Contributors. All rights reserved.
3a59d43fcSGhennadi Procopciuc * Copyright 2025 NXP
48b659130SJun Nie *
58b659130SJun Nie * SPDX-License-Identifier: BSD-3-Clause
68b659130SJun Nie */
78b659130SJun Nie
809d40e0eSAntonio Nino Diaz #include <assert.h>
909d40e0eSAntonio Nino Diaz #include <errno.h>
1009d40e0eSAntonio Nino Diaz #include <string.h>
1109d40e0eSAntonio Nino Diaz
128b659130SJun Nie #include <arch.h>
138b659130SJun Nie #include <arch_helpers.h>
1409d40e0eSAntonio Nino Diaz #include <common/debug.h>
1509d40e0eSAntonio Nino Diaz #include <drivers/delay_timer.h>
1609d40e0eSAntonio Nino Diaz #include <drivers/mmc.h>
17*01d24d69SGhennadi Procopciuc #include <lib/mmio_poll.h>
18cdf002deSGhennadi Procopciuc #include <lib/xlat_tables/xlat_tables_v2.h>
1909d40e0eSAntonio Nino Diaz
208b659130SJun Nie #include <imx_usdhc.h>
218b659130SJun Nie
22a59d43fcSGhennadi Procopciuc /* These masks represent the commands which involve a data transfer. */
23a59d43fcSGhennadi Procopciuc #define ADTC_MASK_SD (BIT_32(6U) | BIT_32(17U) | BIT_32(18U) |\
24a59d43fcSGhennadi Procopciuc BIT_32(24U) | BIT_32(25U))
25a59d43fcSGhennadi Procopciuc #define ADTC_MASK_ACMD (BIT_64(51U))
26a59d43fcSGhennadi Procopciuc
27*01d24d69SGhennadi Procopciuc #define USDHC_TIMEOUT_US (1U * 1000U) /* 1 msec */
28*01d24d69SGhennadi Procopciuc #define USDHC_TRANSFER_TIMEOUT (1U * 1000U * 1000U) /* 1 sec */
29*01d24d69SGhennadi Procopciuc
30b61379fbSGhennadi Procopciuc struct imx_usdhc_device_data {
31b61379fbSGhennadi Procopciuc uint32_t addr;
32b61379fbSGhennadi Procopciuc uint32_t blk_size;
33b61379fbSGhennadi Procopciuc uint32_t blks;
34b61379fbSGhennadi Procopciuc bool valid;
35b61379fbSGhennadi Procopciuc };
36b61379fbSGhennadi Procopciuc
378b659130SJun Nie static void imx_usdhc_initialize(void);
388b659130SJun Nie static int imx_usdhc_send_cmd(struct mmc_cmd *cmd);
398b659130SJun Nie static int imx_usdhc_set_ios(unsigned int clk, unsigned int width);
408b659130SJun Nie static int imx_usdhc_prepare(int lba, uintptr_t buf, size_t size);
418b659130SJun Nie static int imx_usdhc_read(int lba, uintptr_t buf, size_t size);
428b659130SJun Nie static int imx_usdhc_write(int lba, uintptr_t buf, size_t size);
438b659130SJun Nie
448b659130SJun Nie static const struct mmc_ops imx_usdhc_ops = {
458b659130SJun Nie .init = imx_usdhc_initialize,
468b659130SJun Nie .send_cmd = imx_usdhc_send_cmd,
478b659130SJun Nie .set_ios = imx_usdhc_set_ios,
488b659130SJun Nie .prepare = imx_usdhc_prepare,
498b659130SJun Nie .read = imx_usdhc_read,
508b659130SJun Nie .write = imx_usdhc_write,
518b659130SJun Nie };
528b659130SJun Nie
538b659130SJun Nie static imx_usdhc_params_t imx_usdhc_params;
54b61379fbSGhennadi Procopciuc static struct imx_usdhc_device_data imx_usdhc_data;
55b61379fbSGhennadi Procopciuc
imx_usdhc_is_buf_valid(void)56b61379fbSGhennadi Procopciuc static bool imx_usdhc_is_buf_valid(void)
57b61379fbSGhennadi Procopciuc {
58b61379fbSGhennadi Procopciuc return imx_usdhc_data.valid;
59b61379fbSGhennadi Procopciuc }
60b61379fbSGhennadi Procopciuc
imx_usdhc_is_buf_multiblk(void)61b61379fbSGhennadi Procopciuc static bool imx_usdhc_is_buf_multiblk(void)
62b61379fbSGhennadi Procopciuc {
63b61379fbSGhennadi Procopciuc return imx_usdhc_data.blks > 1U;
64b61379fbSGhennadi Procopciuc }
65b61379fbSGhennadi Procopciuc
imx_usdhc_inval_buf_data(void)66b61379fbSGhennadi Procopciuc static void imx_usdhc_inval_buf_data(void)
67b61379fbSGhennadi Procopciuc {
68b61379fbSGhennadi Procopciuc imx_usdhc_data.valid = false;
69b61379fbSGhennadi Procopciuc }
70b61379fbSGhennadi Procopciuc
imx_usdhc_save_buf_data(uintptr_t buf,size_t size)71b61379fbSGhennadi Procopciuc static int imx_usdhc_save_buf_data(uintptr_t buf, size_t size)
72b61379fbSGhennadi Procopciuc {
73b61379fbSGhennadi Procopciuc uint32_t block_size;
74b61379fbSGhennadi Procopciuc uint64_t blks;
75b61379fbSGhennadi Procopciuc
76b61379fbSGhennadi Procopciuc if (size <= MMC_BLOCK_SIZE) {
77b61379fbSGhennadi Procopciuc block_size = (uint32_t)size;
78b61379fbSGhennadi Procopciuc } else {
79b61379fbSGhennadi Procopciuc block_size = MMC_BLOCK_SIZE;
80b61379fbSGhennadi Procopciuc }
81b61379fbSGhennadi Procopciuc
82b61379fbSGhennadi Procopciuc if (buf > UINT32_MAX) {
83b61379fbSGhennadi Procopciuc return -EOVERFLOW;
84b61379fbSGhennadi Procopciuc }
85b61379fbSGhennadi Procopciuc
86b61379fbSGhennadi Procopciuc imx_usdhc_data.addr = (uint32_t)buf;
87b61379fbSGhennadi Procopciuc imx_usdhc_data.blk_size = block_size;
88b61379fbSGhennadi Procopciuc blks = size / block_size;
89b61379fbSGhennadi Procopciuc imx_usdhc_data.blks = (uint32_t)blks;
90b61379fbSGhennadi Procopciuc
91b61379fbSGhennadi Procopciuc imx_usdhc_data.valid = true;
92b61379fbSGhennadi Procopciuc
93b61379fbSGhennadi Procopciuc return 0;
94b61379fbSGhennadi Procopciuc }
95b61379fbSGhennadi Procopciuc
imx_usdhc_write_buf_data(void)96b61379fbSGhennadi Procopciuc static void imx_usdhc_write_buf_data(void)
97b61379fbSGhennadi Procopciuc {
98b61379fbSGhennadi Procopciuc uintptr_t reg_base = imx_usdhc_params.reg_base;
99b61379fbSGhennadi Procopciuc uint32_t addr, blks, blk_size;
100b61379fbSGhennadi Procopciuc
101b61379fbSGhennadi Procopciuc addr = imx_usdhc_data.addr;
102b61379fbSGhennadi Procopciuc blks = imx_usdhc_data.blks;
103b61379fbSGhennadi Procopciuc blk_size = imx_usdhc_data.blk_size;
104b61379fbSGhennadi Procopciuc
105b61379fbSGhennadi Procopciuc mmio_write_32(reg_base + DSADDR, addr);
106b61379fbSGhennadi Procopciuc mmio_write_32(reg_base + BLKATT, BLKATT_BLKCNT(blks) |
107b61379fbSGhennadi Procopciuc BLKATT_BLKSIZE(blk_size));
108b61379fbSGhennadi Procopciuc }
1098b659130SJun Nie
1108b659130SJun Nie #define IMX7_MMC_SRC_CLK_RATE (200 * 1000 * 1000)
imx_usdhc_set_clk(unsigned int clk)111*01d24d69SGhennadi Procopciuc static int imx_usdhc_set_clk(unsigned int clk)
1128b659130SJun Nie {
1138b659130SJun Nie unsigned int sdhc_clk = IMX7_MMC_SRC_CLK_RATE;
1148b659130SJun Nie uintptr_t reg_base = imx_usdhc_params.reg_base;
1152e90f3e6SGhennadi Procopciuc unsigned int pre_div = 1U, div = 1U;
116*01d24d69SGhennadi Procopciuc uint32_t pstate;
117*01d24d69SGhennadi Procopciuc int ret;
1188b659130SJun Nie
1198b659130SJun Nie assert(clk > 0);
1208b659130SJun Nie
1218b659130SJun Nie while (sdhc_clk / (16 * pre_div) > clk && pre_div < 256)
1228b659130SJun Nie pre_div *= 2;
1238b659130SJun Nie
1242e90f3e6SGhennadi Procopciuc while (((sdhc_clk / (div * pre_div)) > clk) && (div < 16U)) {
1258b659130SJun Nie div++;
1262e90f3e6SGhennadi Procopciuc }
1278b659130SJun Nie
1288b659130SJun Nie pre_div >>= 1;
1298b659130SJun Nie div -= 1;
1308b659130SJun Nie clk = (pre_div << 8) | (div << 4);
1318b659130SJun Nie
132*01d24d69SGhennadi Procopciuc ret = mmio_read_32_poll_timeout(reg_base + PSTATE, pstate,
133*01d24d69SGhennadi Procopciuc (pstate & PSTATE_SDSTB) != 0U,
134*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
135*01d24d69SGhennadi Procopciuc if (ret == -ETIMEDOUT) {
136*01d24d69SGhennadi Procopciuc ERROR("Unstable SD clock\n");
137*01d24d69SGhennadi Procopciuc return ret;
138583a544cSGhennadi Procopciuc }
139583a544cSGhennadi Procopciuc
1408b659130SJun Nie mmio_clrbits32(reg_base + VENDSPEC, VENDSPEC_CARD_CLKEN);
1418b659130SJun Nie mmio_clrsetbits32(reg_base + SYSCTRL, SYSCTRL_CLOCK_MASK, clk);
1428b659130SJun Nie udelay(10000);
1438b659130SJun Nie
1448b659130SJun Nie mmio_setbits32(reg_base + VENDSPEC, VENDSPEC_PER_CLKEN | VENDSPEC_CARD_CLKEN);
145*01d24d69SGhennadi Procopciuc
146*01d24d69SGhennadi Procopciuc return 0;
1478b659130SJun Nie }
1488b659130SJun Nie
imx_usdhc_initialize(void)1498b659130SJun Nie static void imx_usdhc_initialize(void)
1508b659130SJun Nie {
1518b659130SJun Nie uintptr_t reg_base = imx_usdhc_params.reg_base;
152*01d24d69SGhennadi Procopciuc uint32_t sysctrl;
153*01d24d69SGhennadi Procopciuc int ret;
1548b659130SJun Nie
1558b659130SJun Nie assert((imx_usdhc_params.reg_base & MMC_BLOCK_MASK) == 0);
1568b659130SJun Nie
1578b659130SJun Nie /* reset the controller */
1588b659130SJun Nie mmio_setbits32(reg_base + SYSCTRL, SYSCTRL_RSTA);
1598b659130SJun Nie
1608b659130SJun Nie /* wait for reset done */
161*01d24d69SGhennadi Procopciuc ret = mmio_read_32_poll_timeout(reg_base + SYSCTRL, sysctrl,
162*01d24d69SGhennadi Procopciuc (sysctrl & SYSCTRL_RSTA) == 0U,
163*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
164*01d24d69SGhennadi Procopciuc if (ret == -ETIMEDOUT) {
165*01d24d69SGhennadi Procopciuc ERROR("Failed to reset the USDHC controller\n");
166*01d24d69SGhennadi Procopciuc panic();
1678b659130SJun Nie }
1688b659130SJun Nie
1698b659130SJun Nie mmio_write_32(reg_base + MMCBOOT, 0);
1708b659130SJun Nie mmio_write_32(reg_base + MIXCTRL, 0);
1718b659130SJun Nie mmio_write_32(reg_base + CLKTUNECTRLSTS, 0);
1728b659130SJun Nie
1738b659130SJun Nie mmio_write_32(reg_base + VENDSPEC, VENDSPEC_INIT);
1748b659130SJun Nie mmio_write_32(reg_base + DLLCTRL, 0);
1758b659130SJun Nie mmio_setbits32(reg_base + VENDSPEC, VENDSPEC_IPG_CLKEN | VENDSPEC_PER_CLKEN);
1768b659130SJun Nie
1778b659130SJun Nie /* Set the initial boot clock rate */
178*01d24d69SGhennadi Procopciuc ret = imx_usdhc_set_clk(MMC_BOOT_CLK_RATE);
179*01d24d69SGhennadi Procopciuc if (ret != 0) {
180*01d24d69SGhennadi Procopciuc panic();
181*01d24d69SGhennadi Procopciuc }
182*01d24d69SGhennadi Procopciuc
1838b659130SJun Nie udelay(100);
1848b659130SJun Nie
1858b659130SJun Nie /* Clear read/write ready status */
1868b659130SJun Nie mmio_clrbits32(reg_base + INTSTATEN, INTSTATEN_BRR | INTSTATEN_BWR);
1878b659130SJun Nie
1888b659130SJun Nie /* configure as little endian */
1898b659130SJun Nie mmio_write_32(reg_base + PROTCTRL, PROTCTRL_LE);
1908b659130SJun Nie
1918b659130SJun Nie /* Set timeout to the maximum value */
1928b659130SJun Nie mmio_clrsetbits32(reg_base + SYSCTRL, SYSCTRL_TIMEOUT_MASK,
1938b659130SJun Nie SYSCTRL_TIMEOUT(15));
1948b659130SJun Nie
1958b659130SJun Nie /* set wartermark level as 16 for safe for MMC */
1968b659130SJun Nie mmio_clrsetbits32(reg_base + WATERMARKLEV, WMKLV_MASK, 16 | (16 << 16));
1978b659130SJun Nie }
1988b659130SJun Nie
is_data_transfer_to_card(const struct mmc_cmd * cmd)19913a839a7SGhennadi Procopciuc static bool is_data_transfer_to_card(const struct mmc_cmd *cmd)
20013a839a7SGhennadi Procopciuc {
20113a839a7SGhennadi Procopciuc unsigned int cmd_idx = cmd->cmd_idx;
20213a839a7SGhennadi Procopciuc
20313a839a7SGhennadi Procopciuc return (cmd_idx == MMC_CMD(24)) || (cmd_idx == MMC_CMD(25));
20413a839a7SGhennadi Procopciuc }
20513a839a7SGhennadi Procopciuc
is_data_transfer_cmd(const struct mmc_cmd * cmd)206a59d43fcSGhennadi Procopciuc static bool is_data_transfer_cmd(const struct mmc_cmd *cmd)
207a59d43fcSGhennadi Procopciuc {
208a59d43fcSGhennadi Procopciuc uintptr_t reg_base = imx_usdhc_params.reg_base;
209a59d43fcSGhennadi Procopciuc unsigned int cmd_idx = cmd->cmd_idx;
210a59d43fcSGhennadi Procopciuc uint32_t xfer_type;
211a59d43fcSGhennadi Procopciuc
212a59d43fcSGhennadi Procopciuc xfer_type = mmio_read_32(reg_base + XFERTYPE);
213a59d43fcSGhennadi Procopciuc
214a59d43fcSGhennadi Procopciuc if (XFERTYPE_GET_CMD(xfer_type) == MMC_CMD(55)) {
215a59d43fcSGhennadi Procopciuc return (ADTC_MASK_ACMD & BIT_64(cmd_idx)) != 0ULL;
216a59d43fcSGhennadi Procopciuc }
217a59d43fcSGhennadi Procopciuc
218a59d43fcSGhennadi Procopciuc if ((ADTC_MASK_SD & BIT_32(cmd->cmd_idx)) != 0U) {
219a59d43fcSGhennadi Procopciuc return true;
220a59d43fcSGhennadi Procopciuc }
221a59d43fcSGhennadi Procopciuc
222a59d43fcSGhennadi Procopciuc return false;
223a59d43fcSGhennadi Procopciuc }
224a59d43fcSGhennadi Procopciuc
get_xfr_type(const struct mmc_cmd * cmd,bool data,uint32_t * xfertype)225a59d43fcSGhennadi Procopciuc static int get_xfr_type(const struct mmc_cmd *cmd, bool data, uint32_t *xfertype)
226a59d43fcSGhennadi Procopciuc {
227a59d43fcSGhennadi Procopciuc *xfertype = XFERTYPE_CMD(cmd->cmd_idx);
228a59d43fcSGhennadi Procopciuc
229a59d43fcSGhennadi Procopciuc switch (cmd->resp_type) {
230a59d43fcSGhennadi Procopciuc case MMC_RESPONSE_R2:
231a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_RSPTYP_136;
232a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_CCCEN;
233a59d43fcSGhennadi Procopciuc break;
234a59d43fcSGhennadi Procopciuc case MMC_RESPONSE_R4:
235a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_RSPTYP_48;
236a59d43fcSGhennadi Procopciuc break;
237a59d43fcSGhennadi Procopciuc case MMC_RESPONSE_R6:
238a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_RSPTYP_48;
239a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_CICEN;
240a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_CCCEN;
241a59d43fcSGhennadi Procopciuc break;
242a59d43fcSGhennadi Procopciuc case MMC_RESPONSE_R1B:
243a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_RSPTYP_48_BUSY;
244a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_CICEN;
245a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_CCCEN;
246a59d43fcSGhennadi Procopciuc break;
24792a7b540SGhennadi Procopciuc case MMC_RESPONSE_NONE:
24892a7b540SGhennadi Procopciuc break;
249a59d43fcSGhennadi Procopciuc default:
250a59d43fcSGhennadi Procopciuc ERROR("Invalid CMD response: %u\n", cmd->resp_type);
251a59d43fcSGhennadi Procopciuc return -EINVAL;
252a59d43fcSGhennadi Procopciuc }
253a59d43fcSGhennadi Procopciuc
254a59d43fcSGhennadi Procopciuc if (data) {
255a59d43fcSGhennadi Procopciuc *xfertype |= XFERTYPE_DPSEL;
256a59d43fcSGhennadi Procopciuc }
257a59d43fcSGhennadi Procopciuc
258a59d43fcSGhennadi Procopciuc return 0;
259a59d43fcSGhennadi Procopciuc }
260a59d43fcSGhennadi Procopciuc
imx_usdhc_send_cmd(struct mmc_cmd * cmd)2618b659130SJun Nie static int imx_usdhc_send_cmd(struct mmc_cmd *cmd)
2628b659130SJun Nie {
2638b659130SJun Nie uintptr_t reg_base = imx_usdhc_params.reg_base;
264*01d24d69SGhennadi Procopciuc unsigned int flags = INTSTATEN_CC | INTSTATEN_CTOE;
265*01d24d69SGhennadi Procopciuc uint32_t xfertype, pstate, intstat, sysctrl;
266f9ed855bSGhennadi Procopciuc unsigned int mixctl = 0;
267*01d24d69SGhennadi Procopciuc int err = 0, ret;
268a59d43fcSGhennadi Procopciuc bool data;
2698b659130SJun Nie
2708b659130SJun Nie assert(cmd);
2718b659130SJun Nie
272a59d43fcSGhennadi Procopciuc data = is_data_transfer_cmd(cmd);
273a59d43fcSGhennadi Procopciuc
274a59d43fcSGhennadi Procopciuc err = get_xfr_type(cmd, data, &xfertype);
275a59d43fcSGhennadi Procopciuc if (err != 0) {
276a59d43fcSGhennadi Procopciuc return err;
277a59d43fcSGhennadi Procopciuc }
278a59d43fcSGhennadi Procopciuc
2798b659130SJun Nie /* clear all irq status */
2808b659130SJun Nie mmio_write_32(reg_base + INTSTAT, 0xffffffff);
2818b659130SJun Nie
2828b659130SJun Nie /* Wait for the bus to be idle */
283*01d24d69SGhennadi Procopciuc err = mmio_read_32_poll_timeout(reg_base + PSTATE, pstate,
284*01d24d69SGhennadi Procopciuc (pstate & (PSTATE_CDIHB | PSTATE_CIHB)) == 0U,
285*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
286*01d24d69SGhennadi Procopciuc if (err == -ETIMEDOUT) {
287*01d24d69SGhennadi Procopciuc ERROR("Failed to wait an idle bus\n");
288*01d24d69SGhennadi Procopciuc return err;
289*01d24d69SGhennadi Procopciuc }
2908b659130SJun Nie
291*01d24d69SGhennadi Procopciuc err = mmio_read_32_poll_timeout(reg_base + PSTATE, pstate,
292*01d24d69SGhennadi Procopciuc (pstate & PSTATE_DLA) == 0U,
293*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
294*01d24d69SGhennadi Procopciuc if (err == -ETIMEDOUT) {
295*01d24d69SGhennadi Procopciuc ERROR("Active data line during the uSDHC init\n");
296*01d24d69SGhennadi Procopciuc return err;
297*01d24d69SGhennadi Procopciuc }
2988b659130SJun Nie
2998b659130SJun Nie mmio_write_32(reg_base + INTSIGEN, 0);
3008b659130SJun Nie
3018b659130SJun Nie if (data) {
3028b659130SJun Nie mixctl |= MIXCTRL_DMAEN;
3038b659130SJun Nie }
3048b659130SJun Nie
30513a839a7SGhennadi Procopciuc if (!is_data_transfer_to_card(cmd)) {
30613a839a7SGhennadi Procopciuc mixctl |= MIXCTRL_DTDSEL;
30713a839a7SGhennadi Procopciuc }
30813a839a7SGhennadi Procopciuc
309b61379fbSGhennadi Procopciuc if ((cmd->cmd_idx != MMC_CMD(55)) && imx_usdhc_is_buf_valid()) {
310b61379fbSGhennadi Procopciuc if (imx_usdhc_is_buf_multiblk()) {
311b61379fbSGhennadi Procopciuc mixctl |= MIXCTRL_MSBSEL | MIXCTRL_BCEN;
312b61379fbSGhennadi Procopciuc }
313b61379fbSGhennadi Procopciuc
314b61379fbSGhennadi Procopciuc imx_usdhc_write_buf_data();
315b61379fbSGhennadi Procopciuc imx_usdhc_inval_buf_data();
316b61379fbSGhennadi Procopciuc }
317b61379fbSGhennadi Procopciuc
3188b659130SJun Nie /* Send the command */
3198b659130SJun Nie mmio_write_32(reg_base + CMDARG, cmd->cmd_arg);
3208b659130SJun Nie mmio_clrsetbits32(reg_base + MIXCTRL, MIXCTRL_DATMASK, mixctl);
3218b659130SJun Nie mmio_write_32(reg_base + XFERTYPE, xfertype);
3228b659130SJun Nie
3238b659130SJun Nie /* Wait for the command done */
324*01d24d69SGhennadi Procopciuc err = mmio_read_32_poll_timeout(reg_base + INTSTAT, intstat,
325*01d24d69SGhennadi Procopciuc (intstat & flags) != 0U,
326*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
327*01d24d69SGhennadi Procopciuc if ((err == -ETIMEDOUT) || ((intstat & (INTSTATEN_CTOE | CMD_ERR)) != 0U)) {
328*01d24d69SGhennadi Procopciuc if ((intstat & (INTSTATEN_CTOE | CMD_ERR)) != 0U) {
3298b659130SJun Nie err = -EIO;
330*01d24d69SGhennadi Procopciuc }
3318b659130SJun Nie ERROR("imx_usdhc mmc cmd %d state 0x%x errno=%d\n",
332*01d24d69SGhennadi Procopciuc cmd->cmd_idx, intstat, err);
3338b659130SJun Nie goto out;
3348b659130SJun Nie }
3358b659130SJun Nie
3368b659130SJun Nie /* Copy the response to the response buffer */
3378b659130SJun Nie if (cmd->resp_type & MMC_RSP_136) {
3388b659130SJun Nie unsigned int cmdrsp3, cmdrsp2, cmdrsp1, cmdrsp0;
3398b659130SJun Nie
3408b659130SJun Nie cmdrsp3 = mmio_read_32(reg_base + CMDRSP3);
3418b659130SJun Nie cmdrsp2 = mmio_read_32(reg_base + CMDRSP2);
3428b659130SJun Nie cmdrsp1 = mmio_read_32(reg_base + CMDRSP1);
3438b659130SJun Nie cmdrsp0 = mmio_read_32(reg_base + CMDRSP0);
3448b659130SJun Nie cmd->resp_data[3] = (cmdrsp3 << 8) | (cmdrsp2 >> 24);
3458b659130SJun Nie cmd->resp_data[2] = (cmdrsp2 << 8) | (cmdrsp1 >> 24);
3468b659130SJun Nie cmd->resp_data[1] = (cmdrsp1 << 8) | (cmdrsp0 >> 24);
3478b659130SJun Nie cmd->resp_data[0] = (cmdrsp0 << 8);
3488b659130SJun Nie } else {
3498b659130SJun Nie cmd->resp_data[0] = mmio_read_32(reg_base + CMDRSP0);
3508b659130SJun Nie }
3518b659130SJun Nie
3528b659130SJun Nie /* Wait until all of the blocks are transferred */
3538b659130SJun Nie if (data) {
3548b659130SJun Nie flags = DATA_COMPLETE;
355*01d24d69SGhennadi Procopciuc err = mmio_read_32_poll_timeout(reg_base + INTSTAT, intstat,
356*01d24d69SGhennadi Procopciuc (((intstat & (INTSTATEN_DTOE | DATA_ERR)) != 0U) ||
357*01d24d69SGhennadi Procopciuc ((intstat & flags) == flags)),
358*01d24d69SGhennadi Procopciuc USDHC_TRANSFER_TIMEOUT);
359*01d24d69SGhennadi Procopciuc if ((intstat & (INTSTATEN_DTOE | DATA_ERR)) != 0U) {
3608b659130SJun Nie err = -EIO;
361*01d24d69SGhennadi Procopciuc ERROR("imx_usdhc mmc data state 0x%x\n", intstat);
3628b659130SJun Nie goto out;
3638b659130SJun Nie }
364*01d24d69SGhennadi Procopciuc
365*01d24d69SGhennadi Procopciuc if (err == -ETIMEDOUT) {
366*01d24d69SGhennadi Procopciuc ERROR("Timeout in block transfer\n");
367*01d24d69SGhennadi Procopciuc goto out;
368*01d24d69SGhennadi Procopciuc }
3698b659130SJun Nie }
3708b659130SJun Nie
3718b659130SJun Nie out:
3728b659130SJun Nie /* Reset CMD and DATA on error */
3738b659130SJun Nie if (err) {
3748b659130SJun Nie mmio_setbits32(reg_base + SYSCTRL, SYSCTRL_RSTC);
375*01d24d69SGhennadi Procopciuc ret = mmio_read_32_poll_timeout(reg_base + SYSCTRL, sysctrl,
376*01d24d69SGhennadi Procopciuc (sysctrl & SYSCTRL_RSTC) == 0U,
377*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
378*01d24d69SGhennadi Procopciuc if (ret == -ETIMEDOUT) {
379*01d24d69SGhennadi Procopciuc ERROR("Failed to reset the CMD line\n");
380*01d24d69SGhennadi Procopciuc }
3818b659130SJun Nie
3828b659130SJun Nie if (data) {
3838b659130SJun Nie mmio_setbits32(reg_base + SYSCTRL, SYSCTRL_RSTD);
384*01d24d69SGhennadi Procopciuc ret = mmio_read_32_poll_timeout(reg_base + SYSCTRL, sysctrl,
385*01d24d69SGhennadi Procopciuc (sysctrl & SYSCTRL_RSTD) == 0U,
386*01d24d69SGhennadi Procopciuc USDHC_TIMEOUT_US);
387*01d24d69SGhennadi Procopciuc if (ret == -ETIMEDOUT) {
388*01d24d69SGhennadi Procopciuc ERROR("Failed to reset the data line\n");
389*01d24d69SGhennadi Procopciuc }
3908b659130SJun Nie }
3918b659130SJun Nie }
3928b659130SJun Nie
3938b659130SJun Nie /* clear all irq status */
3948b659130SJun Nie mmio_write_32(reg_base + INTSTAT, 0xffffffff);
3958b659130SJun Nie
3968b659130SJun Nie return err;
3978b659130SJun Nie }
3988b659130SJun Nie
imx_usdhc_set_ios(unsigned int clk,unsigned int width)3998b659130SJun Nie static int imx_usdhc_set_ios(unsigned int clk, unsigned int width)
4008b659130SJun Nie {
4018b659130SJun Nie uintptr_t reg_base = imx_usdhc_params.reg_base;
402*01d24d69SGhennadi Procopciuc int ret;
4038b659130SJun Nie
404*01d24d69SGhennadi Procopciuc ret = imx_usdhc_set_clk(clk);
405*01d24d69SGhennadi Procopciuc if (ret != 0) {
406*01d24d69SGhennadi Procopciuc return ret;
407*01d24d69SGhennadi Procopciuc }
4088b659130SJun Nie
4098b659130SJun Nie if (width == MMC_BUS_WIDTH_4)
4108b659130SJun Nie mmio_clrsetbits32(reg_base + PROTCTRL, PROTCTRL_WIDTH_MASK,
4118b659130SJun Nie PROTCTRL_WIDTH_4);
4128b659130SJun Nie else if (width == MMC_BUS_WIDTH_8)
4138b659130SJun Nie mmio_clrsetbits32(reg_base + PROTCTRL, PROTCTRL_WIDTH_MASK,
4148b659130SJun Nie PROTCTRL_WIDTH_8);
4158b659130SJun Nie
4168b659130SJun Nie return 0;
4178b659130SJun Nie }
4188b659130SJun Nie
imx_usdhc_prepare(int lba,uintptr_t buf,size_t size)4198b659130SJun Nie static int imx_usdhc_prepare(int lba, uintptr_t buf, size_t size)
4208b659130SJun Nie {
4217e2a4347SGhennadi Procopciuc flush_dcache_range(buf, size);
422b61379fbSGhennadi Procopciuc return imx_usdhc_save_buf_data(buf, size);
4238b659130SJun Nie }
4248b659130SJun Nie
imx_usdhc_read(int lba,uintptr_t buf,size_t size)4258b659130SJun Nie static int imx_usdhc_read(int lba, uintptr_t buf, size_t size)
4268b659130SJun Nie {
4277e2a4347SGhennadi Procopciuc inv_dcache_range(buf, size);
4288b659130SJun Nie return 0;
4298b659130SJun Nie }
4308b659130SJun Nie
imx_usdhc_write(int lba,uintptr_t buf,size_t size)4318b659130SJun Nie static int imx_usdhc_write(int lba, uintptr_t buf, size_t size)
4328b659130SJun Nie {
4338b659130SJun Nie return 0;
4348b659130SJun Nie }
4358b659130SJun Nie
imx_usdhc_init(imx_usdhc_params_t * params,struct mmc_device_info * mmc_dev_info)4368b659130SJun Nie void imx_usdhc_init(imx_usdhc_params_t *params,
4378b659130SJun Nie struct mmc_device_info *mmc_dev_info)
4388b659130SJun Nie {
439cdf002deSGhennadi Procopciuc int ret __maybe_unused;
440cdf002deSGhennadi Procopciuc
4418b659130SJun Nie assert((params != 0) &&
4428b659130SJun Nie ((params->reg_base & MMC_BLOCK_MASK) == 0) &&
4438b659130SJun Nie ((params->bus_width == MMC_BUS_WIDTH_1) ||
4448b659130SJun Nie (params->bus_width == MMC_BUS_WIDTH_4) ||
4458b659130SJun Nie (params->bus_width == MMC_BUS_WIDTH_8)));
4468b659130SJun Nie
447cdf002deSGhennadi Procopciuc #if PLAT_XLAT_TABLES_DYNAMIC
448cdf002deSGhennadi Procopciuc ret = mmap_add_dynamic_region(params->reg_base, params->reg_base,
449cdf002deSGhennadi Procopciuc PAGE_SIZE,
450cdf002deSGhennadi Procopciuc MT_DEVICE | MT_RW | MT_SECURE);
451cdf002deSGhennadi Procopciuc if (ret != 0) {
452cdf002deSGhennadi Procopciuc ERROR("Failed to map the uSDHC registers\n");
453cdf002deSGhennadi Procopciuc panic();
454cdf002deSGhennadi Procopciuc }
455cdf002deSGhennadi Procopciuc #endif
456cdf002deSGhennadi Procopciuc
4578b659130SJun Nie memcpy(&imx_usdhc_params, params, sizeof(imx_usdhc_params_t));
4588b659130SJun Nie mmc_init(&imx_usdhc_ops, params->clk_rate, params->bus_width,
4598b659130SJun Nie params->flags, mmc_dev_info);
4608b659130SJun Nie }
461