clk_rk322x.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. /*
  2. * (C) Copyright 2017 Rockchip Electronics Co., Ltd
  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_rk322x.h>
  14. #include <asm/arch/hardware.h>
  15. #include <dm/lists.h>
  16. #include <dt-bindings/clock/rk3228-cru.h>
  17. #include <linux/log2.h>
  18. DECLARE_GLOBAL_DATA_PTR;
  19. enum {
  20. VCO_MAX_HZ = 3200U * 1000000,
  21. VCO_MIN_HZ = 800 * 1000000,
  22. OUTPUT_MAX_HZ = 3200U * 1000000,
  23. OUTPUT_MIN_HZ = 24 * 1000000,
  24. };
  25. #define DIV_TO_RATE(input_rate, div) ((input_rate) / ((div) + 1))
  26. #define PLL_DIVISORS(hz, _refdiv, _postdiv1, _postdiv2) {\
  27. .refdiv = _refdiv,\
  28. .fbdiv = (u32)((u64)hz * _refdiv * _postdiv1 * _postdiv2 / OSC_HZ), \
  29. .postdiv1 = _postdiv1, .postdiv2 = _postdiv2};\
  30. _Static_assert(((u64)hz * _refdiv * _postdiv1 * _postdiv2 / OSC_HZ) * \
  31. OSC_HZ / (_refdiv * _postdiv1 * _postdiv2) == hz, \
  32. #hz "Hz cannot be hit with PLL "\
  33. "divisors on line " __stringify(__LINE__));
  34. /* use integer mode*/
  35. static const struct pll_div apll_init_cfg = PLL_DIVISORS(APLL_HZ, 1, 3, 1);
  36. static const struct pll_div gpll_init_cfg = PLL_DIVISORS(GPLL_HZ, 2, 2, 1);
  37. static int rkclk_set_pll(struct rk322x_cru *cru, enum rk_clk_id clk_id,
  38. const struct pll_div *div)
  39. {
  40. int pll_id = rk_pll_id(clk_id);
  41. struct rk322x_pll *pll = &cru->pll[pll_id];
  42. /* All PLLs have same VCO and output frequency range restrictions. */
  43. uint vco_hz = OSC_HZ / 1000 * div->fbdiv / div->refdiv * 1000;
  44. uint output_hz = vco_hz / div->postdiv1 / div->postdiv2;
  45. debug("PLL at %p: fb=%d, ref=%d, pst1=%d, pst2=%d, vco=%u Hz, output=%u Hz\n",
  46. pll, div->fbdiv, div->refdiv, div->postdiv1,
  47. div->postdiv2, vco_hz, output_hz);
  48. assert(vco_hz >= VCO_MIN_HZ && vco_hz <= VCO_MAX_HZ &&
  49. output_hz >= OUTPUT_MIN_HZ && output_hz <= OUTPUT_MAX_HZ);
  50. /* use integer mode */
  51. rk_setreg(&pll->con1, 1 << PLL_DSMPD_SHIFT);
  52. /* Power down */
  53. rk_setreg(&pll->con1, 1 << PLL_PD_SHIFT);
  54. rk_clrsetreg(&pll->con0,
  55. PLL_POSTDIV1_MASK | PLL_FBDIV_MASK,
  56. (div->postdiv1 << PLL_POSTDIV1_SHIFT) | div->fbdiv);
  57. rk_clrsetreg(&pll->con1, PLL_POSTDIV2_MASK | PLL_REFDIV_MASK,
  58. (div->postdiv2 << PLL_POSTDIV2_SHIFT |
  59. div->refdiv << PLL_REFDIV_SHIFT));
  60. /* Power Up */
  61. rk_clrreg(&pll->con1, 1 << PLL_PD_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 rk322x_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 | APLL_MODE_MASK,
  75. GPLL_MODE_SLOW << GPLL_MODE_SHIFT |
  76. APLL_MODE_SLOW << APLL_MODE_SHIFT);
  77. /* init pll */
  78. rkclk_set_pll(cru, CLK_ARM, &apll_init_cfg);
  79. rkclk_set_pll(cru, CLK_GENERAL, &gpll_init_cfg);
  80. /*
  81. * select apll as cpu/core clock pll source and
  82. * set up dependent divisors for PERI and ACLK clocks.
  83. * core hz : apll = 1:1
  84. */
  85. aclk_div = APLL_HZ / CORE_ACLK_HZ - 1;
  86. assert((aclk_div + 1) * CORE_ACLK_HZ == APLL_HZ && aclk_div < 0x7);
  87. pclk_div = APLL_HZ / CORE_PERI_HZ - 1;
  88. assert((pclk_div + 1) * CORE_PERI_HZ == APLL_HZ && pclk_div < 0xf);
  89. rk_clrsetreg(&cru->cru_clksel_con[0],
  90. CORE_CLK_PLL_SEL_MASK | CORE_DIV_CON_MASK,
  91. CORE_CLK_PLL_SEL_APLL << CORE_CLK_PLL_SEL_SHIFT |
  92. 0 << CORE_DIV_CON_SHIFT);
  93. rk_clrsetreg(&cru->cru_clksel_con[1],
  94. CORE_ACLK_DIV_MASK | CORE_PERI_DIV_MASK,
  95. aclk_div << CORE_ACLK_DIV_SHIFT |
  96. pclk_div << CORE_PERI_DIV_SHIFT);
  97. /*
  98. * select gpll as pd_bus bus clock source and
  99. * set up dependent divisors for PCLK/HCLK and ACLK clocks.
  100. */
  101. aclk_div = GPLL_HZ / BUS_ACLK_HZ - 1;
  102. assert((aclk_div + 1) * BUS_ACLK_HZ == GPLL_HZ && aclk_div <= 0x1f);
  103. pclk_div = BUS_ACLK_HZ / BUS_PCLK_HZ - 1;
  104. assert((pclk_div + 1) * BUS_PCLK_HZ == GPLL_HZ && pclk_div <= 0x7);
  105. hclk_div = BUS_ACLK_HZ / BUS_HCLK_HZ - 1;
  106. assert((hclk_div + 1) * BUS_HCLK_HZ == GPLL_HZ && hclk_div <= 0x3);
  107. rk_clrsetreg(&cru->cru_clksel_con[0],
  108. BUS_ACLK_PLL_SEL_MASK | BUS_ACLK_DIV_MASK,
  109. BUS_ACLK_PLL_SEL_GPLL << BUS_ACLK_PLL_SEL_SHIFT |
  110. aclk_div << BUS_ACLK_DIV_SHIFT);
  111. rk_clrsetreg(&cru->cru_clksel_con[1],
  112. BUS_PCLK_DIV_MASK | BUS_HCLK_DIV_MASK,
  113. pclk_div << BUS_PCLK_DIV_SHIFT |
  114. hclk_div << BUS_HCLK_DIV_SHIFT);
  115. /*
  116. * select gpll as pd_peri bus clock source and
  117. * set up dependent divisors for PCLK/HCLK and ACLK clocks.
  118. */
  119. aclk_div = GPLL_HZ / PERI_ACLK_HZ - 1;
  120. assert((aclk_div + 1) * PERI_ACLK_HZ == GPLL_HZ && aclk_div < 0x1f);
  121. hclk_div = ilog2(PERI_ACLK_HZ / PERI_HCLK_HZ);
  122. assert((1 << hclk_div) * PERI_HCLK_HZ ==
  123. PERI_ACLK_HZ && (hclk_div < 0x4));
  124. pclk_div = ilog2(PERI_ACLK_HZ / PERI_PCLK_HZ);
  125. assert((1 << pclk_div) * PERI_PCLK_HZ ==
  126. PERI_ACLK_HZ && pclk_div < 0x8);
  127. rk_clrsetreg(&cru->cru_clksel_con[10],
  128. PERI_PLL_SEL_MASK | PERI_PCLK_DIV_MASK |
  129. PERI_HCLK_DIV_MASK | PERI_ACLK_DIV_MASK,
  130. PERI_PLL_GPLL << PERI_PLL_SEL_SHIFT |
  131. pclk_div << PERI_PCLK_DIV_SHIFT |
  132. hclk_div << PERI_HCLK_DIV_SHIFT |
  133. aclk_div << PERI_ACLK_DIV_SHIFT);
  134. /* PLL enter normal-mode */
  135. rk_clrsetreg(&cru->cru_mode_con,
  136. GPLL_MODE_MASK | APLL_MODE_MASK,
  137. GPLL_MODE_NORM << GPLL_MODE_SHIFT |
  138. APLL_MODE_NORM << APLL_MODE_SHIFT);
  139. }
  140. /* Get pll rate by id */
  141. static uint32_t rkclk_pll_get_rate(struct rk322x_cru *cru,
  142. enum rk_clk_id clk_id)
  143. {
  144. uint32_t refdiv, fbdiv, postdiv1, postdiv2;
  145. uint32_t con;
  146. int pll_id = rk_pll_id(clk_id);
  147. struct rk322x_pll *pll = &cru->pll[pll_id];
  148. static u8 clk_shift[CLK_COUNT] = {
  149. 0xff, APLL_MODE_SHIFT, DPLL_MODE_SHIFT, 0xff,
  150. GPLL_MODE_SHIFT, 0xff
  151. };
  152. static u32 clk_mask[CLK_COUNT] = {
  153. 0xff, APLL_MODE_MASK, DPLL_MODE_MASK, 0xff,
  154. GPLL_MODE_MASK, 0xff
  155. };
  156. uint shift;
  157. uint mask;
  158. con = readl(&cru->cru_mode_con);
  159. shift = clk_shift[clk_id];
  160. mask = clk_mask[clk_id];
  161. switch ((con & mask) >> shift) {
  162. case GPLL_MODE_SLOW:
  163. return OSC_HZ;
  164. case GPLL_MODE_NORM:
  165. /* normal mode */
  166. con = readl(&pll->con0);
  167. postdiv1 = (con & PLL_POSTDIV1_MASK) >> PLL_POSTDIV1_SHIFT;
  168. fbdiv = (con & PLL_FBDIV_MASK) >> PLL_FBDIV_SHIFT;
  169. con = readl(&pll->con1);
  170. postdiv2 = (con & PLL_POSTDIV2_MASK) >> PLL_POSTDIV2_SHIFT;
  171. refdiv = (con & PLL_REFDIV_MASK) >> PLL_REFDIV_SHIFT;
  172. return (24 * fbdiv / (refdiv * postdiv1 * postdiv2)) * 1000000;
  173. default:
  174. return 32768;
  175. }
  176. }
  177. static ulong rockchip_mmc_get_clk(struct rk322x_cru *cru, uint clk_general_rate,
  178. int periph)
  179. {
  180. uint src_rate;
  181. uint div, mux;
  182. u32 con;
  183. switch (periph) {
  184. case HCLK_EMMC:
  185. case SCLK_EMMC:
  186. con = readl(&cru->cru_clksel_con[11]);
  187. mux = (con & EMMC_PLL_MASK) >> EMMC_PLL_SHIFT;
  188. con = readl(&cru->cru_clksel_con[12]);
  189. div = (con & EMMC_DIV_MASK) >> EMMC_DIV_SHIFT;
  190. break;
  191. case HCLK_SDMMC:
  192. case SCLK_SDMMC:
  193. con = readl(&cru->cru_clksel_con[11]);
  194. mux = (con & MMC0_PLL_MASK) >> MMC0_PLL_SHIFT;
  195. div = (con & MMC0_DIV_MASK) >> MMC0_DIV_SHIFT;
  196. break;
  197. default:
  198. return -EINVAL;
  199. }
  200. src_rate = mux == EMMC_SEL_24M ? OSC_HZ : clk_general_rate;
  201. return DIV_TO_RATE(src_rate, div) / 2;
  202. }
  203. static ulong rockchip_mmc_set_clk(struct rk322x_cru *cru, uint clk_general_rate,
  204. int periph, uint freq)
  205. {
  206. int src_clk_div;
  207. int mux;
  208. debug("%s: clk_general_rate=%u\n", __func__, clk_general_rate);
  209. /* mmc clock defaulg div 2 internal, need provide double in cru */
  210. src_clk_div = DIV_ROUND_UP(clk_general_rate / 2, freq);
  211. if (src_clk_div > 128) {
  212. src_clk_div = DIV_ROUND_UP(OSC_HZ / 2, freq);
  213. assert(src_clk_div - 1 < 128);
  214. mux = EMMC_SEL_24M;
  215. } else {
  216. mux = EMMC_SEL_GPLL;
  217. }
  218. switch (periph) {
  219. case HCLK_EMMC:
  220. case SCLK_EMMC:
  221. rk_clrsetreg(&cru->cru_clksel_con[11],
  222. EMMC_PLL_MASK,
  223. mux << EMMC_PLL_SHIFT);
  224. rk_clrsetreg(&cru->cru_clksel_con[12],
  225. EMMC_DIV_MASK,
  226. (src_clk_div - 1) << EMMC_DIV_SHIFT);
  227. break;
  228. case HCLK_SDMMC:
  229. case SCLK_SDMMC:
  230. rk_clrsetreg(&cru->cru_clksel_con[11],
  231. MMC0_PLL_MASK | MMC0_DIV_MASK,
  232. mux << MMC0_PLL_SHIFT |
  233. (src_clk_div - 1) << MMC0_DIV_SHIFT);
  234. break;
  235. default:
  236. return -EINVAL;
  237. }
  238. return rockchip_mmc_get_clk(cru, clk_general_rate, periph);
  239. }
  240. static int rk322x_ddr_set_clk(struct rk322x_cru *cru, unsigned int set_rate)
  241. {
  242. struct pll_div dpll_cfg;
  243. /* clk_ddrc == DPLL = 24MHz / refdiv * fbdiv / postdiv1 / postdiv2 */
  244. switch (set_rate) {
  245. case 400*MHz:
  246. dpll_cfg = (struct pll_div)
  247. {.refdiv = 1, .fbdiv = 50, .postdiv1 = 3, .postdiv2 = 1};
  248. break;
  249. case 600*MHz:
  250. dpll_cfg = (struct pll_div)
  251. {.refdiv = 1, .fbdiv = 75, .postdiv1 = 3, .postdiv2 = 1};
  252. break;
  253. case 800*MHz:
  254. dpll_cfg = (struct pll_div)
  255. {.refdiv = 1, .fbdiv = 100, .postdiv1 = 3, .postdiv2 = 1};
  256. break;
  257. }
  258. /* pll enter slow-mode */
  259. rk_clrsetreg(&cru->cru_mode_con, DPLL_MODE_MASK,
  260. DPLL_MODE_SLOW << DPLL_MODE_SHIFT);
  261. rkclk_set_pll(cru, CLK_DDR, &dpll_cfg);
  262. /* PLL enter normal-mode */
  263. rk_clrsetreg(&cru->cru_mode_con, DPLL_MODE_MASK,
  264. DPLL_MODE_NORM << DPLL_MODE_SHIFT);
  265. return set_rate;
  266. }
  267. static ulong rk322x_clk_get_rate(struct clk *clk)
  268. {
  269. struct rk322x_clk_priv *priv = dev_get_priv(clk->dev);
  270. ulong rate, gclk_rate;
  271. gclk_rate = rkclk_pll_get_rate(priv->cru, CLK_GENERAL);
  272. switch (clk->id) {
  273. case 0 ... 63:
  274. rate = rkclk_pll_get_rate(priv->cru, clk->id);
  275. break;
  276. case HCLK_EMMC:
  277. case SCLK_EMMC:
  278. case HCLK_SDMMC:
  279. case SCLK_SDMMC:
  280. rate = rockchip_mmc_get_clk(priv->cru, gclk_rate, clk->id);
  281. break;
  282. default:
  283. return -ENOENT;
  284. }
  285. return rate;
  286. }
  287. static ulong rk322x_clk_set_rate(struct clk *clk, ulong rate)
  288. {
  289. struct rk322x_clk_priv *priv = dev_get_priv(clk->dev);
  290. ulong new_rate, gclk_rate;
  291. gclk_rate = rkclk_pll_get_rate(priv->cru, CLK_GENERAL);
  292. switch (clk->id) {
  293. case HCLK_EMMC:
  294. case SCLK_EMMC:
  295. case HCLK_SDMMC:
  296. case SCLK_SDMMC:
  297. new_rate = rockchip_mmc_set_clk(priv->cru, gclk_rate,
  298. clk->id, rate);
  299. break;
  300. case CLK_DDR:
  301. new_rate = rk322x_ddr_set_clk(priv->cru, rate);
  302. break;
  303. default:
  304. return -ENOENT;
  305. }
  306. return new_rate;
  307. }
  308. static struct clk_ops rk322x_clk_ops = {
  309. .get_rate = rk322x_clk_get_rate,
  310. .set_rate = rk322x_clk_set_rate,
  311. };
  312. static int rk322x_clk_ofdata_to_platdata(struct udevice *dev)
  313. {
  314. struct rk322x_clk_priv *priv = dev_get_priv(dev);
  315. priv->cru = (struct rk322x_cru *)devfdt_get_addr(dev);
  316. return 0;
  317. }
  318. static int rk322x_clk_probe(struct udevice *dev)
  319. {
  320. struct rk322x_clk_priv *priv = dev_get_priv(dev);
  321. rkclk_init(priv->cru);
  322. return 0;
  323. }
  324. static int rk322x_clk_bind(struct udevice *dev)
  325. {
  326. int ret;
  327. struct udevice *sys_child;
  328. struct sysreset_reg *priv;
  329. /* The reset driver does not have a device node, so bind it here */
  330. ret = device_bind_driver(dev, "rockchip_sysreset", "sysreset",
  331. &sys_child);
  332. if (ret) {
  333. debug("Warning: No sysreset driver: ret=%d\n", ret);
  334. } else {
  335. priv = malloc(sizeof(struct sysreset_reg));
  336. priv->glb_srst_fst_value = offsetof(struct rk322x_cru,
  337. cru_glb_srst_fst_value);
  338. priv->glb_srst_snd_value = offsetof(struct rk322x_cru,
  339. cru_glb_srst_snd_value);
  340. sys_child->priv = priv;
  341. }
  342. return 0;
  343. }
  344. static const struct udevice_id rk322x_clk_ids[] = {
  345. { .compatible = "rockchip,rk3228-cru" },
  346. { }
  347. };
  348. U_BOOT_DRIVER(rockchip_rk322x_cru) = {
  349. .name = "clk_rk322x",
  350. .id = UCLASS_CLK,
  351. .of_match = rk322x_clk_ids,
  352. .priv_auto_alloc_size = sizeof(struct rk322x_clk_priv),
  353. .ofdata_to_platdata = rk322x_clk_ofdata_to_platdata,
  354. .ops = &rk322x_clk_ops,
  355. .bind = rk322x_clk_bind,
  356. .probe = rk322x_clk_probe,
  357. };