clock.c 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * [origin: Linux kernel linux/arch/arm/mach-at91/clock.c]
  4. *
  5. * Copyright (C) 2005 David Brownell
  6. * Copyright (C) 2005 Ivan Kokshaysky
  7. * Copyright (C) 2009 Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com>
  8. * Copyright (C) 2013 Bo Shen <voice.shen@atmel.com>
  9. * Copyright (C) 2015 Wenyou Yang <wenyou.yang@atmel.com>
  10. */
  11. #include <common.h>
  12. #include <linux/errno.h>
  13. #include <asm/io.h>
  14. #include <asm/arch/hardware.h>
  15. #include <asm/arch/at91_pmc.h>
  16. #include <asm/arch/clk.h>
  17. #if !defined(CONFIG_AT91FAMILY)
  18. # error You need to define CONFIG_AT91FAMILY in your board config!
  19. #endif
  20. DECLARE_GLOBAL_DATA_PTR;
  21. static unsigned long at91_css_to_rate(unsigned long css)
  22. {
  23. switch (css) {
  24. case AT91_PMC_MCKR_CSS_SLOW:
  25. return CONFIG_SYS_AT91_SLOW_CLOCK;
  26. case AT91_PMC_MCKR_CSS_MAIN:
  27. return gd->arch.main_clk_rate_hz;
  28. case AT91_PMC_MCKR_CSS_PLLA:
  29. return gd->arch.plla_rate_hz;
  30. }
  31. return 0;
  32. }
  33. static u32 at91_pll_rate(u32 freq, u32 reg)
  34. {
  35. unsigned mul, div;
  36. div = reg & 0xff;
  37. mul = (reg >> 18) & 0x7f;
  38. if (div && mul) {
  39. freq /= div;
  40. freq *= mul + 1;
  41. } else {
  42. freq = 0;
  43. }
  44. return freq;
  45. }
  46. int at91_clock_init(unsigned long main_clock)
  47. {
  48. unsigned freq, mckr;
  49. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  50. #ifndef CONFIG_SYS_AT91_MAIN_CLOCK
  51. unsigned tmp;
  52. /*
  53. * When the bootloader initialized the main oscillator correctly,
  54. * there's no problem using the cycle counter. But if it didn't,
  55. * or when using oscillator bypass mode, we must be told the speed
  56. * of the main clock.
  57. */
  58. if (!main_clock) {
  59. do {
  60. tmp = readl(&pmc->mcfr);
  61. } while (!(tmp & AT91_PMC_MCFR_MAINRDY));
  62. tmp &= AT91_PMC_MCFR_MAINF_MASK;
  63. main_clock = tmp * (CONFIG_SYS_AT91_SLOW_CLOCK / 16);
  64. }
  65. #endif
  66. gd->arch.main_clk_rate_hz = main_clock;
  67. /* report if PLLA is more than mildly overclocked */
  68. gd->arch.plla_rate_hz = at91_pll_rate(main_clock, readl(&pmc->pllar));
  69. /*
  70. * MCK and CPU derive from one of those primary clocks.
  71. * For now, assume this parentage won't change.
  72. */
  73. mckr = readl(&pmc->mckr);
  74. /* plla divisor by 2 */
  75. if (mckr & (1 << 12))
  76. gd->arch.plla_rate_hz >>= 1;
  77. gd->arch.mck_rate_hz = at91_css_to_rate(mckr & AT91_PMC_MCKR_CSS_MASK);
  78. freq = gd->arch.mck_rate_hz;
  79. /* prescale */
  80. freq >>= mckr & AT91_PMC_MCKR_PRES_MASK;
  81. switch (mckr & AT91_PMC_MCKR_MDIV_MASK) {
  82. case AT91_PMC_MCKR_MDIV_2:
  83. gd->arch.mck_rate_hz = freq / 2;
  84. break;
  85. case AT91_PMC_MCKR_MDIV_3:
  86. gd->arch.mck_rate_hz = freq / 3;
  87. break;
  88. case AT91_PMC_MCKR_MDIV_4:
  89. gd->arch.mck_rate_hz = freq / 4;
  90. break;
  91. default:
  92. break;
  93. }
  94. gd->arch.cpu_clk_rate_hz = freq;
  95. return 0;
  96. }
  97. void at91_plla_init(u32 pllar)
  98. {
  99. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  100. writel(pllar, &pmc->pllar);
  101. while (!(readl(&pmc->sr) & (AT91_PMC_LOCKA | AT91_PMC_MCKRDY)))
  102. ;
  103. }
  104. void at91_mck_init(u32 mckr)
  105. {
  106. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  107. u32 tmp;
  108. tmp = readl(&pmc->mckr);
  109. tmp &= ~(AT91_PMC_MCKR_CSS_MASK |
  110. AT91_PMC_MCKR_PRES_MASK |
  111. AT91_PMC_MCKR_MDIV_MASK |
  112. AT91_PMC_MCKR_PLLADIV_2);
  113. #ifdef CPU_HAS_H32MXDIV
  114. tmp &= ~AT91_PMC_MCKR_H32MXDIV;
  115. #endif
  116. tmp |= mckr & (AT91_PMC_MCKR_CSS_MASK |
  117. AT91_PMC_MCKR_PRES_MASK |
  118. AT91_PMC_MCKR_MDIV_MASK |
  119. AT91_PMC_MCKR_PLLADIV_2);
  120. #ifdef CPU_HAS_H32MXDIV
  121. tmp |= mckr & AT91_PMC_MCKR_H32MXDIV;
  122. #endif
  123. writel(tmp, &pmc->mckr);
  124. while (!(readl(&pmc->sr) & AT91_PMC_MCKRDY))
  125. ;
  126. }
  127. /*
  128. * For the Master Clock Controller Register(MCKR), while switching
  129. * to a lower clock source, we must switch the clock source first
  130. * instead of last. Otherwise, we could end up with too high frequency
  131. * on the internal bus and peripherals.
  132. */
  133. void at91_mck_init_down(u32 mckr)
  134. {
  135. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  136. u32 tmp;
  137. tmp = readl(&pmc->mckr);
  138. tmp &= (~AT91_PMC_MCKR_CSS_MASK);
  139. tmp |= (mckr & AT91_PMC_MCKR_CSS_MASK);
  140. writel(tmp, &pmc->mckr);
  141. while (!(readl(&pmc->sr) & AT91_PMC_MCKRDY))
  142. ;
  143. #ifdef CPU_HAS_H32MXDIV
  144. tmp = readl(&pmc->mckr);
  145. tmp &= (~AT91_PMC_MCKR_H32MXDIV);
  146. tmp |= (mckr & AT91_PMC_MCKR_H32MXDIV);
  147. writel(tmp, &pmc->mckr);
  148. #endif
  149. tmp = readl(&pmc->mckr);
  150. tmp &= (~AT91_PMC_MCKR_PLLADIV_MASK);
  151. tmp |= (mckr & AT91_PMC_MCKR_PLLADIV_MASK);
  152. writel(tmp, &pmc->mckr);
  153. tmp = readl(&pmc->mckr);
  154. tmp &= (~AT91_PMC_MCKR_MDIV_MASK);
  155. tmp |= (mckr & AT91_PMC_MCKR_MDIV_MASK);
  156. writel(tmp, &pmc->mckr);
  157. tmp = readl(&pmc->mckr);
  158. tmp &= (~AT91_PMC_MCKR_PRES_MASK);
  159. tmp |= (mckr & AT91_PMC_MCKR_PRES_MASK);
  160. writel(tmp, &pmc->mckr);
  161. }
  162. int at91_enable_periph_generated_clk(u32 id, u32 clk_source, u32 div)
  163. {
  164. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  165. u32 regval, status;
  166. u32 timeout = 1000;
  167. if (id > AT91_PMC_PCR_PID_MASK)
  168. return -EINVAL;
  169. if (div > 0xff)
  170. return -EINVAL;
  171. if (clk_source == GCK_CSS_UPLL_CLK) {
  172. if (at91_upll_clk_enable())
  173. return -ENODEV;
  174. }
  175. writel(id, &pmc->pcr);
  176. regval = readl(&pmc->pcr);
  177. regval &= ~AT91_PMC_PCR_GCKCSS;
  178. regval &= ~AT91_PMC_PCR_GCKDIV;
  179. switch (clk_source) {
  180. case GCK_CSS_SLOW_CLK:
  181. regval |= AT91_PMC_PCR_GCKCSS_SLOW_CLK;
  182. break;
  183. case GCK_CSS_MAIN_CLK:
  184. regval |= AT91_PMC_PCR_GCKCSS_MAIN_CLK;
  185. break;
  186. case GCK_CSS_PLLA_CLK:
  187. regval |= AT91_PMC_PCR_GCKCSS_PLLA_CLK;
  188. break;
  189. case GCK_CSS_UPLL_CLK:
  190. regval |= AT91_PMC_PCR_GCKCSS_UPLL_CLK;
  191. break;
  192. case GCK_CSS_MCK_CLK:
  193. regval |= AT91_PMC_PCR_GCKCSS_MCK_CLK;
  194. break;
  195. case GCK_CSS_AUDIO_CLK:
  196. regval |= AT91_PMC_PCR_GCKCSS_AUDIO_CLK;
  197. break;
  198. default:
  199. printf("Error GCK clock source selection!\n");
  200. return -EINVAL;
  201. }
  202. regval |= AT91_PMC_PCR_CMD_WRITE |
  203. AT91_PMC_PCR_GCKDIV_(div) |
  204. AT91_PMC_PCR_GCKEN;
  205. writel(regval, &pmc->pcr);
  206. do {
  207. udelay(1);
  208. status = readl(&pmc->sr);
  209. } while ((!!(--timeout)) && (!(status & AT91_PMC_GCKRDY)));
  210. if (!timeout)
  211. printf("Timeout waiting for GCK ready!\n");
  212. return 0;
  213. }
  214. u32 at91_get_periph_generated_clk(u32 id)
  215. {
  216. struct at91_pmc *pmc = (struct at91_pmc *)ATMEL_BASE_PMC;
  217. u32 regval, clk_source, div;
  218. u32 freq;
  219. if (id > AT91_PMC_PCR_PID_MASK)
  220. return 0;
  221. writel(id, &pmc->pcr);
  222. regval = readl(&pmc->pcr);
  223. clk_source = regval & AT91_PMC_PCR_GCKCSS;
  224. switch (clk_source) {
  225. case AT91_PMC_PCR_GCKCSS_SLOW_CLK:
  226. freq = CONFIG_SYS_AT91_SLOW_CLOCK;
  227. break;
  228. case AT91_PMC_PCR_GCKCSS_MAIN_CLK:
  229. freq = gd->arch.main_clk_rate_hz;
  230. break;
  231. case AT91_PMC_PCR_GCKCSS_PLLA_CLK:
  232. freq = gd->arch.plla_rate_hz;
  233. break;
  234. case AT91_PMC_PCR_GCKCSS_UPLL_CLK:
  235. freq = AT91_UTMI_PLL_CLK_FREQ;
  236. break;
  237. case AT91_PMC_PCR_GCKCSS_MCK_CLK:
  238. freq = gd->arch.mck_rate_hz;
  239. break;
  240. default:
  241. printf("Improper GCK clock source selection!\n");
  242. freq = 0;
  243. break;
  244. }
  245. div = ((regval & AT91_PMC_PCR_GCKDIV) >> AT91_PMC_PCR_GCKDIV_OFFSET);
  246. div += 1;
  247. return freq / div;
  248. }