spl.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * SPL specific code for Compulab CM-FX6 board
  4. *
  5. * Copyright (C) 2014, Compulab Ltd - http://compulab.co.il/
  6. *
  7. * Author: Nikita Kiryanov <nikita@compulab.co.il>
  8. */
  9. #include <common.h>
  10. #include <spl.h>
  11. #include <asm/io.h>
  12. #include <asm/gpio.h>
  13. #include <asm/arch/mx6-ddr.h>
  14. #include <asm/arch/clock.h>
  15. #include <asm/arch/sys_proto.h>
  16. #include <asm/arch/crm_regs.h>
  17. #include <asm/mach-imx/iomux-v3.h>
  18. #include <fsl_esdhc.h>
  19. #include "common.h"
  20. enum ddr_config {
  21. DDR_16BIT_256MB,
  22. DDR_32BIT_512MB,
  23. DDR_32BIT_1GB,
  24. DDR_64BIT_1GB,
  25. DDR_64BIT_2GB,
  26. DDR_64BIT_4GB,
  27. DDR_UNKNOWN,
  28. };
  29. /*
  30. * Below DRAM_RESET[DDR_SEL] = 0 which is incorrect according to
  31. * Freescale QRM, but this is exactly the value used by the automatic
  32. * calibration script and it works also in all our tests, so we leave
  33. * it as is at this point.
  34. */
  35. #define CM_FX6_DDR_IOMUX_CFG \
  36. .dram_sdqs0 = 0x00000038, \
  37. .dram_sdqs1 = 0x00000038, \
  38. .dram_sdqs2 = 0x00000038, \
  39. .dram_sdqs3 = 0x00000038, \
  40. .dram_sdqs4 = 0x00000038, \
  41. .dram_sdqs5 = 0x00000038, \
  42. .dram_sdqs6 = 0x00000038, \
  43. .dram_sdqs7 = 0x00000038, \
  44. .dram_dqm0 = 0x00000038, \
  45. .dram_dqm1 = 0x00000038, \
  46. .dram_dqm2 = 0x00000038, \
  47. .dram_dqm3 = 0x00000038, \
  48. .dram_dqm4 = 0x00000038, \
  49. .dram_dqm5 = 0x00000038, \
  50. .dram_dqm6 = 0x00000038, \
  51. .dram_dqm7 = 0x00000038, \
  52. .dram_cas = 0x00000038, \
  53. .dram_ras = 0x00000038, \
  54. .dram_sdclk_0 = 0x00000038, \
  55. .dram_sdclk_1 = 0x00000038, \
  56. .dram_sdcke0 = 0x00003000, \
  57. .dram_sdcke1 = 0x00003000, \
  58. .dram_reset = 0x00000038, \
  59. .dram_sdba2 = 0x00000000, \
  60. .dram_sdodt0 = 0x00000038, \
  61. .dram_sdodt1 = 0x00000038,
  62. #define CM_FX6_GPR_IOMUX_CFG \
  63. .grp_b0ds = 0x00000038, \
  64. .grp_b1ds = 0x00000038, \
  65. .grp_b2ds = 0x00000038, \
  66. .grp_b3ds = 0x00000038, \
  67. .grp_b4ds = 0x00000038, \
  68. .grp_b5ds = 0x00000038, \
  69. .grp_b6ds = 0x00000038, \
  70. .grp_b7ds = 0x00000038, \
  71. .grp_addds = 0x00000038, \
  72. .grp_ddrmode_ctl = 0x00020000, \
  73. .grp_ddrpke = 0x00000000, \
  74. .grp_ddrmode = 0x00020000, \
  75. .grp_ctlds = 0x00000038, \
  76. .grp_ddr_type = 0x000C0000,
  77. static struct mx6sdl_iomux_ddr_regs ddr_iomux_s = { CM_FX6_DDR_IOMUX_CFG };
  78. static struct mx6sdl_iomux_grp_regs grp_iomux_s = { CM_FX6_GPR_IOMUX_CFG };
  79. static struct mx6dq_iomux_ddr_regs ddr_iomux_q = { CM_FX6_DDR_IOMUX_CFG };
  80. static struct mx6dq_iomux_grp_regs grp_iomux_q = { CM_FX6_GPR_IOMUX_CFG };
  81. static struct mx6_mmdc_calibration cm_fx6_calib_s = {
  82. .p0_mpwldectrl0 = 0x005B0061,
  83. .p0_mpwldectrl1 = 0x004F0055,
  84. .p0_mpdgctrl0 = 0x0314030C,
  85. .p0_mpdgctrl1 = 0x025C0268,
  86. .p0_mprddlctl = 0x42464646,
  87. .p0_mpwrdlctl = 0x36322C34,
  88. };
  89. static struct mx6_ddr_sysinfo cm_fx6_sysinfo_s = {
  90. .cs1_mirror = 1,
  91. .cs_density = 16,
  92. .bi_on = 1,
  93. .rtt_nom = 1,
  94. .rtt_wr = 0,
  95. .ralat = 5,
  96. .walat = 1,
  97. .mif3_mode = 3,
  98. .rst_to_cke = 0x23,
  99. .sde_to_rst = 0x10,
  100. };
  101. static struct mx6_ddr3_cfg cm_fx6_ddr3_cfg_s = {
  102. .mem_speed = 800,
  103. .density = 4,
  104. .rowaddr = 14,
  105. .coladdr = 10,
  106. .pagesz = 2,
  107. .trcd = 1800,
  108. .trcmin = 5200,
  109. .trasmin = 3600,
  110. .SRT = 0,
  111. };
  112. static void spl_mx6s_dram_init(enum ddr_config dram_config, bool reset)
  113. {
  114. if (reset)
  115. ((struct mmdc_p_regs *)MX6_MMDC_P0_MDCTL)->mdmisc = 2;
  116. switch (dram_config) {
  117. case DDR_16BIT_256MB:
  118. cm_fx6_sysinfo_s.dsize = 0;
  119. cm_fx6_sysinfo_s.ncs = 1;
  120. break;
  121. case DDR_32BIT_512MB:
  122. cm_fx6_sysinfo_s.dsize = 1;
  123. cm_fx6_sysinfo_s.ncs = 1;
  124. break;
  125. case DDR_32BIT_1GB:
  126. cm_fx6_sysinfo_s.dsize = 1;
  127. cm_fx6_sysinfo_s.ncs = 2;
  128. break;
  129. default:
  130. puts("Tried to setup invalid DDR configuration\n");
  131. hang();
  132. }
  133. mx6_dram_cfg(&cm_fx6_sysinfo_s, &cm_fx6_calib_s, &cm_fx6_ddr3_cfg_s);
  134. udelay(100);
  135. }
  136. static struct mx6_mmdc_calibration cm_fx6_calib_q = {
  137. .p0_mpwldectrl0 = 0x00630068,
  138. .p0_mpwldectrl1 = 0x0068005D,
  139. .p0_mpdgctrl0 = 0x04140428,
  140. .p0_mpdgctrl1 = 0x037C037C,
  141. .p0_mprddlctl = 0x3C30303A,
  142. .p0_mpwrdlctl = 0x3A344038,
  143. .p1_mpwldectrl0 = 0x0035004C,
  144. .p1_mpwldectrl1 = 0x00170026,
  145. .p1_mpdgctrl0 = 0x0374037C,
  146. .p1_mpdgctrl1 = 0x0350032C,
  147. .p1_mprddlctl = 0x30322A3C,
  148. .p1_mpwrdlctl = 0x48304A3E,
  149. };
  150. static struct mx6_ddr_sysinfo cm_fx6_sysinfo_q = {
  151. .cs_density = 16,
  152. .cs1_mirror = 1,
  153. .bi_on = 1,
  154. .rtt_nom = 1,
  155. .rtt_wr = 0,
  156. .ralat = 5,
  157. .walat = 1,
  158. .mif3_mode = 3,
  159. .rst_to_cke = 0x23,
  160. .sde_to_rst = 0x10,
  161. .refsel = 1, /* Refresh cycles at 32KHz */
  162. .refr = 7, /* 8 refresh commands per refresh cycle */
  163. };
  164. static struct mx6_ddr3_cfg cm_fx6_ddr3_cfg_q = {
  165. .mem_speed = 1066,
  166. .density = 4,
  167. .rowaddr = 14,
  168. .coladdr = 10,
  169. .pagesz = 2,
  170. .trcd = 1324,
  171. .trcmin = 59500,
  172. .trasmin = 9750,
  173. .SRT = 0,
  174. };
  175. static void spl_mx6q_dram_init(enum ddr_config dram_config, bool reset)
  176. {
  177. if (reset)
  178. ((struct mmdc_p_regs *)MX6_MMDC_P0_MDCTL)->mdmisc = 2;
  179. cm_fx6_ddr3_cfg_q.rowaddr = 14;
  180. switch (dram_config) {
  181. case DDR_16BIT_256MB:
  182. cm_fx6_sysinfo_q.dsize = 0;
  183. cm_fx6_sysinfo_q.ncs = 1;
  184. break;
  185. case DDR_32BIT_512MB:
  186. cm_fx6_sysinfo_q.dsize = 1;
  187. cm_fx6_sysinfo_q.ncs = 1;
  188. break;
  189. case DDR_64BIT_1GB:
  190. cm_fx6_sysinfo_q.dsize = 2;
  191. cm_fx6_sysinfo_q.ncs = 1;
  192. break;
  193. case DDR_64BIT_2GB:
  194. cm_fx6_sysinfo_q.dsize = 2;
  195. cm_fx6_sysinfo_q.ncs = 2;
  196. break;
  197. case DDR_64BIT_4GB:
  198. cm_fx6_sysinfo_q.dsize = 2;
  199. cm_fx6_sysinfo_q.ncs = 2;
  200. cm_fx6_ddr3_cfg_q.rowaddr = 15;
  201. break;
  202. default:
  203. puts("Tried to setup invalid DDR configuration\n");
  204. hang();
  205. }
  206. mx6_dram_cfg(&cm_fx6_sysinfo_q, &cm_fx6_calib_q, &cm_fx6_ddr3_cfg_q);
  207. udelay(100);
  208. }
  209. static int cm_fx6_spl_dram_init(void)
  210. {
  211. unsigned long bank1_size, bank2_size;
  212. switch (get_cpu_type()) {
  213. case MXC_CPU_MX6SOLO:
  214. mx6sdl_dram_iocfg(64, &ddr_iomux_s, &grp_iomux_s);
  215. spl_mx6s_dram_init(DDR_32BIT_1GB, false);
  216. bank1_size = get_ram_size((long int *)PHYS_SDRAM_1, 0x80000000);
  217. bank2_size = get_ram_size((long int *)PHYS_SDRAM_2, 0x80000000);
  218. if (bank1_size == 0x20000000) {
  219. if (bank2_size == 0x20000000)
  220. return 0;
  221. spl_mx6s_dram_init(DDR_32BIT_512MB, true);
  222. return 0;
  223. }
  224. spl_mx6s_dram_init(DDR_16BIT_256MB, true);
  225. bank1_size = get_ram_size((long int *)PHYS_SDRAM_1, 0x80000000);
  226. if (bank1_size == 0x10000000)
  227. return 0;
  228. break;
  229. case MXC_CPU_MX6D:
  230. case MXC_CPU_MX6Q:
  231. mx6dq_dram_iocfg(64, &ddr_iomux_q, &grp_iomux_q);
  232. spl_mx6q_dram_init(DDR_64BIT_4GB, false);
  233. bank1_size = get_ram_size((long int *)PHYS_SDRAM_1, 0x80000000);
  234. if (bank1_size == 0x80000000)
  235. return 0;
  236. if (bank1_size == 0x40000000) {
  237. bank2_size = get_ram_size((long int *)PHYS_SDRAM_2,
  238. 0x80000000);
  239. if (bank2_size == 0x40000000) {
  240. /* Don't do a full reset here */
  241. spl_mx6q_dram_init(DDR_64BIT_2GB, false);
  242. } else {
  243. spl_mx6q_dram_init(DDR_64BIT_1GB, true);
  244. }
  245. return 0;
  246. }
  247. spl_mx6q_dram_init(DDR_32BIT_512MB, true);
  248. bank1_size = get_ram_size((long int *)PHYS_SDRAM_1, 0x80000000);
  249. if (bank1_size == 0x20000000)
  250. return 0;
  251. spl_mx6q_dram_init(DDR_16BIT_256MB, true);
  252. bank1_size = get_ram_size((long int *)PHYS_SDRAM_1, 0x80000000);
  253. if (bank1_size == 0x10000000)
  254. return 0;
  255. break;
  256. }
  257. return -1;
  258. }
  259. static iomux_v3_cfg_t const uart4_pads[] = {
  260. IOMUX_PADS(PAD_KEY_COL0__UART4_TX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL)),
  261. IOMUX_PADS(PAD_KEY_ROW0__UART4_RX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL)),
  262. };
  263. static void cm_fx6_setup_uart(void)
  264. {
  265. SETUP_IOMUX_PADS(uart4_pads);
  266. enable_uart_clk(1);
  267. }
  268. #ifdef CONFIG_SPL_SPI_SUPPORT
  269. static void cm_fx6_setup_ecspi(void)
  270. {
  271. cm_fx6_set_ecspi_iomux();
  272. enable_spi_clk(1, 0);
  273. }
  274. #else
  275. static void cm_fx6_setup_ecspi(void) { }
  276. #endif
  277. void board_init_f(ulong dummy)
  278. {
  279. struct mxc_ccm_reg *mxc_ccm = (struct mxc_ccm_reg *)CCM_BASE_ADDR;
  280. /*
  281. * We don't use DMA in SPL, but we do need it in U-Boot. U-Boot
  282. * initializes DMA very early (before all board code), so the only
  283. * opportunity we have to initialize APBHDMA clocks is in SPL.
  284. */
  285. setbits_le32(&mxc_ccm->CCGR0, MXC_CCM_CCGR0_APBHDMA_MASK);
  286. enable_usdhc_clk(1, 2);
  287. arch_cpu_init();
  288. timer_init();
  289. cm_fx6_setup_ecspi();
  290. cm_fx6_setup_uart();
  291. get_clocks();
  292. preloader_console_init();
  293. gpio_direction_output(CM_FX6_GREEN_LED, 1);
  294. if (cm_fx6_spl_dram_init()) {
  295. puts("!!!ERROR!!! DRAM detection failed!!!\n");
  296. hang();
  297. }
  298. }
  299. void board_boot_order(u32 *spl_boot_list)
  300. {
  301. spl_boot_list[0] = spl_boot_device();
  302. switch (spl_boot_list[0]) {
  303. case BOOT_DEVICE_SPI:
  304. spl_boot_list[1] = BOOT_DEVICE_MMC1;
  305. break;
  306. case BOOT_DEVICE_MMC1:
  307. spl_boot_list[1] = BOOT_DEVICE_SPI;
  308. break;
  309. }
  310. }
  311. #ifdef CONFIG_SPL_MMC_SUPPORT
  312. static struct fsl_esdhc_cfg usdhc_cfg = {
  313. .esdhc_base = USDHC3_BASE_ADDR,
  314. .max_bus_width = 4,
  315. };
  316. int board_mmc_init(bd_t *bis)
  317. {
  318. cm_fx6_set_usdhc_iomux();
  319. usdhc_cfg.sdhc_clk = mxc_get_clock(MXC_ESDHC3_CLK);
  320. return fsl_esdhc_initialize(bis, &usdhc_cfg);
  321. }
  322. #endif