blob: 2cec5b9ae3248e544ad02d327e685a56bf9edf01 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Peter Griffinaf0c1be2015-07-30 18:55:22 +01002/*
3 * (C) Copyright 2015 Linaro
4 * peter.griffin <peter.griffin@linaro.org>
Peter Griffinaf0c1be2015-07-30 18:55:22 +01005 */
6
7#include <common.h>
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +05308#include <dm.h>
Peter Griffinaf0c1be2015-07-30 18:55:22 +01009#include <dwmmc.h>
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053010#include <errno.h>
11#include <fdtdec.h>
Peter Griffinaf0c1be2015-07-30 18:55:22 +010012#include <malloc.h>
Simon Glass3ba929a2020-10-30 21:38:53 -060013#include <asm/global_data.h>
Peter Griffinaf0c1be2015-07-30 18:55:22 +010014
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053015DECLARE_GLOBAL_DATA_PTR;
Peter Griffinaf0c1be2015-07-30 18:55:22 +010016
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053017struct hi6220_dwmmc_plat {
18 struct mmc_config cfg;
19 struct mmc mmc;
20};
Peter Griffinaf0c1be2015-07-30 18:55:22 +010021
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053022struct hi6220_dwmmc_priv_data {
23 struct dwmci_host host;
24};
Peter Griffinaf0c1be2015-07-30 18:55:22 +010025
Manivannan Sadhasivam759c45a2019-08-02 20:40:10 +053026struct hisi_mmc_data {
27 unsigned int clock;
28 bool use_fifo;
29};
30
Simon Glassaad29ae2020-12-03 16:55:21 -070031static int hi6220_dwmmc_of_to_plat(struct udevice *dev)
Peter Griffinaf0c1be2015-07-30 18:55:22 +010032{
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053033 struct hi6220_dwmmc_priv_data *priv = dev_get_priv(dev);
34 struct dwmci_host *host = &priv->host;
Peter Griffinaf0c1be2015-07-30 18:55:22 +010035
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053036 host->name = dev->name;
Masahiro Yamada1096ae12020-07-17 14:36:46 +090037 host->ioaddr = dev_read_addr_ptr(dev);
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053038 host->buswidth = fdtdec_get_int(gd->fdt_blob, dev_of_offset(dev),
39 "bus-width", 4);
Peter Griffinaf0c1be2015-07-30 18:55:22 +010040
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053041 /* use non-removable property for differentiating SD card and eMMC */
42 if (dev_read_bool(dev, "non-removable"))
43 host->dev_index = 0;
44 else
45 host->dev_index = 1;
46
47 host->priv = priv;
48
Peter Griffinaf0c1be2015-07-30 18:55:22 +010049 return 0;
50}
51
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053052static int hi6220_dwmmc_probe(struct udevice *dev)
Peter Griffinaf0c1be2015-07-30 18:55:22 +010053{
Simon Glassfa20e932020-12-03 16:55:20 -070054 struct hi6220_dwmmc_plat *plat = dev_get_plat(dev);
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053055 struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
56 struct hi6220_dwmmc_priv_data *priv = dev_get_priv(dev);
57 struct dwmci_host *host = &priv->host;
Manivannan Sadhasivam759c45a2019-08-02 20:40:10 +053058 struct hisi_mmc_data *mmc_data;
59
60 mmc_data = (struct hisi_mmc_data *)dev_get_driver_data(dev);
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053061
62 /* Use default bus speed due to absence of clk driver */
Manivannan Sadhasivam759c45a2019-08-02 20:40:10 +053063 host->bus_hz = mmc_data->clock;
Peter Griffinaf0c1be2015-07-30 18:55:22 +010064
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053065 dwmci_setup_cfg(&plat->cfg, host, host->bus_hz, 400000);
66 host->mmc = &plat->mmc;
Peter Griffinaf0c1be2015-07-30 18:55:22 +010067
Manivannan Sadhasivam759c45a2019-08-02 20:40:10 +053068 host->fifo_mode = mmc_data->use_fifo;
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053069 host->mmc->priv = &priv->host;
70 upriv->mmc = host->mmc;
71 host->mmc->dev = dev;
Peter Griffinaf0c1be2015-07-30 18:55:22 +010072
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053073 return dwmci_probe(dev);
Peter Griffinaf0c1be2015-07-30 18:55:22 +010074}
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053075
76static int hi6220_dwmmc_bind(struct udevice *dev)
77{
Simon Glassfa20e932020-12-03 16:55:20 -070078 struct hi6220_dwmmc_plat *plat = dev_get_plat(dev);
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053079 int ret;
80
81 ret = dwmci_bind(dev, &plat->mmc, &plat->cfg);
82 if (ret)
83 return ret;
84
85 return 0;
86}
87
Manivannan Sadhasivam759c45a2019-08-02 20:40:10 +053088static const struct hisi_mmc_data hi3660_mmc_data = {
89 .clock = 3200000,
90 .use_fifo = true,
91};
92
93static const struct hisi_mmc_data hi6220_mmc_data = {
94 .clock = 50000000,
95 .use_fifo = false,
96};
97
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +053098static const struct udevice_id hi6220_dwmmc_ids[] = {
Manivannan Sadhasivam759c45a2019-08-02 20:40:10 +053099 { .compatible = "hisilicon,hi6220-dw-mshc",
100 .data = (ulong)&hi6220_mmc_data },
101 { .compatible = "hisilicon,hi3798cv200-dw-mshc",
102 .data = (ulong)&hi6220_mmc_data },
103 { .compatible = "hisilicon,hi3660-dw-mshc",
104 .data = (ulong)&hi3660_mmc_data },
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +0530105 { }
106};
107
108U_BOOT_DRIVER(hi6220_dwmmc_drv) = {
109 .name = "hi6220_dwmmc",
110 .id = UCLASS_MMC,
111 .of_match = hi6220_dwmmc_ids,
Simon Glassaad29ae2020-12-03 16:55:21 -0700112 .of_to_plat = hi6220_dwmmc_of_to_plat,
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +0530113 .ops = &dm_dwmci_ops,
114 .bind = hi6220_dwmmc_bind,
115 .probe = hi6220_dwmmc_probe,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700116 .priv_auto = sizeof(struct hi6220_dwmmc_priv_data),
Simon Glass71fa5b42020-12-03 16:55:18 -0700117 .plat_auto = sizeof(struct hi6220_dwmmc_plat),
Manivannan Sadhasivam6ad4f8c2018-12-27 19:04:04 +0530118};