speed.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. /*
  2. *
  3. * Copyright (C) 2004-2007, 2012 Freescale Semiconductor, Inc.
  4. * TsiChung Liew (Tsi-Chung.Liew@freescale.com)
  5. *
  6. * SPDX-License-Identifier: GPL-2.0+
  7. */
  8. #include <common.h>
  9. #include <asm/processor.h>
  10. #include <asm/immap.h>
  11. #include <asm/io.h>
  12. DECLARE_GLOBAL_DATA_PTR;
  13. /*
  14. * Low Power Divider specifications
  15. */
  16. #define CLOCK_LPD_MIN (1 << 0) /* Divider (decoded) */
  17. #define CLOCK_LPD_MAX (1 << 15) /* Divider (decoded) */
  18. #define CLOCK_PLL_FVCO_MAX 540000000
  19. #define CLOCK_PLL_FVCO_MIN 300000000
  20. #define CLOCK_PLL_FSYS_MAX 266666666
  21. #define CLOCK_PLL_FSYS_MIN 100000000
  22. #define MHZ 1000000
  23. void clock_enter_limp(int lpdiv)
  24. {
  25. ccm_t *ccm = (ccm_t *)MMAP_CCM;
  26. int i, j;
  27. /* Check bounds of divider */
  28. if (lpdiv < CLOCK_LPD_MIN)
  29. lpdiv = CLOCK_LPD_MIN;
  30. if (lpdiv > CLOCK_LPD_MAX)
  31. lpdiv = CLOCK_LPD_MAX;
  32. /* Round divider down to nearest power of two */
  33. for (i = 0, j = lpdiv; j != 1; j >>= 1, i++) ;
  34. #ifdef CONFIG_MCF5445x
  35. /* Apply the divider to the system clock */
  36. clrsetbits_be16(&ccm->cdr, 0x0f00, CCM_CDR_LPDIV(i));
  37. #endif
  38. /* Enable Limp Mode */
  39. setbits_be16(&ccm->misccr, CCM_MISCCR_LIMP);
  40. }
  41. /*
  42. * brief Exit Limp mode
  43. * warning The PLL should be set and locked prior to exiting Limp mode
  44. */
  45. void clock_exit_limp(void)
  46. {
  47. ccm_t *ccm = (ccm_t *)MMAP_CCM;
  48. pll_t *pll = (pll_t *)MMAP_PLL;
  49. /* Exit Limp mode */
  50. clrbits_be16(&ccm->misccr, CCM_MISCCR_LIMP);
  51. /* Wait for the PLL to lock */
  52. while (!(in_be32(&pll->psr) & PLL_PSR_LOCK))
  53. ;
  54. }
  55. #ifdef CONFIG_MCF5441x
  56. void setup_5441x_clocks(void)
  57. {
  58. ccm_t *ccm = (ccm_t *)MMAP_CCM;
  59. pll_t *pll = (pll_t *)MMAP_PLL;
  60. int temp, vco = 0, bootmod_ccr, pdr;
  61. bootmod_ccr = (in_be16(&ccm->ccr) & CCM_CCR_BOOTMOD) >> 14;
  62. switch (bootmod_ccr) {
  63. case 0:
  64. out_be32(&pll->pcr, 0x00000013);
  65. out_be32(&pll->pdr, 0x00e70c61);
  66. clock_exit_limp();
  67. break;
  68. case 2:
  69. break;
  70. case 3:
  71. break;
  72. }
  73. /*Change frequency for Modelo SER1 USB host*/
  74. #ifdef CONFIG_LOW_MCFCLK
  75. temp = in_be32(&pll->pcr);
  76. temp &= ~0x3f;
  77. temp |= 5;
  78. out_be32(&pll->pcr, temp);
  79. temp = in_be32(&pll->pdr);
  80. temp &= ~0x001f0000;
  81. temp |= 0x00040000;
  82. out_be32(&pll->pdr, temp);
  83. __asm__("tpf");
  84. #endif
  85. setbits_be16(&ccm->misccr2, 0x02);
  86. vco = ((in_be32(&pll->pcr) & PLL_CR_FBKDIV_BITS) + 1) *
  87. CONFIG_SYS_INPUT_CLKSRC;
  88. gd->arch.vco_clk = vco;
  89. gd->arch.inp_clk = CONFIG_SYS_INPUT_CLKSRC; /* Input clock */
  90. pdr = in_be32(&pll->pdr);
  91. temp = (pdr & PLL_DR_OUTDIV1_BITS) + 1;
  92. gd->cpu_clk = vco / temp; /* cpu clock */
  93. gd->arch.flb_clk = vco / temp; /* FlexBus clock */
  94. gd->arch.flb_clk >>= 1;
  95. if (in_be16(&ccm->misccr2) & 2) /* fsys/4 */
  96. gd->arch.flb_clk >>= 1;
  97. temp = ((pdr & PLL_DR_OUTDIV2_BITS) >> 5) + 1;
  98. gd->bus_clk = vco / temp; /* bus clock */
  99. }
  100. #endif
  101. #ifdef CONFIG_MCF5445x
  102. void setup_5445x_clocks(void)
  103. {
  104. ccm_t *ccm = (ccm_t *)MMAP_CCM;
  105. pll_t *pll = (pll_t *)MMAP_PLL;
  106. int pllmult_nopci[] = { 20, 10, 24, 18, 12, 6, 16, 8 };
  107. int pllmult_pci[] = { 12, 6, 16, 8 };
  108. int vco = 0, temp, fbtemp, pcrvalue;
  109. int *pPllmult = NULL;
  110. u16 fbpll_mask;
  111. #ifdef CONFIG_PCI
  112. int bPci;
  113. #endif
  114. #ifdef CONFIG_M54455EVB
  115. u8 *cpld = (u8 *)(CONFIG_SYS_CS2_BASE + 3);
  116. #endif
  117. u8 bootmode;
  118. /* To determine PCI is present or not */
  119. if (((in_be16(&ccm->ccr) & CCM_CCR_360_FBCONFIG_MASK) == 0x00e0) ||
  120. ((in_be16(&ccm->ccr) & CCM_CCR_360_FBCONFIG_MASK) == 0x0060)) {
  121. pPllmult = &pllmult_pci[0];
  122. fbpll_mask = 3; /* 11b */
  123. #ifdef CONFIG_PCI
  124. bPci = 1;
  125. #endif
  126. } else {
  127. pPllmult = &pllmult_nopci[0];
  128. fbpll_mask = 7; /* 111b */
  129. #ifdef CONFIG_PCI
  130. gd->pci_clk = 0;
  131. bPci = 0;
  132. #endif
  133. }
  134. #ifdef CONFIG_M54455EVB
  135. bootmode = (in_8(cpld) & 0x03);
  136. if (bootmode != 3) {
  137. /* Temporary read from CCR- fixed fb issue, must be the same clock
  138. as pci or input clock, causing cpld/fpga read inconsistancy */
  139. fbtemp = pPllmult[ccm->ccr & fbpll_mask];
  140. /* Break down into small pieces, code still in flex bus */
  141. pcrvalue = in_be32(&pll->pcr) & 0xFFFFF0FF;
  142. temp = fbtemp - 1;
  143. pcrvalue |= PLL_PCR_OUTDIV3(temp);
  144. out_be32(&pll->pcr, pcrvalue);
  145. }
  146. #endif
  147. #ifdef CONFIG_M54451EVB
  148. /* No external logic to read the bootmode, hard coded from built */
  149. #ifdef CONFIG_CF_SBF
  150. bootmode = 3;
  151. #else
  152. bootmode = 2;
  153. /* default value is 16 mul, set to 20 mul */
  154. pcrvalue = (in_be32(&pll->pcr) & 0x00FFFFFF) | 0x14000000;
  155. out_be32(&pll->pcr, pcrvalue);
  156. while ((in_be32(&pll->psr) & PLL_PSR_LOCK) != PLL_PSR_LOCK)
  157. ;
  158. #endif
  159. #endif
  160. if (bootmode == 0) {
  161. /* RCON mode */
  162. vco = pPllmult[ccm->rcon & fbpll_mask] * CONFIG_SYS_INPUT_CLKSRC;
  163. if ((vco < CLOCK_PLL_FVCO_MIN) || (vco > CLOCK_PLL_FVCO_MAX)) {
  164. /* invaild range, re-set in PCR */
  165. int temp = ((in_be32(&pll->pcr) & PLL_PCR_OUTDIV2_MASK) >> 4) + 1;
  166. int i, j, bus;
  167. j = (in_be32(&pll->pcr) & 0xFF000000) >> 24;
  168. for (i = j; i < 0xFF; i++) {
  169. vco = i * CONFIG_SYS_INPUT_CLKSRC;
  170. if (vco >= CLOCK_PLL_FVCO_MIN) {
  171. bus = vco / temp;
  172. if (bus <= CLOCK_PLL_FSYS_MIN - MHZ)
  173. continue;
  174. else
  175. break;
  176. }
  177. }
  178. pcrvalue = in_be32(&pll->pcr) & 0x00FF00FF;
  179. fbtemp = ((i - 1) << 8) | ((i - 1) << 12);
  180. pcrvalue |= ((i << 24) | fbtemp);
  181. out_be32(&pll->pcr, pcrvalue);
  182. }
  183. gd->arch.vco_clk = vco; /* Vco clock */
  184. } else if (bootmode == 2) {
  185. /* Normal mode */
  186. vco = ((in_be32(&pll->pcr) & 0xFF000000) >> 24) * CONFIG_SYS_INPUT_CLKSRC;
  187. if ((vco < CLOCK_PLL_FVCO_MIN) || (vco > CLOCK_PLL_FVCO_MAX)) {
  188. /* Default value */
  189. pcrvalue = (in_be32(&pll->pcr) & 0x00FFFFFF);
  190. pcrvalue |= pPllmult[in_be16(&ccm->ccr) & fbpll_mask] << 24;
  191. out_be32(&pll->pcr, pcrvalue);
  192. vco = ((in_be32(&pll->pcr) & 0xFF000000) >> 24) * CONFIG_SYS_INPUT_CLKSRC;
  193. }
  194. gd->arch.vco_clk = vco; /* Vco clock */
  195. } else if (bootmode == 3) {
  196. /* serial mode */
  197. vco = ((in_be32(&pll->pcr) & 0xFF000000) >> 24) * CONFIG_SYS_INPUT_CLKSRC;
  198. gd->arch.vco_clk = vco; /* Vco clock */
  199. }
  200. if ((in_be16(&ccm->ccr) & CCM_MISCCR_LIMP) == CCM_MISCCR_LIMP) {
  201. /* Limp mode */
  202. } else {
  203. gd->arch.inp_clk = CONFIG_SYS_INPUT_CLKSRC; /* Input clock */
  204. temp = (in_be32(&pll->pcr) & PLL_PCR_OUTDIV1_MASK) + 1;
  205. gd->cpu_clk = vco / temp; /* cpu clock */
  206. temp = ((in_be32(&pll->pcr) & PLL_PCR_OUTDIV2_MASK) >> 4) + 1;
  207. gd->bus_clk = vco / temp; /* bus clock */
  208. temp = ((in_be32(&pll->pcr) & PLL_PCR_OUTDIV3_MASK) >> 8) + 1;
  209. gd->arch.flb_clk = vco / temp; /* FlexBus clock */
  210. #ifdef CONFIG_PCI
  211. if (bPci) {
  212. temp = ((in_be32(&pll->pcr) & PLL_PCR_OUTDIV4_MASK) >> 12) + 1;
  213. gd->pci_clk = vco / temp; /* PCI clock */
  214. }
  215. #endif
  216. }
  217. #ifdef CONFIG_SYS_I2C_FSL
  218. gd->arch.i2c1_clk = gd->bus_clk;
  219. #endif
  220. }
  221. #endif
  222. /* get_clocks() fills in gd->cpu_clock and gd->bus_clk */
  223. int get_clocks(void)
  224. {
  225. #ifdef CONFIG_MCF5441x
  226. setup_5441x_clocks();
  227. #endif
  228. #ifdef CONFIG_MCF5445x
  229. setup_5445x_clocks();
  230. #endif
  231. #ifdef CONFIG_SYS_FSL_I2C
  232. gd->arch.i2c1_clk = gd->bus_clk;
  233. #endif
  234. return (0);
  235. }