generic.c 11 KB

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