blob: d7d5361fd50b29c535f83185a40e475f053e7352 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Simon Glassa8cb4fb2015-08-30 16:55:37 -06002/*
3 * Copyright (c) 2013 Google, Inc
Simon Glassa8cb4fb2015-08-30 16:55:37 -06004 */
5
6#include <common.h>
7#include <clk.h>
8#include <dm.h>
Simon Glassbfeb4432016-07-04 11:58:27 -06009#include <dt-structs.h>
Simon Glassa8cb4fb2015-08-30 16:55:37 -060010#include <dwmmc.h>
11#include <errno.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -060012#include <log.h>
Simon Glassbfeb4432016-07-04 11:58:27 -060013#include <mapmem.h>
Simon Glasse1efec42016-01-21 19:43:34 -070014#include <pwrseq.h>
Simon Glassa8cb4fb2015-08-30 16:55:37 -060015#include <syscon.h>
Simon Glasse1efec42016-01-21 19:43:34 -070016#include <asm/gpio.h>
Kever Yang15f09a12019-03-28 11:01:23 +080017#include <asm/arch-rockchip/clock.h>
18#include <asm/arch-rockchip/periph.h>
Simon Glassc05ed002020-05-10 11:40:11 -060019#include <linux/delay.h>
Simon Glassa8cb4fb2015-08-30 16:55:37 -060020#include <linux/err.h>
21
Simon Glassf6e41d12016-05-14 14:03:08 -060022struct rockchip_mmc_plat {
Simon Glassbfeb4432016-07-04 11:58:27 -060023#if CONFIG_IS_ENABLED(OF_PLATDATA)
24 struct dtd_rockchip_rk3288_dw_mshc dtplat;
25#endif
Simon Glassf6e41d12016-05-14 14:03:08 -060026 struct mmc_config cfg;
27 struct mmc mmc;
28};
29
Simon Glassa8cb4fb2015-08-30 16:55:37 -060030struct rockchip_dwmmc_priv {
Stephen Warren135aa952016-06-17 09:44:00 -060031 struct clk clk;
Simon Glassa8cb4fb2015-08-30 16:55:37 -060032 struct dwmci_host host;
Simon Glass6809b042016-07-04 11:58:26 -060033 int fifo_depth;
34 bool fifo_mode;
35 u32 minmax[2];
Simon Glassa8cb4fb2015-08-30 16:55:37 -060036};
37
38static uint rockchip_dwmmc_get_mmc_clk(struct dwmci_host *host, uint freq)
39{
40 struct udevice *dev = host->priv;
41 struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
42 int ret;
43
Stephen Warren135aa952016-06-17 09:44:00 -060044 ret = clk_set_rate(&priv->clk, freq);
Simon Glassa8cb4fb2015-08-30 16:55:37 -060045 if (ret < 0) {
Kever Yang419b0802017-06-14 16:31:49 +080046 debug("%s: err=%d\n", __func__, ret);
Simon Glassa8cb4fb2015-08-30 16:55:37 -060047 return ret;
48 }
49
50 return freq;
51}
52
Simon Glassd1998a92020-12-03 16:55:21 -070053static int rockchip_dwmmc_of_to_plat(struct udevice *dev)
Simon Glassa8cb4fb2015-08-30 16:55:37 -060054{
Simon Glassbfeb4432016-07-04 11:58:27 -060055#if !CONFIG_IS_ENABLED(OF_PLATDATA)
Simon Glassa8cb4fb2015-08-30 16:55:37 -060056 struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
57 struct dwmci_host *host = &priv->host;
58
59 host->name = dev->name;
Philipp Tomsichbe5f04e2017-09-11 22:04:15 +020060 host->ioaddr = dev_read_addr_ptr(dev);
Philipp Tomsichfd1bf8d2017-06-07 18:46:00 +020061 host->buswidth = dev_read_u32_default(dev, "bus-width", 4);
Simon Glassa8cb4fb2015-08-30 16:55:37 -060062 host->get_mmc_clk = rockchip_dwmmc_get_mmc_clk;
63 host->priv = dev;
64
huang linace21982015-11-18 09:37:25 +080065 /* use non-removeable as sdcard and emmc as judgement */
Philipp Tomsichfd1bf8d2017-06-07 18:46:00 +020066 if (dev_read_bool(dev, "non-removable"))
huang lin65793852016-01-08 14:06:49 +080067 host->dev_index = 0;
68 else
huang linace21982015-11-18 09:37:25 +080069 host->dev_index = 1;
Simon Glassa8cb4fb2015-08-30 16:55:37 -060070
Philipp Tomsichfd1bf8d2017-06-07 18:46:00 +020071 priv->fifo_depth = dev_read_u32_default(dev, "fifo-depth", 0);
72
Simon Glass6809b042016-07-04 11:58:26 -060073 if (priv->fifo_depth < 0)
74 return -EINVAL;
Philipp Tomsichfd1bf8d2017-06-07 18:46:00 +020075 priv->fifo_mode = dev_read_bool(dev, "fifo-mode");
Philipp Tomsichff71f9a2017-04-25 09:52:07 +020076
Heiko Stuebnerc8dd0e42019-11-19 12:04:01 +010077#ifdef CONFIG_SPL_BUILD
78 if (!priv->fifo_mode)
79 priv->fifo_mode = dev_read_bool(dev, "u-boot,spl-fifo-mode");
80#endif
81
Philipp Tomsichff71f9a2017-04-25 09:52:07 +020082 /*
83 * 'clock-freq-min-max' is deprecated
84 * (see https://github.com/torvalds/linux/commit/b023030f10573de738bbe8df63d43acab64c9f7b)
85 */
Philipp Tomsichfd1bf8d2017-06-07 18:46:00 +020086 if (dev_read_u32_array(dev, "clock-freq-min-max", priv->minmax, 2)) {
87 int val = dev_read_u32_default(dev, "max-frequency", -EINVAL);
Philipp Tomsichff71f9a2017-04-25 09:52:07 +020088
89 if (val < 0)
90 return val;
91
92 priv->minmax[0] = 400000; /* 400 kHz */
93 priv->minmax[1] = val;
94 } else {
95 debug("%s: 'clock-freq-min-max' property was deprecated.\n",
96 __func__);
97 }
Simon Glassbfeb4432016-07-04 11:58:27 -060098#endif
Simon Glassa8cb4fb2015-08-30 16:55:37 -060099 return 0;
100}
101
102static int rockchip_dwmmc_probe(struct udevice *dev)
103{
Simon Glassc69cda22020-12-03 16:55:20 -0700104 struct rockchip_mmc_plat *plat = dev_get_plat(dev);
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600105 struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
106 struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
107 struct dwmci_host *host = &priv->host;
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600108 int ret;
109
Simon Glassbfeb4432016-07-04 11:58:27 -0600110#if CONFIG_IS_ENABLED(OF_PLATDATA)
111 struct dtd_rockchip_rk3288_dw_mshc *dtplat = &plat->dtplat;
112
113 host->name = dev->name;
114 host->ioaddr = map_sysmem(dtplat->reg[0], dtplat->reg[1]);
115 host->buswidth = dtplat->bus_width;
116 host->get_mmc_clk = rockchip_dwmmc_get_mmc_clk;
117 host->priv = dev;
118 host->dev_index = 0;
119 priv->fifo_depth = dtplat->fifo_depth;
120 priv->fifo_mode = 0;
Kever Yang80935292017-06-14 16:31:46 +0800121 priv->minmax[0] = 400000; /* 400 kHz */
122 priv->minmax[1] = dtplat->max_frequency;
Simon Glassbfeb4432016-07-04 11:58:27 -0600123
Walter Lozano51f12632020-06-25 01:10:13 -0300124 ret = clk_get_by_driver_info(dev, dtplat->clocks, &priv->clk);
Simon Glassbfeb4432016-07-04 11:58:27 -0600125 if (ret < 0)
126 return ret;
127#else
Kever Yang419b0802017-06-14 16:31:49 +0800128 ret = clk_get_by_index(dev, 0, &priv->clk);
Simon Glass898d6432016-01-21 19:43:38 -0700129 if (ret < 0)
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600130 return ret;
Simon Glassbfeb4432016-07-04 11:58:27 -0600131#endif
huang lin28637242015-11-17 14:20:24 +0800132 host->fifoth_val = MSIZE(0x2) |
Simon Glass6809b042016-07-04 11:58:26 -0600133 RX_WMARK(priv->fifo_depth / 2 - 1) |
134 TX_WMARK(priv->fifo_depth / 2);
huang lin28637242015-11-17 14:20:24 +0800135
Simon Glass6809b042016-07-04 11:58:26 -0600136 host->fifo_mode = priv->fifo_mode;
huang lin28637242015-11-17 14:20:24 +0800137
Jaehoon Chung9d7e6612021-02-16 10:16:54 +0900138#ifdef CONFIG_MMC_PWRSEQ
Simon Glasse1efec42016-01-21 19:43:34 -0700139 /* Enable power if needed */
Jaehoon Chung9d7e6612021-02-16 10:16:54 +0900140 ret = mmc_pwrseq_get_power(dev, &plat->cfg);
Simon Glasse1efec42016-01-21 19:43:34 -0700141 if (!ret) {
Jaehoon Chung9d7e6612021-02-16 10:16:54 +0900142 ret = pwrseq_set_power(plat->cfg.pwr_dev, true);
Simon Glasse1efec42016-01-21 19:43:34 -0700143 if (ret)
144 return ret;
145 }
146#endif
Jaehoon Chunge5113c32016-09-23 19:13:16 +0900147 dwmci_setup_cfg(&plat->cfg, host, priv->minmax[1], priv->minmax[0]);
Simon Glassf6e41d12016-05-14 14:03:08 -0600148 host->mmc = &plat->mmc;
Simon Glassf6e41d12016-05-14 14:03:08 -0600149 host->mmc->priv = &priv->host;
Simon Glasscffe5d82016-05-01 13:52:34 -0600150 host->mmc->dev = dev;
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600151 upriv->mmc = host->mmc;
152
Simon Glass42b37d82016-06-12 23:30:24 -0600153 return dwmci_probe(dev);
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600154}
155
Simon Glassf6e41d12016-05-14 14:03:08 -0600156static int rockchip_dwmmc_bind(struct udevice *dev)
157{
Simon Glassc69cda22020-12-03 16:55:20 -0700158 struct rockchip_mmc_plat *plat = dev_get_plat(dev);
Simon Glassf6e41d12016-05-14 14:03:08 -0600159
Masahiro Yamada24f5aec2016-09-06 22:17:32 +0900160 return dwmci_bind(dev, &plat->mmc, &plat->cfg);
Simon Glassf6e41d12016-05-14 14:03:08 -0600161}
162
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600163static const struct udevice_id rockchip_dwmmc_ids[] = {
Heiko Stuebner26a52f32018-09-21 10:59:46 +0200164 { .compatible = "rockchip,rk2928-dw-mshc" },
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600165 { .compatible = "rockchip,rk3288-dw-mshc" },
166 { }
167};
168
Walter Lozanoe3e24702020-06-25 01:10:04 -0300169U_BOOT_DRIVER(rockchip_rk3288_dw_mshc) = {
Simon Glassbfeb4432016-07-04 11:58:27 -0600170 .name = "rockchip_rk3288_dw_mshc",
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600171 .id = UCLASS_MMC,
172 .of_match = rockchip_dwmmc_ids,
Simon Glassd1998a92020-12-03 16:55:21 -0700173 .of_to_plat = rockchip_dwmmc_of_to_plat,
Simon Glass42b37d82016-06-12 23:30:24 -0600174 .ops = &dm_dwmci_ops,
Simon Glassf6e41d12016-05-14 14:03:08 -0600175 .bind = rockchip_dwmmc_bind,
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600176 .probe = rockchip_dwmmc_probe,
Simon Glass41575d82020-12-03 16:55:17 -0700177 .priv_auto = sizeof(struct rockchip_dwmmc_priv),
Simon Glasscaa4daa2020-12-03 16:55:18 -0700178 .plat_auto = sizeof(struct rockchip_mmc_plat),
Simon Glassa8cb4fb2015-08-30 16:55:37 -0600179};
Simon Glasse1efec42016-01-21 19:43:34 -0700180
Simon Glassbdf8fd72020-12-28 20:34:57 -0700181DM_DRIVER_ALIAS(rockchip_rk3288_dw_mshc, rockchip_rk3328_dw_mshc)
182DM_DRIVER_ALIAS(rockchip_rk3288_dw_mshc, rockchip_rk3368_dw_mshc)