pwm.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. /*
  2. * This file is part of FH8620 BSP for RT-Thread distribution.
  3. *
  4. * Copyright (c) 2016 Shanghai Fullhan Microelectronics Co., Ltd.
  5. * All rights reserved
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License along
  18. * with this program; if not, write to the Free Software Foundation, Inc.,
  19. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  20. *
  21. * Visit http://www.fullhan.com to get contact with Fullhan.
  22. *
  23. * Change Logs:
  24. * Date Author Notes
  25. */
  26. #include "fh_def.h"
  27. #include "pwm.h"
  28. #include "interrupt.h"
  29. #include "board_info.h"
  30. #include "inc/fh_driverlib.h"
  31. #include <rtthread.h>
  32. #include <rtdevice.h>
  33. #ifdef FH_PWM_DEBUG
  34. #define PRINT_PWM_DBG(fmt, args...) \
  35. do \
  36. { \
  37. rt_kprintf("FH_PWM_DEBUG: "); \
  38. rt_kprintf(fmt, ## args); \
  39. } \
  40. while(0)
  41. #else
  42. #define PRINT_PWM_DBG(fmt, args...) do { } while (0)
  43. #endif
  44. static struct pwm_driver pwm_drv =
  45. {
  46. };
  47. static int pwm_get_duty_cycle_ns(struct pwm_device* pwm)
  48. {
  49. struct fh_pwm_obj *pwm_obj = (struct fh_pwm_obj *)pwm_drv.priv;
  50. rt_uint32_t reg, period, duty;
  51. rt_uint32_t clk_rate = 1000000/*todo: clk_get_rate(fh_pwm_ctrl.clk)*/;
  52. reg = PWM_GetPwmCmd(pwm_obj, pwm->id);
  53. period = reg & 0x0fff;
  54. duty = (reg >> 16) & 0xfff;
  55. duty = period - duty; //reverse duty cycle
  56. if(period == 0)
  57. {
  58. period = duty;
  59. }
  60. pwm->counter_ns = duty * 1000000000 / clk_rate;
  61. pwm->period_ns = period * 1000000000 / clk_rate;
  62. PRINT_PWM_DBG("get duty: %d, period: %d, reg: 0x%x\n", duty, period, reg);
  63. return 0;
  64. }
  65. static int pwm_set_duty_cycle_ns(struct pwm_device* pwm)
  66. {
  67. struct fh_pwm_obj *pwm_obj = (struct fh_pwm_obj *)pwm_drv.priv;
  68. rt_uint32_t period, duty, reg, clk_rate, duty_revert;
  69. clk_rate = 1000000/*todo: clk_get_rate(fh_pwm_ctrl.clk)*/;
  70. if(!clk_rate)
  71. {
  72. rt_kprintf("PWM: clock rate is 0\n");
  73. return -RT_EIO;
  74. }
  75. period = pwm->period_ns / (1000000000 / clk_rate);
  76. if(period < 8)
  77. {
  78. rt_kprintf("PWM: min period is 8\n");
  79. return -RT_EIO;
  80. }
  81. duty = pwm->counter_ns / (1000000000 / clk_rate);
  82. if(period < duty)
  83. {
  84. rt_kprintf("PWM: period < duty\n");
  85. return -RT_EIO;
  86. }
  87. duty_revert = period - duty;
  88. if(duty == period)
  89. {
  90. reg = (duty & 0xfff) << 16 | (0 & 0xfff);
  91. }
  92. else
  93. {
  94. reg = (duty_revert & 0xfff) << 16 | (period & 0xfff);
  95. }
  96. PRINT_PWM_DBG("set duty_revert: %d, period: %d, reg: 0x%x\n", duty_revert, period, reg);
  97. PWM_SetPwmCmd(pwm_obj, pwm->id, reg);
  98. return 0;
  99. }
  100. static rt_err_t fh_pwm_open(rt_device_t dev, rt_uint16_t oflag)
  101. {
  102. struct fh_pwm_obj *pwm_obj = (struct fh_pwm_obj *)pwm_drv.priv;
  103. PWM_Enable(pwm_obj, RT_TRUE);
  104. return 0;
  105. }
  106. static rt_err_t fh_pwm_close(rt_device_t dev)
  107. {
  108. struct fh_pwm_obj *pwm_obj = (struct fh_pwm_obj *)pwm_drv.priv;
  109. PWM_Enable(pwm_obj, RT_FALSE);
  110. return 0;
  111. }
  112. static rt_err_t fh_pwm_ioctl(rt_device_t dev, int cmd, void *arg)
  113. {
  114. int ret = 0;
  115. struct pwm_device *pwm;
  116. struct fh_pwm_obj *pwm_obj = (struct fh_pwm_obj *)pwm_drv.priv;
  117. switch(cmd)
  118. {
  119. case ENABLE_PWM:
  120. PWM_Enable(pwm_obj, RT_FALSE);
  121. break;
  122. case DISABLE_PWM:
  123. PWM_Enable(pwm_obj, RT_TRUE);
  124. break;
  125. case SET_PWM_DUTY_CYCLE:
  126. pwm = (struct pwm_device *)arg;
  127. PRINT_PWM_DBG("ioctl: pwm addr: %p, pwm->period: %d ns\n", pwm, pwm->period_ns);
  128. pwm_set_duty_cycle_ns(pwm);
  129. break;
  130. case GET_PWM_DUTY_CYCLE:
  131. pwm = (struct pwm_device *)arg;
  132. PRINT_PWM_DBG("ioctl: pwm->id: %d, pwm->counter: %d, pwm->period: %d\n", pwm->id, pwm->counter_ns, pwm->period_ns);
  133. pwm_get_duty_cycle_ns(pwm);
  134. break;
  135. }
  136. return ret;
  137. }
  138. int fh_pwm_probe(void *priv_data)
  139. {
  140. rt_device_t pwm_dev ;
  141. struct fh_pwm_obj *pwm_obj = (struct fh_pwm_obj *)priv_data;
  142. rt_memset(&pwm_drv, 0, sizeof(struct pwm_driver));
  143. pwm_drv.pwm[0].id = 0;
  144. pwm_drv.pwm[1].id = 1;
  145. pwm_drv.pwm[2].id = 2;
  146. pwm_drv.pwm[0].working = 0;
  147. pwm_drv.pwm[1].working = 0;
  148. pwm_drv.pwm[2].working = 0;
  149. pwm_drv.priv = pwm_obj;
  150. //todo: clk
  151. PWM_Enable(pwm_obj, RT_FALSE);
  152. pwm_dev = rt_calloc(1,sizeof(struct rt_device));
  153. if (pwm_dev == RT_NULL)
  154. {
  155. rt_kprintf("ERROR: %s rt_device calloc failed\n", __func__);
  156. return -RT_ENOMEM;
  157. }
  158. pwm_dev->user_data = &pwm_drv;
  159. pwm_dev->open =fh_pwm_open;
  160. pwm_dev->close = fh_pwm_close;
  161. pwm_dev->control = fh_pwm_ioctl;
  162. pwm_dev->type = RT_Device_Class_Miscellaneous;
  163. rt_device_register(pwm_dev, "pwm", RT_DEVICE_FLAG_RDWR);
  164. return 0;
  165. }
  166. int fh_pwm_exit(void *priv_data)
  167. {
  168. return 0;
  169. }
  170. struct fh_board_ops pwm_driver_ops =
  171. {
  172. .probe = fh_pwm_probe,
  173. .exit = fh_pwm_exit,
  174. };
  175. void rt_hw_pwm_init(void)
  176. {
  177. PRINT_PWM_DBG("%s start\n", __func__);
  178. fh_board_driver_register("pwm", &pwm_driver_ops);
  179. PRINT_PWM_DBG("%s end\n", __func__);
  180. }