clk_rk3036.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360
  1. /*
  2. * (C) Copyright 2015 Google, Inc
  3. *
  4. * SPDX-License-Identifier: GPL-2.0
  5. */
  6. #include <common.h>
  7. #include <clk-uclass.h>
  8. #include <dm.h>
  9. #include <errno.h>
  10. #include <syscon.h>
  11. #include <asm/io.h>
  12. #include <asm/arch/clock.h>
  13. #include <asm/arch/cru_rk3036.h>
  14. #include <asm/arch/hardware.h>
  15. #include <dm/lists.h>
  16. #include <dt-bindings/clock/rk3036-cru.h>
  17. #include <linux/log2.h>
  18. DECLARE_GLOBAL_DATA_PTR;
  19. enum {
  20. VCO_MAX_HZ = 2400U * 1000000,
  21. VCO_MIN_HZ = 600 * 1000000,
  22. OUTPUT_MAX_HZ = 2400U * 1000000,
  23. OUTPUT_MIN_HZ = 24 * 1000000,
  24. };
  25. #define RATE_TO_DIV(input_rate, output_rate) \
  26. ((input_rate) / (output_rate) - 1);
  27. #define DIV_TO_RATE(input_rate, div) ((input_rate) / ((div) + 1))
  28. #define PLL_DIVISORS(hz, _refdiv, _postdiv1, _postdiv2) {\
  29. .refdiv = _refdiv,\
  30. .fbdiv = (u32)((u64)hz * _refdiv * _postdiv1 * _postdiv2 / OSC_HZ),\
  31. .postdiv1 = _postdiv1, .postdiv2 = _postdiv2};\
  32. _Static_assert(((u64)hz * _refdiv * _postdiv1 * _postdiv2 / OSC_HZ) *\
  33. OSC_HZ / (_refdiv * _postdiv1 * _postdiv2) == hz,\
  34. #hz "Hz cannot be hit with PLL "\
  35. "divisors on line " __stringify(__LINE__));
  36. /* use interge mode*/
  37. static const struct pll_div apll_init_cfg = PLL_DIVISORS(APLL_HZ, 1, 3, 1);
  38. static const struct pll_div gpll_init_cfg = PLL_DIVISORS(GPLL_HZ, 2, 2, 1);
  39. static int rkclk_set_pll(struct rk3036_cru *cru, enum rk_clk_id clk_id,
  40. const struct pll_div *div)
  41. {
  42. int pll_id = rk_pll_id(clk_id);
  43. struct rk3036_pll *pll = &cru->pll[pll_id];
  44. /* All PLLs have same VCO and output frequency range restrictions. */
  45. uint vco_hz = OSC_HZ / 1000 * div->fbdiv / div->refdiv * 1000;
  46. uint output_hz = vco_hz / div->postdiv1 / div->postdiv2;
  47. debug("PLL at %p: fbdiv=%d, refdiv=%d, postdiv1=%d, postdiv2=%d,\
  48. vco=%u Hz, output=%u Hz\n",
  49. pll, div->fbdiv, div->refdiv, div->postdiv1,
  50. div->postdiv2, vco_hz, output_hz);
  51. assert(vco_hz >= VCO_MIN_HZ && vco_hz <= VCO_MAX_HZ &&
  52. output_hz >= OUTPUT_MIN_HZ && output_hz <= OUTPUT_MAX_HZ);
  53. /* use interger mode */
  54. rk_clrreg(&pll->con1, 1 << PLL_DSMPD_SHIFT);
  55. rk_clrsetreg(&pll->con0,
  56. PLL_POSTDIV1_MASK << PLL_POSTDIV1_SHIFT | PLL_FBDIV_MASK,
  57. (div->postdiv1 << PLL_POSTDIV1_SHIFT) | div->fbdiv);
  58. rk_clrsetreg(&pll->con1, PLL_POSTDIV2_MASK << PLL_POSTDIV2_SHIFT |
  59. PLL_REFDIV_MASK << PLL_REFDIV_SHIFT,
  60. (div->postdiv2 << PLL_POSTDIV2_SHIFT |
  61. div->refdiv << PLL_REFDIV_SHIFT));
  62. /* waiting for pll lock */
  63. while (readl(&pll->con1) & (1 << PLL_LOCK_STATUS_SHIFT))
  64. udelay(1);
  65. return 0;
  66. }
  67. static void rkclk_init(struct rk3036_cru *cru)
  68. {
  69. u32 aclk_div;
  70. u32 hclk_div;
  71. u32 pclk_div;
  72. /* pll enter slow-mode */
  73. rk_clrsetreg(&cru->cru_mode_con,
  74. GPLL_MODE_MASK << GPLL_MODE_SHIFT |
  75. APLL_MODE_MASK << APLL_MODE_SHIFT,
  76. GPLL_MODE_SLOW << GPLL_MODE_SHIFT |
  77. APLL_MODE_SLOW << APLL_MODE_SHIFT);
  78. /* init pll */
  79. rkclk_set_pll(cru, CLK_ARM, &apll_init_cfg);
  80. rkclk_set_pll(cru, CLK_GENERAL, &gpll_init_cfg);
  81. /*
  82. * select apll as core clock pll source and
  83. * set up dependent divisors for PCLK/HCLK and ACLK clocks.
  84. * core hz : apll = 1:1
  85. */
  86. aclk_div = APLL_HZ / CORE_ACLK_HZ - 1;
  87. assert((aclk_div + 1) * CORE_ACLK_HZ == APLL_HZ && aclk_div < 0x7);
  88. pclk_div = APLL_HZ / CORE_PERI_HZ - 1;
  89. assert((pclk_div + 1) * CORE_PERI_HZ == APLL_HZ && pclk_div < 0xf);
  90. rk_clrsetreg(&cru->cru_clksel_con[0],
  91. CORE_CLK_PLL_SEL_MASK << CORE_CLK_PLL_SEL_SHIFT |
  92. CORE_DIV_CON_MASK << CORE_DIV_CON_SHIFT,
  93. CORE_CLK_PLL_SEL_APLL << CORE_CLK_PLL_SEL_SHIFT |
  94. 0 << CORE_DIV_CON_SHIFT);
  95. rk_clrsetreg(&cru->cru_clksel_con[1],
  96. CORE_ACLK_DIV_MASK << CORE_ACLK_DIV_SHIFT |
  97. CORE_PERI_DIV_MASK << CORE_PERI_DIV_SHIFT,
  98. aclk_div << CORE_ACLK_DIV_SHIFT |
  99. pclk_div << CORE_PERI_DIV_SHIFT);
  100. /*
  101. * select apll as cpu clock pll source and
  102. * set up dependent divisors for PCLK/HCLK and ACLK clocks.
  103. */
  104. aclk_div = APLL_HZ / CPU_ACLK_HZ - 1;
  105. assert((aclk_div + 1) * CPU_ACLK_HZ == APLL_HZ && aclk_div < 0x1f);
  106. pclk_div = APLL_HZ / CPU_PCLK_HZ - 1;
  107. assert((pclk_div + 1) * CPU_PCLK_HZ == APLL_HZ && pclk_div < 0x7);
  108. hclk_div = APLL_HZ / CPU_HCLK_HZ - 1;
  109. assert((hclk_div + 1) * CPU_HCLK_HZ == APLL_HZ && hclk_div < 0x3);
  110. rk_clrsetreg(&cru->cru_clksel_con[0],
  111. CPU_CLK_PLL_SEL_MASK << CPU_CLK_PLL_SEL_SHIFT |
  112. ACLK_CPU_DIV_MASK << ACLK_CPU_DIV_SHIFT,
  113. CPU_CLK_PLL_SEL_APLL << CPU_CLK_PLL_SEL_SHIFT |
  114. aclk_div << ACLK_CPU_DIV_SHIFT);
  115. rk_clrsetreg(&cru->cru_clksel_con[1],
  116. CPU_PCLK_DIV_MASK << CPU_PCLK_DIV_SHIFT |
  117. CPU_HCLK_DIV_MASK << CPU_HCLK_DIV_SHIFT,
  118. pclk_div << CPU_PCLK_DIV_SHIFT |
  119. hclk_div << CPU_HCLK_DIV_SHIFT);
  120. /*
  121. * select gpll as peri clock pll source and
  122. * set up dependent divisors for PCLK/HCLK and ACLK clocks.
  123. */
  124. aclk_div = GPLL_HZ / PERI_ACLK_HZ - 1;
  125. assert((aclk_div + 1) * PERI_ACLK_HZ == GPLL_HZ && aclk_div < 0x1f);
  126. hclk_div = ilog2(PERI_ACLK_HZ / PERI_HCLK_HZ);
  127. assert((1 << hclk_div) * PERI_HCLK_HZ ==
  128. PERI_ACLK_HZ && (pclk_div < 0x4));
  129. pclk_div = ilog2(PERI_ACLK_HZ / PERI_PCLK_HZ);
  130. assert((1 << pclk_div) * PERI_PCLK_HZ ==
  131. PERI_ACLK_HZ && pclk_div < 0x8);
  132. rk_clrsetreg(&cru->cru_clksel_con[10],
  133. PERI_PLL_SEL_MASK << PERI_PLL_SEL_SHIFT |
  134. PERI_PCLK_DIV_MASK << PERI_PCLK_DIV_SHIFT |
  135. PERI_HCLK_DIV_MASK << PERI_HCLK_DIV_SHIFT |
  136. PERI_ACLK_DIV_MASK << PERI_ACLK_DIV_SHIFT,
  137. PERI_PLL_GPLL << PERI_PLL_SEL_SHIFT |
  138. pclk_div << PERI_PCLK_DIV_SHIFT |
  139. hclk_div << PERI_HCLK_DIV_SHIFT |
  140. aclk_div << PERI_ACLK_DIV_SHIFT);
  141. /* PLL enter normal-mode */
  142. rk_clrsetreg(&cru->cru_mode_con,
  143. GPLL_MODE_MASK << GPLL_MODE_SHIFT |
  144. APLL_MODE_MASK << APLL_MODE_SHIFT,
  145. GPLL_MODE_NORM << GPLL_MODE_SHIFT |
  146. APLL_MODE_NORM << APLL_MODE_SHIFT);
  147. }
  148. /* Get pll rate by id */
  149. static uint32_t rkclk_pll_get_rate(struct rk3036_cru *cru,
  150. enum rk_clk_id clk_id)
  151. {
  152. uint32_t refdiv, fbdiv, postdiv1, postdiv2;
  153. uint32_t con;
  154. int pll_id = rk_pll_id(clk_id);
  155. struct rk3036_pll *pll = &cru->pll[pll_id];
  156. static u8 clk_shift[CLK_COUNT] = {
  157. 0xff, APLL_MODE_SHIFT, DPLL_MODE_SHIFT, 0xff,
  158. GPLL_MODE_SHIFT, 0xff
  159. };
  160. static u8 clk_mask[CLK_COUNT] = {
  161. 0xff, APLL_MODE_MASK, DPLL_MODE_MASK, 0xff,
  162. GPLL_MODE_MASK, 0xff
  163. };
  164. uint shift;
  165. uint mask;
  166. con = readl(&cru->cru_mode_con);
  167. shift = clk_shift[clk_id];
  168. mask = clk_mask[clk_id];
  169. switch ((con >> shift) & mask) {
  170. case GPLL_MODE_SLOW:
  171. return OSC_HZ;
  172. case GPLL_MODE_NORM:
  173. /* normal mode */
  174. con = readl(&pll->con0);
  175. postdiv1 = (con >> PLL_POSTDIV1_SHIFT) & PLL_POSTDIV1_MASK;
  176. fbdiv = (con >> PLL_FBDIV_SHIFT) & PLL_FBDIV_MASK;
  177. con = readl(&pll->con1);
  178. postdiv2 = (con >> PLL_POSTDIV2_SHIFT) & PLL_POSTDIV2_MASK;
  179. refdiv = (con >> PLL_REFDIV_SHIFT) & PLL_REFDIV_MASK;
  180. return (24 * fbdiv / (refdiv * postdiv1 * postdiv2)) * 1000000;
  181. case GPLL_MODE_DEEP:
  182. default:
  183. return 32768;
  184. }
  185. }
  186. static ulong rockchip_mmc_get_clk(struct rk3036_cru *cru, uint clk_general_rate,
  187. int periph)
  188. {
  189. uint src_rate;
  190. uint div, mux;
  191. u32 con;
  192. switch (periph) {
  193. case HCLK_EMMC:
  194. con = readl(&cru->cru_clksel_con[12]);
  195. mux = (con >> EMMC_PLL_SHIFT) & EMMC_PLL_MASK;
  196. div = (con >> EMMC_DIV_SHIFT) & EMMC_DIV_MASK;
  197. break;
  198. case HCLK_SDIO:
  199. con = readl(&cru->cru_clksel_con[12]);
  200. mux = (con >> MMC0_PLL_SHIFT) & MMC0_PLL_MASK;
  201. div = (con >> MMC0_DIV_SHIFT) & MMC0_DIV_MASK;
  202. break;
  203. default:
  204. return -EINVAL;
  205. }
  206. src_rate = mux == EMMC_SEL_24M ? OSC_HZ : clk_general_rate;
  207. return DIV_TO_RATE(src_rate, div);
  208. }
  209. static ulong rockchip_mmc_set_clk(struct rk3036_cru *cru, uint clk_general_rate,
  210. int periph, uint freq)
  211. {
  212. int src_clk_div;
  213. int mux;
  214. debug("%s: clk_general_rate=%u\n", __func__, clk_general_rate);
  215. /* mmc clock auto divide 2 in internal */
  216. src_clk_div = (clk_general_rate / 2 + freq - 1) / freq;
  217. if (src_clk_div > 0x7f) {
  218. src_clk_div = (OSC_HZ / 2 + freq - 1) / freq;
  219. mux = EMMC_SEL_24M;
  220. } else {
  221. mux = EMMC_SEL_GPLL;
  222. }
  223. switch (periph) {
  224. case HCLK_EMMC:
  225. rk_clrsetreg(&cru->cru_clksel_con[12],
  226. EMMC_PLL_MASK << EMMC_PLL_SHIFT |
  227. EMMC_DIV_MASK << EMMC_DIV_SHIFT,
  228. mux << EMMC_PLL_SHIFT |
  229. (src_clk_div - 1) << EMMC_DIV_SHIFT);
  230. break;
  231. case HCLK_SDIO:
  232. rk_clrsetreg(&cru->cru_clksel_con[11],
  233. MMC0_PLL_MASK << MMC0_PLL_SHIFT |
  234. MMC0_DIV_MASK << MMC0_DIV_SHIFT,
  235. mux << MMC0_PLL_SHIFT |
  236. (src_clk_div - 1) << MMC0_DIV_SHIFT);
  237. break;
  238. default:
  239. return -EINVAL;
  240. }
  241. return rockchip_mmc_get_clk(cru, clk_general_rate, periph);
  242. }
  243. static ulong rk3036_clk_get_rate(struct clk *clk)
  244. {
  245. struct rk3036_clk_priv *priv = dev_get_priv(clk->dev);
  246. switch (clk->id) {
  247. case 0 ... 63:
  248. return rkclk_pll_get_rate(priv->cru, clk->id);
  249. default:
  250. return -ENOENT;
  251. }
  252. }
  253. static ulong rk3036_clk_set_rate(struct clk *clk, ulong rate)
  254. {
  255. struct rk3036_clk_priv *priv = dev_get_priv(clk->dev);
  256. ulong new_rate, gclk_rate;
  257. gclk_rate = rkclk_pll_get_rate(priv->cru, CLK_GENERAL);
  258. switch (clk->id) {
  259. case 0 ... 63:
  260. return 0;
  261. case HCLK_EMMC:
  262. new_rate = rockchip_mmc_set_clk(priv->cru, gclk_rate,
  263. clk->id, rate);
  264. break;
  265. default:
  266. return -ENOENT;
  267. }
  268. return new_rate;
  269. }
  270. static struct clk_ops rk3036_clk_ops = {
  271. .get_rate = rk3036_clk_get_rate,
  272. .set_rate = rk3036_clk_set_rate,
  273. };
  274. static int rk3036_clk_probe(struct udevice *dev)
  275. {
  276. struct rk3036_clk_priv *priv = dev_get_priv(dev);
  277. priv->cru = (struct rk3036_cru *)dev_get_addr(dev);
  278. rkclk_init(priv->cru);
  279. return 0;
  280. }
  281. static int rk3036_clk_bind(struct udevice *dev)
  282. {
  283. int ret;
  284. /* The reset driver does not have a device node, so bind it here */
  285. ret = device_bind_driver(gd->dm_root, "rk3036_sysreset", "reset", &dev);
  286. if (ret)
  287. debug("Warning: No RK3036 reset driver: ret=%d\n", ret);
  288. return 0;
  289. }
  290. static const struct udevice_id rk3036_clk_ids[] = {
  291. { .compatible = "rockchip,rk3036-cru" },
  292. { }
  293. };
  294. U_BOOT_DRIVER(rockchip_rk3036_cru) = {
  295. .name = "clk_rk3036",
  296. .id = UCLASS_CLK,
  297. .of_match = rk3036_clk_ids,
  298. .priv_auto_alloc_size = sizeof(struct rk3036_clk_priv),
  299. .ops = &rk3036_clk_ops,
  300. .bind = rk3036_clk_bind,
  301. .probe = rk3036_clk_probe,
  302. };