clk_rk3328.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819
  1. /*
  2. * (C) Copyright 2017 Rockchip Electronics Co., Ltd
  3. *
  4. * SPDX-License-Identifier: GPL-2.0
  5. */
  6. #include <common.h>
  7. #include <bitfield.h>
  8. #include <clk-uclass.h>
  9. #include <dm.h>
  10. #include <errno.h>
  11. #include <syscon.h>
  12. #include <asm/arch/clock.h>
  13. #include <asm/arch/cru_rk3328.h>
  14. #include <asm/arch/hardware.h>
  15. #include <asm/arch/grf_rk3328.h>
  16. #include <asm/io.h>
  17. #include <dm/lists.h>
  18. #include <dt-bindings/clock/rk3328-cru.h>
  19. DECLARE_GLOBAL_DATA_PTR;
  20. struct pll_div {
  21. u32 refdiv;
  22. u32 fbdiv;
  23. u32 postdiv1;
  24. u32 postdiv2;
  25. u32 frac;
  26. };
  27. #define RATE_TO_DIV(input_rate, output_rate) \
  28. ((input_rate) / (output_rate) - 1);
  29. #define DIV_TO_RATE(input_rate, div) ((input_rate) / ((div) + 1))
  30. #define PLL_DIVISORS(hz, _refdiv, _postdiv1, _postdiv2) {\
  31. .refdiv = _refdiv,\
  32. .fbdiv = (u32)((u64)hz * _refdiv * _postdiv1 * _postdiv2 / OSC_HZ),\
  33. .postdiv1 = _postdiv1, .postdiv2 = _postdiv2};
  34. static const struct pll_div gpll_init_cfg = PLL_DIVISORS(GPLL_HZ, 1, 4, 1);
  35. static const struct pll_div cpll_init_cfg = PLL_DIVISORS(CPLL_HZ, 2, 2, 1);
  36. static const struct pll_div apll_816_cfg = PLL_DIVISORS(816 * MHz, 1, 2, 1);
  37. static const struct pll_div apll_600_cfg = PLL_DIVISORS(600 * MHz, 1, 3, 1);
  38. static const struct pll_div *apll_cfgs[] = {
  39. [APLL_816_MHZ] = &apll_816_cfg,
  40. [APLL_600_MHZ] = &apll_600_cfg,
  41. };
  42. enum {
  43. /* PLL_CON0 */
  44. PLL_POSTDIV1_SHIFT = 12,
  45. PLL_POSTDIV1_MASK = 0x7 << PLL_POSTDIV1_SHIFT,
  46. PLL_FBDIV_SHIFT = 0,
  47. PLL_FBDIV_MASK = 0xfff,
  48. /* PLL_CON1 */
  49. PLL_DSMPD_SHIFT = 12,
  50. PLL_DSMPD_MASK = 1 << PLL_DSMPD_SHIFT,
  51. PLL_INTEGER_MODE = 1,
  52. PLL_LOCK_STATUS_SHIFT = 10,
  53. PLL_LOCK_STATUS_MASK = 1 << PLL_LOCK_STATUS_SHIFT,
  54. PLL_POSTDIV2_SHIFT = 6,
  55. PLL_POSTDIV2_MASK = 0x7 << PLL_POSTDIV2_SHIFT,
  56. PLL_REFDIV_SHIFT = 0,
  57. PLL_REFDIV_MASK = 0x3f,
  58. /* PLL_CON2 */
  59. PLL_FRACDIV_SHIFT = 0,
  60. PLL_FRACDIV_MASK = 0xffffff,
  61. /* MODE_CON */
  62. APLL_MODE_SHIFT = 0,
  63. NPLL_MODE_SHIFT = 1,
  64. DPLL_MODE_SHIFT = 4,
  65. CPLL_MODE_SHIFT = 8,
  66. GPLL_MODE_SHIFT = 12,
  67. PLL_MODE_SLOW = 0,
  68. PLL_MODE_NORM,
  69. /* CLKSEL_CON0 */
  70. CLK_CORE_PLL_SEL_APLL = 0,
  71. CLK_CORE_PLL_SEL_GPLL,
  72. CLK_CORE_PLL_SEL_DPLL,
  73. CLK_CORE_PLL_SEL_NPLL,
  74. CLK_CORE_PLL_SEL_SHIFT = 6,
  75. CLK_CORE_PLL_SEL_MASK = 3 << CLK_CORE_PLL_SEL_SHIFT,
  76. CLK_CORE_DIV_SHIFT = 0,
  77. CLK_CORE_DIV_MASK = 0x1f,
  78. /* CLKSEL_CON1 */
  79. ACLKM_CORE_DIV_SHIFT = 4,
  80. ACLKM_CORE_DIV_MASK = 0x7 << ACLKM_CORE_DIV_SHIFT,
  81. PCLK_DBG_DIV_SHIFT = 0,
  82. PCLK_DBG_DIV_MASK = 0xF << PCLK_DBG_DIV_SHIFT,
  83. /* CLKSEL_CON27 */
  84. GMAC2IO_PLL_SEL_SHIFT = 7,
  85. GMAC2IO_PLL_SEL_MASK = 1 << GMAC2IO_PLL_SEL_SHIFT,
  86. GMAC2IO_PLL_SEL_CPLL = 0,
  87. GMAC2IO_PLL_SEL_GPLL = 1,
  88. GMAC2IO_CLK_DIV_MASK = 0x1f,
  89. GMAC2IO_CLK_DIV_SHIFT = 0,
  90. /* CLKSEL_CON28 */
  91. ACLK_PERIHP_PLL_SEL_CPLL = 0,
  92. ACLK_PERIHP_PLL_SEL_GPLL,
  93. ACLK_PERIHP_PLL_SEL_HDMIPHY,
  94. ACLK_PERIHP_PLL_SEL_SHIFT = 6,
  95. ACLK_PERIHP_PLL_SEL_MASK = 3 << ACLK_PERIHP_PLL_SEL_SHIFT,
  96. ACLK_PERIHP_DIV_CON_SHIFT = 0,
  97. ACLK_PERIHP_DIV_CON_MASK = 0x1f,
  98. /* CLKSEL_CON29 */
  99. PCLK_PERIHP_DIV_CON_SHIFT = 4,
  100. PCLK_PERIHP_DIV_CON_MASK = 0x7 << PCLK_PERIHP_DIV_CON_SHIFT,
  101. HCLK_PERIHP_DIV_CON_SHIFT = 0,
  102. HCLK_PERIHP_DIV_CON_MASK = 3 << HCLK_PERIHP_DIV_CON_SHIFT,
  103. /* CLKSEL_CON22 */
  104. CLK_TSADC_DIV_CON_SHIFT = 0,
  105. CLK_TSADC_DIV_CON_MASK = 0x3ff,
  106. /* CLKSEL_CON23 */
  107. CLK_SARADC_DIV_CON_SHIFT = 0,
  108. CLK_SARADC_DIV_CON_MASK = GENMASK(9, 0),
  109. CLK_SARADC_DIV_CON_WIDTH = 10,
  110. /* CLKSEL_CON24 */
  111. CLK_PWM_PLL_SEL_CPLL = 0,
  112. CLK_PWM_PLL_SEL_GPLL,
  113. CLK_PWM_PLL_SEL_SHIFT = 15,
  114. CLK_PWM_PLL_SEL_MASK = 1 << CLK_PWM_PLL_SEL_SHIFT,
  115. CLK_PWM_DIV_CON_SHIFT = 8,
  116. CLK_PWM_DIV_CON_MASK = 0x7f << CLK_PWM_DIV_CON_SHIFT,
  117. CLK_SPI_PLL_SEL_CPLL = 0,
  118. CLK_SPI_PLL_SEL_GPLL,
  119. CLK_SPI_PLL_SEL_SHIFT = 7,
  120. CLK_SPI_PLL_SEL_MASK = 1 << CLK_SPI_PLL_SEL_SHIFT,
  121. CLK_SPI_DIV_CON_SHIFT = 0,
  122. CLK_SPI_DIV_CON_MASK = 0x7f << CLK_SPI_DIV_CON_SHIFT,
  123. /* CLKSEL_CON30 */
  124. CLK_SDMMC_PLL_SEL_CPLL = 0,
  125. CLK_SDMMC_PLL_SEL_GPLL,
  126. CLK_SDMMC_PLL_SEL_24M,
  127. CLK_SDMMC_PLL_SEL_USBPHY,
  128. CLK_SDMMC_PLL_SHIFT = 8,
  129. CLK_SDMMC_PLL_MASK = 0x3 << CLK_SDMMC_PLL_SHIFT,
  130. CLK_SDMMC_DIV_CON_SHIFT = 0,
  131. CLK_SDMMC_DIV_CON_MASK = 0xff << CLK_SDMMC_DIV_CON_SHIFT,
  132. /* CLKSEL_CON32 */
  133. CLK_EMMC_PLL_SEL_CPLL = 0,
  134. CLK_EMMC_PLL_SEL_GPLL,
  135. CLK_EMMC_PLL_SEL_24M,
  136. CLK_EMMC_PLL_SEL_USBPHY,
  137. CLK_EMMC_PLL_SHIFT = 8,
  138. CLK_EMMC_PLL_MASK = 0x3 << CLK_EMMC_PLL_SHIFT,
  139. CLK_EMMC_DIV_CON_SHIFT = 0,
  140. CLK_EMMC_DIV_CON_MASK = 0xff << CLK_EMMC_DIV_CON_SHIFT,
  141. /* CLKSEL_CON34 */
  142. CLK_I2C_PLL_SEL_CPLL = 0,
  143. CLK_I2C_PLL_SEL_GPLL,
  144. CLK_I2C_DIV_CON_MASK = 0x7f,
  145. CLK_I2C_PLL_SEL_MASK = 1,
  146. CLK_I2C1_PLL_SEL_SHIFT = 15,
  147. CLK_I2C1_DIV_CON_SHIFT = 8,
  148. CLK_I2C0_PLL_SEL_SHIFT = 7,
  149. CLK_I2C0_DIV_CON_SHIFT = 0,
  150. /* CLKSEL_CON35 */
  151. CLK_I2C3_PLL_SEL_SHIFT = 15,
  152. CLK_I2C3_DIV_CON_SHIFT = 8,
  153. CLK_I2C2_PLL_SEL_SHIFT = 7,
  154. CLK_I2C2_DIV_CON_SHIFT = 0,
  155. };
  156. #define VCO_MAX_KHZ (3200 * (MHz / KHz))
  157. #define VCO_MIN_KHZ (800 * (MHz / KHz))
  158. #define OUTPUT_MAX_KHZ (3200 * (MHz / KHz))
  159. #define OUTPUT_MIN_KHZ (16 * (MHz / KHz))
  160. /*
  161. * the div restructions of pll in integer mode, these are defined in
  162. * * CRU_*PLL_CON0 or PMUCRU_*PLL_CON0
  163. */
  164. #define PLL_DIV_MIN 16
  165. #define PLL_DIV_MAX 3200
  166. /*
  167. * How to calculate the PLL(from TRM V0.3 Part 1 Page 63):
  168. * Formulas also embedded within the Fractional PLL Verilog model:
  169. * If DSMPD = 1 (DSM is disabled, "integer mode")
  170. * FOUTVCO = FREF / REFDIV * FBDIV
  171. * FOUTPOSTDIV = FOUTVCO / POSTDIV1 / POSTDIV2
  172. * Where:
  173. * FOUTVCO = Fractional PLL non-divided output frequency
  174. * FOUTPOSTDIV = Fractional PLL divided output frequency
  175. * (output of second post divider)
  176. * FREF = Fractional PLL input reference frequency, (the OSC_HZ 24MHz input)
  177. * REFDIV = Fractional PLL input reference clock divider
  178. * FBDIV = Integer value programmed into feedback divide
  179. *
  180. */
  181. static void rkclk_set_pll(struct rk3328_cru *cru, enum rk_clk_id clk_id,
  182. const struct pll_div *div)
  183. {
  184. u32 *pll_con;
  185. u32 mode_shift, mode_mask;
  186. pll_con = NULL;
  187. mode_shift = 0;
  188. switch (clk_id) {
  189. case CLK_ARM:
  190. pll_con = cru->apll_con;
  191. mode_shift = APLL_MODE_SHIFT;
  192. break;
  193. case CLK_DDR:
  194. pll_con = cru->dpll_con;
  195. mode_shift = DPLL_MODE_SHIFT;
  196. break;
  197. case CLK_CODEC:
  198. pll_con = cru->cpll_con;
  199. mode_shift = CPLL_MODE_SHIFT;
  200. break;
  201. case CLK_GENERAL:
  202. pll_con = cru->gpll_con;
  203. mode_shift = GPLL_MODE_SHIFT;
  204. break;
  205. case CLK_NEW:
  206. pll_con = cru->npll_con;
  207. mode_shift = NPLL_MODE_SHIFT;
  208. break;
  209. default:
  210. break;
  211. }
  212. mode_mask = 1 << mode_shift;
  213. /* All 8 PLLs have same VCO and output frequency range restrictions. */
  214. u32 vco_khz = OSC_HZ / 1000 * div->fbdiv / div->refdiv;
  215. u32 output_khz = vco_khz / div->postdiv1 / div->postdiv2;
  216. debug("PLL at %p: fbdiv=%d, refdiv=%d, postdiv1=%d, \
  217. postdiv2=%d, vco=%u khz, output=%u khz\n",
  218. pll_con, div->fbdiv, div->refdiv, div->postdiv1,
  219. div->postdiv2, vco_khz, output_khz);
  220. assert(vco_khz >= VCO_MIN_KHZ && vco_khz <= VCO_MAX_KHZ &&
  221. output_khz >= OUTPUT_MIN_KHZ && output_khz <= OUTPUT_MAX_KHZ &&
  222. div->fbdiv >= PLL_DIV_MIN && div->fbdiv <= PLL_DIV_MAX);
  223. /*
  224. * When power on or changing PLL setting,
  225. * we must force PLL into slow mode to ensure output stable clock.
  226. */
  227. rk_clrsetreg(&cru->mode_con, mode_mask, PLL_MODE_SLOW << mode_shift);
  228. /* use integer mode */
  229. rk_clrsetreg(&pll_con[1], PLL_DSMPD_MASK,
  230. PLL_INTEGER_MODE << PLL_DSMPD_SHIFT);
  231. rk_clrsetreg(&pll_con[0],
  232. PLL_FBDIV_MASK | PLL_POSTDIV1_MASK,
  233. (div->fbdiv << PLL_FBDIV_SHIFT) |
  234. (div->postdiv1 << PLL_POSTDIV1_SHIFT));
  235. rk_clrsetreg(&pll_con[1],
  236. PLL_POSTDIV2_MASK | PLL_REFDIV_MASK,
  237. (div->postdiv2 << PLL_POSTDIV2_SHIFT) |
  238. (div->refdiv << PLL_REFDIV_SHIFT));
  239. /* waiting for pll lock */
  240. while (!(readl(&pll_con[1]) & (1 << PLL_LOCK_STATUS_SHIFT)))
  241. udelay(1);
  242. /* pll enter normal mode */
  243. rk_clrsetreg(&cru->mode_con, mode_mask, PLL_MODE_NORM << mode_shift);
  244. }
  245. static void rkclk_init(struct rk3328_cru *cru)
  246. {
  247. u32 aclk_div;
  248. u32 hclk_div;
  249. u32 pclk_div;
  250. /* configure gpll cpll */
  251. rkclk_set_pll(cru, CLK_GENERAL, &gpll_init_cfg);
  252. rkclk_set_pll(cru, CLK_CODEC, &cpll_init_cfg);
  253. /* configure perihp aclk, hclk, pclk */
  254. aclk_div = GPLL_HZ / PERIHP_ACLK_HZ - 1;
  255. hclk_div = PERIHP_ACLK_HZ / PERIHP_HCLK_HZ - 1;
  256. pclk_div = PERIHP_ACLK_HZ / PERIHP_PCLK_HZ - 1;
  257. rk_clrsetreg(&cru->clksel_con[28],
  258. ACLK_PERIHP_PLL_SEL_MASK | ACLK_PERIHP_DIV_CON_MASK,
  259. ACLK_PERIHP_PLL_SEL_GPLL << ACLK_PERIHP_PLL_SEL_SHIFT |
  260. aclk_div << ACLK_PERIHP_DIV_CON_SHIFT);
  261. rk_clrsetreg(&cru->clksel_con[29],
  262. PCLK_PERIHP_DIV_CON_MASK | HCLK_PERIHP_DIV_CON_MASK,
  263. pclk_div << PCLK_PERIHP_DIV_CON_SHIFT |
  264. hclk_div << HCLK_PERIHP_DIV_CON_SHIFT);
  265. }
  266. void rk3328_configure_cpu(struct rk3328_cru *cru,
  267. enum apll_frequencies apll_freq)
  268. {
  269. u32 clk_core_div;
  270. u32 aclkm_div;
  271. u32 pclk_dbg_div;
  272. rkclk_set_pll(cru, CLK_ARM, apll_cfgs[apll_freq]);
  273. clk_core_div = APLL_HZ / CLK_CORE_HZ - 1;
  274. aclkm_div = APLL_HZ / ACLKM_CORE_HZ / (clk_core_div + 1) - 1;
  275. pclk_dbg_div = APLL_HZ / PCLK_DBG_HZ / (clk_core_div + 1) - 1;
  276. rk_clrsetreg(&cru->clksel_con[0],
  277. CLK_CORE_PLL_SEL_MASK | CLK_CORE_DIV_MASK,
  278. CLK_CORE_PLL_SEL_APLL << CLK_CORE_PLL_SEL_SHIFT |
  279. clk_core_div << CLK_CORE_DIV_SHIFT);
  280. rk_clrsetreg(&cru->clksel_con[1],
  281. PCLK_DBG_DIV_MASK | ACLKM_CORE_DIV_MASK,
  282. pclk_dbg_div << PCLK_DBG_DIV_SHIFT |
  283. aclkm_div << ACLKM_CORE_DIV_SHIFT);
  284. }
  285. static ulong rk3328_i2c_get_clk(struct rk3328_cru *cru, ulong clk_id)
  286. {
  287. u32 div, con;
  288. switch (clk_id) {
  289. case SCLK_I2C0:
  290. con = readl(&cru->clksel_con[34]);
  291. div = con >> CLK_I2C0_DIV_CON_SHIFT & CLK_I2C_DIV_CON_MASK;
  292. break;
  293. case SCLK_I2C1:
  294. con = readl(&cru->clksel_con[34]);
  295. div = con >> CLK_I2C1_DIV_CON_SHIFT & CLK_I2C_DIV_CON_MASK;
  296. break;
  297. case SCLK_I2C2:
  298. con = readl(&cru->clksel_con[35]);
  299. div = con >> CLK_I2C2_DIV_CON_SHIFT & CLK_I2C_DIV_CON_MASK;
  300. break;
  301. case SCLK_I2C3:
  302. con = readl(&cru->clksel_con[35]);
  303. div = con >> CLK_I2C3_DIV_CON_SHIFT & CLK_I2C_DIV_CON_MASK;
  304. break;
  305. default:
  306. printf("do not support this i2c bus\n");
  307. return -EINVAL;
  308. }
  309. return DIV_TO_RATE(GPLL_HZ, div);
  310. }
  311. static ulong rk3328_i2c_set_clk(struct rk3328_cru *cru, ulong clk_id, uint hz)
  312. {
  313. int src_clk_div;
  314. src_clk_div = GPLL_HZ / hz;
  315. assert(src_clk_div - 1 < 127);
  316. switch (clk_id) {
  317. case SCLK_I2C0:
  318. rk_clrsetreg(&cru->clksel_con[34],
  319. CLK_I2C_DIV_CON_MASK << CLK_I2C0_DIV_CON_SHIFT |
  320. CLK_I2C_PLL_SEL_MASK << CLK_I2C0_PLL_SEL_SHIFT,
  321. (src_clk_div - 1) << CLK_I2C0_DIV_CON_SHIFT |
  322. CLK_I2C_PLL_SEL_GPLL << CLK_I2C0_PLL_SEL_SHIFT);
  323. break;
  324. case SCLK_I2C1:
  325. rk_clrsetreg(&cru->clksel_con[34],
  326. CLK_I2C_DIV_CON_MASK << CLK_I2C1_DIV_CON_SHIFT |
  327. CLK_I2C_PLL_SEL_MASK << CLK_I2C1_PLL_SEL_SHIFT,
  328. (src_clk_div - 1) << CLK_I2C1_DIV_CON_SHIFT |
  329. CLK_I2C_PLL_SEL_GPLL << CLK_I2C1_PLL_SEL_SHIFT);
  330. break;
  331. case SCLK_I2C2:
  332. rk_clrsetreg(&cru->clksel_con[35],
  333. CLK_I2C_DIV_CON_MASK << CLK_I2C2_DIV_CON_SHIFT |
  334. CLK_I2C_PLL_SEL_MASK << CLK_I2C2_PLL_SEL_SHIFT,
  335. (src_clk_div - 1) << CLK_I2C2_DIV_CON_SHIFT |
  336. CLK_I2C_PLL_SEL_GPLL << CLK_I2C2_PLL_SEL_SHIFT);
  337. break;
  338. case SCLK_I2C3:
  339. rk_clrsetreg(&cru->clksel_con[35],
  340. CLK_I2C_DIV_CON_MASK << CLK_I2C3_DIV_CON_SHIFT |
  341. CLK_I2C_PLL_SEL_MASK << CLK_I2C3_PLL_SEL_SHIFT,
  342. (src_clk_div - 1) << CLK_I2C3_DIV_CON_SHIFT |
  343. CLK_I2C_PLL_SEL_GPLL << CLK_I2C3_PLL_SEL_SHIFT);
  344. break;
  345. default:
  346. printf("do not support this i2c bus\n");
  347. return -EINVAL;
  348. }
  349. return DIV_TO_RATE(GPLL_HZ, src_clk_div);
  350. }
  351. static ulong rk3328_gmac2io_set_clk(struct rk3328_cru *cru, ulong rate)
  352. {
  353. struct rk3328_grf_regs *grf;
  354. ulong ret;
  355. grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF);
  356. /*
  357. * The RGMII CLK can be derived either from an external "clkin"
  358. * or can be generated from internally by a divider from SCLK_MAC.
  359. */
  360. if (readl(&grf->mac_con[1]) & BIT(10) &&
  361. readl(&grf->soc_con[4]) & BIT(14)) {
  362. /* An external clock will always generate the right rate... */
  363. ret = rate;
  364. } else {
  365. u32 con = readl(&cru->clksel_con[27]);
  366. ulong pll_rate;
  367. u8 div;
  368. if ((con >> GMAC2IO_PLL_SEL_SHIFT) & GMAC2IO_PLL_SEL_GPLL)
  369. pll_rate = GPLL_HZ;
  370. else
  371. pll_rate = CPLL_HZ;
  372. div = DIV_ROUND_UP(pll_rate, rate) - 1;
  373. if (div <= 0x1f)
  374. rk_clrsetreg(&cru->clksel_con[27], GMAC2IO_CLK_DIV_MASK,
  375. div << GMAC2IO_CLK_DIV_SHIFT);
  376. else
  377. debug("Unsupported div for gmac:%d\n", div);
  378. return DIV_TO_RATE(pll_rate, div);
  379. }
  380. return ret;
  381. }
  382. static ulong rk3328_mmc_get_clk(struct rk3328_cru *cru, uint clk_id)
  383. {
  384. u32 div, con, con_id;
  385. switch (clk_id) {
  386. case HCLK_SDMMC:
  387. case SCLK_SDMMC:
  388. con_id = 30;
  389. break;
  390. case HCLK_EMMC:
  391. case SCLK_EMMC:
  392. con_id = 32;
  393. break;
  394. default:
  395. return -EINVAL;
  396. }
  397. con = readl(&cru->clksel_con[con_id]);
  398. div = (con & CLK_EMMC_DIV_CON_MASK) >> CLK_EMMC_DIV_CON_SHIFT;
  399. if ((con & CLK_EMMC_PLL_MASK) >> CLK_EMMC_PLL_SHIFT
  400. == CLK_EMMC_PLL_SEL_24M)
  401. return DIV_TO_RATE(OSC_HZ, div) / 2;
  402. else
  403. return DIV_TO_RATE(GPLL_HZ, div) / 2;
  404. }
  405. static ulong rk3328_mmc_set_clk(struct rk3328_cru *cru,
  406. ulong clk_id, ulong set_rate)
  407. {
  408. int src_clk_div;
  409. u32 con_id;
  410. switch (clk_id) {
  411. case HCLK_SDMMC:
  412. case SCLK_SDMMC:
  413. con_id = 30;
  414. break;
  415. case HCLK_EMMC:
  416. case SCLK_EMMC:
  417. con_id = 32;
  418. break;
  419. default:
  420. return -EINVAL;
  421. }
  422. /* Select clk_sdmmc/emmc source from GPLL by default */
  423. /* mmc clock defaulg div 2 internal, need provide double in cru */
  424. src_clk_div = DIV_ROUND_UP(GPLL_HZ / 2, set_rate);
  425. if (src_clk_div > 127) {
  426. /* use 24MHz source for 400KHz clock */
  427. src_clk_div = DIV_ROUND_UP(OSC_HZ / 2, set_rate);
  428. rk_clrsetreg(&cru->clksel_con[con_id],
  429. CLK_EMMC_PLL_MASK | CLK_EMMC_DIV_CON_MASK,
  430. CLK_EMMC_PLL_SEL_24M << CLK_EMMC_PLL_SHIFT |
  431. (src_clk_div - 1) << CLK_EMMC_DIV_CON_SHIFT);
  432. } else {
  433. rk_clrsetreg(&cru->clksel_con[con_id],
  434. CLK_EMMC_PLL_MASK | CLK_EMMC_DIV_CON_MASK,
  435. CLK_EMMC_PLL_SEL_GPLL << CLK_EMMC_PLL_SHIFT |
  436. (src_clk_div - 1) << CLK_EMMC_DIV_CON_SHIFT);
  437. }
  438. return rk3328_mmc_get_clk(cru, clk_id);
  439. }
  440. static ulong rk3328_pwm_get_clk(struct rk3328_cru *cru)
  441. {
  442. u32 div, con;
  443. con = readl(&cru->clksel_con[24]);
  444. div = (con & CLK_PWM_DIV_CON_MASK) >> CLK_PWM_DIV_CON_SHIFT;
  445. return DIV_TO_RATE(GPLL_HZ, div);
  446. }
  447. static ulong rk3328_pwm_set_clk(struct rk3328_cru *cru, uint hz)
  448. {
  449. u32 div = GPLL_HZ / hz;
  450. rk_clrsetreg(&cru->clksel_con[24],
  451. CLK_PWM_PLL_SEL_MASK | CLK_PWM_DIV_CON_MASK,
  452. CLK_PWM_PLL_SEL_GPLL << CLK_PWM_PLL_SEL_SHIFT |
  453. (div - 1) << CLK_PWM_DIV_CON_SHIFT);
  454. return DIV_TO_RATE(GPLL_HZ, div);
  455. }
  456. static ulong rk3328_saradc_get_clk(struct rk3328_cru *cru)
  457. {
  458. u32 div, val;
  459. val = readl(&cru->clksel_con[23]);
  460. div = bitfield_extract(val, CLK_SARADC_DIV_CON_SHIFT,
  461. CLK_SARADC_DIV_CON_WIDTH);
  462. return DIV_TO_RATE(OSC_HZ, div);
  463. }
  464. static ulong rk3328_saradc_set_clk(struct rk3328_cru *cru, uint hz)
  465. {
  466. int src_clk_div;
  467. src_clk_div = DIV_ROUND_UP(OSC_HZ, hz) - 1;
  468. assert(src_clk_div < 128);
  469. rk_clrsetreg(&cru->clksel_con[23],
  470. CLK_SARADC_DIV_CON_MASK,
  471. src_clk_div << CLK_SARADC_DIV_CON_SHIFT);
  472. return rk3328_saradc_get_clk(cru);
  473. }
  474. static ulong rk3328_clk_get_rate(struct clk *clk)
  475. {
  476. struct rk3328_clk_priv *priv = dev_get_priv(clk->dev);
  477. ulong rate = 0;
  478. switch (clk->id) {
  479. case 0 ... 29:
  480. return 0;
  481. case HCLK_SDMMC:
  482. case HCLK_EMMC:
  483. case SCLK_SDMMC:
  484. case SCLK_EMMC:
  485. rate = rk3328_mmc_get_clk(priv->cru, clk->id);
  486. break;
  487. case SCLK_I2C0:
  488. case SCLK_I2C1:
  489. case SCLK_I2C2:
  490. case SCLK_I2C3:
  491. rate = rk3328_i2c_get_clk(priv->cru, clk->id);
  492. break;
  493. case SCLK_PWM:
  494. rate = rk3328_pwm_get_clk(priv->cru);
  495. break;
  496. case SCLK_SARADC:
  497. rate = rk3328_saradc_get_clk(priv->cru);
  498. break;
  499. default:
  500. return -ENOENT;
  501. }
  502. return rate;
  503. }
  504. static ulong rk3328_clk_set_rate(struct clk *clk, ulong rate)
  505. {
  506. struct rk3328_clk_priv *priv = dev_get_priv(clk->dev);
  507. ulong ret = 0;
  508. switch (clk->id) {
  509. case 0 ... 29:
  510. return 0;
  511. case HCLK_SDMMC:
  512. case HCLK_EMMC:
  513. case SCLK_SDMMC:
  514. case SCLK_EMMC:
  515. ret = rk3328_mmc_set_clk(priv->cru, clk->id, rate);
  516. break;
  517. case SCLK_I2C0:
  518. case SCLK_I2C1:
  519. case SCLK_I2C2:
  520. case SCLK_I2C3:
  521. ret = rk3328_i2c_set_clk(priv->cru, clk->id, rate);
  522. break;
  523. case SCLK_MAC2IO:
  524. ret = rk3328_gmac2io_set_clk(priv->cru, rate);
  525. break;
  526. case SCLK_PWM:
  527. ret = rk3328_pwm_set_clk(priv->cru, rate);
  528. break;
  529. case SCLK_SARADC:
  530. ret = rk3328_saradc_set_clk(priv->cru, rate);
  531. break;
  532. case DCLK_LCDC:
  533. case SCLK_PDM:
  534. case SCLK_RTC32K:
  535. case SCLK_UART0:
  536. case SCLK_UART1:
  537. case SCLK_UART2:
  538. case SCLK_SDIO:
  539. case SCLK_TSP:
  540. case SCLK_WIFI:
  541. case ACLK_BUS_PRE:
  542. case HCLK_BUS_PRE:
  543. case PCLK_BUS_PRE:
  544. case ACLK_PERI_PRE:
  545. case HCLK_PERI:
  546. case PCLK_PERI:
  547. case ACLK_VIO_PRE:
  548. case HCLK_VIO_PRE:
  549. case ACLK_RGA_PRE:
  550. case SCLK_RGA:
  551. case ACLK_VOP_PRE:
  552. case ACLK_RKVDEC_PRE:
  553. case ACLK_RKVENC:
  554. case ACLK_VPU_PRE:
  555. case SCLK_VDEC_CABAC:
  556. case SCLK_VDEC_CORE:
  557. case SCLK_VENC_CORE:
  558. case SCLK_VENC_DSP:
  559. case SCLK_EFUSE:
  560. case PCLK_DDR:
  561. case ACLK_GMAC:
  562. case PCLK_GMAC:
  563. case SCLK_USB3OTG_SUSPEND:
  564. return 0;
  565. default:
  566. return -ENOENT;
  567. }
  568. return ret;
  569. }
  570. static int rk3328_gmac2io_set_parent(struct clk *clk, struct clk *parent)
  571. {
  572. struct rk3328_grf_regs *grf;
  573. const char *clock_output_name;
  574. int ret;
  575. grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF);
  576. /*
  577. * If the requested parent is in the same clock-controller and the id
  578. * is SCLK_MAC2IO_SRC ("clk_mac2io_src"), switch to the internal clock.
  579. */
  580. if ((parent->dev == clk->dev) && (parent->id == SCLK_MAC2IO_SRC)) {
  581. debug("%s: switching RGMII to SCLK_MAC2IO_SRC\n", __func__);
  582. rk_clrreg(&grf->mac_con[1], BIT(10));
  583. return 0;
  584. }
  585. /*
  586. * Otherwise, we need to check the clock-output-names of the
  587. * requested parent to see if the requested id is "gmac_clkin".
  588. */
  589. ret = dev_read_string_index(parent->dev, "clock-output-names",
  590. parent->id, &clock_output_name);
  591. if (ret < 0)
  592. return -ENODATA;
  593. /* If this is "gmac_clkin", switch to the external clock input */
  594. if (!strcmp(clock_output_name, "gmac_clkin")) {
  595. debug("%s: switching RGMII to CLKIN\n", __func__);
  596. rk_setreg(&grf->mac_con[1], BIT(10));
  597. return 0;
  598. }
  599. return -EINVAL;
  600. }
  601. static int rk3328_gmac2io_ext_set_parent(struct clk *clk, struct clk *parent)
  602. {
  603. struct rk3328_grf_regs *grf;
  604. const char *clock_output_name;
  605. int ret;
  606. grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF);
  607. /*
  608. * If the requested parent is in the same clock-controller and the id
  609. * is SCLK_MAC2IO ("clk_mac2io"), switch to the internal clock.
  610. */
  611. if ((parent->dev == clk->dev) && (parent->id == SCLK_MAC2IO)) {
  612. debug("%s: switching RGMII to SCLK_MAC2IO\n", __func__);
  613. rk_clrreg(&grf->soc_con[4], BIT(14));
  614. return 0;
  615. }
  616. /*
  617. * Otherwise, we need to check the clock-output-names of the
  618. * requested parent to see if the requested id is "gmac_clkin".
  619. */
  620. ret = dev_read_string_index(parent->dev, "clock-output-names",
  621. parent->id, &clock_output_name);
  622. if (ret < 0)
  623. return -ENODATA;
  624. /* If this is "gmac_clkin", switch to the external clock input */
  625. if (!strcmp(clock_output_name, "gmac_clkin")) {
  626. debug("%s: switching RGMII to CLKIN\n", __func__);
  627. rk_setreg(&grf->soc_con[4], BIT(14));
  628. return 0;
  629. }
  630. return -EINVAL;
  631. }
  632. static int rk3328_clk_set_parent(struct clk *clk, struct clk *parent)
  633. {
  634. switch (clk->id) {
  635. case SCLK_MAC2IO:
  636. return rk3328_gmac2io_set_parent(clk, parent);
  637. case SCLK_MAC2IO_EXT:
  638. return rk3328_gmac2io_ext_set_parent(clk, parent);
  639. case DCLK_LCDC:
  640. case SCLK_PDM:
  641. case SCLK_RTC32K:
  642. case SCLK_UART0:
  643. case SCLK_UART1:
  644. case SCLK_UART2:
  645. return 0;
  646. }
  647. debug("%s: unsupported clk %ld\n", __func__, clk->id);
  648. return -ENOENT;
  649. }
  650. static struct clk_ops rk3328_clk_ops = {
  651. .get_rate = rk3328_clk_get_rate,
  652. .set_rate = rk3328_clk_set_rate,
  653. .set_parent = rk3328_clk_set_parent,
  654. };
  655. static int rk3328_clk_probe(struct udevice *dev)
  656. {
  657. struct rk3328_clk_priv *priv = dev_get_priv(dev);
  658. rkclk_init(priv->cru);
  659. return 0;
  660. }
  661. static int rk3328_clk_ofdata_to_platdata(struct udevice *dev)
  662. {
  663. struct rk3328_clk_priv *priv = dev_get_priv(dev);
  664. priv->cru = (struct rk3328_cru *)devfdt_get_addr(dev);
  665. return 0;
  666. }
  667. static int rk3328_clk_bind(struct udevice *dev)
  668. {
  669. int ret;
  670. struct udevice *sys_child;
  671. struct sysreset_reg *priv;
  672. /* The reset driver does not have a device node, so bind it here */
  673. ret = device_bind_driver(dev, "rockchip_sysreset", "sysreset",
  674. &sys_child);
  675. if (ret) {
  676. debug("Warning: No sysreset driver: ret=%d\n", ret);
  677. } else {
  678. priv = malloc(sizeof(struct sysreset_reg));
  679. priv->glb_srst_fst_value = offsetof(struct rk3328_cru,
  680. glb_srst_fst_value);
  681. priv->glb_srst_snd_value = offsetof(struct rk3328_cru,
  682. glb_srst_snd_value);
  683. sys_child->priv = priv;
  684. }
  685. #if CONFIG_IS_ENABLED(CONFIG_RESET_ROCKCHIP)
  686. ret = offsetof(struct rk3328_cru, softrst_con[0]);
  687. ret = rockchip_reset_bind(dev, ret, 12);
  688. if (ret)
  689. debug("Warning: software reset driver bind faile\n");
  690. #endif
  691. return ret;
  692. }
  693. static const struct udevice_id rk3328_clk_ids[] = {
  694. { .compatible = "rockchip,rk3328-cru" },
  695. { }
  696. };
  697. U_BOOT_DRIVER(rockchip_rk3328_cru) = {
  698. .name = "rockchip_rk3328_cru",
  699. .id = UCLASS_CLK,
  700. .of_match = rk3328_clk_ids,
  701. .priv_auto_alloc_size = sizeof(struct rk3328_clk_priv),
  702. .ofdata_to_platdata = rk3328_clk_ofdata_to_platdata,
  703. .ops = &rk3328_clk_ops,
  704. .bind = rk3328_clk_bind,
  705. .probe = rk3328_clk_probe,
  706. };