pwm-imx.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * (C) Copyright 2014
  4. * Heiko Schocher, DENX Software Engineering, hs@denx.de.
  5. *
  6. * Basic support for the pwm module on imx6.
  7. */
  8. #include <common.h>
  9. #include <div64.h>
  10. #include <dm.h>
  11. #include <log.h>
  12. #include <pwm.h>
  13. #include <asm/arch/imx-regs.h>
  14. #include <asm/io.h>
  15. #include <clk.h>
  16. int pwm_config_internal(struct pwm_regs *pwm, unsigned long period_cycles,
  17. unsigned long duty_cycles, unsigned long prescale)
  18. {
  19. u32 cr;
  20. writel(0, &pwm->ir);
  21. cr = PWMCR_PRESCALER(prescale) |
  22. PWMCR_DOZEEN | PWMCR_WAITEN |
  23. PWMCR_DBGEN | PWMCR_CLKSRC_IPG_HIGH;
  24. writel(cr, &pwm->cr);
  25. /* set duty cycles */
  26. writel(duty_cycles, &pwm->sar);
  27. /* set period cycles */
  28. writel(period_cycles, &pwm->pr);
  29. return 0;
  30. }
  31. #ifndef CONFIG_DM_PWM
  32. /* pwm_id from 0..7 */
  33. struct pwm_regs *pwm_id_to_reg(int pwm_id)
  34. {
  35. switch (pwm_id) {
  36. case 0:
  37. return (struct pwm_regs *)PWM1_BASE_ADDR;
  38. case 1:
  39. return (struct pwm_regs *)PWM2_BASE_ADDR;
  40. #ifdef CONFIG_MX6
  41. case 2:
  42. return (struct pwm_regs *)PWM3_BASE_ADDR;
  43. case 3:
  44. return (struct pwm_regs *)PWM4_BASE_ADDR;
  45. #endif
  46. #ifdef CONFIG_MX6SX
  47. case 4:
  48. return (struct pwm_regs *)PWM5_BASE_ADDR;
  49. case 5:
  50. return (struct pwm_regs *)PWM6_BASE_ADDR;
  51. case 6:
  52. return (struct pwm_regs *)PWM7_BASE_ADDR;
  53. case 7:
  54. return (struct pwm_regs *)PWM8_BASE_ADDR;
  55. #endif
  56. default:
  57. printf("unknown pwm_id: %d\n", pwm_id);
  58. break;
  59. }
  60. return NULL;
  61. }
  62. int pwm_imx_get_parms(int period_ns, int duty_ns, unsigned long *period_c,
  63. unsigned long *duty_c, unsigned long *prescale)
  64. {
  65. unsigned long long c;
  66. /*
  67. * we have not yet a clock framework for imx6, so add the clock
  68. * value here as a define. Replace it when we have the clock
  69. * framework.
  70. */
  71. c = CFG_IMX6_PWM_PER_CLK;
  72. c = c * period_ns;
  73. do_div(c, 1000000000);
  74. *period_c = c;
  75. *prescale = *period_c / 0x10000 + 1;
  76. *period_c /= *prescale;
  77. c = *period_c * (unsigned long long)duty_ns;
  78. do_div(c, period_ns);
  79. *duty_c = c;
  80. /*
  81. * according to imx pwm RM, the real period value should be
  82. * PERIOD value in PWMPR plus 2.
  83. */
  84. if (*period_c > 2)
  85. *period_c -= 2;
  86. else
  87. *period_c = 0;
  88. return 0;
  89. }
  90. int pwm_init(int pwm_id, int div, int invert)
  91. {
  92. struct pwm_regs *pwm = (struct pwm_regs *)pwm_id_to_reg(pwm_id);
  93. if (!pwm)
  94. return -1;
  95. writel(0, &pwm->ir);
  96. return 0;
  97. }
  98. int pwm_config(int pwm_id, int duty_ns, int period_ns)
  99. {
  100. struct pwm_regs *pwm = (struct pwm_regs *)pwm_id_to_reg(pwm_id);
  101. unsigned long period_cycles, duty_cycles, prescale;
  102. if (!pwm)
  103. return -1;
  104. pwm_imx_get_parms(period_ns, duty_ns, &period_cycles, &duty_cycles,
  105. &prescale);
  106. return pwm_config_internal(pwm, period_cycles, duty_cycles, prescale);
  107. }
  108. int pwm_enable(int pwm_id)
  109. {
  110. struct pwm_regs *pwm = (struct pwm_regs *)pwm_id_to_reg(pwm_id);
  111. if (!pwm)
  112. return -1;
  113. setbits_le32(&pwm->cr, PWMCR_EN);
  114. return 0;
  115. }
  116. void pwm_disable(int pwm_id)
  117. {
  118. struct pwm_regs *pwm = (struct pwm_regs *)pwm_id_to_reg(pwm_id);
  119. if (!pwm)
  120. return;
  121. clrbits_le32(&pwm->cr, PWMCR_EN);
  122. }
  123. #else
  124. struct imx_pwm_priv {
  125. struct pwm_regs *regs;
  126. bool invert;
  127. struct clk per_clk;
  128. struct clk ipg_clk;
  129. };
  130. int pwm_dm_imx_get_parms(struct imx_pwm_priv *priv, int period_ns,
  131. int duty_ns, unsigned long *period_c, unsigned long *duty_c,
  132. unsigned long *prescale)
  133. {
  134. unsigned long long c;
  135. c = clk_get_rate(&priv->per_clk);
  136. c = c * period_ns;
  137. do_div(c, 1000000000);
  138. *period_c = c;
  139. *prescale = *period_c / 0x10000 + 1;
  140. *period_c /= *prescale;
  141. c = *period_c * (unsigned long long)duty_ns;
  142. do_div(c, period_ns);
  143. *duty_c = c;
  144. /*
  145. * according to imx pwm RM, the real period value should be
  146. * PERIOD value in PWMPR plus 2.
  147. */
  148. if (*period_c > 2)
  149. *period_c -= 2;
  150. else
  151. *period_c = 0;
  152. return 0;
  153. }
  154. static int imx_pwm_set_invert(struct udevice *dev, uint channel,
  155. bool polarity)
  156. {
  157. struct imx_pwm_priv *priv = dev_get_priv(dev);
  158. debug("%s: polarity=%u\n", __func__, polarity);
  159. priv->invert = polarity;
  160. return 0;
  161. }
  162. static int imx_pwm_set_config(struct udevice *dev, uint channel,
  163. uint period_ns, uint duty_ns)
  164. {
  165. struct imx_pwm_priv *priv = dev_get_priv(dev);
  166. struct pwm_regs *regs = priv->regs;
  167. unsigned long period_cycles, duty_cycles, prescale;
  168. debug("%s: Config '%s' channel: %d\n", __func__, dev->name, channel);
  169. pwm_dm_imx_get_parms(priv, period_ns, duty_ns, &period_cycles, &duty_cycles,
  170. &prescale);
  171. return pwm_config_internal(regs, period_cycles, duty_cycles, prescale);
  172. };
  173. static int imx_pwm_set_enable(struct udevice *dev, uint channel, bool enable)
  174. {
  175. struct imx_pwm_priv *priv = dev_get_priv(dev);
  176. struct pwm_regs *regs = priv->regs;
  177. debug("%s: Enable '%s' state: %d\n", __func__, dev->name, enable);
  178. if (enable)
  179. setbits_le32(&regs->cr, PWMCR_EN);
  180. else
  181. clrbits_le32(&regs->cr, PWMCR_EN);
  182. return 0;
  183. };
  184. static int imx_pwm_of_to_plat(struct udevice *dev)
  185. {
  186. int ret;
  187. struct imx_pwm_priv *priv = dev_get_priv(dev);
  188. priv->regs = dev_read_addr_ptr(dev);
  189. ret = clk_get_by_name(dev, "per", &priv->per_clk);
  190. if (ret) {
  191. printf("Failed to get per_clk\n");
  192. return ret;
  193. }
  194. ret = clk_get_by_name(dev, "ipg", &priv->ipg_clk);
  195. if (ret) {
  196. printf("Failed to get ipg_clk\n");
  197. return ret;
  198. }
  199. return 0;
  200. }
  201. static int imx_pwm_probe(struct udevice *dev)
  202. {
  203. int ret;
  204. struct imx_pwm_priv *priv = dev_get_priv(dev);
  205. ret = clk_enable(&priv->per_clk);
  206. if (ret) {
  207. printf("Failed to enable per_clk\n");
  208. return ret;
  209. }
  210. ret = clk_enable(&priv->ipg_clk);
  211. if (ret) {
  212. printf("Failed to enable ipg_clk\n");
  213. return ret;
  214. }
  215. return 0;
  216. }
  217. static const struct pwm_ops imx_pwm_ops = {
  218. .set_invert = imx_pwm_set_invert,
  219. .set_config = imx_pwm_set_config,
  220. .set_enable = imx_pwm_set_enable,
  221. };
  222. static const struct udevice_id imx_pwm_ids[] = {
  223. { .compatible = "fsl,imx27-pwm" },
  224. { }
  225. };
  226. U_BOOT_DRIVER(imx_pwm) = {
  227. .name = "imx_pwm",
  228. .id = UCLASS_PWM,
  229. .of_match = imx_pwm_ids,
  230. .ops = &imx_pwm_ops,
  231. .of_to_plat = imx_pwm_of_to_plat,
  232. .probe = imx_pwm_probe,
  233. .priv_auto = sizeof(struct imx_pwm_priv),
  234. };
  235. #endif