blob: 0481e276c0f1ef598c9c4a1926ff2a9efeed17c6 [file] [log] [blame]
Simon Glass1e8eb1b2015-06-23 15:38:48 -06001/*
2 * Copyright (C) 2015 Google, Inc
3 * Written by Simon Glass <sjg@chromium.org>
4 *
5 * SPDX-License-Identifier: GPL-2.0+
6 */
7
8#include <common.h>
9#include <mmc.h>
10#include <dm.h>
Simon Glass2f1ea902016-06-12 23:30:16 -060011#include <dm/device-internal.h>
Simon Glass1e8eb1b2015-06-23 15:38:48 -060012#include <dm/lists.h>
Simon Glass2f1ea902016-06-12 23:30:16 -060013#include "mmc_private.h"
Simon Glass1e8eb1b2015-06-23 15:38:48 -060014
Simon Glass394dfc02016-06-12 23:30:22 -060015int dm_mmc_send_cmd(struct udevice *dev, struct mmc_cmd *cmd,
16 struct mmc_data *data)
17{
18 struct mmc *mmc = mmc_get_mmc_dev(dev);
19 struct dm_mmc_ops *ops = mmc_get_ops(dev);
20 int ret;
21
22 mmmc_trace_before_send(mmc, cmd);
23 if (ops->send_cmd)
24 ret = ops->send_cmd(dev, cmd, data);
25 else
26 ret = -ENOSYS;
27 mmmc_trace_after_send(mmc, cmd, ret);
28
29 return ret;
30}
31
32int mmc_send_cmd(struct mmc *mmc, struct mmc_cmd *cmd, struct mmc_data *data)
33{
34 return dm_mmc_send_cmd(mmc->dev, cmd, data);
35}
36
37int dm_mmc_set_ios(struct udevice *dev)
38{
39 struct dm_mmc_ops *ops = mmc_get_ops(dev);
40
41 if (!ops->set_ios)
42 return -ENOSYS;
43 return ops->set_ios(dev);
44}
45
46int mmc_set_ios(struct mmc *mmc)
47{
48 return dm_mmc_set_ios(mmc->dev);
49}
50
Jean-Jacques Hiblot5f23d872017-09-21 16:30:01 +020051void dm_mmc_send_init_stream(struct udevice *dev)
52{
53 struct dm_mmc_ops *ops = mmc_get_ops(dev);
54
55 if (ops->send_init_stream)
56 ops->send_init_stream(dev);
57}
58
59void mmc_send_init_stream(struct mmc *mmc)
60{
61 dm_mmc_send_init_stream(mmc->dev);
62}
63
Jean-Jacques Hiblot6051e782017-11-30 17:44:01 +010064#if CONFIG_IS_ENABLED(MMC_UHS_SUPPORT)
Jean-Jacques Hiblotf4d5b3e2017-09-21 16:30:07 +020065int dm_mmc_wait_dat0(struct udevice *dev, int state, int timeout)
66{
67 struct dm_mmc_ops *ops = mmc_get_ops(dev);
68
69 if (!ops->wait_dat0)
70 return -ENOSYS;
71 return ops->wait_dat0(dev, state, timeout);
72}
73
74int mmc_wait_dat0(struct mmc *mmc, int state, int timeout)
75{
76 return dm_mmc_wait_dat0(mmc->dev, state, timeout);
77}
Jean-Jacques Hiblot6051e782017-11-30 17:44:01 +010078#endif
Jean-Jacques Hiblotf4d5b3e2017-09-21 16:30:07 +020079
Simon Glass394dfc02016-06-12 23:30:22 -060080int dm_mmc_get_wp(struct udevice *dev)
81{
82 struct dm_mmc_ops *ops = mmc_get_ops(dev);
83
84 if (!ops->get_wp)
85 return -ENOSYS;
86 return ops->get_wp(dev);
87}
88
89int mmc_getwp(struct mmc *mmc)
90{
91 return dm_mmc_get_wp(mmc->dev);
92}
93
94int dm_mmc_get_cd(struct udevice *dev)
95{
96 struct dm_mmc_ops *ops = mmc_get_ops(dev);
97
98 if (!ops->get_cd)
99 return -ENOSYS;
100 return ops->get_cd(dev);
101}
102
103int mmc_getcd(struct mmc *mmc)
104{
105 return dm_mmc_get_cd(mmc->dev);
106}
Simon Glass394dfc02016-06-12 23:30:22 -0600107
Jean-Jacques Hiblot6051e782017-11-30 17:44:01 +0100108#ifdef MMC_SUPPORTS_TUNING
Kishon Vijay Abraham Iae7174f2017-09-21 16:30:05 +0200109int dm_mmc_execute_tuning(struct udevice *dev, uint opcode)
110{
111 struct dm_mmc_ops *ops = mmc_get_ops(dev);
112
113 if (!ops->execute_tuning)
114 return -ENOSYS;
115 return ops->execute_tuning(dev, opcode);
116}
117
118int mmc_execute_tuning(struct mmc *mmc, uint opcode)
119{
120 return dm_mmc_execute_tuning(mmc->dev, opcode);
121}
Jean-Jacques Hiblot6051e782017-11-30 17:44:01 +0100122#endif
Kishon Vijay Abraham Iae7174f2017-09-21 16:30:05 +0200123
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100124int mmc_of_parse(struct udevice *dev, struct mmc_config *cfg)
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200125{
126 int val;
127
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100128 val = dev_read_u32_default(dev, "bus-width", 1);
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200129
130 switch (val) {
131 case 0x8:
132 cfg->host_caps |= MMC_MODE_8BIT;
133 /* fall through */
134 case 0x4:
135 cfg->host_caps |= MMC_MODE_4BIT;
136 /* fall through */
137 case 0x1:
138 cfg->host_caps |= MMC_MODE_1BIT;
139 break;
140 default:
Masahiro Yamada57b12d82017-12-30 02:00:07 +0900141 dev_err(dev, "Invalid \"bus-width\" value %u!\n", val);
142 return -EINVAL;
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200143 }
144
Masahiro Yamada191cd132017-12-30 02:00:06 +0900145 /* f_max is obtained from the optional "max-frequency" property */
146 dev_read_u32(dev, "max-frequency", &cfg->f_max);
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200147
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100148 if (dev_read_bool(dev, "cap-sd-highspeed"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200149 cfg->host_caps |= MMC_CAP(SD_HS);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100150 if (dev_read_bool(dev, "cap-mmc-highspeed"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200151 cfg->host_caps |= MMC_CAP(MMC_HS);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100152 if (dev_read_bool(dev, "sd-uhs-sdr12"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200153 cfg->host_caps |= MMC_CAP(UHS_SDR12);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100154 if (dev_read_bool(dev, "sd-uhs-sdr25"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200155 cfg->host_caps |= MMC_CAP(UHS_SDR25);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100156 if (dev_read_bool(dev, "sd-uhs-sdr50"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200157 cfg->host_caps |= MMC_CAP(UHS_SDR50);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100158 if (dev_read_bool(dev, "sd-uhs-sdr104"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200159 cfg->host_caps |= MMC_CAP(UHS_SDR104);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100160 if (dev_read_bool(dev, "sd-uhs-ddr50"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200161 cfg->host_caps |= MMC_CAP(UHS_DDR50);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100162 if (dev_read_bool(dev, "mmc-ddr-1_8v"))
163 cfg->host_caps |= MMC_CAP(MMC_DDR_52);
164 if (dev_read_bool(dev, "mmc-ddr-1_2v"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200165 cfg->host_caps |= MMC_CAP(MMC_DDR_52);
Jean-Jacques Hiblotd39be652017-11-30 17:43:55 +0100166 if (dev_read_bool(dev, "mmc-hs200-1_8v"))
167 cfg->host_caps |= MMC_CAP(MMC_HS_200);
168 if (dev_read_bool(dev, "mmc-hs200-1_2v"))
Kishon Vijay Abraham I422b2942017-09-21 16:30:13 +0200169 cfg->host_caps |= MMC_CAP(MMC_HS_200);
170
171 return 0;
172}
173
Simon Glass1e8eb1b2015-06-23 15:38:48 -0600174struct mmc *mmc_get_mmc_dev(struct udevice *dev)
175{
176 struct mmc_uclass_priv *upriv;
177
178 if (!device_active(dev))
179 return NULL;
180 upriv = dev_get_uclass_priv(dev);
181 return upriv->mmc;
182}
183
Simon Glass5f4bd8c2017-07-04 13:31:19 -0600184#if CONFIG_IS_ENABLED(BLK)
Simon Glass4bca6662016-05-01 13:52:39 -0600185struct mmc *find_mmc_device(int dev_num)
186{
187 struct udevice *dev, *mmc_dev;
188 int ret;
189
Simon Glassbcad6042017-05-27 11:37:19 -0600190 ret = blk_find_device(IF_TYPE_MMC, dev_num, &dev);
Simon Glass4bca6662016-05-01 13:52:39 -0600191
192 if (ret) {
193#if !defined(CONFIG_SPL_BUILD) || defined(CONFIG_SPL_LIBCOMMON_SUPPORT)
194 printf("MMC Device %d not found\n", dev_num);
195#endif
196 return NULL;
197 }
198
199 mmc_dev = dev_get_parent(dev);
200
Simon Glassbcad6042017-05-27 11:37:19 -0600201 struct mmc *mmc = mmc_get_mmc_dev(mmc_dev);
202
203 return mmc;
Simon Glass4bca6662016-05-01 13:52:39 -0600204}
205
206int get_mmc_num(void)
207{
Kever Yang38456602016-07-22 17:22:50 +0800208 return max((blk_find_max_devnum(IF_TYPE_MMC) + 1), 0);
Simon Glass4bca6662016-05-01 13:52:39 -0600209}
210
211int mmc_get_next_devnum(void)
212{
Masahiro Yamadaa6f07e52016-10-14 00:13:18 +0900213 return blk_find_max_devnum(IF_TYPE_MMC);
Simon Glass4bca6662016-05-01 13:52:39 -0600214}
215
216struct blk_desc *mmc_get_blk_desc(struct mmc *mmc)
217{
218 struct blk_desc *desc;
219 struct udevice *dev;
220
221 device_find_first_child(mmc->dev, &dev);
222 if (!dev)
223 return NULL;
224 desc = dev_get_uclass_platdata(dev);
225
226 return desc;
227}
228
229void mmc_do_preinit(void)
230{
231 struct udevice *dev;
232 struct uclass *uc;
233 int ret;
234
235 ret = uclass_get(UCLASS_MMC, &uc);
236 if (ret)
237 return;
238 uclass_foreach_dev(dev, uc) {
239 struct mmc *m = mmc_get_mmc_dev(dev);
240
241 if (!m)
242 continue;
243#ifdef CONFIG_FSL_ESDHC_ADAPTER_IDENT
244 mmc_set_preinit(m, 1);
245#endif
246 if (m->preinit)
247 mmc_start_init(m);
248 }
249}
250
251#if !defined(CONFIG_SPL_BUILD) || defined(CONFIG_SPL_LIBCOMMON_SUPPORT)
252void print_mmc_devices(char separator)
253{
254 struct udevice *dev;
255 char *mmc_type;
256 bool first = true;
257
258 for (uclass_first_device(UCLASS_MMC, &dev);
259 dev;
Xu Ziyuan77d747e2016-07-23 11:11:22 +0800260 uclass_next_device(&dev), first = false) {
Simon Glass4bca6662016-05-01 13:52:39 -0600261 struct mmc *m = mmc_get_mmc_dev(dev);
262
263 if (!first) {
264 printf("%c", separator);
265 if (separator != '\n')
266 puts(" ");
267 }
268 if (m->has_init)
269 mmc_type = IS_SD(m) ? "SD" : "eMMC";
270 else
271 mmc_type = NULL;
272
273 printf("%s: %d", m->cfg->name, mmc_get_blk_desc(m)->devnum);
274 if (mmc_type)
275 printf(" (%s)", mmc_type);
276 }
277
278 printf("\n");
279}
280
281#else
282void print_mmc_devices(char separator) { }
283#endif
Simon Glass2f1ea902016-06-12 23:30:16 -0600284
285int mmc_bind(struct udevice *dev, struct mmc *mmc, const struct mmc_config *cfg)
286{
287 struct blk_desc *bdesc;
288 struct udevice *bdev;
Jaehoon Chung2910da82017-02-02 13:41:14 +0900289 int ret, devnum = -1;
290
Simon Glass9e720b02017-04-23 20:02:09 -0600291 if (!mmc_get_ops(dev))
292 return -ENOSYS;
Jaehoon Chung2910da82017-02-02 13:41:14 +0900293#ifndef CONFIG_SPL_BUILD
294 /* Use the fixed index with aliase node's index */
Simon Glass6f4187c2017-05-18 20:09:36 -0600295 ret = dev_read_alias_seq(dev, &devnum);
296 debug("%s: alias ret=%d, devnum=%d\n", __func__, ret, devnum);
Jaehoon Chung2910da82017-02-02 13:41:14 +0900297#endif
Simon Glass2f1ea902016-06-12 23:30:16 -0600298
Jaehoon Chung2910da82017-02-02 13:41:14 +0900299 ret = blk_create_devicef(dev, "mmc_blk", "blk", IF_TYPE_MMC,
300 devnum, 512, 0, &bdev);
Simon Glass2f1ea902016-06-12 23:30:16 -0600301 if (ret) {
302 debug("Cannot create block device\n");
303 return ret;
304 }
305 bdesc = dev_get_uclass_platdata(bdev);
306 mmc->cfg = cfg;
307 mmc->priv = dev;
308
309 /* the following chunk was from mmc_register() */
310
311 /* Setup dsr related values */
312 mmc->dsr_imp = 0;
313 mmc->dsr = 0xffffffff;
314 /* Setup the universal parts of the block interface just once */
315 bdesc->removable = 1;
316
317 /* setup initial part type */
318 bdesc->part_type = cfg->part_type;
319 mmc->dev = dev;
320
321 return 0;
322}
323
324int mmc_unbind(struct udevice *dev)
325{
326 struct udevice *bdev;
327
328 device_find_first_child(dev, &bdev);
329 if (bdev) {
Stefan Roese80b5bc92017-03-20 12:51:48 +0100330 device_remove(bdev, DM_REMOVE_NORMAL);
Simon Glass2f1ea902016-06-12 23:30:16 -0600331 device_unbind(bdev);
332 }
333
334 return 0;
335}
336
337static int mmc_select_hwpart(struct udevice *bdev, int hwpart)
338{
339 struct udevice *mmc_dev = dev_get_parent(bdev);
340 struct mmc *mmc = mmc_get_mmc_dev(mmc_dev);
341 struct blk_desc *desc = dev_get_uclass_platdata(bdev);
Simon Glass2f1ea902016-06-12 23:30:16 -0600342
343 if (desc->hwpart == hwpart)
344 return 0;
345
346 if (mmc->part_config == MMCPART_NOAVAILABLE)
347 return -EMEDIUMTYPE;
348
Masahiro Yamadaa6f07e52016-10-14 00:13:18 +0900349 return mmc_switch_part(mmc, hwpart);
Simon Glass2f1ea902016-06-12 23:30:16 -0600350}
351
Fiach Antaw7cdbe502017-01-25 19:00:24 +1000352static int mmc_blk_probe(struct udevice *dev)
353{
Simon Glasse11b2392017-04-23 20:02:10 -0600354 struct udevice *mmc_dev = dev_get_parent(dev);
355 struct mmc_uclass_priv *upriv = dev_get_uclass_priv(mmc_dev);
356 struct mmc *mmc = upriv->mmc;
357 int ret;
358
359 ret = mmc_init(mmc);
360 if (ret) {
361 debug("%s: mmc_init() failed (err=%d)\n", __func__, ret);
362 return ret;
363 }
Fiach Antaw7cdbe502017-01-25 19:00:24 +1000364
Simon Glasse11b2392017-04-23 20:02:10 -0600365 return 0;
Fiach Antaw7cdbe502017-01-25 19:00:24 +1000366}
367
Simon Glass2f1ea902016-06-12 23:30:16 -0600368static const struct blk_ops mmc_blk_ops = {
369 .read = mmc_bread,
Jean-Jacques Hiblotd0531672018-01-04 15:23:32 +0100370#if CONFIG_IS_ENABLED(MMC_WRITE)
Simon Glass2f1ea902016-06-12 23:30:16 -0600371 .write = mmc_bwrite,
Simon Glassfcc53c12016-10-01 14:43:17 -0600372 .erase = mmc_berase,
Simon Glass2f1ea902016-06-12 23:30:16 -0600373#endif
374 .select_hwpart = mmc_select_hwpart,
375};
376
377U_BOOT_DRIVER(mmc_blk) = {
378 .name = "mmc_blk",
379 .id = UCLASS_BLK,
380 .ops = &mmc_blk_ops,
Fiach Antaw7cdbe502017-01-25 19:00:24 +1000381 .probe = mmc_blk_probe,
Simon Glass2f1ea902016-06-12 23:30:16 -0600382};
Simon Glass4bca6662016-05-01 13:52:39 -0600383#endif /* CONFIG_BLK */
384
Simon Glass1e8eb1b2015-06-23 15:38:48 -0600385U_BOOT_DRIVER(mmc) = {
386 .name = "mmc",
387 .id = UCLASS_MMC,
388};
389
390UCLASS_DRIVER(mmc) = {
391 .id = UCLASS_MMC,
392 .name = "mmc",
393 .flags = DM_UC_FLAG_SEQ_ALIAS,
394 .per_device_auto_alloc_size = sizeof(struct mmc_uclass_priv),
395};