bcm2835_sdhci.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. /*
  2. * This code was extracted from:
  3. * git://github.com/gonzoua/u-boot-pi.git master
  4. * and hence presumably (C) 2012 Oleksandr Tymoshenko
  5. *
  6. * Tweaks for U-Boot upstreaming
  7. * (C) 2012 Stephen Warren
  8. *
  9. * Portions (e.g. read/write macros, concepts for back-to-back register write
  10. * timing workarounds) obviously extracted from the Linux kernel at:
  11. * https://github.com/raspberrypi/linux.git rpi-3.6.y
  12. *
  13. * The Linux kernel code has the following (c) and license, which is hence
  14. * propagated to Oleksandr's tree and here:
  15. *
  16. * Support for SDHCI device on 2835
  17. * Based on sdhci-bcm2708.c (c) 2010 Broadcom
  18. *
  19. * This program is free software; you can redistribute it and/or modify
  20. * it under the terms of the GNU General Public License version 2 as
  21. * published by the Free Software Foundation.
  22. *
  23. * This program is distributed in the hope that it will be useful,
  24. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  25. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  26. * GNU General Public License for more details.
  27. *
  28. * You should have received a copy of the GNU General Public License
  29. * along with this program; if not, write to the Free Software
  30. * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  31. */
  32. /* Supports:
  33. * SDHCI platform device - Arasan SD controller in BCM2708
  34. *
  35. * Inspired by sdhci-pci.c, by Pierre Ossman
  36. */
  37. #include <common.h>
  38. #include <dm.h>
  39. #include <malloc.h>
  40. #include <memalign.h>
  41. #include <sdhci.h>
  42. #include <asm/arch/msg.h>
  43. #include <asm/arch/mbox.h>
  44. #include <mach/sdhci.h>
  45. #include <mach/timer.h>
  46. /* 400KHz is max freq for card ID etc. Use that as min */
  47. #define MIN_FREQ 400000
  48. #define SDHCI_BUFFER 0x20
  49. struct bcm2835_sdhci_plat {
  50. struct mmc_config cfg;
  51. struct mmc mmc;
  52. };
  53. struct bcm2835_sdhci_host {
  54. struct sdhci_host host;
  55. uint twoticks_delay;
  56. ulong last_write;
  57. };
  58. static inline struct bcm2835_sdhci_host *to_bcm(struct sdhci_host *host)
  59. {
  60. return (struct bcm2835_sdhci_host *)host;
  61. }
  62. static inline void bcm2835_sdhci_raw_writel(struct sdhci_host *host, u32 val,
  63. int reg)
  64. {
  65. struct bcm2835_sdhci_host *bcm_host = to_bcm(host);
  66. /*
  67. * The Arasan has a bugette whereby it may lose the content of
  68. * successive writes to registers that are within two SD-card clock
  69. * cycles of each other (a clock domain crossing problem).
  70. * It seems, however, that the data register does not have this problem.
  71. * (Which is just as well - otherwise we'd have to nobble the DMA engine
  72. * too)
  73. */
  74. if (reg != SDHCI_BUFFER) {
  75. while (timer_get_us() - bcm_host->last_write <
  76. bcm_host->twoticks_delay)
  77. ;
  78. }
  79. writel(val, host->ioaddr + reg);
  80. bcm_host->last_write = timer_get_us();
  81. }
  82. static inline u32 bcm2835_sdhci_raw_readl(struct sdhci_host *host, int reg)
  83. {
  84. return readl(host->ioaddr + reg);
  85. }
  86. static void bcm2835_sdhci_writel(struct sdhci_host *host, u32 val, int reg)
  87. {
  88. bcm2835_sdhci_raw_writel(host, val, reg);
  89. }
  90. static void bcm2835_sdhci_writew(struct sdhci_host *host, u16 val, int reg)
  91. {
  92. static u32 shadow;
  93. u32 oldval = (reg == SDHCI_COMMAND) ? shadow :
  94. bcm2835_sdhci_raw_readl(host, reg & ~3);
  95. u32 word_num = (reg >> 1) & 1;
  96. u32 word_shift = word_num * 16;
  97. u32 mask = 0xffff << word_shift;
  98. u32 newval = (oldval & ~mask) | (val << word_shift);
  99. if (reg == SDHCI_TRANSFER_MODE)
  100. shadow = newval;
  101. else
  102. bcm2835_sdhci_raw_writel(host, newval, reg & ~3);
  103. }
  104. static void bcm2835_sdhci_writeb(struct sdhci_host *host, u8 val, int reg)
  105. {
  106. u32 oldval = bcm2835_sdhci_raw_readl(host, reg & ~3);
  107. u32 byte_num = reg & 3;
  108. u32 byte_shift = byte_num * 8;
  109. u32 mask = 0xff << byte_shift;
  110. u32 newval = (oldval & ~mask) | (val << byte_shift);
  111. bcm2835_sdhci_raw_writel(host, newval, reg & ~3);
  112. }
  113. static u32 bcm2835_sdhci_readl(struct sdhci_host *host, int reg)
  114. {
  115. u32 val = bcm2835_sdhci_raw_readl(host, reg);
  116. return val;
  117. }
  118. static u16 bcm2835_sdhci_readw(struct sdhci_host *host, int reg)
  119. {
  120. u32 val = bcm2835_sdhci_raw_readl(host, (reg & ~3));
  121. u32 word_num = (reg >> 1) & 1;
  122. u32 word_shift = word_num * 16;
  123. u32 word = (val >> word_shift) & 0xffff;
  124. return word;
  125. }
  126. static u8 bcm2835_sdhci_readb(struct sdhci_host *host, int reg)
  127. {
  128. u32 val = bcm2835_sdhci_raw_readl(host, (reg & ~3));
  129. u32 byte_num = reg & 3;
  130. u32 byte_shift = byte_num * 8;
  131. u32 byte = (val >> byte_shift) & 0xff;
  132. return byte;
  133. }
  134. static const struct sdhci_ops bcm2835_ops = {
  135. .write_l = bcm2835_sdhci_writel,
  136. .write_w = bcm2835_sdhci_writew,
  137. .write_b = bcm2835_sdhci_writeb,
  138. .read_l = bcm2835_sdhci_readl,
  139. .read_w = bcm2835_sdhci_readw,
  140. .read_b = bcm2835_sdhci_readb,
  141. };
  142. static int bcm2835_sdhci_bind(struct udevice *dev)
  143. {
  144. struct bcm2835_sdhci_plat *plat = dev_get_platdata(dev);
  145. return sdhci_bind(dev, &plat->mmc, &plat->cfg);
  146. }
  147. static int bcm2835_sdhci_probe(struct udevice *dev)
  148. {
  149. struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
  150. struct bcm2835_sdhci_plat *plat = dev_get_platdata(dev);
  151. struct bcm2835_sdhci_host *priv = dev_get_priv(dev);
  152. struct sdhci_host *host = &priv->host;
  153. fdt_addr_t base;
  154. int emmc_freq;
  155. int ret;
  156. base = devfdt_get_addr(dev);
  157. if (base == FDT_ADDR_T_NONE)
  158. return -EINVAL;
  159. ret = bcm2835_get_mmc_clock();
  160. if (ret < 0) {
  161. debug("%s: Failed to set MMC clock (err=%d)\n", __func__, ret);
  162. return ret;
  163. }
  164. emmc_freq = ret;
  165. /*
  166. * See the comments in bcm2835_sdhci_raw_writel().
  167. *
  168. * This should probably be dynamically calculated based on the actual
  169. * frequency. However, this is the longest we'll have to wait, and
  170. * doesn't seem to slow access down too much, so the added complexity
  171. * doesn't seem worth it for now.
  172. *
  173. * 1/MIN_FREQ is (max) time per tick of eMMC clock.
  174. * 2/MIN_FREQ is time for two ticks.
  175. * Multiply by 1000000 to get uS per two ticks.
  176. * +1 for hack rounding.
  177. */
  178. priv->twoticks_delay = ((2 * 1000000) / MIN_FREQ) + 1;
  179. priv->last_write = 0;
  180. host->name = dev->name;
  181. host->ioaddr = (void *)base;
  182. host->quirks = SDHCI_QUIRK_BROKEN_VOLTAGE | SDHCI_QUIRK_BROKEN_R1B |
  183. SDHCI_QUIRK_WAIT_SEND_CMD | SDHCI_QUIRK_NO_HISPD_BIT;
  184. host->max_clk = emmc_freq;
  185. host->voltages = MMC_VDD_32_33 | MMC_VDD_33_34 | MMC_VDD_165_195;
  186. host->ops = &bcm2835_ops;
  187. ret = sdhci_setup_cfg(&plat->cfg, host, emmc_freq, MIN_FREQ);
  188. if (ret) {
  189. debug("%s: Failed to setup SDHCI (err=%d)\n", __func__, ret);
  190. return ret;
  191. }
  192. upriv->mmc = &plat->mmc;
  193. host->mmc = &plat->mmc;
  194. host->mmc->priv = host;
  195. return sdhci_probe(dev);
  196. }
  197. static const struct udevice_id bcm2835_sdhci_match[] = {
  198. { .compatible = "brcm,bcm2835-sdhci" },
  199. { /* sentinel */ }
  200. };
  201. U_BOOT_DRIVER(sdhci_cdns) = {
  202. .name = "sdhci-bcm2835",
  203. .id = UCLASS_MMC,
  204. .of_match = bcm2835_sdhci_match,
  205. .bind = bcm2835_sdhci_bind,
  206. .probe = bcm2835_sdhci_probe,
  207. .priv_auto_alloc_size = sizeof(struct bcm2835_sdhci_host),
  208. .platdata_auto_alloc_size = sizeof(struct bcm2835_sdhci_plat),
  209. .ops = &sdhci_ops,
  210. };