generic.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  1. /*
  2. * (C) Copyright 2007
  3. * Sascha Hauer, Pengutronix
  4. *
  5. * (C) Copyright 2008-2010 Freescale Semiconductor, Inc.
  6. *
  7. * See file CREDITS for list of people who contributed to this
  8. * project.
  9. *
  10. * This program is free software; you can redistribute it and/or
  11. * modify it under the terms of the GNU General Public License as
  12. * published by the Free Software Foundation; either version 2 of
  13. * the License, or (at your option) any later version.
  14. *
  15. * This program is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. * GNU General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU General Public License
  21. * along with this program; if not, write to the Free Software
  22. * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
  23. * MA 02111-1307 USA
  24. */
  25. #include <common.h>
  26. #include <div64.h>
  27. #include <asm/io.h>
  28. #include <asm/errno.h>
  29. #include <asm/arch/imx-regs.h>
  30. #include <asm/arch/crm_regs.h>
  31. #include <asm/arch/clock.h>
  32. #include <asm/arch/sys_proto.h>
  33. #ifdef CONFIG_FSL_ESDHC
  34. #include <fsl_esdhc.h>
  35. #endif
  36. #include <netdev.h>
  37. #define CLK_CODE(arm, ahb, sel) (((arm) << 16) + ((ahb) << 8) + (sel))
  38. #define CLK_CODE_ARM(c) (((c) >> 16) & 0xFF)
  39. #define CLK_CODE_AHB(c) (((c) >> 8) & 0xFF)
  40. #define CLK_CODE_PATH(c) ((c) & 0xFF)
  41. #define CCM_GET_DIVIDER(x, m, o) (((x) & (m)) >> (o))
  42. #ifdef CONFIG_FSL_ESDHC
  43. DECLARE_GLOBAL_DATA_PTR;
  44. #endif
  45. static int g_clk_mux_auto[8] = {
  46. CLK_CODE(1, 3, 0), CLK_CODE(1, 2, 1), CLK_CODE(2, 1, 1), -1,
  47. CLK_CODE(1, 6, 0), CLK_CODE(1, 4, 1), CLK_CODE(2, 2, 1), -1,
  48. };
  49. static int g_clk_mux_consumer[16] = {
  50. CLK_CODE(1, 4, 0), CLK_CODE(1, 3, 1), CLK_CODE(1, 3, 1), -1,
  51. -1, -1, CLK_CODE(4, 1, 0), CLK_CODE(1, 5, 0),
  52. CLK_CODE(1, 8, 1), CLK_CODE(1, 6, 1), CLK_CODE(2, 4, 0), -1,
  53. -1, -1, CLK_CODE(4, 2, 0), -1,
  54. };
  55. static int hsp_div_table[3][16] = {
  56. {4, 3, 2, -1, -1, -1, 1, 5, 4, 3, 2, -1, -1, -1, 1, -1},
  57. {-1, -1, -1, -1, -1, -1, -1, -1, 8, 6, 4, -1, -1, -1, 2, -1},
  58. {3, -1, -1, -1, -1, -1, -1, -1, 3, -1, -1, -1, -1, -1, -1, -1},
  59. };
  60. u32 get_cpu_rev(void)
  61. {
  62. int reg;
  63. struct iim_regs *iim =
  64. (struct iim_regs *)IIM_BASE_ADDR;
  65. reg = readl(&iim->iim_srev);
  66. if (!reg) {
  67. reg = readw(ROMPATCH_REV);
  68. reg <<= 4;
  69. } else {
  70. reg += CHIP_REV_1_0;
  71. }
  72. return 0x35000 + (reg & 0xFF);
  73. }
  74. static u32 get_arm_div(u32 pdr0, u32 *fi, u32 *fd)
  75. {
  76. int *pclk_mux;
  77. if (pdr0 & MXC_CCM_PDR0_AUTO_CON) {
  78. pclk_mux = g_clk_mux_consumer +
  79. ((pdr0 & MXC_CCM_PDR0_CON_MUX_DIV_MASK) >>
  80. MXC_CCM_PDR0_CON_MUX_DIV_OFFSET);
  81. } else {
  82. pclk_mux = g_clk_mux_auto +
  83. ((pdr0 & MXC_CCM_PDR0_AUTO_MUX_DIV_MASK) >>
  84. MXC_CCM_PDR0_AUTO_MUX_DIV_OFFSET);
  85. }
  86. if ((*pclk_mux) == -1)
  87. return -1;
  88. if (fi && fd) {
  89. if (!CLK_CODE_PATH(*pclk_mux)) {
  90. *fi = *fd = 1;
  91. return CLK_CODE_ARM(*pclk_mux);
  92. }
  93. if (pdr0 & MXC_CCM_PDR0_AUTO_CON) {
  94. *fi = 3;
  95. *fd = 4;
  96. } else {
  97. *fi = 2;
  98. *fd = 3;
  99. }
  100. }
  101. return CLK_CODE_ARM(*pclk_mux);
  102. }
  103. static int get_ahb_div(u32 pdr0)
  104. {
  105. int *pclk_mux;
  106. pclk_mux = g_clk_mux_consumer +
  107. ((pdr0 & MXC_CCM_PDR0_CON_MUX_DIV_MASK) >>
  108. MXC_CCM_PDR0_CON_MUX_DIV_OFFSET);
  109. if ((*pclk_mux) == -1)
  110. return -1;
  111. return CLK_CODE_AHB(*pclk_mux);
  112. }
  113. static u32 decode_pll(u32 reg, u32 infreq)
  114. {
  115. u32 mfi = (reg >> 10) & 0xf;
  116. s32 mfn = reg & 0x3ff;
  117. u32 mfd = (reg >> 16) & 0x3ff;
  118. u32 pd = (reg >> 26) & 0xf;
  119. mfi = mfi <= 5 ? 5 : mfi;
  120. mfn = mfn >= 512 ? mfn - 1024 : mfn;
  121. mfd += 1;
  122. pd += 1;
  123. return lldiv(2 * (u64)infreq * (mfi * mfd + mfn),
  124. mfd * pd);
  125. }
  126. static u32 get_mcu_main_clk(void)
  127. {
  128. u32 arm_div = 0, fi = 0, fd = 0;
  129. struct ccm_regs *ccm =
  130. (struct ccm_regs *)IMX_CCM_BASE;
  131. arm_div = get_arm_div(readl(&ccm->pdr0), &fi, &fd);
  132. fi *= decode_pll(readl(&ccm->mpctl), MXC_HCLK);
  133. return fi / (arm_div * fd);
  134. }
  135. static u32 get_ipg_clk(void)
  136. {
  137. u32 freq = get_mcu_main_clk();
  138. struct ccm_regs *ccm =
  139. (struct ccm_regs *)IMX_CCM_BASE;
  140. u32 pdr0 = readl(&ccm->pdr0);
  141. return freq / (get_ahb_div(pdr0) * 2);
  142. }
  143. static u32 get_ipg_per_clk(void)
  144. {
  145. u32 freq = get_mcu_main_clk();
  146. struct ccm_regs *ccm =
  147. (struct ccm_regs *)IMX_CCM_BASE;
  148. u32 pdr0 = readl(&ccm->pdr0);
  149. u32 pdr4 = readl(&ccm->pdr4);
  150. u32 div;
  151. if (pdr0 & MXC_CCM_PDR0_PER_SEL) {
  152. div = CCM_GET_DIVIDER(pdr4,
  153. MXC_CCM_PDR4_PER0_PODF_MASK,
  154. MXC_CCM_PDR4_PER0_PODF_OFFSET) + 1;
  155. } else {
  156. div = CCM_GET_DIVIDER(pdr0,
  157. MXC_CCM_PDR0_PER_PODF_MASK,
  158. MXC_CCM_PDR0_PER_PODF_OFFSET) + 1;
  159. div *= get_ahb_div(pdr0);
  160. }
  161. return freq / div;
  162. }
  163. u32 imx_get_uartclk(void)
  164. {
  165. u32 freq;
  166. struct ccm_regs *ccm =
  167. (struct ccm_regs *)IMX_CCM_BASE;
  168. u32 pdr4 = readl(&ccm->pdr4);
  169. if (readl(&ccm->pdr3) & MXC_CCM_PDR3_UART_M_U)
  170. freq = get_mcu_main_clk();
  171. else
  172. freq = decode_pll(readl(&ccm->ppctl), MXC_HCLK);
  173. freq /= CCM_GET_DIVIDER(pdr4,
  174. MXC_CCM_PDR4_UART_PODF_MASK,
  175. MXC_CCM_PDR4_UART_PODF_OFFSET) + 1;
  176. return freq;
  177. }
  178. unsigned int mxc_get_main_clock(enum mxc_main_clock clk)
  179. {
  180. u32 nfc_pdf, hsp_podf;
  181. u32 pll, ret_val = 0, usb_podf;
  182. struct ccm_regs *ccm =
  183. (struct ccm_regs *)IMX_CCM_BASE;
  184. u32 reg = readl(&ccm->pdr0);
  185. u32 reg4 = readl(&ccm->pdr4);
  186. reg |= 0x1;
  187. switch (clk) {
  188. case CPU_CLK:
  189. ret_val = get_mcu_main_clk();
  190. break;
  191. case AHB_CLK:
  192. ret_val = get_mcu_main_clk();
  193. break;
  194. case HSP_CLK:
  195. if (reg & CLKMODE_CONSUMER) {
  196. hsp_podf = (reg >> 20) & 0x3;
  197. pll = get_mcu_main_clk();
  198. hsp_podf = hsp_div_table[hsp_podf][(reg>>16)&0xF];
  199. if (hsp_podf > 0) {
  200. ret_val = pll / hsp_podf;
  201. } else {
  202. puts("mismatch HSP with ARM clock setting\n");
  203. ret_val = 0;
  204. }
  205. } else {
  206. ret_val = get_mcu_main_clk();
  207. }
  208. break;
  209. case IPG_CLK:
  210. ret_val = get_ipg_clk();
  211. break;
  212. case IPG_PER_CLK:
  213. ret_val = get_ipg_per_clk();
  214. break;
  215. case NFC_CLK:
  216. nfc_pdf = (reg4 >> 28) & 0xF;
  217. pll = get_mcu_main_clk();
  218. /* AHB/nfc_pdf */
  219. ret_val = pll / (nfc_pdf + 1);
  220. break;
  221. case USB_CLK:
  222. usb_podf = (reg4 >> 22) & 0x3F;
  223. if (reg4 & 0x200)
  224. pll = get_mcu_main_clk();
  225. else
  226. pll = decode_pll(readl(&ccm->ppctl), MXC_HCLK);
  227. ret_val = pll / (usb_podf + 1);
  228. break;
  229. default:
  230. printf("Unknown clock: %d\n", clk);
  231. break;
  232. }
  233. return ret_val;
  234. }
  235. unsigned int mxc_get_peri_clock(enum mxc_peri_clock clk)
  236. {
  237. u32 ret_val = 0, pdf, pre_pdf, clk_sel;
  238. struct ccm_regs *ccm =
  239. (struct ccm_regs *)IMX_CCM_BASE;
  240. u32 mpdr2 = readl(&ccm->pdr2);
  241. u32 mpdr3 = readl(&ccm->pdr3);
  242. u32 mpdr4 = readl(&ccm->pdr4);
  243. switch (clk) {
  244. case UART1_BAUD:
  245. case UART2_BAUD:
  246. case UART3_BAUD:
  247. clk_sel = mpdr3 & (1 << 14);
  248. pdf = (mpdr4 >> 10) & 0x3F;
  249. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  250. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) / (pdf + 1);
  251. break;
  252. case SSI1_BAUD:
  253. pre_pdf = (mpdr2 >> 24) & 0x7;
  254. pdf = mpdr2 & 0x3F;
  255. clk_sel = mpdr2 & (1 << 6);
  256. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  257. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) /
  258. ((pre_pdf + 1) * (pdf + 1));
  259. break;
  260. case SSI2_BAUD:
  261. pre_pdf = (mpdr2 >> 27) & 0x7;
  262. pdf = (mpdr2 >> 8) & 0x3F;
  263. clk_sel = mpdr2 & (1 << 6);
  264. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  265. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) /
  266. ((pre_pdf + 1) * (pdf + 1));
  267. break;
  268. case CSI_BAUD:
  269. clk_sel = mpdr2 & (1 << 7);
  270. pdf = (mpdr2 >> 16) & 0x3F;
  271. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  272. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) / (pdf + 1);
  273. break;
  274. case MSHC_CLK:
  275. pre_pdf = readl(&ccm->pdr1);
  276. clk_sel = (pre_pdf & 0x80);
  277. pdf = (pre_pdf >> 22) & 0x3F;
  278. pre_pdf = (pre_pdf >> 28) & 0x7;
  279. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  280. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) /
  281. ((pre_pdf + 1) * (pdf + 1));
  282. break;
  283. case ESDHC1_CLK:
  284. clk_sel = mpdr3 & 0x40;
  285. pdf = mpdr3 & 0x3F;
  286. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  287. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) / (pdf + 1);
  288. break;
  289. case ESDHC2_CLK:
  290. clk_sel = mpdr3 & 0x40;
  291. pdf = (mpdr3 >> 8) & 0x3F;
  292. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  293. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) / (pdf + 1);
  294. break;
  295. case ESDHC3_CLK:
  296. clk_sel = mpdr3 & 0x40;
  297. pdf = (mpdr3 >> 16) & 0x3F;
  298. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  299. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) / (pdf + 1);
  300. break;
  301. case SPDIF_CLK:
  302. clk_sel = mpdr3 & 0x400000;
  303. pre_pdf = (mpdr3 >> 29) & 0x7;
  304. pdf = (mpdr3 >> 23) & 0x3F;
  305. ret_val = ((clk_sel != 0) ? mxc_get_main_clock(CPU_CLK) :
  306. decode_pll(readl(&ccm->ppctl), MXC_HCLK)) /
  307. ((pre_pdf + 1) * (pdf + 1));
  308. break;
  309. default:
  310. printf("%s(): This clock: %d not supported yet\n",
  311. __func__, clk);
  312. break;
  313. }
  314. return ret_val;
  315. }
  316. unsigned int mxc_get_clock(enum mxc_clock clk)
  317. {
  318. switch (clk) {
  319. case MXC_ARM_CLK:
  320. return get_mcu_main_clk();
  321. case MXC_AHB_CLK:
  322. break;
  323. case MXC_IPG_CLK:
  324. return get_ipg_clk();
  325. case MXC_IPG_PERCLK:
  326. case MXC_I2C_CLK:
  327. return get_ipg_per_clk();
  328. case MXC_UART_CLK:
  329. return imx_get_uartclk();
  330. case MXC_ESDHC1_CLK:
  331. return mxc_get_peri_clock(ESDHC1_CLK);
  332. case MXC_ESDHC2_CLK:
  333. return mxc_get_peri_clock(ESDHC2_CLK);
  334. case MXC_ESDHC3_CLK:
  335. return mxc_get_peri_clock(ESDHC3_CLK);
  336. case MXC_USB_CLK:
  337. return mxc_get_main_clock(USB_CLK);
  338. case MXC_FEC_CLK:
  339. return get_ipg_clk();
  340. case MXC_CSPI_CLK:
  341. return get_ipg_clk();
  342. }
  343. return -1;
  344. }
  345. #ifdef CONFIG_FEC_MXC
  346. /*
  347. * The MX35 has no fuse for MAC, return a NULL MAC
  348. */
  349. void imx_get_mac_from_fuse(int dev_id, unsigned char *mac)
  350. {
  351. memset(mac, 0, 6);
  352. }
  353. u32 imx_get_fecclk(void)
  354. {
  355. return mxc_get_clock(MXC_IPG_CLK);
  356. }
  357. #endif
  358. int do_mx35_showclocks(cmd_tbl_t *cmdtp,
  359. int flag, int argc, char * const argv[])
  360. {
  361. u32 cpufreq = get_mcu_main_clk();
  362. printf("mx35 cpu clock: %dMHz\n", cpufreq / 1000000);
  363. printf("ipg clock : %dHz\n", get_ipg_clk());
  364. printf("ipg per clock : %dHz\n", get_ipg_per_clk());
  365. printf("uart clock : %dHz\n", mxc_get_clock(MXC_UART_CLK));
  366. return 0;
  367. }
  368. U_BOOT_CMD(
  369. clocks, CONFIG_SYS_MAXARGS, 1, do_mx35_showclocks,
  370. "display clocks",
  371. ""
  372. );
  373. #if defined(CONFIG_DISPLAY_CPUINFO)
  374. static char *get_reset_cause(void)
  375. {
  376. /* read RCSR register from CCM module */
  377. struct ccm_regs *ccm =
  378. (struct ccm_regs *)IMX_CCM_BASE;
  379. u32 cause = readl(&ccm->rcsr) & 0x0F;
  380. switch (cause) {
  381. case 0x0000:
  382. return "POR";
  383. case 0x0002:
  384. return "JTAG";
  385. case 0x0004:
  386. return "RST";
  387. case 0x0008:
  388. return "WDOG";
  389. default:
  390. return "unknown reset";
  391. }
  392. }
  393. int print_cpuinfo(void)
  394. {
  395. u32 srev = get_cpu_rev();
  396. printf("CPU: Freescale i.MX35 rev %d.%d at %d MHz.\n",
  397. (srev & 0xF0) >> 4, (srev & 0x0F),
  398. get_mcu_main_clk() / 1000000);
  399. printf("Reset cause: %s\n", get_reset_cause());
  400. return 0;
  401. }
  402. #endif
  403. /*
  404. * Initializes on-chip ethernet controllers.
  405. * to override, implement board_eth_init()
  406. */
  407. int cpu_eth_init(bd_t *bis)
  408. {
  409. int rc = -ENODEV;
  410. #if defined(CONFIG_FEC_MXC)
  411. rc = fecmxc_initialize(bis);
  412. #endif
  413. return rc;
  414. }
  415. #ifdef CONFIG_FSL_ESDHC
  416. /*
  417. * Initializes on-chip MMC controllers.
  418. * to override, implement board_mmc_init()
  419. */
  420. int cpu_mmc_init(bd_t *bis)
  421. {
  422. return fsl_esdhc_mmc_init(bis);
  423. }
  424. #endif
  425. int get_clocks(void)
  426. {
  427. #ifdef CONFIG_FSL_ESDHC
  428. #if CONFIG_SYS_FSL_ESDHC_ADDR == MMC_SDHC2_BASE_ADDR
  429. gd->sdhc_clk = mxc_get_clock(MXC_ESDHC2_CLK);
  430. #elif CONFIG_SYS_FSL_ESDHC_ADDR == MMC_SDHC3_BASE_ADDR
  431. gd->sdhc_clk = mxc_get_clock(MXC_ESDHC3_CLK);
  432. #else
  433. gd->sdhc_clk = mxc_get_clock(MXC_ESDHC1_CLK);
  434. #endif
  435. #endif
  436. return 0;
  437. }
  438. void reset_cpu(ulong addr)
  439. {
  440. struct wdog_regs *wdog = (struct wdog_regs *)WDOG_BASE_ADDR;
  441. writew(4, &wdog->wcr);
  442. }