sdio_chip.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. /*
  2. * Copyright (c) 2011 Broadcom Corporation
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for any
  5. * purpose with or without fee is hereby granted, provided that the above
  6. * copyright notice and this permission notice appear in all copies.
  7. *
  8. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  9. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  10. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
  11. * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
  13. * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
  14. * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. /* ***** SDIO interface chip backplane handle functions ***** */
  17. #include <linux/types.h>
  18. #include <linux/netdevice.h>
  19. #include <linux/mmc/card.h>
  20. #include <chipcommon.h>
  21. #include <brcm_hw_ids.h>
  22. #include <brcmu_wifi.h>
  23. #include <brcmu_utils.h>
  24. #include <soc.h>
  25. #include "dhd.h"
  26. #include "dhd_dbg.h"
  27. #include "sdio_host.h"
  28. #include "sdio_chip.h"
  29. /* chip core base & ramsize */
  30. /* bcm4329 */
  31. /* SDIO device core, ID 0x829 */
  32. #define BCM4329_CORE_BUS_BASE 0x18011000
  33. /* internal memory core, ID 0x80e */
  34. #define BCM4329_CORE_SOCRAM_BASE 0x18003000
  35. /* ARM Cortex M3 core, ID 0x82a */
  36. #define BCM4329_CORE_ARM_BASE 0x18002000
  37. #define BCM4329_RAMSIZE 0x48000
  38. /* SB regs */
  39. /* sbidhigh */
  40. #define SBIDH_RC_MASK 0x000f /* revision code */
  41. #define SBIDH_RCE_MASK 0x7000 /* revision code extension field */
  42. #define SBIDH_RCE_SHIFT 8
  43. #define SBCOREREV(sbidh) \
  44. ((((sbidh) & SBIDH_RCE_MASK) >> SBIDH_RCE_SHIFT) | \
  45. ((sbidh) & SBIDH_RC_MASK))
  46. #define SBIDH_CC_MASK 0x8ff0 /* core code */
  47. #define SBIDH_CC_SHIFT 4
  48. #define SBIDH_VC_MASK 0xffff0000 /* vendor code */
  49. #define SBIDH_VC_SHIFT 16
  50. bool
  51. brcmf_sdio_chip_iscoreup(struct brcmf_sdio_dev *sdiodev,
  52. u32 corebase)
  53. {
  54. u32 regdata;
  55. regdata = brcmf_sdcard_reg_read(sdiodev,
  56. CORE_SB(corebase, sbtmstatelow), 4);
  57. regdata &= (SBTML_RESET | SBTML_REJ_MASK |
  58. (SICF_CLOCK_EN << SBTML_SICF_SHIFT));
  59. return ((SICF_CLOCK_EN << SBTML_SICF_SHIFT) == regdata);
  60. }
  61. void
  62. brcmf_sdio_chip_coredisable(struct brcmf_sdio_dev *sdiodev, u32 corebase)
  63. {
  64. u32 regdata;
  65. regdata = brcmf_sdcard_reg_read(sdiodev,
  66. CORE_SB(corebase, sbtmstatelow), 4);
  67. if (regdata & SBTML_RESET)
  68. return;
  69. regdata = brcmf_sdcard_reg_read(sdiodev,
  70. CORE_SB(corebase, sbtmstatelow), 4);
  71. if ((regdata & (SICF_CLOCK_EN << SBTML_SICF_SHIFT)) != 0) {
  72. /*
  73. * set target reject and spin until busy is clear
  74. * (preserve core-specific bits)
  75. */
  76. regdata = brcmf_sdcard_reg_read(sdiodev,
  77. CORE_SB(corebase, sbtmstatelow), 4);
  78. brcmf_sdcard_reg_write(sdiodev, CORE_SB(corebase, sbtmstatelow),
  79. 4, regdata | SBTML_REJ);
  80. regdata = brcmf_sdcard_reg_read(sdiodev,
  81. CORE_SB(corebase, sbtmstatelow), 4);
  82. udelay(1);
  83. SPINWAIT((brcmf_sdcard_reg_read(sdiodev,
  84. CORE_SB(corebase, sbtmstatehigh), 4) &
  85. SBTMH_BUSY), 100000);
  86. regdata = brcmf_sdcard_reg_read(sdiodev,
  87. CORE_SB(corebase, sbtmstatehigh), 4);
  88. if (regdata & SBTMH_BUSY)
  89. brcmf_dbg(ERROR, "core state still busy\n");
  90. regdata = brcmf_sdcard_reg_read(sdiodev,
  91. CORE_SB(corebase, sbidlow), 4);
  92. if (regdata & SBIDL_INIT) {
  93. regdata = brcmf_sdcard_reg_read(sdiodev,
  94. CORE_SB(corebase, sbimstate), 4) |
  95. SBIM_RJ;
  96. brcmf_sdcard_reg_write(sdiodev,
  97. CORE_SB(corebase, sbimstate), 4,
  98. regdata);
  99. regdata = brcmf_sdcard_reg_read(sdiodev,
  100. CORE_SB(corebase, sbimstate), 4);
  101. udelay(1);
  102. SPINWAIT((brcmf_sdcard_reg_read(sdiodev,
  103. CORE_SB(corebase, sbimstate), 4) &
  104. SBIM_BY), 100000);
  105. }
  106. /* set reset and reject while enabling the clocks */
  107. brcmf_sdcard_reg_write(sdiodev,
  108. CORE_SB(corebase, sbtmstatelow), 4,
  109. (((SICF_FGC | SICF_CLOCK_EN) << SBTML_SICF_SHIFT) |
  110. SBTML_REJ | SBTML_RESET));
  111. regdata = brcmf_sdcard_reg_read(sdiodev,
  112. CORE_SB(corebase, sbtmstatelow), 4);
  113. udelay(10);
  114. /* clear the initiator reject bit */
  115. regdata = brcmf_sdcard_reg_read(sdiodev,
  116. CORE_SB(corebase, sbidlow), 4);
  117. if (regdata & SBIDL_INIT) {
  118. regdata = brcmf_sdcard_reg_read(sdiodev,
  119. CORE_SB(corebase, sbimstate), 4) &
  120. ~SBIM_RJ;
  121. brcmf_sdcard_reg_write(sdiodev,
  122. CORE_SB(corebase, sbimstate), 4,
  123. regdata);
  124. }
  125. }
  126. /* leave reset and reject asserted */
  127. brcmf_sdcard_reg_write(sdiodev, CORE_SB(corebase, sbtmstatelow), 4,
  128. (SBTML_REJ | SBTML_RESET));
  129. udelay(1);
  130. }
  131. static int brcmf_sdio_chip_recognition(struct brcmf_sdio_dev *sdiodev,
  132. struct chip_info *ci, u32 regs)
  133. {
  134. u32 regdata;
  135. /*
  136. * Get CC core rev
  137. * Chipid is assume to be at offset 0 from regs arg
  138. * For different chiptypes or old sdio hosts w/o chipcommon,
  139. * other ways of recognition should be added here.
  140. */
  141. ci->cccorebase = regs;
  142. regdata = brcmf_sdcard_reg_read(sdiodev,
  143. CORE_CC_REG(ci->cccorebase, chipid), 4);
  144. ci->chip = regdata & CID_ID_MASK;
  145. ci->chiprev = (regdata & CID_REV_MASK) >> CID_REV_SHIFT;
  146. brcmf_dbg(INFO, "chipid=0x%x chiprev=%d\n", ci->chip, ci->chiprev);
  147. /* Address of cores for new chips should be added here */
  148. switch (ci->chip) {
  149. case BCM4329_CHIP_ID:
  150. ci->buscorebase = BCM4329_CORE_BUS_BASE;
  151. ci->ramcorebase = BCM4329_CORE_SOCRAM_BASE;
  152. ci->armcorebase = BCM4329_CORE_ARM_BASE;
  153. ci->ramsize = BCM4329_RAMSIZE;
  154. break;
  155. default:
  156. brcmf_dbg(ERROR, "chipid 0x%x is not supported\n", ci->chip);
  157. return -ENODEV;
  158. }
  159. return 0;
  160. }
  161. static int
  162. brcmf_sdio_chip_buscoreprep(struct brcmf_sdio_dev *sdiodev)
  163. {
  164. int err = 0;
  165. u8 clkval, clkset;
  166. /* Try forcing SDIO core to do ALPAvail request only */
  167. clkset = SBSDIO_FORCE_HW_CLKREQ_OFF | SBSDIO_ALP_AVAIL_REQ;
  168. brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1,
  169. SBSDIO_FUNC1_CHIPCLKCSR, clkset, &err);
  170. if (err) {
  171. brcmf_dbg(ERROR, "error writing for HT off\n");
  172. return err;
  173. }
  174. /* If register supported, wait for ALPAvail and then force ALP */
  175. /* This may take up to 15 milliseconds */
  176. clkval = brcmf_sdcard_cfg_read(sdiodev, SDIO_FUNC_1,
  177. SBSDIO_FUNC1_CHIPCLKCSR, NULL);
  178. if ((clkval & ~SBSDIO_AVBITS) != clkset) {
  179. brcmf_dbg(ERROR, "ChipClkCSR access: wrote 0x%02x read 0x%02x\n",
  180. clkset, clkval);
  181. return -EACCES;
  182. }
  183. SPINWAIT(((clkval = brcmf_sdcard_cfg_read(sdiodev, SDIO_FUNC_1,
  184. SBSDIO_FUNC1_CHIPCLKCSR, NULL)),
  185. !SBSDIO_ALPAV(clkval)),
  186. PMU_MAX_TRANSITION_DLY);
  187. if (!SBSDIO_ALPAV(clkval)) {
  188. brcmf_dbg(ERROR, "timeout on ALPAV wait, clkval 0x%02x\n",
  189. clkval);
  190. return -EBUSY;
  191. }
  192. clkset = SBSDIO_FORCE_HW_CLKREQ_OFF | SBSDIO_FORCE_ALP;
  193. brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1,
  194. SBSDIO_FUNC1_CHIPCLKCSR, clkset, &err);
  195. udelay(65);
  196. /* Also, disable the extra SDIO pull-ups */
  197. brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1,
  198. SBSDIO_FUNC1_SDIOPULLUP, 0, NULL);
  199. return 0;
  200. }
  201. static void
  202. brcmf_sdio_chip_buscoresetup(struct brcmf_sdio_dev *sdiodev,
  203. struct chip_info *ci)
  204. {
  205. u32 regdata;
  206. /* get chipcommon rev */
  207. regdata = brcmf_sdcard_reg_read(sdiodev,
  208. CORE_SB(ci->cccorebase, sbidhigh), 4);
  209. ci->ccrev = SBCOREREV(regdata);
  210. /* get chipcommon capabilites */
  211. ci->cccaps = brcmf_sdcard_reg_read(sdiodev,
  212. CORE_CC_REG(ci->cccorebase, capabilities), 4);
  213. /* get pmu caps & rev */
  214. if (ci->cccaps & CC_CAP_PMU) {
  215. ci->pmucaps = brcmf_sdcard_reg_read(sdiodev,
  216. CORE_CC_REG(ci->cccorebase, pmucapabilities), 4);
  217. ci->pmurev = ci->pmucaps & PCAP_REV_MASK;
  218. }
  219. regdata = brcmf_sdcard_reg_read(sdiodev,
  220. CORE_SB(ci->buscorebase, sbidhigh), 4);
  221. ci->buscorerev = SBCOREREV(regdata);
  222. ci->buscoretype = (regdata & SBIDH_CC_MASK) >> SBIDH_CC_SHIFT;
  223. brcmf_dbg(INFO, "ccrev=%d, pmurev=%d, buscore rev/type=%d/0x%x\n",
  224. ci->ccrev, ci->pmurev, ci->buscorerev, ci->buscoretype);
  225. /*
  226. * Make sure any on-chip ARM is off (in case strapping is wrong),
  227. * or downloaded code was already running.
  228. */
  229. brcmf_sdio_chip_coredisable(sdiodev, ci->armcorebase);
  230. }
  231. int brcmf_sdio_chip_attach(struct brcmf_sdio_dev *sdiodev,
  232. struct chip_info **ci_ptr, u32 regs)
  233. {
  234. int ret;
  235. struct chip_info *ci;
  236. brcmf_dbg(TRACE, "Enter\n");
  237. /* alloc chip_info_t */
  238. ci = kzalloc(sizeof(struct chip_info), GFP_ATOMIC);
  239. if (!ci)
  240. return -ENOMEM;
  241. ret = brcmf_sdio_chip_buscoreprep(sdiodev);
  242. if (ret != 0)
  243. goto err;
  244. ret = brcmf_sdio_chip_recognition(sdiodev, ci, regs);
  245. if (ret != 0)
  246. goto err;
  247. brcmf_sdio_chip_buscoresetup(sdiodev, ci);
  248. brcmf_sdcard_reg_write(sdiodev,
  249. CORE_CC_REG(ci->cccorebase, gpiopullup), 4, 0);
  250. brcmf_sdcard_reg_write(sdiodev,
  251. CORE_CC_REG(ci->cccorebase, gpiopulldown), 4, 0);
  252. *ci_ptr = ci;
  253. return 0;
  254. err:
  255. kfree(ci);
  256. return ret;
  257. }