pcm052.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright 2013 Freescale Semiconductor, Inc.
  4. */
  5. #include <common.h>
  6. #include <asm/io.h>
  7. #include <asm/arch/imx-regs.h>
  8. #include <asm/arch/iomux-vf610.h>
  9. #include <asm/arch/ddrmc-vf610.h>
  10. #include <asm/arch/crm_regs.h>
  11. #include <asm/arch/clock.h>
  12. #include <mmc.h>
  13. #include <fsl_esdhc.h>
  14. #include <miiphy.h>
  15. #include <netdev.h>
  16. #include <i2c.h>
  17. DECLARE_GLOBAL_DATA_PTR;
  18. /*
  19. * Default DDR pad settings in arch/arm/include/asm/arch-vf610/iomux-vf610.h
  20. * do not match our settings. Let us (re)define our own settings here.
  21. */
  22. #define PCM052_VF610_DDR_PAD_CTRL PAD_CTL_DSE_20ohm
  23. #define PCM052_VF610_DDR_PAD_CTRL_1 (PAD_CTL_DSE_20ohm | \
  24. PAD_CTL_INPUT_DIFFERENTIAL)
  25. #define PCM052_VF610_DDR_RESET_PAD_CTL (PAD_CTL_DSE_150ohm | \
  26. PAD_CTL_PUS_100K_UP | \
  27. PAD_CTL_INPUT_DIFFERENTIAL)
  28. enum {
  29. PCM052_VF610_PAD_DDR_RESETB = IOMUX_PAD(0x021c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_RESET_PAD_CTL),
  30. PCM052_VF610_PAD_DDR_A15__DDR_A_15 = IOMUX_PAD(0x0220, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  31. PCM052_VF610_PAD_DDR_A14__DDR_A_14 = IOMUX_PAD(0x0224, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  32. PCM052_VF610_PAD_DDR_A13__DDR_A_13 = IOMUX_PAD(0x0228, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  33. PCM052_VF610_PAD_DDR_A12__DDR_A_12 = IOMUX_PAD(0x022c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  34. PCM052_VF610_PAD_DDR_A11__DDR_A_11 = IOMUX_PAD(0x0230, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  35. PCM052_VF610_PAD_DDR_A10__DDR_A_10 = IOMUX_PAD(0x0234, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  36. PCM052_VF610_PAD_DDR_A9__DDR_A_9 = IOMUX_PAD(0x0238, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  37. PCM052_VF610_PAD_DDR_A8__DDR_A_8 = IOMUX_PAD(0x023c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  38. PCM052_VF610_PAD_DDR_A7__DDR_A_7 = IOMUX_PAD(0x0240, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  39. PCM052_VF610_PAD_DDR_A6__DDR_A_6 = IOMUX_PAD(0x0244, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  40. PCM052_VF610_PAD_DDR_A5__DDR_A_5 = IOMUX_PAD(0x0248, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  41. PCM052_VF610_PAD_DDR_A4__DDR_A_4 = IOMUX_PAD(0x024c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  42. PCM052_VF610_PAD_DDR_A3__DDR_A_3 = IOMUX_PAD(0x0250, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  43. PCM052_VF610_PAD_DDR_A2__DDR_A_2 = IOMUX_PAD(0x0254, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  44. PCM052_VF610_PAD_DDR_A1__DDR_A_1 = IOMUX_PAD(0x0258, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  45. PCM052_VF610_PAD_DDR_A0__DDR_A_0 = IOMUX_PAD(0x025c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  46. PCM052_VF610_PAD_DDR_BA2__DDR_BA_2 = IOMUX_PAD(0x0260, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  47. PCM052_VF610_PAD_DDR_BA1__DDR_BA_1 = IOMUX_PAD(0x0264, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  48. PCM052_VF610_PAD_DDR_BA0__DDR_BA_0 = IOMUX_PAD(0x0268, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  49. PCM052_VF610_PAD_DDR_CAS__DDR_CAS_B = IOMUX_PAD(0x026c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  50. PCM052_VF610_PAD_DDR_CKE__DDR_CKE_0 = IOMUX_PAD(0x0270, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  51. PCM052_VF610_PAD_DDR_CLK__DDR_CLK_0 = IOMUX_PAD(0x0274, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL_1),
  52. PCM052_VF610_PAD_DDR_CS__DDR_CS_B_0 = IOMUX_PAD(0x0278, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  53. PCM052_VF610_PAD_DDR_D15__DDR_D_15 = IOMUX_PAD(0x027c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  54. PCM052_VF610_PAD_DDR_D14__DDR_D_14 = IOMUX_PAD(0x0280, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  55. PCM052_VF610_PAD_DDR_D13__DDR_D_13 = IOMUX_PAD(0x0284, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  56. PCM052_VF610_PAD_DDR_D12__DDR_D_12 = IOMUX_PAD(0x0288, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  57. PCM052_VF610_PAD_DDR_D11__DDR_D_11 = IOMUX_PAD(0x028c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  58. PCM052_VF610_PAD_DDR_D10__DDR_D_10 = IOMUX_PAD(0x0290, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  59. PCM052_VF610_PAD_DDR_D9__DDR_D_9 = IOMUX_PAD(0x0294, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  60. PCM052_VF610_PAD_DDR_D8__DDR_D_8 = IOMUX_PAD(0x0298, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  61. PCM052_VF610_PAD_DDR_D7__DDR_D_7 = IOMUX_PAD(0x029c, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  62. PCM052_VF610_PAD_DDR_D6__DDR_D_6 = IOMUX_PAD(0x02a0, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  63. PCM052_VF610_PAD_DDR_D5__DDR_D_5 = IOMUX_PAD(0x02a4, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  64. PCM052_VF610_PAD_DDR_D4__DDR_D_4 = IOMUX_PAD(0x02a8, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  65. PCM052_VF610_PAD_DDR_D3__DDR_D_3 = IOMUX_PAD(0x02ac, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  66. PCM052_VF610_PAD_DDR_D2__DDR_D_2 = IOMUX_PAD(0x02b0, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  67. PCM052_VF610_PAD_DDR_D1__DDR_D_1 = IOMUX_PAD(0x02b4, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  68. PCM052_VF610_PAD_DDR_D0__DDR_D_0 = IOMUX_PAD(0x02b8, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  69. PCM052_VF610_PAD_DDR_DQM1__DDR_DQM_1 = IOMUX_PAD(0x02bc, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  70. PCM052_VF610_PAD_DDR_DQM0__DDR_DQM_0 = IOMUX_PAD(0x02c0, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  71. PCM052_VF610_PAD_DDR_DQS1__DDR_DQS_1 = IOMUX_PAD(0x02c4, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL_1),
  72. PCM052_VF610_PAD_DDR_DQS0__DDR_DQS_0 = IOMUX_PAD(0x02c8, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL_1),
  73. PCM052_VF610_PAD_DDR_RAS__DDR_RAS_B = IOMUX_PAD(0x02cc, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  74. PCM052_VF610_PAD_DDR_WE__DDR_WE_B = IOMUX_PAD(0x02d0, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  75. PCM052_VF610_PAD_DDR_ODT1__DDR_ODT_0 = IOMUX_PAD(0x02d4, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  76. PCM052_VF610_PAD_DDR_ODT0__DDR_ODT_1 = IOMUX_PAD(0x02d8, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  77. PCM052_VF610_PAD_DDR_DDRBYTE1__DDR_DDRBYTE1 = IOMUX_PAD(0x02dc, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  78. PCM052_VF610_PAD_DDR_DDRBYTE0__DDR_DDRBYTE0 = IOMUX_PAD(0x02e0, __NA_, 0, __NA_, 0, PCM052_VF610_DDR_PAD_CTRL),
  79. };
  80. static struct ddrmc_cr_setting pcm052_cr_settings[] = {
  81. /* not in the datasheets, but in the original code */
  82. { 0x00002000, 105 },
  83. { 0x00000020, 110 },
  84. /* AXI */
  85. { DDRMC_CR117_AXI0_W_PRI(1) | DDRMC_CR117_AXI0_R_PRI(1), 117 },
  86. { DDRMC_CR118_AXI1_W_PRI(1) | DDRMC_CR118_AXI1_R_PRI(1), 118 },
  87. { DDRMC_CR120_AXI0_PRI1_RPRI(2) |
  88. DDRMC_CR120_AXI0_PRI0_RPRI(2), 120 },
  89. { DDRMC_CR121_AXI0_PRI3_RPRI(2) |
  90. DDRMC_CR121_AXI0_PRI2_RPRI(2), 121 },
  91. { DDRMC_CR122_AXI1_PRI1_RPRI(1) | DDRMC_CR122_AXI1_PRI0_RPRI(1) |
  92. DDRMC_CR122_AXI0_PRIRLX(100), 122 },
  93. { DDRMC_CR123_AXI1_P_ODR_EN | DDRMC_CR123_AXI1_PRI3_RPRI(1) |
  94. DDRMC_CR123_AXI1_PRI2_RPRI(1), 123 },
  95. { DDRMC_CR124_AXI1_PRIRLX(100), 124 },
  96. { DDRMC_CR126_PHY_RDLAT(11), 126 },
  97. { DDRMC_CR132_WRLAT_ADJ(5) | DDRMC_CR132_RDLAT_ADJ(6), 132 },
  98. { DDRMC_CR137_PHYCTL_DL(2), 137 },
  99. { DDRMC_CR139_PHY_WRLV_RESPLAT(4) | DDRMC_CR139_PHY_WRLV_LOAD(7) |
  100. DDRMC_CR139_PHY_WRLV_DLL(3) |
  101. DDRMC_CR139_PHY_WRLV_EN(3), 139 },
  102. { DDRMC_CR154_PAD_ZQ_EARLY_CMP_EN_TIMER(13) |
  103. DDRMC_CR154_PAD_ZQ_MODE(1) |
  104. DDRMC_CR154_DDR_SEL_PAD_CONTR(3) |
  105. DDRMC_CR154_PAD_ZQ_HW_FOR(0), 154 },
  106. { DDRMC_CR155_PAD_ODT_BYTE1(5) | DDRMC_CR155_PAD_ODT_BYTE0(5), 155 },
  107. { DDRMC_CR158_TWR(6), 158 },
  108. { DDRMC_CR161_ODT_EN(0) | DDRMC_CR161_TODTH_RD(0) |
  109. DDRMC_CR161_TODTH_WR(6), 161 },
  110. /* end marker */
  111. { 0, -1 }
  112. };
  113. /* PHY settings -- most of them differ from default in imx-regs.h */
  114. #define PCM052_DDRMC_PHY_DQ_TIMING 0x00002213
  115. #define PCM052_DDRMC_PHY_CTRL 0x00290000
  116. #define PCM052_DDRMC_PHY_SLAVE_CTRL 0x00002c00
  117. #define PCM052_DDRMC_PHY_PROC_PAD_ODT 0x00010020
  118. static struct ddrmc_phy_setting pcm052_phy_settings[] = {
  119. { PCM052_DDRMC_PHY_DQ_TIMING, 0 },
  120. { PCM052_DDRMC_PHY_DQ_TIMING, 16 },
  121. { PCM052_DDRMC_PHY_DQ_TIMING, 32 },
  122. { PCM052_DDRMC_PHY_DQ_TIMING, 48 },
  123. { DDRMC_PHY_DQS_TIMING, 1 },
  124. { DDRMC_PHY_DQS_TIMING, 17 },
  125. { DDRMC_PHY_DQS_TIMING, 33 },
  126. { DDRMC_PHY_DQS_TIMING, 49 },
  127. { PCM052_DDRMC_PHY_CTRL, 2 },
  128. { PCM052_DDRMC_PHY_CTRL, 18 },
  129. { PCM052_DDRMC_PHY_CTRL, 34 },
  130. { DDRMC_PHY_MASTER_CTRL, 3 },
  131. { DDRMC_PHY_MASTER_CTRL, 19 },
  132. { DDRMC_PHY_MASTER_CTRL, 35 },
  133. { PCM052_DDRMC_PHY_SLAVE_CTRL, 4 },
  134. { PCM052_DDRMC_PHY_SLAVE_CTRL, 20 },
  135. { PCM052_DDRMC_PHY_SLAVE_CTRL, 36 },
  136. { DDRMC_PHY50_DDR3_MODE | DDRMC_PHY50_EN_SW_HALF_CYCLE, 50 },
  137. { PCM052_DDRMC_PHY_PROC_PAD_ODT, 52 },
  138. /* end marker */
  139. { 0, -1 }
  140. };
  141. int dram_init(void)
  142. {
  143. static const iomux_v3_cfg_t pcm052_pads[] = {
  144. PCM052_VF610_PAD_DDR_A15__DDR_A_15,
  145. PCM052_VF610_PAD_DDR_A14__DDR_A_14,
  146. PCM052_VF610_PAD_DDR_A13__DDR_A_13,
  147. PCM052_VF610_PAD_DDR_A12__DDR_A_12,
  148. PCM052_VF610_PAD_DDR_A11__DDR_A_11,
  149. PCM052_VF610_PAD_DDR_A10__DDR_A_10,
  150. PCM052_VF610_PAD_DDR_A9__DDR_A_9,
  151. PCM052_VF610_PAD_DDR_A8__DDR_A_8,
  152. PCM052_VF610_PAD_DDR_A7__DDR_A_7,
  153. PCM052_VF610_PAD_DDR_A6__DDR_A_6,
  154. PCM052_VF610_PAD_DDR_A5__DDR_A_5,
  155. PCM052_VF610_PAD_DDR_A4__DDR_A_4,
  156. PCM052_VF610_PAD_DDR_A3__DDR_A_3,
  157. PCM052_VF610_PAD_DDR_A2__DDR_A_2,
  158. PCM052_VF610_PAD_DDR_A1__DDR_A_1,
  159. PCM052_VF610_PAD_DDR_A0__DDR_A_0,
  160. PCM052_VF610_PAD_DDR_BA2__DDR_BA_2,
  161. PCM052_VF610_PAD_DDR_BA1__DDR_BA_1,
  162. PCM052_VF610_PAD_DDR_BA0__DDR_BA_0,
  163. PCM052_VF610_PAD_DDR_CAS__DDR_CAS_B,
  164. PCM052_VF610_PAD_DDR_CKE__DDR_CKE_0,
  165. PCM052_VF610_PAD_DDR_CLK__DDR_CLK_0,
  166. PCM052_VF610_PAD_DDR_CS__DDR_CS_B_0,
  167. PCM052_VF610_PAD_DDR_D15__DDR_D_15,
  168. PCM052_VF610_PAD_DDR_D14__DDR_D_14,
  169. PCM052_VF610_PAD_DDR_D13__DDR_D_13,
  170. PCM052_VF610_PAD_DDR_D12__DDR_D_12,
  171. PCM052_VF610_PAD_DDR_D11__DDR_D_11,
  172. PCM052_VF610_PAD_DDR_D10__DDR_D_10,
  173. PCM052_VF610_PAD_DDR_D9__DDR_D_9,
  174. PCM052_VF610_PAD_DDR_D8__DDR_D_8,
  175. PCM052_VF610_PAD_DDR_D7__DDR_D_7,
  176. PCM052_VF610_PAD_DDR_D6__DDR_D_6,
  177. PCM052_VF610_PAD_DDR_D5__DDR_D_5,
  178. PCM052_VF610_PAD_DDR_D4__DDR_D_4,
  179. PCM052_VF610_PAD_DDR_D3__DDR_D_3,
  180. PCM052_VF610_PAD_DDR_D2__DDR_D_2,
  181. PCM052_VF610_PAD_DDR_D1__DDR_D_1,
  182. PCM052_VF610_PAD_DDR_D0__DDR_D_0,
  183. PCM052_VF610_PAD_DDR_DQM1__DDR_DQM_1,
  184. PCM052_VF610_PAD_DDR_DQM0__DDR_DQM_0,
  185. PCM052_VF610_PAD_DDR_DQS1__DDR_DQS_1,
  186. PCM052_VF610_PAD_DDR_DQS0__DDR_DQS_0,
  187. PCM052_VF610_PAD_DDR_RAS__DDR_RAS_B,
  188. PCM052_VF610_PAD_DDR_WE__DDR_WE_B,
  189. PCM052_VF610_PAD_DDR_ODT1__DDR_ODT_0,
  190. PCM052_VF610_PAD_DDR_ODT0__DDR_ODT_1,
  191. PCM052_VF610_PAD_DDR_DDRBYTE1__DDR_DDRBYTE1,
  192. PCM052_VF610_PAD_DDR_DDRBYTE0__DDR_DDRBYTE0,
  193. PCM052_VF610_PAD_DDR_RESETB,
  194. };
  195. #if defined(CONFIG_TARGET_PCM052)
  196. static const struct ddr3_jedec_timings pcm052_ddr_timings = {
  197. .tinit = 5,
  198. .trst_pwron = 80000,
  199. .cke_inactive = 200000,
  200. .wrlat = 5,
  201. .caslat_lin = 12,
  202. .trc = 6,
  203. .trrd = 4,
  204. .tccd = 4,
  205. .tbst_int_interval = 4,
  206. .tfaw = 18,
  207. .trp = 6,
  208. .twtr = 4,
  209. .tras_min = 15,
  210. .tmrd = 4,
  211. .trtp = 4,
  212. .tras_max = 14040,
  213. .tmod = 12,
  214. .tckesr = 4,
  215. .tcke = 3,
  216. .trcd_int = 6,
  217. .tras_lockout = 1,
  218. .tdal = 10,
  219. .bstlen = 3,
  220. .tdll = 512,
  221. .trp_ab = 6,
  222. .tref = 1542,
  223. .trfc = 64,
  224. .tref_int = 5,
  225. .tpdex = 3,
  226. .txpdll = 10,
  227. .txsnr = 68,
  228. .txsr = 506,
  229. .cksrx = 5,
  230. .cksre = 5,
  231. .freq_chg_en = 1,
  232. .zqcl = 256,
  233. .zqinit = 512,
  234. .zqcs = 64,
  235. .ref_per_zq = 64,
  236. .zqcs_rotate = 1,
  237. .aprebit = 10,
  238. .cmd_age_cnt = 255,
  239. .age_cnt = 255,
  240. .q_fullness = 0,
  241. .odt_rd_mapcs0 = 1,
  242. .odt_wr_mapcs0 = 1,
  243. .wlmrd = 40,
  244. .wldqsen = 25,
  245. };
  246. const int row_diff = 2;
  247. #elif defined(CONFIG_TARGET_BK4R1)
  248. static const struct ddr3_jedec_timings pcm052_ddr_timings = {
  249. .tinit = 5,
  250. .trst_pwron = 80000,
  251. .cke_inactive = 200000,
  252. .wrlat = 5,
  253. .caslat_lin = 12,
  254. .trc = 6,
  255. .trrd = 4,
  256. .tccd = 4,
  257. .tbst_int_interval = 0,
  258. .tfaw = 16,
  259. .trp = 6,
  260. .twtr = 4,
  261. .tras_min = 15,
  262. .tmrd = 4,
  263. .trtp = 4,
  264. .tras_max = 28080,
  265. .tmod = 12,
  266. .tckesr = 4,
  267. .tcke = 3,
  268. .trcd_int = 6,
  269. .tras_lockout = 1,
  270. .tdal = 12,
  271. .bstlen = 3,
  272. .tdll = 512,
  273. .trp_ab = 6,
  274. .tref = 3120,
  275. .trfc = 104,
  276. .tref_int = 0,
  277. .tpdex = 3,
  278. .txpdll = 10,
  279. .txsnr = 108,
  280. .txsr = 512,
  281. .cksrx = 5,
  282. .cksre = 5,
  283. .freq_chg_en = 1,
  284. .zqcl = 256,
  285. .zqinit = 512,
  286. .zqcs = 64,
  287. .ref_per_zq = 64,
  288. .zqcs_rotate = 1,
  289. .aprebit = 10,
  290. .cmd_age_cnt = 255,
  291. .age_cnt = 255,
  292. .q_fullness = 0,
  293. .odt_rd_mapcs0 = 1,
  294. .odt_wr_mapcs0 = 1,
  295. .wlmrd = 40,
  296. .wldqsen = 25,
  297. };
  298. const int row_diff = 1;
  299. #else /* Unknown PCM052 variant */
  300. #error DDR characteristics undefined for this target. Please define them.
  301. #endif
  302. imx_iomux_v3_setup_multiple_pads(pcm052_pads, ARRAY_SIZE(pcm052_pads));
  303. ddrmc_ctrl_init_ddr3(&pcm052_ddr_timings, pcm052_cr_settings,
  304. pcm052_phy_settings, 1, row_diff);
  305. gd->ram_size = get_ram_size((void *)PHYS_SDRAM, PHYS_SDRAM_SIZE);
  306. return 0;
  307. }
  308. static void setup_iomux_uart(void)
  309. {
  310. static const iomux_v3_cfg_t uart1_pads[] = {
  311. NEW_PAD_CTRL(VF610_PAD_PTB4__UART1_TX, VF610_UART_PAD_CTRL),
  312. NEW_PAD_CTRL(VF610_PAD_PTB5__UART1_RX, VF610_UART_PAD_CTRL),
  313. };
  314. imx_iomux_v3_setup_multiple_pads(uart1_pads, ARRAY_SIZE(uart1_pads));
  315. }
  316. #define ENET_PAD_CTRL (PAD_CTL_PUS_47K_UP | PAD_CTL_SPEED_HIGH | \
  317. PAD_CTL_DSE_50ohm | PAD_CTL_OBE_IBE_ENABLE)
  318. static void setup_iomux_enet(void)
  319. {
  320. static const iomux_v3_cfg_t enet0_pads[] = {
  321. NEW_PAD_CTRL(VF610_PAD_PTA6__RMII0_CLKIN, ENET_PAD_CTRL),
  322. NEW_PAD_CTRL(VF610_PAD_PTC1__RMII0_MDIO, ENET_PAD_CTRL),
  323. NEW_PAD_CTRL(VF610_PAD_PTC0__RMII0_MDC, ENET_PAD_CTRL),
  324. NEW_PAD_CTRL(VF610_PAD_PTC2__RMII0_CRS_DV, ENET_PAD_CTRL),
  325. NEW_PAD_CTRL(VF610_PAD_PTC3__RMII0_RD1, ENET_PAD_CTRL),
  326. NEW_PAD_CTRL(VF610_PAD_PTC4__RMII0_RD0, ENET_PAD_CTRL),
  327. NEW_PAD_CTRL(VF610_PAD_PTC5__RMII0_RXER, ENET_PAD_CTRL),
  328. NEW_PAD_CTRL(VF610_PAD_PTC6__RMII0_TD1, ENET_PAD_CTRL),
  329. NEW_PAD_CTRL(VF610_PAD_PTC7__RMII0_TD0, ENET_PAD_CTRL),
  330. NEW_PAD_CTRL(VF610_PAD_PTC8__RMII0_TXEN, ENET_PAD_CTRL),
  331. };
  332. imx_iomux_v3_setup_multiple_pads(enet0_pads, ARRAY_SIZE(enet0_pads));
  333. }
  334. /*
  335. * I2C2 is the only I2C used, on pads PTA22/PTA23.
  336. */
  337. static void setup_iomux_i2c(void)
  338. {
  339. static const iomux_v3_cfg_t i2c_pads[] = {
  340. VF610_PAD_PTA22__I2C2_SCL,
  341. VF610_PAD_PTA23__I2C2_SDA,
  342. };
  343. imx_iomux_v3_setup_multiple_pads(i2c_pads, ARRAY_SIZE(i2c_pads));
  344. }
  345. #ifdef CONFIG_NAND_VF610_NFC
  346. static void setup_iomux_nfc(void)
  347. {
  348. static const iomux_v3_cfg_t nfc_pads[] = {
  349. VF610_PAD_PTD31__NF_IO15,
  350. VF610_PAD_PTD30__NF_IO14,
  351. VF610_PAD_PTD29__NF_IO13,
  352. VF610_PAD_PTD28__NF_IO12,
  353. VF610_PAD_PTD27__NF_IO11,
  354. VF610_PAD_PTD26__NF_IO10,
  355. VF610_PAD_PTD25__NF_IO9,
  356. VF610_PAD_PTD24__NF_IO8,
  357. VF610_PAD_PTD23__NF_IO7,
  358. VF610_PAD_PTD22__NF_IO6,
  359. VF610_PAD_PTD21__NF_IO5,
  360. VF610_PAD_PTD20__NF_IO4,
  361. VF610_PAD_PTD19__NF_IO3,
  362. VF610_PAD_PTD18__NF_IO2,
  363. VF610_PAD_PTD17__NF_IO1,
  364. VF610_PAD_PTD16__NF_IO0,
  365. VF610_PAD_PTB24__NF_WE_B,
  366. VF610_PAD_PTB25__NF_CE0_B,
  367. VF610_PAD_PTB27__NF_RE_B,
  368. VF610_PAD_PTC26__NF_RB_B,
  369. VF610_PAD_PTC27__NF_ALE,
  370. VF610_PAD_PTC28__NF_CLE
  371. };
  372. imx_iomux_v3_setup_multiple_pads(nfc_pads, ARRAY_SIZE(nfc_pads));
  373. }
  374. #endif
  375. static void setup_iomux_qspi(void)
  376. {
  377. static const iomux_v3_cfg_t qspi0_pads[] = {
  378. VF610_PAD_PTD0__QSPI0_A_QSCK,
  379. VF610_PAD_PTD1__QSPI0_A_CS0,
  380. VF610_PAD_PTD2__QSPI0_A_DATA3,
  381. VF610_PAD_PTD3__QSPI0_A_DATA2,
  382. VF610_PAD_PTD4__QSPI0_A_DATA1,
  383. VF610_PAD_PTD5__QSPI0_A_DATA0,
  384. VF610_PAD_PTD7__QSPI0_B_QSCK,
  385. VF610_PAD_PTD8__QSPI0_B_CS0,
  386. VF610_PAD_PTD9__QSPI0_B_DATA3,
  387. VF610_PAD_PTD10__QSPI0_B_DATA2,
  388. VF610_PAD_PTD11__QSPI0_B_DATA1,
  389. VF610_PAD_PTD12__QSPI0_B_DATA0,
  390. };
  391. imx_iomux_v3_setup_multiple_pads(qspi0_pads, ARRAY_SIZE(qspi0_pads));
  392. }
  393. #define ESDHC_PAD_CTRL (PAD_CTL_PUS_100K_UP | PAD_CTL_SPEED_HIGH | \
  394. PAD_CTL_DSE_20ohm | PAD_CTL_OBE_IBE_ENABLE)
  395. struct fsl_esdhc_cfg esdhc_cfg[1] = {
  396. {ESDHC1_BASE_ADDR},
  397. };
  398. int board_mmc_getcd(struct mmc *mmc)
  399. {
  400. /* eSDHC1 is always present */
  401. return 1;
  402. }
  403. int board_mmc_init(bd_t *bis)
  404. {
  405. static const iomux_v3_cfg_t esdhc1_pads[] = {
  406. NEW_PAD_CTRL(VF610_PAD_PTA24__ESDHC1_CLK, ESDHC_PAD_CTRL),
  407. NEW_PAD_CTRL(VF610_PAD_PTA25__ESDHC1_CMD, ESDHC_PAD_CTRL),
  408. NEW_PAD_CTRL(VF610_PAD_PTA26__ESDHC1_DAT0, ESDHC_PAD_CTRL),
  409. NEW_PAD_CTRL(VF610_PAD_PTA27__ESDHC1_DAT1, ESDHC_PAD_CTRL),
  410. NEW_PAD_CTRL(VF610_PAD_PTA28__ESDHC1_DAT2, ESDHC_PAD_CTRL),
  411. NEW_PAD_CTRL(VF610_PAD_PTA29__ESDHC1_DAT3, ESDHC_PAD_CTRL),
  412. };
  413. esdhc_cfg[0].sdhc_clk = mxc_get_clock(MXC_ESDHC_CLK);
  414. imx_iomux_v3_setup_multiple_pads(
  415. esdhc1_pads, ARRAY_SIZE(esdhc1_pads));
  416. return fsl_esdhc_initialize(bis, &esdhc_cfg[0]);
  417. }
  418. static void clock_init(void)
  419. {
  420. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  421. struct anadig_reg *anadig = (struct anadig_reg *)ANADIG_BASE_ADDR;
  422. clrsetbits_le32(&ccm->ccgr0, CCM_REG_CTRL_MASK,
  423. CCM_CCGR0_UART1_CTRL_MASK);
  424. clrsetbits_le32(&ccm->ccgr1, CCM_REG_CTRL_MASK,
  425. CCM_CCGR1_PIT_CTRL_MASK | CCM_CCGR1_WDOGA5_CTRL_MASK);
  426. clrsetbits_le32(&ccm->ccgr2, CCM_REG_CTRL_MASK,
  427. CCM_CCGR2_IOMUXC_CTRL_MASK | CCM_CCGR2_PORTA_CTRL_MASK |
  428. CCM_CCGR2_PORTB_CTRL_MASK | CCM_CCGR2_PORTC_CTRL_MASK |
  429. CCM_CCGR2_PORTD_CTRL_MASK | CCM_CCGR2_PORTE_CTRL_MASK |
  430. CCM_CCGR2_QSPI0_CTRL_MASK);
  431. clrsetbits_le32(&ccm->ccgr3, CCM_REG_CTRL_MASK,
  432. CCM_CCGR3_ANADIG_CTRL_MASK | CCM_CCGR3_SCSC_CTRL_MASK);
  433. clrsetbits_le32(&ccm->ccgr4, CCM_REG_CTRL_MASK,
  434. CCM_CCGR4_WKUP_CTRL_MASK | CCM_CCGR4_CCM_CTRL_MASK |
  435. CCM_CCGR4_GPC_CTRL_MASK);
  436. clrsetbits_le32(&ccm->ccgr6, CCM_REG_CTRL_MASK,
  437. CCM_CCGR6_OCOTP_CTRL_MASK | CCM_CCGR6_DDRMC_CTRL_MASK);
  438. clrsetbits_le32(&ccm->ccgr7, CCM_REG_CTRL_MASK,
  439. CCM_CCGR7_SDHC1_CTRL_MASK);
  440. clrsetbits_le32(&ccm->ccgr9, CCM_REG_CTRL_MASK,
  441. CCM_CCGR9_FEC0_CTRL_MASK | CCM_CCGR9_FEC1_CTRL_MASK);
  442. clrsetbits_le32(&ccm->ccgr10, CCM_REG_CTRL_MASK,
  443. CCM_CCGR10_NFC_CTRL_MASK | CCM_CCGR10_I2C2_CTRL_MASK);
  444. clrsetbits_le32(&anadig->pll2_ctrl, ANADIG_PLL2_CTRL_POWERDOWN,
  445. ANADIG_PLL2_CTRL_ENABLE | ANADIG_PLL2_CTRL_DIV_SELECT);
  446. clrsetbits_le32(&anadig->pll1_ctrl, ANADIG_PLL1_CTRL_POWERDOWN,
  447. ANADIG_PLL1_CTRL_ENABLE | ANADIG_PLL1_CTRL_DIV_SELECT);
  448. clrsetbits_le32(&ccm->ccr, CCM_CCR_OSCNT_MASK,
  449. CCM_CCR_FIRC_EN | CCM_CCR_OSCNT(5));
  450. clrsetbits_le32(&ccm->ccsr, CCM_REG_CTRL_MASK,
  451. CCM_CCSR_PLL1_PFD_CLK_SEL(3) | CCM_CCSR_PLL2_PFD4_EN |
  452. CCM_CCSR_PLL2_PFD3_EN | CCM_CCSR_PLL2_PFD2_EN |
  453. CCM_CCSR_PLL2_PFD1_EN | CCM_CCSR_PLL1_PFD4_EN |
  454. CCM_CCSR_PLL1_PFD3_EN | CCM_CCSR_PLL1_PFD2_EN |
  455. CCM_CCSR_PLL1_PFD1_EN | CCM_CCSR_DDRC_CLK_SEL(1) |
  456. CCM_CCSR_FAST_CLK_SEL(1) | CCM_CCSR_SYS_CLK_SEL(4));
  457. clrsetbits_le32(&ccm->cacrr, CCM_REG_CTRL_MASK,
  458. CCM_CACRR_IPG_CLK_DIV(1) | CCM_CACRR_BUS_CLK_DIV(2) |
  459. CCM_CACRR_ARM_CLK_DIV(0));
  460. clrsetbits_le32(&ccm->cscmr1, CCM_REG_CTRL_MASK,
  461. CCM_CSCMR1_ESDHC1_CLK_SEL(3) |
  462. CCM_CSCMR1_QSPI0_CLK_SEL(3) |
  463. CCM_CSCMR1_NFC_CLK_SEL(0));
  464. clrsetbits_le32(&ccm->cscdr1, CCM_REG_CTRL_MASK,
  465. CCM_CSCDR1_RMII_CLK_EN);
  466. clrsetbits_le32(&ccm->cscdr2, CCM_REG_CTRL_MASK,
  467. CCM_CSCDR2_ESDHC1_EN | CCM_CSCDR2_ESDHC1_CLK_DIV(0) |
  468. CCM_CSCDR2_NFC_EN);
  469. clrsetbits_le32(&ccm->cscdr3, CCM_REG_CTRL_MASK,
  470. CCM_CSCDR3_QSPI0_EN | CCM_CSCDR3_QSPI0_DIV(1) |
  471. CCM_CSCDR3_QSPI0_X2_DIV(1) |
  472. CCM_CSCDR3_QSPI0_X4_DIV(3) |
  473. CCM_CSCDR3_NFC_PRE_DIV(5));
  474. clrsetbits_le32(&ccm->cscmr2, CCM_REG_CTRL_MASK,
  475. CCM_CSCMR2_RMII_CLK_SEL(0));
  476. }
  477. static void mscm_init(void)
  478. {
  479. struct mscm_ir *mscmir = (struct mscm_ir *)MSCM_IR_BASE_ADDR;
  480. int i;
  481. for (i = 0; i < MSCM_IRSPRC_NUM; i++)
  482. writew(MSCM_IRSPRC_CP0_EN, &mscmir->irsprc[i]);
  483. }
  484. int board_phy_config(struct phy_device *phydev)
  485. {
  486. if (phydev->drv->config)
  487. phydev->drv->config(phydev);
  488. return 0;
  489. }
  490. int board_early_init_f(void)
  491. {
  492. clock_init();
  493. mscm_init();
  494. setup_iomux_uart();
  495. setup_iomux_enet();
  496. setup_iomux_i2c();
  497. setup_iomux_qspi();
  498. setup_iomux_nfc();
  499. return 0;
  500. }
  501. int board_init(void)
  502. {
  503. struct scsc_reg *scsc = (struct scsc_reg *)SCSC_BASE_ADDR;
  504. /* address of boot parameters */
  505. gd->bd->bi_boot_params = PHYS_SDRAM + 0x100;
  506. /*
  507. * Enable external 32K Oscillator
  508. *
  509. * The internal clock experiences significant drift
  510. * so we must use the external oscillator in order
  511. * to maintain correct time in the hwclock
  512. */
  513. setbits_le32(&scsc->sosc_ctr, SCSC_SOSC_CTR_SOSC_EN);
  514. return 0;
  515. }
  516. int checkboard(void)
  517. {
  518. puts("Board: PCM-052\n");
  519. return 0;
  520. }
  521. static int do_m4go(cmd_tbl_t *cmdtp, int flag, int argc,
  522. char * const argv[])
  523. {
  524. ulong addr;
  525. /* Consume 'm4go' */
  526. argc--; argv++;
  527. /*
  528. * Parse provided address - default to load_addr in case not provided.
  529. */
  530. if (argc)
  531. addr = simple_strtoul(argv[0], NULL, 16);
  532. else
  533. addr = load_addr;
  534. /*
  535. * Write boot address in PERSISTENT_ENTRY1[31:0] aka SRC_GPR2[31:0]
  536. */
  537. writel(addr + 0x401, 0x4006E028);
  538. /*
  539. * Start secondary processor by enabling its clock
  540. */
  541. writel(0x15a5a, 0x4006B08C);
  542. return 1;
  543. }
  544. U_BOOT_CMD(
  545. m4go, 2 /* one arg max */, 1 /* repeatable */, do_m4go,
  546. "start the secondary Cortex-M4 from scatter file image",
  547. "[<addr>]\n"
  548. " - start secondary Cortex-M4 core using a scatter file image\n"
  549. "The argument needs to be a scatter file\n"
  550. );