rockchip_dw_mmc.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. /*
  2. * Copyright (c) 2013 Google, Inc
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. */
  6. #include <common.h>
  7. #include <clk.h>
  8. #include <dm.h>
  9. #include <dt-structs.h>
  10. #include <dwmmc.h>
  11. #include <errno.h>
  12. #include <mapmem.h>
  13. #include <pwrseq.h>
  14. #include <syscon.h>
  15. #include <asm/gpio.h>
  16. #include <asm/arch/clock.h>
  17. #include <asm/arch/periph.h>
  18. #include <linux/err.h>
  19. DECLARE_GLOBAL_DATA_PTR;
  20. struct rockchip_mmc_plat {
  21. #if CONFIG_IS_ENABLED(OF_PLATDATA)
  22. struct dtd_rockchip_rk3288_dw_mshc dtplat;
  23. #endif
  24. struct mmc_config cfg;
  25. struct mmc mmc;
  26. };
  27. struct rockchip_dwmmc_priv {
  28. struct clk clk;
  29. struct dwmci_host host;
  30. int fifo_depth;
  31. bool fifo_mode;
  32. u32 minmax[2];
  33. };
  34. static uint rockchip_dwmmc_get_mmc_clk(struct dwmci_host *host, uint freq)
  35. {
  36. struct udevice *dev = host->priv;
  37. struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
  38. int ret;
  39. ret = clk_set_rate(&priv->clk, freq);
  40. if (ret < 0) {
  41. debug("%s: err=%d\n", __func__, ret);
  42. return ret;
  43. }
  44. return freq;
  45. }
  46. static int rockchip_dwmmc_ofdata_to_platdata(struct udevice *dev)
  47. {
  48. #if !CONFIG_IS_ENABLED(OF_PLATDATA)
  49. struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
  50. struct dwmci_host *host = &priv->host;
  51. host->name = dev->name;
  52. host->ioaddr = (void *)dev_get_addr(dev);
  53. host->buswidth = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
  54. "bus-width", 4);
  55. host->get_mmc_clk = rockchip_dwmmc_get_mmc_clk;
  56. host->priv = dev;
  57. /* use non-removeable as sdcard and emmc as judgement */
  58. if (fdtdec_get_bool(gd->fdt_blob, dev->of_offset, "non-removable"))
  59. host->dev_index = 0;
  60. else
  61. host->dev_index = 1;
  62. priv->fifo_depth = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
  63. "fifo-depth", 0);
  64. if (priv->fifo_depth < 0)
  65. return -EINVAL;
  66. priv->fifo_mode = fdtdec_get_bool(gd->fdt_blob, dev->of_offset,
  67. "fifo-mode");
  68. if (fdtdec_get_int_array(gd->fdt_blob, dev->of_offset,
  69. "clock-freq-min-max", priv->minmax, 2))
  70. return -EINVAL;
  71. #endif
  72. return 0;
  73. }
  74. static int rockchip_dwmmc_probe(struct udevice *dev)
  75. {
  76. struct rockchip_mmc_plat *plat = dev_get_platdata(dev);
  77. struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
  78. struct rockchip_dwmmc_priv *priv = dev_get_priv(dev);
  79. struct dwmci_host *host = &priv->host;
  80. struct udevice *pwr_dev __maybe_unused;
  81. int ret;
  82. #if CONFIG_IS_ENABLED(OF_PLATDATA)
  83. struct dtd_rockchip_rk3288_dw_mshc *dtplat = &plat->dtplat;
  84. host->name = dev->name;
  85. host->ioaddr = map_sysmem(dtplat->reg[0], dtplat->reg[1]);
  86. host->buswidth = dtplat->bus_width;
  87. host->get_mmc_clk = rockchip_dwmmc_get_mmc_clk;
  88. host->priv = dev;
  89. host->dev_index = 0;
  90. priv->fifo_depth = dtplat->fifo_depth;
  91. priv->fifo_mode = 0;
  92. memcpy(priv->minmax, dtplat->clock_freq_min_max, sizeof(priv->minmax));
  93. ret = clk_get_by_index_platdata(dev, 0, dtplat->clocks, &priv->clk);
  94. if (ret < 0)
  95. return ret;
  96. #else
  97. ret = clk_get_by_index(dev, 0, &priv->clk);
  98. if (ret < 0)
  99. return ret;
  100. #endif
  101. host->fifoth_val = MSIZE(0x2) |
  102. RX_WMARK(priv->fifo_depth / 2 - 1) |
  103. TX_WMARK(priv->fifo_depth / 2);
  104. host->fifo_mode = priv->fifo_mode;
  105. #ifdef CONFIG_PWRSEQ
  106. /* Enable power if needed */
  107. ret = uclass_get_device_by_phandle(UCLASS_PWRSEQ, dev, "mmc-pwrseq",
  108. &pwr_dev);
  109. if (!ret) {
  110. ret = pwrseq_set_power(pwr_dev, true);
  111. if (ret)
  112. return ret;
  113. }
  114. #endif
  115. dwmci_setup_cfg(&plat->cfg, host, priv->minmax[1], priv->minmax[0]);
  116. host->mmc = &plat->mmc;
  117. host->mmc->priv = &priv->host;
  118. host->mmc->dev = dev;
  119. upriv->mmc = host->mmc;
  120. return dwmci_probe(dev);
  121. }
  122. static int rockchip_dwmmc_bind(struct udevice *dev)
  123. {
  124. struct rockchip_mmc_plat *plat = dev_get_platdata(dev);
  125. return dwmci_bind(dev, &plat->mmc, &plat->cfg);
  126. }
  127. static const struct udevice_id rockchip_dwmmc_ids[] = {
  128. { .compatible = "rockchip,rk3288-dw-mshc" },
  129. { }
  130. };
  131. U_BOOT_DRIVER(rockchip_dwmmc_drv) = {
  132. .name = "rockchip_rk3288_dw_mshc",
  133. .id = UCLASS_MMC,
  134. .of_match = rockchip_dwmmc_ids,
  135. .ofdata_to_platdata = rockchip_dwmmc_ofdata_to_platdata,
  136. .ops = &dm_dwmci_ops,
  137. .bind = rockchip_dwmmc_bind,
  138. .probe = rockchip_dwmmc_probe,
  139. .priv_auto_alloc_size = sizeof(struct rockchip_dwmmc_priv),
  140. .platdata_auto_alloc_size = sizeof(struct rockchip_mmc_plat),
  141. };
  142. #ifdef CONFIG_PWRSEQ
  143. static int rockchip_dwmmc_pwrseq_set_power(struct udevice *dev, bool enable)
  144. {
  145. struct gpio_desc reset;
  146. int ret;
  147. ret = gpio_request_by_name(dev, "reset-gpios", 0, &reset, GPIOD_IS_OUT);
  148. if (ret)
  149. return ret;
  150. dm_gpio_set_value(&reset, 1);
  151. udelay(1);
  152. dm_gpio_set_value(&reset, 0);
  153. udelay(200);
  154. return 0;
  155. }
  156. static const struct pwrseq_ops rockchip_dwmmc_pwrseq_ops = {
  157. .set_power = rockchip_dwmmc_pwrseq_set_power,
  158. };
  159. static const struct udevice_id rockchip_dwmmc_pwrseq_ids[] = {
  160. { .compatible = "mmc-pwrseq-emmc" },
  161. { }
  162. };
  163. U_BOOT_DRIVER(rockchip_dwmmc_pwrseq_drv) = {
  164. .name = "mmc_pwrseq_emmc",
  165. .id = UCLASS_PWRSEQ,
  166. .of_match = rockchip_dwmmc_pwrseq_ids,
  167. .ops = &rockchip_dwmmc_pwrseq_ops,
  168. };
  169. #endif