generic.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  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/clock.h>
  9. #include <asm/arch/crm_regs.h>
  10. #include <asm/mach-imx/sys_proto.h>
  11. #include <netdev.h>
  12. #ifdef CONFIG_FSL_ESDHC
  13. #include <fsl_esdhc.h>
  14. #endif
  15. #ifdef CONFIG_FSL_ESDHC
  16. DECLARE_GLOBAL_DATA_PTR;
  17. #endif
  18. static char soc_type[] = "xx0";
  19. #ifdef CONFIG_MXC_OCOTP
  20. void enable_ocotp_clk(unsigned char enable)
  21. {
  22. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  23. u32 reg;
  24. reg = readl(&ccm->ccgr6);
  25. if (enable)
  26. reg |= CCM_CCGR6_OCOTP_CTRL_MASK;
  27. else
  28. reg &= ~CCM_CCGR6_OCOTP_CTRL_MASK;
  29. writel(reg, &ccm->ccgr6);
  30. }
  31. #endif
  32. static u32 get_mcu_main_clk(void)
  33. {
  34. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  35. u32 ccm_ccsr, ccm_cacrr, armclk_div;
  36. u32 sysclk_sel, pll_pfd_sel = 0;
  37. u32 freq = 0;
  38. ccm_ccsr = readl(&ccm->ccsr);
  39. sysclk_sel = ccm_ccsr & CCM_CCSR_SYS_CLK_SEL_MASK;
  40. sysclk_sel >>= CCM_CCSR_SYS_CLK_SEL_OFFSET;
  41. ccm_cacrr = readl(&ccm->cacrr);
  42. armclk_div = ccm_cacrr & CCM_CACRR_ARM_CLK_DIV_MASK;
  43. armclk_div >>= CCM_CACRR_ARM_CLK_DIV_OFFSET;
  44. armclk_div += 1;
  45. switch (sysclk_sel) {
  46. case 0:
  47. freq = FASE_CLK_FREQ;
  48. break;
  49. case 1:
  50. freq = SLOW_CLK_FREQ;
  51. break;
  52. case 2:
  53. pll_pfd_sel = ccm_ccsr & CCM_CCSR_PLL2_PFD_CLK_SEL_MASK;
  54. pll_pfd_sel >>= CCM_CCSR_PLL2_PFD_CLK_SEL_OFFSET;
  55. if (pll_pfd_sel == 0)
  56. freq = PLL2_MAIN_FREQ;
  57. else if (pll_pfd_sel == 1)
  58. freq = PLL2_PFD1_FREQ;
  59. else if (pll_pfd_sel == 2)
  60. freq = PLL2_PFD2_FREQ;
  61. else if (pll_pfd_sel == 3)
  62. freq = PLL2_PFD3_FREQ;
  63. else if (pll_pfd_sel == 4)
  64. freq = PLL2_PFD4_FREQ;
  65. break;
  66. case 3:
  67. freq = PLL2_MAIN_FREQ;
  68. break;
  69. case 4:
  70. pll_pfd_sel = ccm_ccsr & CCM_CCSR_PLL1_PFD_CLK_SEL_MASK;
  71. pll_pfd_sel >>= CCM_CCSR_PLL1_PFD_CLK_SEL_OFFSET;
  72. if (pll_pfd_sel == 0)
  73. freq = PLL1_MAIN_FREQ;
  74. else if (pll_pfd_sel == 1)
  75. freq = PLL1_PFD1_FREQ;
  76. else if (pll_pfd_sel == 2)
  77. freq = PLL1_PFD2_FREQ;
  78. else if (pll_pfd_sel == 3)
  79. freq = PLL1_PFD3_FREQ;
  80. else if (pll_pfd_sel == 4)
  81. freq = PLL1_PFD4_FREQ;
  82. break;
  83. case 5:
  84. freq = PLL3_MAIN_FREQ;
  85. break;
  86. default:
  87. printf("unsupported system clock select\n");
  88. }
  89. return freq / armclk_div;
  90. }
  91. static u32 get_bus_clk(void)
  92. {
  93. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  94. u32 ccm_cacrr, busclk_div;
  95. ccm_cacrr = readl(&ccm->cacrr);
  96. busclk_div = ccm_cacrr & CCM_CACRR_BUS_CLK_DIV_MASK;
  97. busclk_div >>= CCM_CACRR_BUS_CLK_DIV_OFFSET;
  98. busclk_div += 1;
  99. return get_mcu_main_clk() / busclk_div;
  100. }
  101. static u32 get_ipg_clk(void)
  102. {
  103. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  104. u32 ccm_cacrr, ipgclk_div;
  105. ccm_cacrr = readl(&ccm->cacrr);
  106. ipgclk_div = ccm_cacrr & CCM_CACRR_IPG_CLK_DIV_MASK;
  107. ipgclk_div >>= CCM_CACRR_IPG_CLK_DIV_OFFSET;
  108. ipgclk_div += 1;
  109. return get_bus_clk() / ipgclk_div;
  110. }
  111. static u32 get_uart_clk(void)
  112. {
  113. return get_ipg_clk();
  114. }
  115. static u32 get_sdhc_clk(void)
  116. {
  117. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  118. u32 ccm_cscmr1, ccm_cscdr2, sdhc_clk_sel, sdhc_clk_div;
  119. u32 freq = 0;
  120. ccm_cscmr1 = readl(&ccm->cscmr1);
  121. sdhc_clk_sel = ccm_cscmr1 & CCM_CSCMR1_ESDHC1_CLK_SEL_MASK;
  122. sdhc_clk_sel >>= CCM_CSCMR1_ESDHC1_CLK_SEL_OFFSET;
  123. ccm_cscdr2 = readl(&ccm->cscdr2);
  124. sdhc_clk_div = ccm_cscdr2 & CCM_CSCDR2_ESDHC1_CLK_DIV_MASK;
  125. sdhc_clk_div >>= CCM_CSCDR2_ESDHC1_CLK_DIV_OFFSET;
  126. sdhc_clk_div += 1;
  127. switch (sdhc_clk_sel) {
  128. case 0:
  129. freq = PLL3_MAIN_FREQ;
  130. break;
  131. case 1:
  132. freq = PLL3_PFD3_FREQ;
  133. break;
  134. case 2:
  135. freq = PLL1_PFD3_FREQ;
  136. break;
  137. case 3:
  138. freq = get_bus_clk();
  139. break;
  140. }
  141. return freq / sdhc_clk_div;
  142. }
  143. u32 get_fec_clk(void)
  144. {
  145. struct ccm_reg *ccm = (struct ccm_reg *)CCM_BASE_ADDR;
  146. u32 ccm_cscmr2, rmii_clk_sel;
  147. u32 freq = 0;
  148. ccm_cscmr2 = readl(&ccm->cscmr2);
  149. rmii_clk_sel = ccm_cscmr2 & CCM_CSCMR2_RMII_CLK_SEL_MASK;
  150. rmii_clk_sel >>= CCM_CSCMR2_RMII_CLK_SEL_OFFSET;
  151. switch (rmii_clk_sel) {
  152. case 0:
  153. freq = ENET_EXTERNAL_CLK;
  154. break;
  155. case 1:
  156. freq = AUDIO_EXTERNAL_CLK;
  157. break;
  158. case 2:
  159. freq = PLL5_MAIN_FREQ;
  160. break;
  161. case 3:
  162. freq = PLL5_MAIN_FREQ / 2;
  163. break;
  164. }
  165. return freq;
  166. }
  167. static u32 get_i2c_clk(void)
  168. {
  169. return get_ipg_clk();
  170. }
  171. static u32 get_dspi_clk(void)
  172. {
  173. return get_ipg_clk();
  174. }
  175. u32 get_lpuart_clk(void)
  176. {
  177. return get_uart_clk();
  178. }
  179. unsigned int mxc_get_clock(enum mxc_clock clk)
  180. {
  181. switch (clk) {
  182. case MXC_ARM_CLK:
  183. return get_mcu_main_clk();
  184. case MXC_BUS_CLK:
  185. return get_bus_clk();
  186. case MXC_IPG_CLK:
  187. return get_ipg_clk();
  188. case MXC_UART_CLK:
  189. return get_uart_clk();
  190. case MXC_ESDHC_CLK:
  191. return get_sdhc_clk();
  192. case MXC_FEC_CLK:
  193. return get_fec_clk();
  194. case MXC_I2C_CLK:
  195. return get_i2c_clk();
  196. case MXC_DSPI_CLK:
  197. return get_dspi_clk();
  198. default:
  199. break;
  200. }
  201. return -1;
  202. }
  203. /* Dump some core clocks */
  204. int do_vf610_showclocks(cmd_tbl_t *cmdtp, int flag, int argc,
  205. char * const argv[])
  206. {
  207. printf("\n");
  208. printf("cpu clock : %8d MHz\n", mxc_get_clock(MXC_ARM_CLK) / 1000000);
  209. printf("bus clock : %8d MHz\n", mxc_get_clock(MXC_BUS_CLK) / 1000000);
  210. printf("ipg clock : %8d MHz\n", mxc_get_clock(MXC_IPG_CLK) / 1000000);
  211. return 0;
  212. }
  213. U_BOOT_CMD(
  214. clocks, CONFIG_SYS_MAXARGS, 1, do_vf610_showclocks,
  215. "display clocks",
  216. ""
  217. );
  218. #ifdef CONFIG_FEC_MXC
  219. void imx_get_mac_from_fuse(int dev_id, unsigned char *mac)
  220. {
  221. struct ocotp_regs *ocotp = (struct ocotp_regs *)OCOTP_BASE_ADDR;
  222. struct fuse_bank *bank = &ocotp->bank[4];
  223. struct fuse_bank4_regs *fuse =
  224. (struct fuse_bank4_regs *)bank->fuse_regs;
  225. u32 value = readl(&fuse->mac_addr0);
  226. mac[0] = (value >> 8);
  227. mac[1] = value;
  228. value = readl(&fuse->mac_addr1);
  229. mac[2] = value >> 24;
  230. mac[3] = value >> 16;
  231. mac[4] = value >> 8;
  232. mac[5] = value;
  233. }
  234. #endif
  235. u32 get_cpu_rev(void)
  236. {
  237. return MXC_CPU_VF610 << 12;
  238. }
  239. #if defined(CONFIG_DISPLAY_CPUINFO)
  240. static char *get_reset_cause(void)
  241. {
  242. u32 cause;
  243. struct src *src_regs = (struct src *)SRC_BASE_ADDR;
  244. cause = readl(&src_regs->srsr);
  245. writel(cause, &src_regs->srsr);
  246. if (cause & SRC_SRSR_POR_RST)
  247. return "POWER ON RESET";
  248. else if (cause & SRC_SRSR_WDOG_A5)
  249. return "WDOG A5";
  250. else if (cause & SRC_SRSR_WDOG_M4)
  251. return "WDOG M4";
  252. else if (cause & SRC_SRSR_JTAG_RST)
  253. return "JTAG HIGH-Z";
  254. else if (cause & SRC_SRSR_SW_RST)
  255. return "SW RESET";
  256. else if (cause & SRC_SRSR_RESETB)
  257. return "EXTERNAL RESET";
  258. else
  259. return "unknown reset";
  260. }
  261. int print_cpuinfo(void)
  262. {
  263. printf("CPU: Freescale Vybrid VF%s at %d MHz\n",
  264. soc_type, mxc_get_clock(MXC_ARM_CLK) / 1000000);
  265. printf("Reset cause: %s\n", get_reset_cause());
  266. return 0;
  267. }
  268. #endif
  269. int arch_cpu_init(void)
  270. {
  271. struct mscm *mscm = (struct mscm *)MSCM_BASE_ADDR;
  272. soc_type[0] = mscm->cpxcount ? '6' : '5'; /*Dual Core => VF6x0 */
  273. soc_type[1] = mscm->cpxcfg1 ? '1' : '0'; /* L2 Cache => VFx10 */
  274. return 0;
  275. }
  276. #ifdef CONFIG_ARCH_MISC_INIT
  277. int arch_misc_init(void)
  278. {
  279. char soc[6];
  280. strcpy(soc, "vf");
  281. strcat(soc, soc_type);
  282. env_set("soc", soc);
  283. return 0;
  284. }
  285. #endif
  286. int cpu_eth_init(bd_t *bis)
  287. {
  288. int rc = -ENODEV;
  289. #if defined(CONFIG_FEC_MXC)
  290. rc = fecmxc_initialize(bis);
  291. #endif
  292. return rc;
  293. }
  294. #ifdef CONFIG_FSL_ESDHC
  295. int cpu_mmc_init(bd_t *bis)
  296. {
  297. return fsl_esdhc_mmc_init(bis);
  298. }
  299. #endif
  300. int get_clocks(void)
  301. {
  302. #ifdef CONFIG_FSL_ESDHC
  303. gd->arch.sdhc_clk = mxc_get_clock(MXC_ESDHC_CLK);
  304. #endif
  305. return 0;
  306. }
  307. #ifndef CONFIG_SYS_DCACHE_OFF
  308. void enable_caches(void)
  309. {
  310. #if defined(CONFIG_SYS_ARM_CACHE_WRITETHROUGH)
  311. enum dcache_option option = DCACHE_WRITETHROUGH;
  312. #else
  313. enum dcache_option option = DCACHE_WRITEBACK;
  314. #endif
  315. dcache_enable();
  316. icache_enable();
  317. /* Enable caching on OCRAM */
  318. mmu_set_region_dcache_behaviour(IRAM_BASE_ADDR, IRAM_SIZE, option);
  319. }
  320. #endif