clk-pllv2.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/kernel.h>
  3. #include <linux/clk.h>
  4. #include <linux/io.h>
  5. #include <linux/errno.h>
  6. #include <linux/delay.h>
  7. #include <linux/slab.h>
  8. #include <linux/err.h>
  9. #include <asm/div64.h>
  10. #include "clk.h"
  11. #define to_clk_pllv2(clk) (container_of(clk, struct clk_pllv2, clk))
  12. /* PLL Register Offsets */
  13. #define MXC_PLL_DP_CTL 0x00
  14. #define MXC_PLL_DP_CONFIG 0x04
  15. #define MXC_PLL_DP_OP 0x08
  16. #define MXC_PLL_DP_MFD 0x0C
  17. #define MXC_PLL_DP_MFN 0x10
  18. #define MXC_PLL_DP_MFNMINUS 0x14
  19. #define MXC_PLL_DP_MFNPLUS 0x18
  20. #define MXC_PLL_DP_HFS_OP 0x1C
  21. #define MXC_PLL_DP_HFS_MFD 0x20
  22. #define MXC_PLL_DP_HFS_MFN 0x24
  23. #define MXC_PLL_DP_MFN_TOGC 0x28
  24. #define MXC_PLL_DP_DESTAT 0x2c
  25. /* PLL Register Bit definitions */
  26. #define MXC_PLL_DP_CTL_MUL_CTRL 0x2000
  27. #define MXC_PLL_DP_CTL_DPDCK0_2_EN 0x1000
  28. #define MXC_PLL_DP_CTL_DPDCK0_2_OFFSET 12
  29. #define MXC_PLL_DP_CTL_ADE 0x800
  30. #define MXC_PLL_DP_CTL_REF_CLK_DIV 0x400
  31. #define MXC_PLL_DP_CTL_REF_CLK_SEL_MASK (3 << 8)
  32. #define MXC_PLL_DP_CTL_REF_CLK_SEL_OFFSET 8
  33. #define MXC_PLL_DP_CTL_HFSM 0x80
  34. #define MXC_PLL_DP_CTL_PRE 0x40
  35. #define MXC_PLL_DP_CTL_UPEN 0x20
  36. #define MXC_PLL_DP_CTL_RST 0x10
  37. #define MXC_PLL_DP_CTL_RCP 0x8
  38. #define MXC_PLL_DP_CTL_PLM 0x4
  39. #define MXC_PLL_DP_CTL_BRM0 0x2
  40. #define MXC_PLL_DP_CTL_LRF 0x1
  41. #define MXC_PLL_DP_CONFIG_BIST 0x8
  42. #define MXC_PLL_DP_CONFIG_SJC_CE 0x4
  43. #define MXC_PLL_DP_CONFIG_AREN 0x2
  44. #define MXC_PLL_DP_CONFIG_LDREQ 0x1
  45. #define MXC_PLL_DP_OP_MFI_OFFSET 4
  46. #define MXC_PLL_DP_OP_MFI_MASK (0xF << 4)
  47. #define MXC_PLL_DP_OP_PDF_OFFSET 0
  48. #define MXC_PLL_DP_OP_PDF_MASK 0xF
  49. #define MXC_PLL_DP_MFD_OFFSET 0
  50. #define MXC_PLL_DP_MFD_MASK 0x07FFFFFF
  51. #define MXC_PLL_DP_MFN_OFFSET 0x0
  52. #define MXC_PLL_DP_MFN_MASK 0x07FFFFFF
  53. #define MXC_PLL_DP_MFN_TOGC_TOG_DIS (1 << 17)
  54. #define MXC_PLL_DP_MFN_TOGC_TOG_EN (1 << 16)
  55. #define MXC_PLL_DP_MFN_TOGC_CNT_OFFSET 0x0
  56. #define MXC_PLL_DP_MFN_TOGC_CNT_MASK 0xFFFF
  57. #define MXC_PLL_DP_DESTAT_TOG_SEL (1 << 31)
  58. #define MXC_PLL_DP_DESTAT_MFN 0x07FFFFFF
  59. #define MAX_DPLL_WAIT_TRIES 1000 /* 1000 * udelay(1) = 1ms */
  60. struct clk_pllv2 {
  61. struct clk_hw hw;
  62. void __iomem *base;
  63. };
  64. static unsigned long __clk_pllv2_recalc_rate(unsigned long parent_rate,
  65. u32 dp_ctl, u32 dp_op, u32 dp_mfd, u32 dp_mfn)
  66. {
  67. long mfi, mfn, mfd, pdf, ref_clk;
  68. unsigned long dbl;
  69. u64 temp;
  70. dbl = dp_ctl & MXC_PLL_DP_CTL_DPDCK0_2_EN;
  71. pdf = dp_op & MXC_PLL_DP_OP_PDF_MASK;
  72. mfi = (dp_op & MXC_PLL_DP_OP_MFI_MASK) >> MXC_PLL_DP_OP_MFI_OFFSET;
  73. mfi = (mfi <= 5) ? 5 : mfi;
  74. mfd = dp_mfd & MXC_PLL_DP_MFD_MASK;
  75. mfn = dp_mfn & MXC_PLL_DP_MFN_MASK;
  76. mfn = sign_extend32(mfn, 26);
  77. ref_clk = 2 * parent_rate;
  78. if (dbl != 0)
  79. ref_clk *= 2;
  80. ref_clk /= (pdf + 1);
  81. temp = (u64) ref_clk * abs(mfn);
  82. do_div(temp, mfd + 1);
  83. if (mfn < 0)
  84. temp = (ref_clk * mfi) - temp;
  85. else
  86. temp = (ref_clk * mfi) + temp;
  87. return temp;
  88. }
  89. static unsigned long clk_pllv2_recalc_rate(struct clk_hw *hw,
  90. unsigned long parent_rate)
  91. {
  92. u32 dp_op, dp_mfd, dp_mfn, dp_ctl;
  93. void __iomem *pllbase;
  94. struct clk_pllv2 *pll = to_clk_pllv2(hw);
  95. pllbase = pll->base;
  96. dp_ctl = __raw_readl(pllbase + MXC_PLL_DP_CTL);
  97. dp_op = __raw_readl(pllbase + MXC_PLL_DP_OP);
  98. dp_mfd = __raw_readl(pllbase + MXC_PLL_DP_MFD);
  99. dp_mfn = __raw_readl(pllbase + MXC_PLL_DP_MFN);
  100. return __clk_pllv2_recalc_rate(parent_rate, dp_ctl, dp_op, dp_mfd, dp_mfn);
  101. }
  102. static int __clk_pllv2_set_rate(unsigned long rate, unsigned long parent_rate,
  103. u32 *dp_op, u32 *dp_mfd, u32 *dp_mfn)
  104. {
  105. u32 reg;
  106. long mfi, pdf, mfn, mfd = 999999;
  107. u64 temp64;
  108. unsigned long quad_parent_rate;
  109. quad_parent_rate = 4 * parent_rate;
  110. pdf = mfi = -1;
  111. while (++pdf < 16 && mfi < 5)
  112. mfi = rate * (pdf+1) / quad_parent_rate;
  113. if (mfi > 15)
  114. return -EINVAL;
  115. pdf--;
  116. temp64 = rate * (pdf + 1) - quad_parent_rate * mfi;
  117. do_div(temp64, quad_parent_rate / 1000000);
  118. mfn = (long)temp64;
  119. reg = mfi << 4 | pdf;
  120. *dp_op = reg;
  121. *dp_mfd = mfd;
  122. *dp_mfn = mfn;
  123. return 0;
  124. }
  125. static int clk_pllv2_set_rate(struct clk_hw *hw, unsigned long rate,
  126. unsigned long parent_rate)
  127. {
  128. struct clk_pllv2 *pll = to_clk_pllv2(hw);
  129. void __iomem *pllbase;
  130. u32 dp_ctl, dp_op, dp_mfd, dp_mfn;
  131. int ret;
  132. pllbase = pll->base;
  133. ret = __clk_pllv2_set_rate(rate, parent_rate, &dp_op, &dp_mfd, &dp_mfn);
  134. if (ret)
  135. return ret;
  136. dp_ctl = __raw_readl(pllbase + MXC_PLL_DP_CTL);
  137. /* use dpdck0_2 */
  138. __raw_writel(dp_ctl | 0x1000L, pllbase + MXC_PLL_DP_CTL);
  139. __raw_writel(dp_op, pllbase + MXC_PLL_DP_OP);
  140. __raw_writel(dp_mfd, pllbase + MXC_PLL_DP_MFD);
  141. __raw_writel(dp_mfn, pllbase + MXC_PLL_DP_MFN);
  142. return 0;
  143. }
  144. static int clk_pllv2_determine_rate(struct clk_hw *hw,
  145. struct clk_rate_request *req)
  146. {
  147. u32 dp_op, dp_mfd, dp_mfn;
  148. int ret;
  149. ret = __clk_pllv2_set_rate(req->rate, req->best_parent_rate, &dp_op,
  150. &dp_mfd, &dp_mfn);
  151. if (ret) {
  152. req->rate = ret;
  153. return 0;
  154. }
  155. req->rate = __clk_pllv2_recalc_rate(req->best_parent_rate,
  156. MXC_PLL_DP_CTL_DPDCK0_2_EN, dp_op,
  157. dp_mfd, dp_mfn);
  158. return 0;
  159. }
  160. static int clk_pllv2_prepare(struct clk_hw *hw)
  161. {
  162. struct clk_pllv2 *pll = to_clk_pllv2(hw);
  163. u32 reg;
  164. void __iomem *pllbase;
  165. int i = 0;
  166. pllbase = pll->base;
  167. reg = __raw_readl(pllbase + MXC_PLL_DP_CTL) | MXC_PLL_DP_CTL_UPEN;
  168. __raw_writel(reg, pllbase + MXC_PLL_DP_CTL);
  169. /* Wait for lock */
  170. do {
  171. reg = __raw_readl(pllbase + MXC_PLL_DP_CTL);
  172. if (reg & MXC_PLL_DP_CTL_LRF)
  173. break;
  174. udelay(1);
  175. } while (++i < MAX_DPLL_WAIT_TRIES);
  176. if (i == MAX_DPLL_WAIT_TRIES) {
  177. pr_err("MX5: pll locking failed\n");
  178. return -EINVAL;
  179. }
  180. return 0;
  181. }
  182. static void clk_pllv2_unprepare(struct clk_hw *hw)
  183. {
  184. struct clk_pllv2 *pll = to_clk_pllv2(hw);
  185. u32 reg;
  186. void __iomem *pllbase;
  187. pllbase = pll->base;
  188. reg = __raw_readl(pllbase + MXC_PLL_DP_CTL) & ~MXC_PLL_DP_CTL_UPEN;
  189. __raw_writel(reg, pllbase + MXC_PLL_DP_CTL);
  190. }
  191. static const struct clk_ops clk_pllv2_ops = {
  192. .prepare = clk_pllv2_prepare,
  193. .unprepare = clk_pllv2_unprepare,
  194. .recalc_rate = clk_pllv2_recalc_rate,
  195. .determine_rate = clk_pllv2_determine_rate,
  196. .set_rate = clk_pllv2_set_rate,
  197. };
  198. struct clk_hw *imx_clk_hw_pllv2(const char *name, const char *parent,
  199. void __iomem *base)
  200. {
  201. struct clk_pllv2 *pll;
  202. struct clk_hw *hw;
  203. struct clk_init_data init;
  204. int ret;
  205. pll = kzalloc_obj(*pll);
  206. if (!pll)
  207. return ERR_PTR(-ENOMEM);
  208. pll->base = base;
  209. init.name = name;
  210. init.ops = &clk_pllv2_ops;
  211. init.flags = 0;
  212. init.parent_names = &parent;
  213. init.num_parents = 1;
  214. pll->hw.init = &init;
  215. hw = &pll->hw;
  216. ret = clk_hw_register(NULL, hw);
  217. if (ret) {
  218. kfree(pll);
  219. return ERR_PTR(ret);
  220. }
  221. return hw;
  222. }