898d64395c
At present we use the same peripheral ID for clocks and pinctrl. While this works it is probably better to use the device tree clock binding ID for clocks. We can use the clk_get_by_index() function to find this. Update the clock drivers and the code that uses them. Signed-off-by: Simon Glass <sjg@chromium.org>
159 lines
3.5 KiB
C
159 lines
3.5 KiB
C
/*
|
|
* Copyright (c) 2013 Google, Inc
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <clk.h>
|
|
#include <dm.h>
|
|
#include <dwmmc.h>
|
|
#include <errno.h>
|
|
#include <pwrseq.h>
|
|
#include <syscon.h>
|
|
#include <asm/gpio.h>
|
|
#include <asm/arch/clock.h>
|
|
#include <asm/arch/periph.h>
|
|
#include <linux/err.h>
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
struct rockchip_dwmmc_priv {
|
|
struct udevice *clk;
|
|
int periph;
|
|
struct dwmci_host host;
|
|
};
|
|
|
|
static uint rockchip_dwmmc_get_mmc_clk(struct dwmci_host *host, uint freq)
|
|
{
|
|
struct udevice *dev = host->priv;
|
|
struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
|
|
int ret;
|
|
|
|
ret = clk_set_periph_rate(priv->clk, priv->periph, freq);
|
|
if (ret < 0) {
|
|
debug("%s: err=%d\n", __func__, ret);
|
|
return ret;
|
|
}
|
|
|
|
return freq;
|
|
}
|
|
|
|
static int rockchip_dwmmc_ofdata_to_platdata(struct udevice *dev)
|
|
{
|
|
struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
|
|
struct dwmci_host *host = &priv->host;
|
|
|
|
host->name = dev->name;
|
|
host->ioaddr = (void *)dev_get_addr(dev);
|
|
host->buswidth = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
|
|
"bus-width", 4);
|
|
host->get_mmc_clk = rockchip_dwmmc_get_mmc_clk;
|
|
host->priv = dev;
|
|
|
|
/* use non-removeable as sdcard and emmc as judgement */
|
|
if (fdtdec_get_bool(gd->fdt_blob, dev->of_offset, "non-removable"))
|
|
host->dev_index = 0;
|
|
else
|
|
host->dev_index = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rockchip_dwmmc_probe(struct udevice *dev)
|
|
{
|
|
struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
|
|
struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
|
|
struct dwmci_host *host = &priv->host;
|
|
struct udevice *pwr_dev __maybe_unused;
|
|
u32 minmax[2];
|
|
int ret;
|
|
int fifo_depth;
|
|
|
|
ret = clk_get_by_index(dev, 0, &priv->clk);
|
|
if (ret < 0)
|
|
return ret;
|
|
priv->periph = ret;
|
|
|
|
if (fdtdec_get_int_array(gd->fdt_blob, dev->of_offset,
|
|
"clock-freq-min-max", minmax, 2))
|
|
return -EINVAL;
|
|
|
|
fifo_depth = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
|
|
"fifo-depth", 0);
|
|
if (fifo_depth < 0)
|
|
return -EINVAL;
|
|
|
|
host->fifoth_val = MSIZE(0x2) |
|
|
RX_WMARK(fifo_depth / 2 - 1) | TX_WMARK(fifo_depth / 2);
|
|
|
|
if (fdtdec_get_bool(gd->fdt_blob, dev->of_offset, "fifo-mode"))
|
|
host->fifo_mode = true;
|
|
|
|
#ifdef CONFIG_PWRSEQ
|
|
/* Enable power if needed */
|
|
ret = uclass_get_device_by_phandle(UCLASS_PWRSEQ, dev, "mmc-pwrseq",
|
|
&pwr_dev);
|
|
if (!ret) {
|
|
ret = pwrseq_set_power(pwr_dev, true);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
#endif
|
|
ret = add_dwmci(host, minmax[1], minmax[0]);
|
|
if (ret)
|
|
return ret;
|
|
|
|
upriv->mmc = host->mmc;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct udevice_id rockchip_dwmmc_ids[] = {
|
|
{ .compatible = "rockchip,rk3288-dw-mshc" },
|
|
{ }
|
|
};
|
|
|
|
U_BOOT_DRIVER(rockchip_dwmmc_drv) = {
|
|
.name = "rockchip_dwmmc",
|
|
.id = UCLASS_MMC,
|
|
.of_match = rockchip_dwmmc_ids,
|
|
.ofdata_to_platdata = rockchip_dwmmc_ofdata_to_platdata,
|
|
.probe = rockchip_dwmmc_probe,
|
|
.priv_auto_alloc_size = sizeof(struct rockchip_dwmmc_priv),
|
|
};
|
|
|
|
#ifdef CONFIG_PWRSEQ
|
|
static int rockchip_dwmmc_pwrseq_set_power(struct udevice *dev, bool enable)
|
|
{
|
|
struct gpio_desc reset;
|
|
int ret;
|
|
|
|
ret = gpio_request_by_name(dev, "reset-gpios", 0, &reset, GPIOD_IS_OUT);
|
|
if (ret)
|
|
return ret;
|
|
dm_gpio_set_value(&reset, 1);
|
|
udelay(1);
|
|
dm_gpio_set_value(&reset, 0);
|
|
udelay(200);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct pwrseq_ops rockchip_dwmmc_pwrseq_ops = {
|
|
.set_power = rockchip_dwmmc_pwrseq_set_power,
|
|
};
|
|
|
|
static const struct udevice_id rockchip_dwmmc_pwrseq_ids[] = {
|
|
{ .compatible = "mmc-pwrseq-emmc" },
|
|
{ }
|
|
};
|
|
|
|
U_BOOT_DRIVER(rockchip_dwmmc_pwrseq_drv) = {
|
|
.name = "mmc_pwrseq_emmc",
|
|
.id = UCLASS_PWRSEQ,
|
|
.of_match = rockchip_dwmmc_pwrseq_ids,
|
|
.ops = &rockchip_dwmmc_pwrseq_ops,
|
|
};
|
|
#endif
|