clock.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. /*
  2. * [origin: Linux kernel linux/arch/arm/mach-at91/clock.c]
  3. *
  4. * Copyright (C) 2005 David Brownell
  5. * Copyright (C) 2005 Ivan Kokshaysky
  6. * Copyright (C) 2009 Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com>
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation; either version 2 of the License, or
  11. * (at your option) any later version.
  12. */
  13. #include <common.h>
  14. #include <asm/io.h>
  15. #include <asm/arch/hardware.h>
  16. #include <asm/arch/at91_pmc.h>
  17. #include <asm/arch/clk.h>
  18. #if !defined(CONFIG_AT91FAMILY)
  19. # error You need to define CONFIG_AT91FAMILY in your board config!
  20. #endif
  21. DECLARE_GLOBAL_DATA_PTR;
  22. unsigned long get_cpu_clk_rate(void)
  23. {
  24. return gd->cpu_clk_rate_hz;
  25. }
  26. unsigned long get_main_clk_rate(void)
  27. {
  28. return gd->main_clk_rate_hz;
  29. }
  30. unsigned long get_mck_clk_rate(void)
  31. {
  32. return gd->mck_rate_hz;
  33. }
  34. unsigned long get_plla_clk_rate(void)
  35. {
  36. return gd->plla_rate_hz;
  37. }
  38. unsigned long get_pllb_clk_rate(void)
  39. {
  40. return gd->pllb_rate_hz;
  41. }
  42. u32 get_pllb_init(void)
  43. {
  44. return gd->at91_pllb_usb_init;
  45. }
  46. static unsigned long at91_css_to_rate(unsigned long css)
  47. {
  48. switch (css) {
  49. case AT91_PMC_MCKR_CSS_SLOW:
  50. return CONFIG_SYS_AT91_SLOW_CLOCK;
  51. case AT91_PMC_MCKR_CSS_MAIN:
  52. return gd->main_clk_rate_hz;
  53. case AT91_PMC_MCKR_CSS_PLLA:
  54. return gd->plla_rate_hz;
  55. case AT91_PMC_MCKR_CSS_PLLB:
  56. return gd->pllb_rate_hz;
  57. }
  58. return 0;
  59. }
  60. #ifdef CONFIG_USB_ATMEL
  61. static unsigned at91_pll_calc(unsigned main_freq, unsigned out_freq)
  62. {
  63. unsigned i, div = 0, mul = 0, diff = 1 << 30;
  64. unsigned ret = (out_freq > 155000000) ? 0xbe00 : 0x3e00;
  65. /* PLL output max 240 MHz (or 180 MHz per errata) */
  66. if (out_freq > 240000000)
  67. goto fail;
  68. for (i = 1; i < 256; i++) {
  69. int diff1;
  70. unsigned input, mul1;
  71. /*
  72. * PLL input between 1MHz and 32MHz per spec, but lower
  73. * frequences seem necessary in some cases so allow 100K.
  74. * Warning: some newer products need 2MHz min.
  75. */
  76. input = main_freq / i;
  77. #if defined(CONFIG_AT91SAM9G20)
  78. if (input < 2000000)
  79. continue;
  80. #endif
  81. if (input < 100000)
  82. continue;
  83. if (input > 32000000)
  84. continue;
  85. mul1 = out_freq / input;
  86. #if defined(CONFIG_AT91SAM9G20)
  87. if (mul > 63)
  88. continue;
  89. #endif
  90. if (mul1 > 2048)
  91. continue;
  92. if (mul1 < 2)
  93. goto fail;
  94. diff1 = out_freq - input * mul1;
  95. if (diff1 < 0)
  96. diff1 = -diff1;
  97. if (diff > diff1) {
  98. diff = diff1;
  99. div = i;
  100. mul = mul1;
  101. if (diff == 0)
  102. break;
  103. }
  104. }
  105. if (i == 256 && diff > (out_freq >> 5))
  106. goto fail;
  107. return ret | ((mul - 1) << 16) | div;
  108. fail:
  109. return 0;
  110. }
  111. #endif
  112. static u32 at91_pll_rate(u32 freq, u32 reg)
  113. {
  114. unsigned mul, div;
  115. div = reg & 0xff;
  116. mul = (reg >> 16) & 0x7ff;
  117. if (div && mul) {
  118. freq /= div;
  119. freq *= mul + 1;
  120. } else
  121. freq = 0;
  122. return freq;
  123. }
  124. int at91_clock_init(unsigned long main_clock)
  125. {
  126. unsigned freq, mckr;
  127. at91_pmc_t *pmc = (at91_pmc_t *) ATMEL_BASE_PMC;
  128. #ifndef CONFIG_SYS_AT91_MAIN_CLOCK
  129. unsigned tmp;
  130. /*
  131. * When the bootloader initialized the main oscillator correctly,
  132. * there's no problem using the cycle counter. But if it didn't,
  133. * or when using oscillator bypass mode, we must be told the speed
  134. * of the main clock.
  135. */
  136. if (!main_clock) {
  137. do {
  138. tmp = readl(&pmc->mcfr);
  139. } while (!(tmp & AT91_PMC_MCFR_MAINRDY));
  140. tmp &= AT91_PMC_MCFR_MAINF_MASK;
  141. main_clock = tmp * (CONFIG_SYS_AT91_SLOW_CLOCK / 16);
  142. }
  143. #endif
  144. gd->main_clk_rate_hz = main_clock;
  145. /* report if PLLA is more than mildly overclocked */
  146. gd->plla_rate_hz = at91_pll_rate(main_clock, readl(&pmc->pllar));
  147. #ifdef CONFIG_USB_ATMEL
  148. /*
  149. * USB clock init: choose 48 MHz PLLB value,
  150. * disable 48MHz clock during usb peripheral suspend.
  151. *
  152. * REVISIT: assumes MCK doesn't derive from PLLB!
  153. */
  154. gd->at91_pllb_usb_init = at91_pll_calc(main_clock, 48000000 * 2) |
  155. AT91_PMC_PLLBR_USBDIV_2;
  156. gd->pllb_rate_hz = at91_pll_rate(main_clock, gd->at91_pllb_usb_init);
  157. #endif
  158. /*
  159. * MCK and CPU derive from one of those primary clocks.
  160. * For now, assume this parentage won't change.
  161. */
  162. mckr = readl(&pmc->mckr);
  163. #if defined(CONFIG_AT91SAM9G45) || defined(CONFIG_AT91SAM9M10G45)
  164. /* plla divisor by 2 */
  165. gd->plla_rate_hz /= (1 << ((mckr & 1 << 12) >> 12));
  166. #endif
  167. gd->mck_rate_hz = at91_css_to_rate(mckr & AT91_PMC_MCKR_CSS_MASK);
  168. freq = gd->mck_rate_hz;
  169. freq /= (1 << ((mckr & AT91_PMC_MCKR_PRES_MASK) >> 2)); /* prescale */
  170. #if defined(CONFIG_AT91RM9200)
  171. /* mdiv */
  172. gd->mck_rate_hz = freq / (1 + ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
  173. #elif defined(CONFIG_AT91SAM9G20)
  174. /* mdiv ; (x >> 7) = ((x >> 8) * 2) */
  175. gd->mck_rate_hz = (mckr & AT91_PMC_MCKR_MDIV_MASK) ?
  176. freq / ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 7) : freq;
  177. if (mckr & AT91_PMC_MCKR_MDIV_MASK)
  178. freq /= 2; /* processor clock division */
  179. #elif defined(CONFIG_AT91SAM9G45) || defined(CONFIG_AT91SAM9M10G45)
  180. gd->mck_rate_hz = (mckr & AT91_PMC_MCKR_MDIV_MASK) ==
  181. (AT91_PMC_MCKR_MDIV_2 | AT91_PMC_MCKR_MDIV_4)
  182. ? freq / 3
  183. : freq / (1 << ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
  184. #else
  185. gd->mck_rate_hz = freq / (1 << ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
  186. #endif
  187. gd->cpu_clk_rate_hz = freq;
  188. return 0;
  189. }