dsi_pll_28hpm.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2012-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #define pr_fmt(fmt) "%s: " fmt, __func__
  6. #include <linux/kernel.h>
  7. #include <linux/err.h>
  8. #include <linux/delay.h>
  9. #include <linux/clk/msm-clk-provider.h>
  10. #include <linux/clk/msm-clk.h>
  11. #include <linux/clk/msm-clock-generic.h>
  12. #include <dt-bindings/clock/msm-clocks-8974.h>
  13. #include "pll_drv.h"
  14. #include "dsi_pll.h"
  15. #define VCO_DELAY_USEC 1
  16. static struct clk_div_ops fixed_2div_ops;
  17. static const struct clk_ops byte_mux_clk_ops;
  18. static const struct clk_ops pixel_clk_src_ops;
  19. static const struct clk_ops byte_clk_src_ops;
  20. static const struct clk_ops analog_postdiv_clk_ops;
  21. static struct lpfr_cfg lpfr_lut_struct[] = {
  22. {479500000, 8},
  23. {480000000, 11},
  24. {575500000, 8},
  25. {576000000, 12},
  26. {610500000, 8},
  27. {659500000, 9},
  28. {671500000, 10},
  29. {672000000, 14},
  30. {708500000, 10},
  31. {750000000, 11},
  32. };
  33. static void dsi_pll_software_reset(struct mdss_pll_resources *dsi_pll_res)
  34. {
  35. /*
  36. * Add HW recommended delays after toggling the software
  37. * reset bit off and back on.
  38. */
  39. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  40. DSI_PHY_PLL_UNIPHY_PLL_TEST_CFG, 0x01);
  41. udelay(1);
  42. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  43. DSI_PHY_PLL_UNIPHY_PLL_TEST_CFG, 0x00);
  44. udelay(1);
  45. }
  46. static int vco_set_rate_hpm(struct clk *c, unsigned long rate)
  47. {
  48. int rc;
  49. struct dsi_pll_vco_clk *vco = to_vco_clk(c);
  50. struct mdss_pll_resources *dsi_pll_res = vco->priv;
  51. rc = mdss_pll_resource_enable(dsi_pll_res, true);
  52. if (rc) {
  53. pr_err("Failed to enable mdss dsi pll resources\n");
  54. return rc;
  55. }
  56. rc = vco_set_rate(vco, rate);
  57. mdss_pll_resource_enable(dsi_pll_res, false);
  58. return rc;
  59. }
  60. static int dsi_pll_enable_seq_8974(struct mdss_pll_resources *dsi_pll_res)
  61. {
  62. int i, rc = 0;
  63. int pll_locked;
  64. dsi_pll_software_reset(dsi_pll_res);
  65. /*
  66. * PLL power up sequence.
  67. * Add necessary delays recommeded by hardware.
  68. */
  69. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  70. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x01);
  71. udelay(1);
  72. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  73. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x05);
  74. udelay(200);
  75. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  76. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x07);
  77. udelay(500);
  78. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  79. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x0f);
  80. udelay(500);
  81. for (i = 0; i < 2; i++) {
  82. udelay(100);
  83. /* DSI Uniphy lock detect setting */
  84. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  85. DSI_PHY_PLL_UNIPHY_PLL_LKDET_CFG2, 0x0c);
  86. udelay(100);
  87. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  88. DSI_PHY_PLL_UNIPHY_PLL_LKDET_CFG2, 0x0d);
  89. pll_locked = dsi_pll_lock_status(dsi_pll_res);
  90. if (pll_locked)
  91. break;
  92. dsi_pll_software_reset(dsi_pll_res);
  93. /*
  94. * PLL power up sequence.
  95. * Add necessary delays recommeded by hardware.
  96. */
  97. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  98. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x1);
  99. udelay(1);
  100. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  101. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x5);
  102. udelay(200);
  103. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  104. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x7);
  105. udelay(250);
  106. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  107. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x5);
  108. udelay(200);
  109. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  110. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0x7);
  111. udelay(500);
  112. MDSS_PLL_REG_W(dsi_pll_res->pll_base,
  113. DSI_PHY_PLL_UNIPHY_PLL_GLB_CFG, 0xf);
  114. udelay(500);
  115. }
  116. if (!pll_locked) {
  117. pr_err("DSI PLL lock failed\n");
  118. rc = -EINVAL;
  119. } else {
  120. pr_debug("DSI PLL Lock success\n");
  121. }
  122. return rc;
  123. }
  124. /* Op structures */
  125. static const struct clk_ops clk_ops_dsi_vco = {
  126. .set_rate = vco_set_rate_hpm,
  127. .round_rate = vco_round_rate,
  128. .handoff = vco_handoff,
  129. .prepare = vco_prepare,
  130. .unprepare = vco_unprepare,
  131. };
  132. static struct clk_div_ops fixed_4div_ops = {
  133. .set_div = fixed_4div_set_div,
  134. .get_div = fixed_4div_get_div,
  135. };
  136. static struct clk_div_ops analog_postdiv_ops = {
  137. .set_div = analog_set_div,
  138. .get_div = analog_get_div,
  139. };
  140. static struct clk_div_ops digital_postdiv_ops = {
  141. .set_div = digital_set_div,
  142. .get_div = digital_get_div,
  143. };
  144. static struct clk_mux_ops byte_mux_ops = {
  145. .set_mux_sel = set_byte_mux_sel,
  146. .get_mux_sel = get_byte_mux_sel,
  147. };
  148. static struct dsi_pll_vco_clk dsi_vco_clk_8974 = {
  149. .ref_clk_rate = 19200000,
  150. .min_rate = 350000000,
  151. .max_rate = 750000000,
  152. .pll_en_seq_cnt = 3,
  153. .pll_enable_seqs[0] = dsi_pll_enable_seq_8974,
  154. .pll_enable_seqs[1] = dsi_pll_enable_seq_8974,
  155. .pll_enable_seqs[2] = dsi_pll_enable_seq_8974,
  156. .lpfr_lut_size = 10,
  157. .lpfr_lut = lpfr_lut_struct,
  158. .c = {
  159. .dbg_name = "dsi_vco_clk_8974",
  160. .ops = &clk_ops_dsi_vco,
  161. CLK_INIT(dsi_vco_clk_8974.c),
  162. },
  163. };
  164. static struct div_clk analog_postdiv_clk_8974 = {
  165. .data = {
  166. .max_div = 255,
  167. .min_div = 1,
  168. },
  169. .ops = &analog_postdiv_ops,
  170. .c = {
  171. .parent = &dsi_vco_clk_8974.c,
  172. .dbg_name = "analog_postdiv_clk",
  173. .ops = &analog_postdiv_clk_ops,
  174. .flags = CLKFLAG_NO_RATE_CACHE,
  175. CLK_INIT(analog_postdiv_clk_8974.c),
  176. },
  177. };
  178. static struct div_clk indirect_path_div2_clk_8974 = {
  179. .ops = &fixed_2div_ops,
  180. .data = {
  181. .div = 2,
  182. .min_div = 2,
  183. .max_div = 2,
  184. },
  185. .c = {
  186. .parent = &analog_postdiv_clk_8974.c,
  187. .dbg_name = "indirect_path_div2_clk",
  188. .ops = &clk_ops_div,
  189. .flags = CLKFLAG_NO_RATE_CACHE,
  190. CLK_INIT(indirect_path_div2_clk_8974.c),
  191. },
  192. };
  193. static struct div_clk pixel_clk_src_8974 = {
  194. .data = {
  195. .max_div = 255,
  196. .min_div = 1,
  197. },
  198. .ops = &digital_postdiv_ops,
  199. .c = {
  200. .parent = &dsi_vco_clk_8974.c,
  201. .dbg_name = "pixel_clk_src_8974",
  202. .ops = &pixel_clk_src_ops,
  203. .flags = CLKFLAG_NO_RATE_CACHE,
  204. CLK_INIT(pixel_clk_src_8974.c),
  205. },
  206. };
  207. static struct mux_clk byte_mux_8974 = {
  208. .num_parents = 2,
  209. .parents = (struct clk_src[]){
  210. {&dsi_vco_clk_8974.c, 0},
  211. {&indirect_path_div2_clk_8974.c, 1},
  212. },
  213. .ops = &byte_mux_ops,
  214. .c = {
  215. .parent = &dsi_vco_clk_8974.c,
  216. .dbg_name = "byte_mux_8974",
  217. .ops = &byte_mux_clk_ops,
  218. CLK_INIT(byte_mux_8974.c),
  219. },
  220. };
  221. static struct div_clk byte_clk_src_8974 = {
  222. .ops = &fixed_4div_ops,
  223. .data = {
  224. .min_div = 4,
  225. .max_div = 4,
  226. },
  227. .c = {
  228. .parent = &byte_mux_8974.c,
  229. .dbg_name = "byte_clk_src_8974",
  230. .ops = &byte_clk_src_ops,
  231. CLK_INIT(byte_clk_src_8974.c),
  232. },
  233. };
  234. static struct clk_lookup mdss_dsi_pllcc_8974[] = {
  235. CLK_LOOKUP_OF("pixel_src", pixel_clk_src_8974,
  236. "fd8c0000.qcom,mmsscc-mdss"),
  237. CLK_LOOKUP_OF("byte_src", byte_clk_src_8974,
  238. "fd8c0000.qcom,mmsscc-mdss"),
  239. };
  240. int dsi_pll_clock_register_hpm(struct platform_device *pdev,
  241. struct mdss_pll_resources *pll_res)
  242. {
  243. int rc;
  244. /* Set client data to mux, div and vco clocks */
  245. byte_clk_src_8974.priv = pll_res;
  246. pixel_clk_src_8974.priv = pll_res;
  247. byte_mux_8974.priv = pll_res;
  248. indirect_path_div2_clk_8974.priv = pll_res;
  249. analog_postdiv_clk_8974.priv = pll_res;
  250. dsi_vco_clk_8974.priv = pll_res;
  251. pll_res->vco_delay = VCO_DELAY_USEC;
  252. /* Set clock source operations */
  253. pixel_clk_src_ops = clk_ops_slave_div;
  254. pixel_clk_src_ops.prepare = dsi_pll_div_prepare;
  255. analog_postdiv_clk_ops = clk_ops_div;
  256. analog_postdiv_clk_ops.prepare = dsi_pll_div_prepare;
  257. byte_clk_src_ops = clk_ops_div;
  258. byte_clk_src_ops.prepare = dsi_pll_div_prepare;
  259. byte_mux_clk_ops = clk_ops_gen_mux;
  260. byte_mux_clk_ops.prepare = dsi_pll_mux_prepare;
  261. if (pll_res->target_id == MDSS_PLL_TARGET_8974) {
  262. rc = of_msm_clock_register(pdev->dev.of_node,
  263. mdss_dsi_pllcc_8974, ARRAY_SIZE(mdss_dsi_pllcc_8974));
  264. if (rc) {
  265. pr_err("Clock register failed\n");
  266. rc = -EPROBE_DEFER;
  267. }
  268. } else {
  269. pr_err("Invalid target ID\n");
  270. rc = -EINVAL;
  271. }
  272. if (!rc)
  273. pr_info("Registered DSI PLL clocks successfully\n");
  274. return rc;
  275. }