clock.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  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. * Copyright (C) 2013 Bo Shen <voice.shen@atmel.com>
  8. * Copyright (C) 2015 Wenyou Yang <wenyou.yang@atmel.com>
  9. *
  10. * SPDX-License-Identifier: GPL-2.0+
  11. */
  12. #include <common.h>
  13. #include <asm/errno.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. static unsigned long at91_css_to_rate(unsigned long css)
  23. {
  24. switch (css) {
  25. case AT91_PMC_MCKR_CSS_SLOW:
  26. return CONFIG_SYS_AT91_SLOW_CLOCK;
  27. case AT91_PMC_MCKR_CSS_MAIN:
  28. return gd->arch.main_clk_rate_hz;
  29. case AT91_PMC_MCKR_CSS_PLLA:
  30. return gd->arch.plla_rate_hz;
  31. }
  32. return 0;
  33. }
  34. static u32 at91_pll_rate(u32 freq, u32 reg)
  35. {
  36. unsigned mul, div;
  37. div = reg & 0xff;
  38. mul = (reg >> 18) & 0x7f;
  39. if (div && mul) {
  40. freq /= div;
  41. freq *= mul + 1;
  42. } else {
  43. freq = 0;
  44. }
  45. return freq;
  46. }
  47. int at91_clock_init(unsigned long main_clock)
  48. {
  49. unsigned freq, mckr;
  50. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  51. #ifndef CONFIG_SYS_AT91_MAIN_CLOCK
  52. unsigned tmp;
  53. /*
  54. * When the bootloader initialized the main oscillator correctly,
  55. * there's no problem using the cycle counter. But if it didn't,
  56. * or when using oscillator bypass mode, we must be told the speed
  57. * of the main clock.
  58. */
  59. if (!main_clock) {
  60. do {
  61. tmp = readl(&pmc->mcfr);
  62. } while (!(tmp & AT91_PMC_MCFR_MAINRDY));
  63. tmp &= AT91_PMC_MCFR_MAINF_MASK;
  64. main_clock = tmp * (CONFIG_SYS_AT91_SLOW_CLOCK / 16);
  65. }
  66. #endif
  67. gd->arch.main_clk_rate_hz = main_clock;
  68. /* report if PLLA is more than mildly overclocked */
  69. gd->arch.plla_rate_hz = at91_pll_rate(main_clock, readl(&pmc->pllar));
  70. /*
  71. * MCK and CPU derive from one of those primary clocks.
  72. * For now, assume this parentage won't change.
  73. */
  74. mckr = readl(&pmc->mckr);
  75. /* plla divisor by 2 */
  76. if (mckr & (1 << 12))
  77. gd->arch.plla_rate_hz >>= 1;
  78. gd->arch.mck_rate_hz = at91_css_to_rate(mckr & AT91_PMC_MCKR_CSS_MASK);
  79. freq = gd->arch.mck_rate_hz;
  80. /* prescale */
  81. freq >>= mckr & AT91_PMC_MCKR_PRES_MASK;
  82. switch (mckr & AT91_PMC_MCKR_MDIV_MASK) {
  83. case AT91_PMC_MCKR_MDIV_2:
  84. gd->arch.mck_rate_hz = freq / 2;
  85. break;
  86. case AT91_PMC_MCKR_MDIV_3:
  87. gd->arch.mck_rate_hz = freq / 3;
  88. break;
  89. case AT91_PMC_MCKR_MDIV_4:
  90. gd->arch.mck_rate_hz = freq / 4;
  91. break;
  92. default:
  93. break;
  94. }
  95. gd->arch.cpu_clk_rate_hz = freq;
  96. return 0;
  97. }
  98. void at91_plla_init(u32 pllar)
  99. {
  100. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  101. writel(pllar, &pmc->pllar);
  102. while (!(readl(&pmc->sr) & (AT91_PMC_LOCKA | AT91_PMC_MCKRDY)))
  103. ;
  104. }
  105. void at91_mck_init(u32 mckr)
  106. {
  107. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  108. u32 tmp;
  109. tmp = readl(&pmc->mckr);
  110. tmp &= ~(AT91_PMC_MCKR_CSS_MASK |
  111. AT91_PMC_MCKR_PRES_MASK |
  112. AT91_PMC_MCKR_MDIV_MASK |
  113. AT91_PMC_MCKR_PLLADIV_2);
  114. #ifdef CPU_HAS_H32MXDIV
  115. tmp &= ~AT91_PMC_MCKR_H32MXDIV;
  116. #endif
  117. tmp |= mckr & (AT91_PMC_MCKR_CSS_MASK |
  118. AT91_PMC_MCKR_PRES_MASK |
  119. AT91_PMC_MCKR_MDIV_MASK |
  120. AT91_PMC_MCKR_PLLADIV_2);
  121. #ifdef CPU_HAS_H32MXDIV
  122. tmp |= mckr & AT91_PMC_MCKR_H32MXDIV;
  123. #endif
  124. writel(tmp, &pmc->mckr);
  125. while (!(readl(&pmc->sr) & AT91_PMC_MCKRDY))
  126. ;
  127. }
  128. int at91_enable_periph_generated_clk(u32 id, u32 clk_source, u32 div)
  129. {
  130. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  131. u32 regval, status;
  132. u32 timeout = 1000;
  133. if (id > AT91_PMC_PCR_PID_MASK)
  134. return -EINVAL;
  135. if (div > 0xff)
  136. return -EINVAL;
  137. writel(id, &pmc->pcr);
  138. regval = readl(&pmc->pcr);
  139. regval &= ~AT91_PMC_PCR_GCKCSS;
  140. regval &= ~AT91_PMC_PCR_GCKDIV;
  141. switch (clk_source) {
  142. case GCK_CSS_SLOW_CLK:
  143. regval |= AT91_PMC_PCR_GCKCSS_SLOW_CLK;
  144. break;
  145. case GCK_CSS_MAIN_CLK:
  146. regval |= AT91_PMC_PCR_GCKCSS_MAIN_CLK;
  147. break;
  148. case GCK_CSS_PLLA_CLK:
  149. regval |= AT91_PMC_PCR_GCKCSS_PLLA_CLK;
  150. break;
  151. case GCK_CSS_UPLL_CLK:
  152. regval |= AT91_PMC_PCR_GCKCSS_UPLL_CLK;
  153. break;
  154. case GCK_CSS_MCK_CLK:
  155. regval |= AT91_PMC_PCR_GCKCSS_MCK_CLK;
  156. break;
  157. case GCK_CSS_AUDIO_CLK:
  158. regval |= AT91_PMC_PCR_GCKCSS_AUDIO_CLK;
  159. break;
  160. default:
  161. printf("Error GCK clock source selection!\n");
  162. return -EINVAL;
  163. }
  164. regval |= AT91_PMC_PCR_CMD_WRITE |
  165. AT91_PMC_PCR_GCKDIV_(div) |
  166. AT91_PMC_PCR_GCKEN;
  167. writel(regval, &pmc->pcr);
  168. do {
  169. udelay(1);
  170. status = readl(&pmc->sr);
  171. } while ((!!(--timeout)) && (!(status & AT91_PMC_GCKRDY)));
  172. if (!timeout)
  173. printf("Timeout waiting for GCK ready!\n");
  174. return 0;
  175. }
  176. u32 at91_get_periph_generated_clk(u32 id)
  177. {
  178. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  179. u32 regval, clk_source, div;
  180. u32 freq;
  181. if (id > AT91_PMC_PCR_PID_MASK)
  182. return 0;
  183. writel(id, &pmc->pcr);
  184. regval = readl(&pmc->pcr);
  185. clk_source = regval & AT91_PMC_PCR_GCKCSS;
  186. switch (clk_source) {
  187. case AT91_PMC_PCR_GCKCSS_SLOW_CLK:
  188. freq = CONFIG_SYS_AT91_SLOW_CLOCK;
  189. break;
  190. case AT91_PMC_PCR_GCKCSS_MAIN_CLK:
  191. freq = gd->arch.main_clk_rate_hz;
  192. break;
  193. case AT91_PMC_PCR_GCKCSS_PLLA_CLK:
  194. freq = gd->arch.plla_rate_hz;
  195. break;
  196. default:
  197. printf("Improper GCK clock source selection!\n");
  198. freq = 0;
  199. break;
  200. }
  201. div = ((regval & AT91_PMC_PCR_GCKDIV) >> AT91_PMC_PCR_GCKDIV_OFFSET);
  202. div += 1;
  203. return freq / div;
  204. }