clock.c 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367
  1. /*
  2. * Copyright (C) 2016 Freescale Semiconductor, Inc.
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. */
  6. #include <common.h>
  7. #include <div64.h>
  8. #include <asm/io.h>
  9. #include <errno.h>
  10. #include <asm/arch/clock.h>
  11. #include <asm/arch/sys_proto.h>
  12. DECLARE_GLOBAL_DATA_PTR;
  13. int get_clocks(void)
  14. {
  15. #ifdef CONFIG_FSL_ESDHC
  16. #if CONFIG_SYS_FSL_ESDHC_ADDR == USDHC0_RBASE
  17. gd->arch.sdhc_clk = mxc_get_clock(MXC_ESDHC_CLK);
  18. #elif CONFIG_SYS_FSL_ESDHC_ADDR == USDHC1_RBASE
  19. gd->arch.sdhc_clk = mxc_get_clock(MXC_ESDHC2_CLK);
  20. #endif
  21. #endif
  22. return 0;
  23. }
  24. static u32 get_fast_plat_clk(void)
  25. {
  26. return scg_clk_get_rate(SCG_NIC0_CLK);
  27. }
  28. static u32 get_slow_plat_clk(void)
  29. {
  30. return scg_clk_get_rate(SCG_NIC1_CLK);
  31. }
  32. static u32 get_ipg_clk(void)
  33. {
  34. return scg_clk_get_rate(SCG_NIC1_BUS_CLK);
  35. }
  36. u32 get_lpuart_clk(void)
  37. {
  38. int index = 0;
  39. const u32 lpuart_array[] = {
  40. LPUART0_RBASE,
  41. LPUART1_RBASE,
  42. LPUART2_RBASE,
  43. LPUART3_RBASE,
  44. LPUART4_RBASE,
  45. LPUART5_RBASE,
  46. LPUART6_RBASE,
  47. LPUART7_RBASE,
  48. };
  49. const enum pcc_clk lpuart_pcc_clks[] = {
  50. PER_CLK_LPUART4,
  51. PER_CLK_LPUART5,
  52. PER_CLK_LPUART6,
  53. PER_CLK_LPUART7,
  54. };
  55. for (index = 0; index < 8; index++) {
  56. if (lpuart_array[index] == LPUART_BASE)
  57. break;
  58. }
  59. if (index < 4 || index > 7)
  60. return 0;
  61. return pcc_clock_get_rate(lpuart_pcc_clks[index - 4]);
  62. }
  63. #ifdef CONFIG_SYS_LPI2C_IMX
  64. int enable_i2c_clk(unsigned char enable, unsigned i2c_num)
  65. {
  66. /* Set parent to FIRC DIV2 clock */
  67. const enum pcc_clk lpi2c_pcc_clks[] = {
  68. PER_CLK_LPI2C4,
  69. PER_CLK_LPI2C5,
  70. PER_CLK_LPI2C6,
  71. PER_CLK_LPI2C7,
  72. };
  73. if (i2c_num < 4 || i2c_num > 7)
  74. return -EINVAL;
  75. if (enable) {
  76. pcc_clock_enable(lpi2c_pcc_clks[i2c_num - 4], false);
  77. pcc_clock_sel(lpi2c_pcc_clks[i2c_num - 4], SCG_FIRC_DIV2_CLK);
  78. pcc_clock_enable(lpi2c_pcc_clks[i2c_num - 4], true);
  79. } else {
  80. pcc_clock_enable(lpi2c_pcc_clks[i2c_num - 4], false);
  81. }
  82. return 0;
  83. }
  84. u32 imx_get_i2cclk(unsigned i2c_num)
  85. {
  86. const enum pcc_clk lpi2c_pcc_clks[] = {
  87. PER_CLK_LPI2C4,
  88. PER_CLK_LPI2C5,
  89. PER_CLK_LPI2C6,
  90. PER_CLK_LPI2C7,
  91. };
  92. if (i2c_num < 4 || i2c_num > 7)
  93. return 0;
  94. return pcc_clock_get_rate(lpi2c_pcc_clks[i2c_num - 4]);
  95. }
  96. #endif
  97. unsigned int mxc_get_clock(enum mxc_clock clk)
  98. {
  99. switch (clk) {
  100. case MXC_ARM_CLK:
  101. return scg_clk_get_rate(SCG_CORE_CLK);
  102. case MXC_AXI_CLK:
  103. return get_fast_plat_clk();
  104. case MXC_AHB_CLK:
  105. return get_slow_plat_clk();
  106. case MXC_IPG_CLK:
  107. return get_ipg_clk();
  108. case MXC_I2C_CLK:
  109. return pcc_clock_get_rate(PER_CLK_LPI2C4);
  110. case MXC_UART_CLK:
  111. return get_lpuart_clk();
  112. case MXC_ESDHC_CLK:
  113. return pcc_clock_get_rate(PER_CLK_USDHC0);
  114. case MXC_ESDHC2_CLK:
  115. return pcc_clock_get_rate(PER_CLK_USDHC1);
  116. case MXC_DDR_CLK:
  117. return scg_clk_get_rate(SCG_DDR_CLK);
  118. default:
  119. printf("Unsupported mxc_clock %d\n", clk);
  120. break;
  121. }
  122. return 0;
  123. }
  124. void init_clk_usdhc(u32 index)
  125. {
  126. switch (index) {
  127. case 0:
  128. /*Disable the clock before configure it */
  129. pcc_clock_enable(PER_CLK_USDHC0, false);
  130. /* 158MHz / 1 = 158MHz */
  131. pcc_clock_sel(PER_CLK_USDHC0, SCG_NIC1_CLK);
  132. pcc_clock_div_config(PER_CLK_USDHC0, false, 1);
  133. pcc_clock_enable(PER_CLK_USDHC0, true);
  134. break;
  135. case 1:
  136. /*Disable the clock before configure it */
  137. pcc_clock_enable(PER_CLK_USDHC1, false);
  138. /* 158MHz / 1 = 158MHz */
  139. pcc_clock_sel(PER_CLK_USDHC1, SCG_NIC1_CLK);
  140. pcc_clock_div_config(PER_CLK_USDHC1, false, 1);
  141. pcc_clock_enable(PER_CLK_USDHC1, true);
  142. break;
  143. default:
  144. printf("Invalid index for USDHC %d\n", index);
  145. break;
  146. }
  147. }
  148. #ifdef CONFIG_MXC_OCOTP
  149. #define OCOTP_CTRL_PCC1_SLOT (38)
  150. #define OCOTP_CTRL_HIGH4K_PCC1_SLOT (39)
  151. void enable_ocotp_clk(unsigned char enable)
  152. {
  153. u32 val;
  154. /*
  155. * Seems the OCOTP CLOCKs have been enabled at default,
  156. * check its inuse flag
  157. */
  158. val = readl(PCC1_RBASE + 4 * OCOTP_CTRL_PCC1_SLOT);
  159. if (!(val & PCC_INUSE_MASK))
  160. writel(PCC_CGC_MASK, (PCC1_RBASE + 4 * OCOTP_CTRL_PCC1_SLOT));
  161. val = readl(PCC1_RBASE + 4 * OCOTP_CTRL_HIGH4K_PCC1_SLOT);
  162. if (!(val & PCC_INUSE_MASK))
  163. writel(PCC_CGC_MASK,
  164. (PCC1_RBASE + 4 * OCOTP_CTRL_HIGH4K_PCC1_SLOT));
  165. }
  166. #endif
  167. void enable_usboh3_clk(unsigned char enable)
  168. {
  169. if (enable) {
  170. pcc_clock_enable(PER_CLK_USB0, false);
  171. pcc_clock_sel(PER_CLK_USB0, SCG_NIC1_BUS_CLK);
  172. pcc_clock_enable(PER_CLK_USB0, true);
  173. #ifdef CONFIG_USB_MAX_CONTROLLER_COUNT
  174. if (CONFIG_USB_MAX_CONTROLLER_COUNT > 1) {
  175. pcc_clock_enable(PER_CLK_USB1, false);
  176. pcc_clock_sel(PER_CLK_USB1, SCG_NIC1_BUS_CLK);
  177. pcc_clock_enable(PER_CLK_USB1, true);
  178. }
  179. #endif
  180. pcc_clock_enable(PER_CLK_USB_PHY, true);
  181. pcc_clock_enable(PER_CLK_USB_PL301, true);
  182. } else {
  183. pcc_clock_enable(PER_CLK_USB0, false);
  184. pcc_clock_enable(PER_CLK_USB1, false);
  185. pcc_clock_enable(PER_CLK_USB_PHY, false);
  186. pcc_clock_enable(PER_CLK_USB_PL301, false);
  187. }
  188. }
  189. static void lpuart_set_clk(uint32_t index, enum scg_clk clk)
  190. {
  191. const enum pcc_clk lpuart_pcc_clks[] = {
  192. PER_CLK_LPUART4,
  193. PER_CLK_LPUART5,
  194. PER_CLK_LPUART6,
  195. PER_CLK_LPUART7,
  196. };
  197. if (index < 4 || index > 7)
  198. return;
  199. #ifndef CONFIG_CLK_DEBUG
  200. pcc_clock_enable(lpuart_pcc_clks[index - 4], false);
  201. #endif
  202. pcc_clock_sel(lpuart_pcc_clks[index - 4], clk);
  203. pcc_clock_enable(lpuart_pcc_clks[index - 4], true);
  204. }
  205. static void init_clk_lpuart(void)
  206. {
  207. u32 index = 0, i;
  208. const u32 lpuart_array[] = {
  209. LPUART0_RBASE,
  210. LPUART1_RBASE,
  211. LPUART2_RBASE,
  212. LPUART3_RBASE,
  213. LPUART4_RBASE,
  214. LPUART5_RBASE,
  215. LPUART6_RBASE,
  216. LPUART7_RBASE,
  217. };
  218. for (i = 0; i < 8; i++) {
  219. if (lpuart_array[i] == LPUART_BASE) {
  220. index = i;
  221. break;
  222. }
  223. }
  224. lpuart_set_clk(index, SCG_SOSC_DIV2_CLK);
  225. }
  226. static void init_clk_rgpio2p(void)
  227. {
  228. /*Enable RGPIO2P1 clock */
  229. pcc_clock_enable(PER_CLK_RGPIO2P1, true);
  230. /*
  231. * Hard code to enable RGPIO2P0 clock since it is not
  232. * in clock frame for A7 domain
  233. */
  234. writel(PCC_CGC_MASK, (PCC0_RBASE + 0x3C));
  235. }
  236. /* Configure PLL/PFD freq */
  237. void clock_init(void)
  238. {
  239. /*
  240. * ROM has enabled clocks:
  241. * A4 side: SIRC 16Mhz (DIV1-3 off), FIRC 48Mhz (DIV1-2 on),
  242. * Non-LP-boot: SOSC, SPLL PFD0 (scs selected)
  243. * A7 side: SPLL PFD0 (scs selected, 413Mhz),
  244. * APLL PFD0 (352Mhz), DDRCLK, all NIC clocks
  245. * A7 Plat0 (NIC0) = 176Mhz, Plat1 (NIC1) = 176Mhz,
  246. * IP BUS (NIC1_BUS) = 58.6Mhz
  247. *
  248. * In u-boot:
  249. * 1. Enable PFD1-3 of APLL for A7 side. Enable FIRC and DIVs.
  250. * 2. Enable USB PLL
  251. * 3. Init the clocks of peripherals used in u-boot bu
  252. * without set rate interface.The clocks for these
  253. * peripherals are enabled in this intialization.
  254. * 4.Other peripherals with set clock rate interface
  255. * does not be set in this function.
  256. */
  257. scg_a7_firc_init();
  258. scg_a7_soscdiv_init();
  259. /* APLL PFD1 = 270Mhz, PFD2=480Mhz, PFD3=800Mhz */
  260. scg_enable_pll_pfd(SCG_APLL_PFD1_CLK, 35);
  261. scg_enable_pll_pfd(SCG_APLL_PFD2_CLK, 20);
  262. scg_enable_pll_pfd(SCG_APLL_PFD3_CLK, 12);
  263. init_clk_lpuart();
  264. init_clk_rgpio2p();
  265. enable_usboh3_clk(1);
  266. }
  267. #ifdef CONFIG_SECURE_BOOT
  268. void hab_caam_clock_enable(unsigned char enable)
  269. {
  270. if (enable)
  271. pcc_clock_enable(PER_CLK_CAAM, true);
  272. else
  273. pcc_clock_enable(PER_CLK_CAAM, false);
  274. }
  275. #endif
  276. #ifndef CONFIG_SPL_BUILD
  277. /*
  278. * Dump some core clockes.
  279. */
  280. int do_mx7_showclocks(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  281. {
  282. u32 addr = 0;
  283. u32 freq;
  284. freq = decode_pll(PLL_A7_SPLL);
  285. printf("PLL_A7_SPLL %8d MHz\n", freq / 1000000);
  286. freq = decode_pll(PLL_A7_APLL);
  287. printf("PLL_A7_APLL %8d MHz\n", freq / 1000000);
  288. freq = decode_pll(PLL_USB);
  289. printf("PLL_USB %8d MHz\n", freq / 1000000);
  290. printf("\n");
  291. printf("CORE %8d kHz\n", scg_clk_get_rate(SCG_CORE_CLK) / 1000);
  292. printf("IPG %8d kHz\n", mxc_get_clock(MXC_IPG_CLK) / 1000);
  293. printf("UART %8d kHz\n", mxc_get_clock(MXC_UART_CLK) / 1000);
  294. printf("AHB %8d kHz\n", mxc_get_clock(MXC_AHB_CLK) / 1000);
  295. printf("AXI %8d kHz\n", mxc_get_clock(MXC_AXI_CLK) / 1000);
  296. printf("DDR %8d kHz\n", mxc_get_clock(MXC_DDR_CLK) / 1000);
  297. printf("USDHC1 %8d kHz\n", mxc_get_clock(MXC_ESDHC_CLK) / 1000);
  298. printf("USDHC2 %8d kHz\n", mxc_get_clock(MXC_ESDHC2_CLK) / 1000);
  299. printf("I2C4 %8d kHz\n", mxc_get_clock(MXC_I2C_CLK) / 1000);
  300. addr = (u32) clock_init;
  301. printf("[%s] addr = 0x%08X\r\n", __func__, addr);
  302. scg_a7_info();
  303. return 0;
  304. }
  305. U_BOOT_CMD(
  306. clocks, CONFIG_SYS_MAXARGS, 1, do_mx7_showclocks,
  307. "display clocks",
  308. ""
  309. );
  310. #endif