blob: 451fe4a4e5a28bd5ff290cec5e4ad9289cb469b4 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glass8e6cc462015-07-06 12:54:32 -06002/*
3 * Copyright (c) 2015 Google, Inc
4 * Written by Simon Glass <sjg@chromium.org>
Simon Glass8e6cc462015-07-06 12:54:32 -06005 */
6
7#include <common.h>
8#include <dm.h>
9#include <errno.h>
Simon Glassf376a3c2016-05-01 13:52:42 -060010#include <fdtdec.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -060011#include <log.h>
Simon Glass0bf61ac2021-10-23 17:25:59 -060012#include <malloc.h>
Simon Glass8e6cc462015-07-06 12:54:32 -060013#include <mmc.h>
Simon Glass0bf61ac2021-10-23 17:25:59 -060014#include <os.h>
Simon Glass8e6cc462015-07-06 12:54:32 -060015#include <asm/test.h>
16
Simon Glassf376a3c2016-05-01 13:52:42 -060017struct sandbox_mmc_plat {
18 struct mmc_config cfg;
19 struct mmc mmc;
Simon Glass0bf61ac2021-10-23 17:25:59 -060020 const char *fname;
Simon Glassf376a3c2016-05-01 13:52:42 -060021};
22
Simon Glass0bf61ac2021-10-23 17:25:59 -060023#define MMC_CMULT 8 /* 8 because the card is high-capacity */
24#define MMC_BL_LEN_SHIFT 10
25#define MMC_BL_LEN BIT(MMC_BL_LEN_SHIFT)
26#define SIZE_MULTIPLE ((1 << (MMC_CMULT + 2)) * MMC_BL_LEN)
Sean Anderson3f6fb772021-02-05 09:38:53 -050027
28struct sandbox_mmc_priv {
Simon Glass0bf61ac2021-10-23 17:25:59 -060029 char *buf;
30 int csize; /* CSIZE value to report */
31 int size;
Sean Anderson3f6fb772021-02-05 09:38:53 -050032};
33
Simon Glassf376a3c2016-05-01 13:52:42 -060034/**
35 * sandbox_mmc_send_cmd() - Emulate SD commands
36 *
37 * This emulate an SD card version 2. Single-block reads result in zero data.
38 * Multiple-block reads return a test string.
39 */
Simon Glass9a46bd32016-06-12 23:30:26 -060040static int sandbox_mmc_send_cmd(struct udevice *dev, struct mmc_cmd *cmd,
Simon Glassf376a3c2016-05-01 13:52:42 -060041 struct mmc_data *data)
42{
Sean Anderson3f6fb772021-02-05 09:38:53 -050043 struct sandbox_mmc_priv *priv = dev_get_priv(dev);
44 struct mmc *mmc = mmc_get_mmc_dev(dev);
45 static ulong erase_start, erase_end;
46
Simon Glassf376a3c2016-05-01 13:52:42 -060047 switch (cmd->cmdidx) {
48 case MMC_CMD_ALL_SEND_CID:
Simon Glass5abd58f2019-09-25 08:56:00 -060049 memset(cmd->response, '\0', sizeof(cmd->response));
Simon Glassf376a3c2016-05-01 13:52:42 -060050 break;
51 case SD_CMD_SEND_RELATIVE_ADDR:
52 cmd->response[0] = 0 << 16; /* mmc->rca */
53 case MMC_CMD_GO_IDLE_STATE:
54 break;
55 case SD_CMD_SEND_IF_COND:
56 cmd->response[0] = 0xaa;
57 break;
58 case MMC_CMD_SEND_STATUS:
59 cmd->response[0] = MMC_STATUS_RDY_FOR_DATA;
60 break;
61 case MMC_CMD_SELECT_CARD:
62 break;
63 case MMC_CMD_SEND_CSD:
64 cmd->response[0] = 0;
Sean Anderson3f6fb772021-02-05 09:38:53 -050065 cmd->response[1] = (MMC_BL_LEN_SHIFT << 16) |
Simon Glass0bf61ac2021-10-23 17:25:59 -060066 ((priv->csize >> 16) & 0x3f);
67 cmd->response[2] = (priv->csize & 0xffff) << 16;
Simon Glass5abd58f2019-09-25 08:56:00 -060068 cmd->response[3] = 0;
Simon Glassf376a3c2016-05-01 13:52:42 -060069 break;
70 case SD_CMD_SWITCH_FUNC: {
Jean-Jacques Hiblot49f89252017-12-14 11:47:14 +010071 if (!data)
72 break;
Simon Glassf376a3c2016-05-01 13:52:42 -060073 u32 *resp = (u32 *)data->dest;
Simon Glass5abd58f2019-09-25 08:56:00 -060074 resp[3] = 0;
Simon Glassf376a3c2016-05-01 13:52:42 -060075 resp[7] = cpu_to_be32(SD_HIGHSPEED_BUSY);
Jean-Jacques Hiblot49f89252017-12-14 11:47:14 +010076 if ((cmd->cmdarg & 0xF) == UHS_SDR12_BUS_SPEED)
77 resp[4] = (cmd->cmdarg & 0xF) << 24;
Simon Glassf376a3c2016-05-01 13:52:42 -060078 break;
79 }
80 case MMC_CMD_READ_SINGLE_BLOCK:
Simon Glassf376a3c2016-05-01 13:52:42 -060081 case MMC_CMD_READ_MULTIPLE_BLOCK:
Sean Anderson3f6fb772021-02-05 09:38:53 -050082 memcpy(data->dest, &priv->buf[cmd->cmdarg * data->blocksize],
83 data->blocks * data->blocksize);
84 break;
85 case MMC_CMD_WRITE_SINGLE_BLOCK:
86 case MMC_CMD_WRITE_MULTIPLE_BLOCK:
87 memcpy(&priv->buf[cmd->cmdarg * data->blocksize], data->src,
88 data->blocks * data->blocksize);
Simon Glassf376a3c2016-05-01 13:52:42 -060089 break;
90 case MMC_CMD_STOP_TRANSMISSION:
91 break;
Sean Anderson3f6fb772021-02-05 09:38:53 -050092 case SD_CMD_ERASE_WR_BLK_START:
93 erase_start = cmd->cmdarg;
94 break;
95 case SD_CMD_ERASE_WR_BLK_END:
96 erase_end = cmd->cmdarg;
97 break;
98 case MMC_CMD_ERASE:
99 memset(&priv->buf[erase_start * mmc->write_bl_len], '\0',
100 (erase_end - erase_start + 1) * mmc->write_bl_len);
101 break;
Simon Glassf376a3c2016-05-01 13:52:42 -0600102 case SD_CMD_APP_SEND_OP_COND:
103 cmd->response[0] = OCR_BUSY | OCR_HCS;
104 cmd->response[1] = 0;
105 cmd->response[2] = 0;
106 break;
107 case MMC_CMD_APP_CMD:
108 break;
109 case MMC_CMD_SET_BLOCKLEN:
110 debug("block len %d\n", cmd->cmdarg);
111 break;
112 case SD_CMD_APP_SEND_SCR: {
113 u32 *scr = (u32 *)data->dest;
114
115 scr[0] = cpu_to_be32(2 << 24 | 1 << 15); /* SD version 3 */
116 break;
117 }
118 default:
119 debug("%s: Unknown command %d\n", __func__, cmd->cmdidx);
120 break;
121 }
122
123 return 0;
124}
125
Simon Glass9a46bd32016-06-12 23:30:26 -0600126static int sandbox_mmc_set_ios(struct udevice *dev)
Simon Glassf376a3c2016-05-01 13:52:42 -0600127{
128 return 0;
129}
130
Simon Glass9a46bd32016-06-12 23:30:26 -0600131static int sandbox_mmc_get_cd(struct udevice *dev)
Simon Glassf376a3c2016-05-01 13:52:42 -0600132{
133 return 1;
134}
135
Simon Glass9a46bd32016-06-12 23:30:26 -0600136static const struct dm_mmc_ops sandbox_mmc_ops = {
Simon Glassf376a3c2016-05-01 13:52:42 -0600137 .send_cmd = sandbox_mmc_send_cmd,
138 .set_ios = sandbox_mmc_set_ios,
Simon Glass9a46bd32016-06-12 23:30:26 -0600139 .get_cd = sandbox_mmc_get_cd,
Simon Glassf376a3c2016-05-01 13:52:42 -0600140};
141
Simon Glass6b165ab2021-07-05 16:32:58 -0600142static int sandbox_mmc_of_to_plat(struct udevice *dev)
143{
144 struct sandbox_mmc_plat *plat = dev_get_plat(dev);
145 struct mmc_config *cfg = &plat->cfg;
146 struct blk_desc *blk;
147 int ret;
148
Simon Glass0bf61ac2021-10-23 17:25:59 -0600149 plat->fname = dev_read_string(dev, "filename");
150
Simon Glass6b165ab2021-07-05 16:32:58 -0600151 ret = mmc_of_parse(dev, cfg);
152 if (ret)
153 return ret;
154 blk = mmc_get_blk_desc(&plat->mmc);
155 if (blk)
156 blk->removable = !(cfg->host_caps & MMC_CAP_NONREMOVABLE);
157
158 return 0;
159}
160
161static int sandbox_mmc_probe(struct udevice *dev)
Simon Glassf376a3c2016-05-01 13:52:42 -0600162{
Simon Glassc69cda22020-12-03 16:55:20 -0700163 struct sandbox_mmc_plat *plat = dev_get_plat(dev);
Simon Glass0bf61ac2021-10-23 17:25:59 -0600164 struct sandbox_mmc_priv *priv = dev_get_priv(dev);
165 int ret;
166
167 if (plat->fname) {
168 ret = os_map_file(plat->fname, OS_O_RDWR | OS_O_CREAT,
169 (void **)&priv->buf, &priv->size);
170 if (ret) {
171 log_err("%s: Unable to map file '%s'\n", dev->name,
172 plat->fname);
173 return ret;
174 }
175 priv->csize = priv->size / SIZE_MULTIPLE - 1;
176 } else {
177 priv->csize = 0;
178 priv->size = (priv->csize + 1) * SIZE_MULTIPLE; /* 1 MiB */
179
180 priv->buf = malloc(priv->size);
181 if (!priv->buf) {
182 log_err("%s: Not enough memory (%x bytes)\n",
183 dev->name, priv->size);
184 return -ENOMEM;
185 }
186 }
Simon Glassf376a3c2016-05-01 13:52:42 -0600187
188 return mmc_init(&plat->mmc);
189}
190
Simon Glass0bf61ac2021-10-23 17:25:59 -0600191static int sandbox_mmc_remove(struct udevice *dev)
192{
193 struct sandbox_mmc_plat *plat = dev_get_plat(dev);
194 struct sandbox_mmc_priv *priv = dev_get_priv(dev);
195
196 if (plat->fname)
197 os_unmap(priv->buf, priv->size);
198 else
199 free(priv->buf);
200
201 return 0;
202}
203
Simon Glass6b165ab2021-07-05 16:32:58 -0600204static int sandbox_mmc_bind(struct udevice *dev)
Simon Glassf376a3c2016-05-01 13:52:42 -0600205{
Simon Glassc69cda22020-12-03 16:55:20 -0700206 struct sandbox_mmc_plat *plat = dev_get_plat(dev);
Simon Glassf376a3c2016-05-01 13:52:42 -0600207 struct mmc_config *cfg = &plat->cfg;
Simon Glassf376a3c2016-05-01 13:52:42 -0600208
209 cfg->name = dev->name;
Simon Glassf376a3c2016-05-01 13:52:42 -0600210 cfg->host_caps = MMC_MODE_HS_52MHz | MMC_MODE_HS | MMC_MODE_8BIT;
211 cfg->voltages = MMC_VDD_165_195 | MMC_VDD_32_33 | MMC_VDD_33_34;
212 cfg->f_min = 1000000;
213 cfg->f_max = 52000000;
214 cfg->b_max = U32_MAX;
215
Masahiro Yamada24f5aec2016-09-06 22:17:32 +0900216 return mmc_bind(dev, &plat->mmc, cfg);
Simon Glassf376a3c2016-05-01 13:52:42 -0600217}
218
Simon Glass6b165ab2021-07-05 16:32:58 -0600219static int sandbox_mmc_unbind(struct udevice *dev)
Simon Glassf376a3c2016-05-01 13:52:42 -0600220{
221 mmc_unbind(dev);
222
223 return 0;
224}
225
Simon Glass8e6cc462015-07-06 12:54:32 -0600226static const struct udevice_id sandbox_mmc_ids[] = {
227 { .compatible = "sandbox,mmc" },
228 { }
229};
230
Simon Glassf376a3c2016-05-01 13:52:42 -0600231U_BOOT_DRIVER(mmc_sandbox) = {
Simon Glass8e6cc462015-07-06 12:54:32 -0600232 .name = "mmc_sandbox",
233 .id = UCLASS_MMC,
234 .of_match = sandbox_mmc_ids,
Simon Glass9a46bd32016-06-12 23:30:26 -0600235 .ops = &sandbox_mmc_ops,
Simon Glassf376a3c2016-05-01 13:52:42 -0600236 .bind = sandbox_mmc_bind,
237 .unbind = sandbox_mmc_unbind,
Simon Glass6b165ab2021-07-05 16:32:58 -0600238 .of_to_plat = sandbox_mmc_of_to_plat,
Simon Glassf376a3c2016-05-01 13:52:42 -0600239 .probe = sandbox_mmc_probe,
Simon Glass0bf61ac2021-10-23 17:25:59 -0600240 .remove = sandbox_mmc_remove,
Sean Anderson3f6fb772021-02-05 09:38:53 -0500241 .priv_auto = sizeof(struct sandbox_mmc_priv),
242 .plat_auto = sizeof(struct sandbox_mmc_plat),
Simon Glass8e6cc462015-07-06 12:54:32 -0600243};