pwm.h 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848
  1. /*
  2. * Copyright (c) 2016 Intel Corporation.
  3. * Copyright (c) 2020-2021 Vestas Wind Systems A/S
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. /**
  8. * @file
  9. * @brief Public PWM Driver APIs
  10. */
  11. #ifndef ZEPHYR_INCLUDE_DRIVERS_PWM_H_
  12. #define ZEPHYR_INCLUDE_DRIVERS_PWM_H_
  13. /**
  14. * @brief PWM Interface
  15. * @defgroup pwm_interface PWM Interface
  16. * @ingroup io_interfaces
  17. * @{
  18. */
  19. #include <errno.h>
  20. #include <zephyr/types.h>
  21. #include <stddef.h>
  22. #include <sys/math_extras.h>
  23. #include <device.h>
  24. #include <dt-bindings/pwm/pwm.h>
  25. #ifdef __cplusplus
  26. extern "C" {
  27. #endif
  28. /**
  29. * @name PWM capture configuration flags
  30. * @{
  31. */
  32. /** @cond INTERNAL_HIDDEN */
  33. /* Bit 0 is used for PWM_POLARITY_NORMAL/PWM_POLARITY_INVERTED */
  34. #define PWM_CAPTURE_TYPE_SHIFT 1U
  35. #define PWM_CAPTURE_TYPE_MASK (3U << PWM_CAPTURE_TYPE_SHIFT)
  36. #define PWM_CAPTURE_MODE_SHIFT 3U
  37. #define PWM_CAPTURE_MODE_MASK (1U << PWM_CAPTURE_MODE_SHIFT)
  38. /** @endcond */
  39. /** PWM pin capture captures period. */
  40. #define PWM_CAPTURE_TYPE_PERIOD (1U << PWM_CAPTURE_TYPE_SHIFT)
  41. /** PWM pin capture captures pulse width. */
  42. #define PWM_CAPTURE_TYPE_PULSE (2U << PWM_CAPTURE_TYPE_SHIFT)
  43. /** PWM pin capture captures both period and pulse width. */
  44. #define PWM_CAPTURE_TYPE_BOTH (PWM_CAPTURE_TYPE_PERIOD | \
  45. PWM_CAPTURE_TYPE_PULSE)
  46. /** PWM pin capture captures a single period/pulse width. */
  47. #define PWM_CAPTURE_MODE_SINGLE (0U << PWM_CAPTURE_MODE_SHIFT)
  48. /** PWM pin capture captures period/pulse width continuously. */
  49. #define PWM_CAPTURE_MODE_CONTINUOUS (1U << PWM_CAPTURE_MODE_SHIFT)
  50. typedef struct {
  51. #define PWM_BREATH_RISE_TIME_DEFAULT (500)
  52. #define PWM_BREATH_DOWN_TIME_DEFAULT (500)
  53. #define PWM_BREATH_HIGH_TIME_DEFAULT (500)
  54. #define PWM_BREATH_LOW_TIME_DEFAULT (2000)
  55. /* lark */
  56. u16_t rise_time_ms;
  57. u16_t down_time_ms;
  58. u16_t high_time_ms;
  59. u16_t low_time_ms;
  60. /* pearlriver */
  61. u16_t pwm_count_max;
  62. u8_t stage_high_wait;
  63. u8_t stage_low_wait;
  64. u8_t stage_a_step;
  65. u8_t stage_b_step;
  66. u8_t stage_c_step;
  67. u8_t stage_d_step;
  68. u8_t stage_e_step;
  69. u8_t stage_f_step;
  70. u8_t stage_a_repeat;
  71. u8_t stage_b_repeat;
  72. u8_t stage_c_repeat;
  73. u8_t stage_d_repeat;
  74. u8_t stage_e_repeat;
  75. u8_t stage_f_repeat;
  76. u8_t stage_a_pwm;
  77. u8_t stage_b_pwm;
  78. u8_t stage_c_pwm;
  79. u8_t stage_d_pwm;
  80. u8_t stage_e_pwm;
  81. u8_t stage_f_pwm;
  82. u8_t start_pwm;
  83. u8_t start_dir;
  84. } pwm_breath_ctrl_t;
  85. typedef struct {
  86. #define PWM_PROGRAM_DMA_IRQ_HF (1 << 0)
  87. #define PWM_PROGRAM_DMA_IRQ_TC (1 << 1)
  88. u32_t period_cycles;
  89. u16_t *ram_buf;
  90. u32_t ram_buf_len;
  91. int (*program_callback)(void *cb_data, u8_t reason);
  92. void *cb_data;
  93. u8_t reload_en : 1;
  94. u16_t repeat;
  95. u16_t cntmax;
  96. } pwm_program_ctrl_t;
  97. typedef struct {
  98. volatile u32_t ir_period;
  99. volatile u32_t ir_duty;
  100. volatile u32_t ir_lc;
  101. volatile u32_t ir_pl0_pre;
  102. volatile u32_t ir_pl0_post;
  103. volatile u32_t ir_pl1_pre;
  104. volatile u32_t ir_pl1_post;
  105. volatile u32_t ir_ll;
  106. volatile u32_t ir_ld;
  107. volatile u32_t ir_pl;
  108. volatile u32_t ir_pd0;
  109. volatile u32_t ir_pd1;
  110. volatile u32_t ir_sl;
  111. volatile u32_t ir_asc;
  112. volatile u32_t buf_num;
  113. u16_t mode;
  114. } pwm_ir_ctrl_t;
  115. /** @} */
  116. /**
  117. * @brief Provides a type to hold PWM configuration flags.
  118. */
  119. typedef uint8_t pwm_flags_t;
  120. /**
  121. * @typedef pwm_pin_set_t
  122. * @brief Callback API upon setting the pin
  123. * See @a pwm_pin_set_cycles() for argument description
  124. */
  125. typedef int (*pwm_pin_set_t)(const struct device *dev, uint32_t pwm,
  126. uint32_t period_cycles, uint32_t pulse_cycles,
  127. pwm_flags_t flags);
  128. /**
  129. * @typedef pwm_set_breath_t
  130. * @brief Callback API upon setting pwm breath mode.
  131. * See @a pwm_set_breath_mode() for argument description
  132. */
  133. typedef int (*pwm_set_breath_t)(const struct device *dev, u32_t pwm, pwm_breath_ctrl_t *ctrl);
  134. /**
  135. * @typedef pwm_set_program_t
  136. * @brief Callback API upon setting pwm program mode.
  137. * See @a pwm_set_program_mode() for argument description
  138. */
  139. typedef int (*pwm_set_program_t)(const struct device *dev, u32_t pwm, pwm_program_ctrl_t *ctrl);
  140. /**
  141. * @typedef pwm_ir_transfer_t
  142. * @brief Callback API upon setting pwm ir mode.
  143. * See @a pwm_ir_transfer_mode() for argument description
  144. */
  145. typedef int (*pwm_ir_transfer_t)(const struct device *dev, u32_t pwm, pwm_ir_ctrl_t *ctrl);
  146. /**
  147. * @typedef pwm_pin_stop_t
  148. * @brief Callback API upon stopping the specified pwm.
  149. * See @a pwm_pin_stop() for argument description
  150. */
  151. typedef int (*pwm_pin_stop_t)(const struct device *dev, u32_t pwm);
  152. /**
  153. * @typedef pwm_pin_mfp_set_t
  154. * @brief Callback API upon init the specified pwm.
  155. * See @a pwm_pin_mfp_set() for argument description
  156. */
  157. typedef int (*pwm_pin_mfp_set_t)(const struct device *dev, u32_t pwm, u32_t pin);
  158. /**
  159. * @typedef pwm_pin_freq_duty_set_t
  160. * @brief Callback API upon setting the pin
  161. * See @a pwm_pin_freq_duty_set() for argument description
  162. */
  163. typedef int (*pwm_pin_freq_duty_set_t)(const struct device *dev, uint32_t pwm,
  164. uint32_t freq, uint8_t duty, pwm_flags_t flags);
  165. /**
  166. * @typedef pwm_capture_callback_handler_t
  167. * @brief PWM capture callback handler function signature
  168. *
  169. * @note The callback handler will be called in interrupt context.
  170. *
  171. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected to enable PWM capture
  172. * support.
  173. *
  174. * @param dev Pointer to the device structure for the driver instance.
  175. * @param pwm PWM pin.
  176. * @param period_cycles Captured PWM period width (in clock cycles). HW
  177. * specific.
  178. * @param pulse_cycles Captured PWM pulse width (in clock cycles). HW specific.
  179. * @param status Status for the PWM capture (0 if no error, negative errno
  180. * otherwise. See @a pwm_pin_capture_cycles() return value
  181. * descriptions for details).
  182. * @param user_data User data passed to @a pwm_pin_configure_capture()
  183. */
  184. typedef void (*pwm_capture_callback_handler_t)(const struct device *dev,
  185. uint32_t pwm,
  186. uint32_t period_cycles,
  187. uint32_t pulse_cycles,
  188. int status,
  189. void *user_data);
  190. /**
  191. * @typedef pwm_pin_configure_capture_t
  192. * @brief Callback API upon configuring PWM pin capture
  193. * See @a pwm_pin_configure_capture() for argument description
  194. */
  195. typedef int (*pwm_pin_configure_capture_t)(const struct device *dev,
  196. uint32_t pwm,
  197. pwm_flags_t flags,
  198. pwm_capture_callback_handler_t cb,
  199. void *user_data);
  200. /**
  201. * @typedef pwm_pin_enable_capture_t
  202. * @brief Callback API upon enabling PWM pin capture
  203. * See @a pwm_pin_enable_capture() for argument description
  204. */
  205. typedef int (*pwm_pin_enable_capture_t)(const struct device *dev,
  206. uint32_t pwm);
  207. /**
  208. * @typedef pwm_pin_disable_capture_t
  209. * @brief Callback API upon disabling PWM pin capture
  210. * See @a pwm_pin_disable_capture() for argument description
  211. */
  212. typedef int (*pwm_pin_disable_capture_t)(const struct device *dev,
  213. uint32_t pwm);
  214. /**
  215. * @typedef pwm_get_cycles_per_sec_t
  216. * @brief Callback API upon getting cycles per second
  217. * See @a pwm_get_cycles_per_sec() for argument description
  218. */
  219. typedef int (*pwm_get_cycles_per_sec_t)(const struct device *dev,
  220. uint32_t pwm,
  221. uint64_t *cycles);
  222. /** @brief PWM driver API definition. */
  223. __subsystem struct pwm_driver_api {
  224. pwm_pin_set_t pin_set;
  225. pwm_set_breath_t set_breath;
  226. pwm_set_program_t set_program;
  227. pwm_ir_transfer_t ir_transfer;
  228. pwm_pin_stop_t pin_stop;
  229. pwm_pin_mfp_set_t pin_mfp;
  230. pwm_pin_freq_duty_set_t pin_set_freq_duty;
  231. #ifdef CONFIG_PWM_CAPTURE
  232. pwm_pin_configure_capture_t pin_configure_capture;
  233. pwm_pin_enable_capture_t pin_enable_capture;
  234. pwm_pin_disable_capture_t pin_disable_capture;
  235. #endif /* CONFIG_PWM_CAPTURE */
  236. pwm_get_cycles_per_sec_t get_cycles_per_sec;
  237. };
  238. /**
  239. * @brief Set the period and pulse width for a single PWM output.
  240. *
  241. * The PWM period and pulse width will synchronously be set to the new values
  242. * without glitches in the PWM signal, but the call will not block for the
  243. * change to take effect.
  244. *
  245. * @note Not all PWM controllers support synchronous, glitch-free updates of the
  246. * PWM period and pulse width. Depending on the hardware, changing the PWM
  247. * period and/or pulse width may cause a glitch in the generated PWM signal.
  248. *
  249. * @note Some multi-channel PWM controllers share the PWM period across all
  250. * channels. Depending on the hardware, changing the PWM period for one channel
  251. * may affect the PWM period for the other channels of the same PWM controller.
  252. *
  253. * Passing 0 as @p pulse will cause the pin to be driven to a constant
  254. * inactive level.
  255. * Passing a non-zero @p pulse equal to @p period will cause the pin
  256. * to be driven to a constant active level.
  257. *
  258. * @param dev Pointer to the device structure for the driver instance.
  259. * @param pwm PWM pin.
  260. * @param period Period (in clock cycle) set to the PWM. HW specific.
  261. * @param pulse Pulse width (in clock cycle) set to the PWM. HW specific.
  262. * @param flags Flags for pin configuration (polarity).
  263. *
  264. * @retval 0 If successful.
  265. * @retval Negative errno code if failure.
  266. */
  267. __syscall int pwm_pin_set_cycles(const struct device *dev, uint32_t pwm,
  268. uint32_t period, uint32_t pulse, pwm_flags_t flags);
  269. static inline int z_impl_pwm_pin_set_cycles(const struct device *dev,
  270. uint32_t pwm,
  271. uint32_t period, uint32_t pulse,
  272. pwm_flags_t flags)
  273. {
  274. struct pwm_driver_api *api;
  275. api = (struct pwm_driver_api *)dev->api;
  276. return api->pin_set(dev, pwm, period, pulse, flags);
  277. }
  278. /**
  279. * @brief Set the pwm breath mode for for a single PWM output.
  280. *
  281. * @param dev Pointer to the device structure for the driver instance.
  282. * @param pwm PWM pin.
  283. * @param ctrl Pointer to the control parameters of pwm breath mode.
  284. * @retval 0 If successful.
  285. * @retval Negative errno code if failure.
  286. */
  287. static inline int pwm_set_breath_mode(const struct device *dev, u32_t pwm, pwm_breath_ctrl_t *ctrl)
  288. {
  289. struct pwm_driver_api *api;
  290. api = (struct pwm_driver_api *)dev->api;
  291. return api->set_breath(dev, pwm, ctrl);
  292. }
  293. /**
  294. * @brief Set the pwm program mode for for a single PWM output.
  295. *
  296. * @param dev Pointer to the device structure for the driver instance.
  297. * @param pwm PWM pin.
  298. * @param ctrl Pointer to the control parameters of pwm program mode.
  299. * @retval 0 If successful.
  300. * @retval Negative errno code if failure.
  301. */
  302. static inline int pwm_set_program_mode(const struct device *dev, u32_t pwm, pwm_program_ctrl_t *ctrl)
  303. {
  304. struct pwm_driver_api *api;
  305. api = (struct pwm_driver_api *)dev->api;
  306. return api->set_program(dev, pwm, ctrl);
  307. }
  308. /**
  309. * @brief Set the pwm ir transfer for for a single PWM output.
  310. *
  311. * @param dev Pointer to the device structure for the driver instance.
  312. * @param pwm PWM pin.
  313. * @param ctrl Pointer to the control parameters of pwm ir mode.
  314. * @retval 0 If successful.
  315. * @retval Negative errno code if failure.
  316. */
  317. static inline int pwm_ir_transfer(const struct device *dev, u32_t pwm, pwm_ir_ctrl_t *ctrl)
  318. {
  319. struct pwm_driver_api *api;
  320. api = (struct pwm_driver_api *)dev->api;
  321. return api->ir_transfer(dev, pwm, ctrl);
  322. }
  323. /**
  324. * @brief Stop the PWM output by the specified pwm pin
  325. * @note This API cooperates with pwm_set_program_mode() function.
  326. *
  327. * @param dev Pointer to the device structure for the driver instance.
  328. * @param pwm PWM pin.
  329. * @retval 0 If successful.
  330. * @retval Negative errno code if failure.
  331. */
  332. static inline int pwm_pin_stop(const struct device *dev, u32_t pwm)
  333. {
  334. struct pwm_driver_api *api;
  335. api = (struct pwm_driver_api *)dev->api;
  336. return api->pin_stop(dev, pwm);
  337. }
  338. /**
  339. * @brief Stop the PWM output by the specified pwm pin
  340. * @note This API cooperates with pwm_set_program_mode() function.
  341. *
  342. * @param dev Pointer to the device structure for the driver instance.
  343. * @param pwm PWM chan.
  344. * @param pwm PWM pin.
  345. * @retval 0 If successful.
  346. * @retval Negative errno code if failure.
  347. */
  348. static inline int pwm_pin_mfp_set(const struct device *dev, u32_t pwm, u32_t pin)
  349. {
  350. struct pwm_driver_api *api;
  351. api = (struct pwm_driver_api *)dev->api;
  352. return api->pin_mfp(dev, pwm, pin);
  353. }
  354. /**
  355. * @brief Set the frequency and duty cycle for a single PWM output.
  356. * @param dev Pointer to the device structure for the driver instance.
  357. * @param pwm PWM pin.
  358. * @param freq frequency of PWM.
  359. * @param duty duty cycle of PWM.
  360. * @param flags Flags for pin configuration (polarity).
  361. *
  362. * @retval 0 If successful.
  363. * @retval Negative errno code if failure.
  364. */
  365. static inline int pwm_pin_set_freq_duty(const struct device *dev,
  366. uint32_t pwm, uint32_t freq, uint8_t duty, pwm_flags_t flags)
  367. {
  368. struct pwm_driver_api *api;
  369. api = (struct pwm_driver_api *)dev->api;
  370. return api->pin_set_freq_duty(dev, pwm, freq, duty, flags);
  371. }
  372. /**
  373. * @brief Configure PWM period/pulse width capture for a single PWM input.
  374. *
  375. * After configuring PWM capture using this function, the capture can be
  376. * enabled/disabled using @a pwm_pin_enable_capture() and @a
  377. * pwm_pin_disable_capture().
  378. *
  379. * @note This API function cannot be invoked from user space due to the use of a
  380. * function callback. In user space, one of the simpler API functions (@a
  381. * pwm_pin_capture_cycles(), @a pwm_pin_capture_usec(), or @a
  382. * pwm_pin_capture_nsec()) can be used instead.
  383. *
  384. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected for this function to be
  385. * available.
  386. *
  387. * @param dev Pointer to the device structure for the driver instance.
  388. * @param pwm PWM pin.
  389. * @param flags PWM capture flags
  390. * @param cb Application callback handler function to be called upon capture
  391. * @param user_data User data to pass to the application callback handler
  392. * function
  393. *
  394. * @retval -EINVAL if invalid function parameters were given
  395. * @retval -ENOSYS if PWM capture is not supported or the given flags are not
  396. * supported
  397. * @retval -EIO if IO error occurred while configuring
  398. * @retval -EBUSY if PWM capture is already in progress
  399. */
  400. #ifdef CONFIG_PWM_CAPTURE
  401. static inline int pwm_pin_configure_capture(const struct device *dev,
  402. uint32_t pwm,
  403. pwm_flags_t flags,
  404. pwm_capture_callback_handler_t cb,
  405. void *user_data)
  406. {
  407. const struct pwm_driver_api *api = (struct pwm_driver_api *)dev->api;
  408. if (api->pin_configure_capture == NULL) {
  409. return -ENOSYS;
  410. }
  411. return api->pin_configure_capture(dev, pwm, flags, cb, user_data);
  412. }
  413. #endif /* CONFIG_PWM_CAPTURE */
  414. /**
  415. * @brief Enable PWM period/pulse width capture for a single PWM input.
  416. *
  417. * The PWM pin must be configured using @a pwm_pin_configure_capture() prior to
  418. * calling this function.
  419. *
  420. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected for this function to be
  421. * available.
  422. *
  423. * @param dev Pointer to the device structure for the driver instance.
  424. * @param pwm PWM pin.
  425. *
  426. * @retval 0 If successful.
  427. * @retval -EINVAL if invalid function parameters were given
  428. * @retval -ENOSYS if PWM capture is not supported
  429. * @retval -EIO if IO error occurred while enabling PWM capture
  430. * @retval -EBUSY if PWM capture is already in progress
  431. */
  432. __syscall int pwm_pin_enable_capture(const struct device *dev, uint32_t pwm);
  433. #ifdef CONFIG_PWM_CAPTURE
  434. static inline int z_impl_pwm_pin_enable_capture(const struct device *dev,
  435. uint32_t pwm)
  436. {
  437. const struct pwm_driver_api *api = (struct pwm_driver_api *)dev->api;
  438. if (api->pin_enable_capture == NULL) {
  439. return -ENOSYS;
  440. }
  441. return api->pin_enable_capture(dev, pwm);
  442. }
  443. #endif /* CONFIG_PWM_CAPTURE */
  444. /**
  445. * @brief Disable PWM period/pulse width capture for a single PWM input.
  446. *
  447. *
  448. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected for this function to be
  449. * available.
  450. *
  451. * @param dev Pointer to the device structure for the driver instance.
  452. * @param pwm PWM pin.
  453. *
  454. * @retval 0 If successful.
  455. * @retval -EINVAL if invalid function parameters were given
  456. * @retval -ENOSYS if PWM capture is not supported
  457. * @retval -EIO if IO error occurred while disabling PWM capture
  458. */
  459. __syscall int pwm_pin_disable_capture(const struct device *dev, uint32_t pwm);
  460. #ifdef CONFIG_PWM_CAPTURE
  461. static inline int z_impl_pwm_pin_disable_capture(const struct device *dev,
  462. uint32_t pwm)
  463. {
  464. const struct pwm_driver_api *api = (struct pwm_driver_api *)dev->api;
  465. if (api->pin_disable_capture == NULL) {
  466. return -ENOSYS;
  467. }
  468. return api->pin_disable_capture(dev, pwm);
  469. }
  470. #endif /* CONFIG_PWM_CAPTURE */
  471. /**
  472. * @brief Capture a single PWM period/pulse width in clock cycles for a single
  473. * PWM input.
  474. *
  475. * This API function wraps calls to @a pwm_pin_configure_capture(), @a
  476. * pwm_pin_enable_capture(), and @a pwm_pin_disable_capture() and passes the
  477. * capture result to the caller. The function is blocking until either the PWM
  478. * capture is completed or a timeout occurs.
  479. *
  480. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected for this function to be
  481. * available.
  482. *
  483. * @param dev Pointer to the device structure for the driver instance.
  484. * @param pwm PWM pin.
  485. * @param flags PWM capture flags.
  486. * @param period Pointer to the memory to store the captured PWM period width
  487. * (in clock cycles). HW specific.
  488. * @param pulse Pointer to the memory to store the captured PWM pulse width (in
  489. * clock cycles). HW specific.
  490. * @param timeout Waiting period for the capture to complete.
  491. *
  492. * @retval 0 If successful.
  493. * @retval -EBUSY PWM capture already in progress.
  494. * @retval -EAGAIN Waiting period timed out.
  495. * @retval -EIO IO error while capturing.
  496. * @retval -ERANGE If result is too large.
  497. */
  498. __syscall int pwm_pin_capture_cycles(const struct device *dev, uint32_t pwm,
  499. pwm_flags_t flags,
  500. uint32_t *period,
  501. uint32_t *pulse,
  502. k_timeout_t timeout);
  503. /**
  504. * @brief Get the clock rate (cycles per second) for a single PWM output.
  505. *
  506. * @param dev Pointer to the device structure for the driver instance.
  507. * @param pwm PWM pin.
  508. * @param cycles Pointer to the memory to store clock rate (cycles per sec).
  509. * HW specific.
  510. *
  511. * @retval 0 If successful.
  512. * @retval Negative errno code if failure.
  513. */
  514. __syscall int pwm_get_cycles_per_sec(const struct device *dev, uint32_t pwm,
  515. uint64_t *cycles);
  516. static inline int z_impl_pwm_get_cycles_per_sec(const struct device *dev,
  517. uint32_t pwm,
  518. uint64_t *cycles)
  519. {
  520. struct pwm_driver_api *api;
  521. api = (struct pwm_driver_api *)dev->api;
  522. return api->get_cycles_per_sec(dev, pwm, cycles);
  523. }
  524. /**
  525. * @brief Set the period and pulse width for a single PWM output.
  526. *
  527. * @param dev Pointer to the device structure for the driver instance.
  528. * @param pwm PWM pin.
  529. * @param period Period (in microseconds) set to the PWM.
  530. * @param pulse Pulse width (in microseconds) set to the PWM.
  531. * @param flags Flags for pin configuration (polarity).
  532. *
  533. * @retval 0 If successful.
  534. * @retval Negative errno code if failure.
  535. */
  536. static inline int pwm_pin_set_usec(const struct device *dev, uint32_t pwm,
  537. uint32_t period, uint32_t pulse,
  538. pwm_flags_t flags)
  539. {
  540. uint64_t period_cycles, pulse_cycles, cycles_per_sec;
  541. if (pwm_get_cycles_per_sec(dev, pwm, &cycles_per_sec) != 0) {
  542. return -EIO;
  543. }
  544. period_cycles = (period * cycles_per_sec) / USEC_PER_SEC;
  545. if (period_cycles >= ((uint64_t)1 << 32)) {
  546. return -ENOTSUP;
  547. }
  548. pulse_cycles = (pulse * cycles_per_sec) / USEC_PER_SEC;
  549. if (pulse_cycles >= ((uint64_t)1 << 32)) {
  550. return -ENOTSUP;
  551. }
  552. return pwm_pin_set_cycles(dev, pwm, (uint32_t)period_cycles,
  553. (uint32_t)pulse_cycles, flags);
  554. }
  555. /**
  556. * @brief Set the period and pulse width for a single PWM output.
  557. *
  558. * @param dev Pointer to the device structure for the driver instance.
  559. * @param pwm PWM pin.
  560. * @param period Period (in nanoseconds) set to the PWM.
  561. * @param pulse Pulse width (in nanoseconds) set to the PWM.
  562. * @param flags Flags for pin configuration (polarity).
  563. *
  564. * @retval 0 If successful.
  565. * @retval Negative errno code if failure.
  566. */
  567. static inline int pwm_pin_set_nsec(const struct device *dev, uint32_t pwm,
  568. uint32_t period, uint32_t pulse,
  569. pwm_flags_t flags)
  570. {
  571. uint64_t period_cycles, pulse_cycles, cycles_per_sec;
  572. if (pwm_get_cycles_per_sec(dev, pwm, &cycles_per_sec) != 0) {
  573. return -EIO;
  574. }
  575. period_cycles = (period * cycles_per_sec) / NSEC_PER_SEC;
  576. if (period_cycles >= ((uint64_t)1 << 32)) {
  577. return -ENOTSUP;
  578. }
  579. pulse_cycles = (pulse * cycles_per_sec) / NSEC_PER_SEC;
  580. if (pulse_cycles >= ((uint64_t)1 << 32)) {
  581. return -ENOTSUP;
  582. }
  583. return pwm_pin_set_cycles(dev, pwm, (uint32_t)period_cycles,
  584. (uint32_t)pulse_cycles, flags);
  585. }
  586. /**
  587. * @brief Convert from PWM cycles to microseconds.
  588. *
  589. * @param dev Pointer to the device structure for the driver instance.
  590. * @param pwm PWM pin.
  591. * @param cycles Cycles to be converted.
  592. * @param usec Pointer to the memory to store calculated usec.
  593. *
  594. * @retval 0 If successful.
  595. * @retval -EIO If cycles per second cannot be determined.
  596. * @retval -ERANGE If result is too large.
  597. */
  598. static inline int pwm_pin_cycles_to_usec(const struct device *dev, uint32_t pwm,
  599. uint32_t cycles, uint64_t *usec)
  600. {
  601. uint64_t cycles_per_sec;
  602. uint64_t temp;
  603. if (pwm_get_cycles_per_sec(dev, pwm, &cycles_per_sec) != 0) {
  604. return -EIO;
  605. }
  606. if (u64_mul_overflow(cycles, (uint64_t)USEC_PER_SEC, &temp)) {
  607. return -ERANGE;
  608. }
  609. *usec = temp / cycles_per_sec;
  610. return 0;
  611. }
  612. /**
  613. * @brief Convert from PWM cycles to nanoseconds.
  614. *
  615. * @param dev Pointer to the device structure for the driver instance.
  616. * @param pwm PWM pin.
  617. * @param cycles Cycles to be converted.
  618. * @param nsec Pointer to the memory to store the calculated nsec.
  619. *
  620. * @retval 0 If successful.
  621. * @retval -EIO If cycles per second cannot be determined.
  622. * @retval -ERANGE If result is too large.
  623. */
  624. static inline int pwm_pin_cycles_to_nsec(const struct device *dev, uint32_t pwm,
  625. uint32_t cycles, uint64_t *nsec)
  626. {
  627. uint64_t cycles_per_sec;
  628. uint64_t temp;
  629. if (pwm_get_cycles_per_sec(dev, pwm, &cycles_per_sec) != 0) {
  630. return -EIO;
  631. }
  632. if (u64_mul_overflow(cycles, (uint64_t)NSEC_PER_SEC, &temp)) {
  633. return -ERANGE;
  634. }
  635. *nsec = temp / cycles_per_sec;
  636. return 0;
  637. }
  638. /**
  639. * @brief Capture a single PWM period/pulse width in microseconds for a single
  640. * PWM input.
  641. *
  642. * This API function wraps calls to @a pwm_pin_capture_cycles() and @a
  643. * pwm_pin_cycles_to_usec() and passes the capture result to the caller. The
  644. * function is blocking until either the PWM capture is completed or a timeout
  645. * occurs.
  646. *
  647. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected for this function to be
  648. * available.
  649. *
  650. * @param dev Pointer to the device structure for the driver instance.
  651. * @param pwm PWM pin.
  652. * @param flags PWM capture flags.
  653. * @param period Pointer to the memory to store the captured PWM period width
  654. * (in usec).
  655. * @param pulse Pointer to the memory to store the captured PWM pulse width (in
  656. * usec).
  657. * @param timeout Waiting period for the capture to complete.
  658. *
  659. * @retval 0 If successful.
  660. * @retval -EBUSY PWM capture already in progress.
  661. * @retval -EAGAIN Waiting period timed out.
  662. * @retval -EIO IO error while capturing.
  663. * @retval -ERANGE If result is too large.
  664. */
  665. static inline int pwm_pin_capture_usec(const struct device *dev, uint32_t pwm,
  666. pwm_flags_t flags,
  667. uint64_t *period,
  668. uint64_t *pulse,
  669. k_timeout_t timeout)
  670. {
  671. uint32_t period_cycles;
  672. uint32_t pulse_cycles;
  673. int err;
  674. err = pwm_pin_capture_cycles(dev, pwm, flags, &period_cycles,
  675. &pulse_cycles, timeout);
  676. if (err) {
  677. return err;
  678. }
  679. err = pwm_pin_cycles_to_usec(dev, pwm, period_cycles, period);
  680. if (err) {
  681. return err;
  682. }
  683. err = pwm_pin_cycles_to_usec(dev, pwm, pulse_cycles, pulse);
  684. if (err) {
  685. return err;
  686. }
  687. return 0;
  688. }
  689. /**
  690. * @brief Capture a single PWM period/pulse width in nanoseconds for a single
  691. * PWM input.
  692. *
  693. * This API function wraps calls to @a pwm_pin_capture_cycles() and @a
  694. * pwm_pin_cycles_to_nsec() and passes the capture result to the caller. The
  695. * function is blocking until either the PWM capture is completed or a timeout
  696. * occurs.
  697. *
  698. * @note @kconfig{CONFIG_PWM_CAPTURE} must be selected for this function to be
  699. * available.
  700. *
  701. * @param dev Pointer to the device structure for the driver instance.
  702. * @param pwm PWM pin.
  703. * @param flags PWM capture flags.
  704. * @param period Pointer to the memory to store the captured PWM period width
  705. * (in nsec).
  706. * @param pulse Pointer to the memory to store the captured PWM pulse width (in
  707. * nsec).
  708. * @param timeout Waiting period for the capture to complete.
  709. *
  710. * @retval 0 If successful.
  711. * @retval -EBUSY PWM capture already in progress.
  712. * @retval -EAGAIN Waiting period timed out.
  713. * @retval -EIO IO error while capturing.
  714. * @retval -ERANGE If result is too large.
  715. */
  716. static inline int pwm_pin_capture_nsec(const struct device *dev, uint32_t pwm,
  717. pwm_flags_t flags,
  718. uint64_t *period,
  719. uint64_t *pulse,
  720. k_timeout_t timeout)
  721. {
  722. uint32_t period_cycles;
  723. uint32_t pulse_cycles;
  724. int err;
  725. err = pwm_pin_capture_cycles(dev, pwm, flags, &period_cycles,
  726. &pulse_cycles, timeout);
  727. if (err) {
  728. return err;
  729. }
  730. err = pwm_pin_cycles_to_nsec(dev, pwm, period_cycles, period);
  731. if (err) {
  732. return err;
  733. }
  734. err = pwm_pin_cycles_to_nsec(dev, pwm, pulse_cycles, pulse);
  735. if (err) {
  736. return err;
  737. }
  738. return 0;
  739. }
  740. #ifdef __cplusplus
  741. }
  742. #endif
  743. /**
  744. * @}
  745. */
  746. #include <syscalls/pwm.h>
  747. #endif /* ZEPHYR_INCLUDE_DRIVERS_PWM_H_ */