counter.h 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. * Copyright (c) 2016 Intel Corporation
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. /**
  8. * @file
  9. * @brief Public API for counter and timer drivers
  10. */
  11. #ifndef ZEPHYR_INCLUDE_DRIVERS_COUNTER_H_
  12. #define ZEPHYR_INCLUDE_DRIVERS_COUNTER_H_
  13. /**
  14. * @brief Counter Interface
  15. * @defgroup counter_interface Counter Interface
  16. * @ingroup io_interfaces
  17. * @{
  18. */
  19. #include <zephyr/types.h>
  20. #include <stddef.h>
  21. #include <device.h>
  22. #include <stdbool.h>
  23. #ifdef __cplusplus
  24. extern "C" {
  25. #endif
  26. /**@defgroup COUNTER_FLAGS Counter device capabilities
  27. * @{ */
  28. /**
  29. * @brief Counter count up flag.
  30. */
  31. #define COUNTER_CONFIG_INFO_COUNT_UP BIT(0)
  32. /**@} */
  33. /**@defgroup COUNTER_TOP_FLAGS Flags used by @ref counter_top_cfg.
  34. * @{
  35. */
  36. /**
  37. * @brief Flag preventing counter reset when top value is changed.
  38. *
  39. * If flags is set then counter is free running while top value is updated,
  40. * otherwise counter is reset (see @ref counter_set_top_value()).
  41. */
  42. #define COUNTER_TOP_CFG_DONT_RESET BIT(0)
  43. /**
  44. * @brief Flag instructing counter to reset itself if changing top value
  45. * results in counter going out of new top value bound.
  46. *
  47. * See @ref COUNTER_TOP_CFG_DONT_RESET.
  48. */
  49. #define COUNTER_TOP_CFG_RESET_WHEN_LATE BIT(1)
  50. /**@} */
  51. /**@defgroup COUNTER_ALARM_FLAGS Alarm configuration flags
  52. *
  53. * @brief Used in alarm configuration structure (@ref counter_alarm_cfg).
  54. * @{ */
  55. /**
  56. * @brief Counter alarm absolute value flag.
  57. *
  58. * Ticks relation to counter value. If set ticks are treated as absolute value,
  59. * else it is relative to the counter reading performed during the call.
  60. */
  61. #define COUNTER_ALARM_CFG_ABSOLUTE BIT(0)
  62. /**
  63. * @brief Alarm flag enabling immediate expiration when driver detects that
  64. * absolute alarm was set too late.
  65. *
  66. * Alarm callback must be called from the same context as if it was set on time.
  67. */
  68. #define COUNTER_ALARM_CFG_EXPIRE_WHEN_LATE BIT(1)
  69. /**@} */
  70. /**@defgroup COUNTER_GUARD_PERIOD_FLAGS Counter guard period flags
  71. *
  72. * @brief Used by @ref counter_set_guard_period and
  73. * @ref counter_get_guard_period.
  74. * @{ */
  75. /**
  76. * @brief Identifies guard period needed for detection of late setting of
  77. * absolute alarm (see @ref counter_set_channel_alarm).
  78. */
  79. #define COUNTER_GUARD_PERIOD_LATE_TO_SET BIT(0)
  80. /**@} */
  81. /** @brief Alarm callback
  82. *
  83. * @param dev Pointer to the device structure for the driver instance.
  84. * @param chan_id Channel ID.
  85. * @param ticks Counter value that triggered the alarm.
  86. * @param user_data User data.
  87. */
  88. typedef void (*counter_alarm_callback_t)(const struct device *dev,
  89. uint8_t chan_id, uint32_t ticks,
  90. void *user_data);
  91. /** @brief Alarm callback structure.
  92. *
  93. * @param callback Callback called on alarm (cannot be NULL).
  94. * @param ticks Number of ticks that triggers the alarm. It can be relative (to
  95. * now) or absolute value (see @ref COUNTER_ALARM_CFG_ABSOLUTE).
  96. * Absolute alarm cannot be set further in future than top_value
  97. * decremented by the guard period. Relative alarm ticks cannot
  98. * exceed current top value (see @ref counter_get_top_value).
  99. * If counter is clock driven then ticks can be converted to
  100. * microseconds (see @ref counter_ticks_to_us). Alternatively,
  101. * counter implementation may count asynchronous events.
  102. * @param user_data User data returned in callback.
  103. * @param flags Alarm flags. See @ref COUNTER_ALARM_FLAGS.
  104. */
  105. struct counter_alarm_cfg {
  106. counter_alarm_callback_t callback;
  107. uint32_t ticks;
  108. void *user_data;
  109. uint32_t flags;
  110. };
  111. /** @brief Callback called when counter turns around.
  112. *
  113. * @param dev Pointer to the device structure for the driver instance.
  114. * @param user_data User data provided in @ref counter_set_top_value.
  115. */
  116. typedef void (*counter_top_callback_t)(const struct device *dev,
  117. void *user_data);
  118. /** @brief Top value configuration structure.
  119. *
  120. * @param ticks Top value.
  121. * @param callback Callback function. Can be NULL.
  122. * @param user_data User data passed to callback function. Not valid if
  123. * callback is NULL.
  124. * @param flags Flags. See @ref COUNTER_TOP_FLAGS.
  125. */
  126. struct counter_top_cfg {
  127. uint32_t ticks;
  128. counter_top_callback_t callback;
  129. void *user_data;
  130. uint32_t flags;
  131. };
  132. /** @brief Structure with generic counter features.
  133. *
  134. * @param max_top_value Maximal (default) top value on which counter is reset
  135. * (cleared or reloaded).
  136. * @param freq Frequency of the source clock if synchronous events are
  137. * counted.
  138. * @param flags Flags. See @ref COUNTER_FLAGS.
  139. * @param channels Number of channels that can be used for setting alarm,
  140. * see @ref counter_set_channel_alarm.
  141. */
  142. struct counter_config_info {
  143. uint32_t max_top_value;
  144. uint32_t freq;
  145. uint8_t flags;
  146. uint8_t channels;
  147. };
  148. typedef int (*counter_api_start)(const struct device *dev);
  149. typedef int (*counter_api_stop)(const struct device *dev);
  150. typedef int (*counter_api_get_value)(const struct device *dev,
  151. uint32_t *ticks);
  152. typedef int (*counter_api_set_alarm)(const struct device *dev,
  153. uint8_t chan_id,
  154. const struct counter_alarm_cfg *alarm_cfg);
  155. typedef int (*counter_api_cancel_alarm)(const struct device *dev,
  156. uint8_t chan_id);
  157. typedef int (*counter_api_set_top_value)(const struct device *dev,
  158. const struct counter_top_cfg *cfg);
  159. typedef uint32_t (*counter_api_get_pending_int)(const struct device *dev);
  160. typedef uint32_t (*counter_api_get_top_value)(const struct device *dev);
  161. typedef uint32_t (*counter_api_get_guard_period)(const struct device *dev,
  162. uint32_t flags);
  163. typedef int (*counter_api_set_guard_period)(const struct device *dev,
  164. uint32_t ticks,
  165. uint32_t flags);
  166. __subsystem struct counter_driver_api {
  167. counter_api_start start;
  168. counter_api_stop stop;
  169. counter_api_get_value get_value;
  170. counter_api_set_alarm set_alarm;
  171. counter_api_cancel_alarm cancel_alarm;
  172. counter_api_set_top_value set_top_value;
  173. counter_api_get_pending_int get_pending_int;
  174. counter_api_get_top_value get_top_value;
  175. counter_api_get_guard_period get_guard_period;
  176. counter_api_set_guard_period set_guard_period;
  177. };
  178. /**
  179. * @brief Function to check if counter is counting up.
  180. *
  181. * @param[in] dev Pointer to the device structure for the driver instance.
  182. *
  183. * @retval true if counter is counting up.
  184. * @retval false if counter is counting down.
  185. */
  186. __syscall bool counter_is_counting_up(const struct device *dev);
  187. static inline bool z_impl_counter_is_counting_up(const struct device *dev)
  188. {
  189. const struct counter_config_info *config =
  190. (const struct counter_config_info *)dev->config;
  191. return config->flags & COUNTER_CONFIG_INFO_COUNT_UP;
  192. }
  193. /**
  194. * @brief Function to get number of alarm channels.
  195. *
  196. * @param[in] dev Pointer to the device structure for the driver instance.
  197. *
  198. * @return Number of alarm channels.
  199. */
  200. __syscall uint8_t counter_get_num_of_channels(const struct device *dev);
  201. static inline uint8_t z_impl_counter_get_num_of_channels(const struct device *dev)
  202. {
  203. const struct counter_config_info *config =
  204. (const struct counter_config_info *)dev->config;
  205. return config->channels;
  206. }
  207. /**
  208. * @brief Function to get counter frequency.
  209. *
  210. * @param[in] dev Pointer to the device structure for the driver instance.
  211. *
  212. * @return Frequency of the counter in Hz, or zero if the counter does
  213. * not have a fixed frequency.
  214. */
  215. __syscall uint32_t counter_get_frequency(const struct device *dev);
  216. static inline uint32_t z_impl_counter_get_frequency(const struct device *dev)
  217. {
  218. const struct counter_config_info *config =
  219. (const struct counter_config_info *)dev->config;
  220. return config->freq;
  221. }
  222. /**
  223. * @brief Function to convert microseconds to ticks.
  224. *
  225. * @param[in] dev Pointer to the device structure for the driver instance.
  226. * @param[in] us Microseconds.
  227. *
  228. * @return Converted ticks. Ticks will be saturated if exceed 32 bits.
  229. */
  230. __syscall uint32_t counter_us_to_ticks(const struct device *dev, uint64_t us);
  231. static inline uint32_t z_impl_counter_us_to_ticks(const struct device *dev,
  232. uint64_t us)
  233. {
  234. const struct counter_config_info *config =
  235. (const struct counter_config_info *)dev->config;
  236. uint64_t ticks = (us * config->freq) / USEC_PER_SEC;
  237. return (ticks > (uint64_t)UINT32_MAX) ? UINT32_MAX : ticks;
  238. }
  239. /**
  240. * @brief Function to convert ticks to microseconds.
  241. *
  242. * @param[in] dev Pointer to the device structure for the driver instance.
  243. * @param[in] ticks Ticks.
  244. *
  245. * @return Converted microseconds.
  246. */
  247. __syscall uint64_t counter_ticks_to_us(const struct device *dev, uint32_t ticks);
  248. static inline uint64_t z_impl_counter_ticks_to_us(const struct device *dev,
  249. uint32_t ticks)
  250. {
  251. const struct counter_config_info *config =
  252. (const struct counter_config_info *)dev->config;
  253. return ((uint64_t)ticks * USEC_PER_SEC) / config->freq;
  254. }
  255. /**
  256. * @brief Function to retrieve maximum top value that can be set.
  257. *
  258. * @param[in] dev Pointer to the device structure for the driver instance.
  259. *
  260. * @return Max top value.
  261. */
  262. __syscall uint32_t counter_get_max_top_value(const struct device *dev);
  263. static inline uint32_t z_impl_counter_get_max_top_value(const struct device *dev)
  264. {
  265. const struct counter_config_info *config =
  266. (const struct counter_config_info *)dev->config;
  267. return config->max_top_value;
  268. }
  269. /**
  270. * @brief Start counter device in free running mode.
  271. *
  272. * @param dev Pointer to the device structure for the driver instance.
  273. *
  274. * @retval 0 If successful.
  275. * @retval Negative errno code if failure.
  276. */
  277. __syscall int counter_start(const struct device *dev);
  278. static inline int z_impl_counter_start(const struct device *dev)
  279. {
  280. const struct counter_driver_api *api =
  281. (struct counter_driver_api *)dev->api;
  282. return api->start(dev);
  283. }
  284. /**
  285. * @brief Stop counter device.
  286. *
  287. * @param dev Pointer to the device structure for the driver instance.
  288. *
  289. * @retval 0 If successful.
  290. * @retval -ENOTSUP if the device doesn't support stopping the
  291. * counter.
  292. */
  293. __syscall int counter_stop(const struct device *dev);
  294. static inline int z_impl_counter_stop(const struct device *dev)
  295. {
  296. const struct counter_driver_api *api =
  297. (struct counter_driver_api *)dev->api;
  298. return api->stop(dev);
  299. }
  300. /**
  301. * @brief Get current counter value.
  302. * @param dev Pointer to the device structure for the driver instance.
  303. * @param ticks Pointer to where to store the current counter value
  304. *
  305. * @retval 0 If successful.
  306. * @retval Negative error code on failure getting the counter value
  307. */
  308. __syscall int counter_get_value(const struct device *dev, uint32_t *ticks);
  309. static inline int z_impl_counter_get_value(const struct device *dev,
  310. uint32_t *ticks)
  311. {
  312. const struct counter_driver_api *api =
  313. (struct counter_driver_api *)dev->api;
  314. return api->get_value(dev, ticks);
  315. }
  316. /**
  317. * @brief Set a single shot alarm on a channel.
  318. *
  319. * After expiration alarm can be set again, disabling is not needed. When alarm
  320. * expiration handler is called, channel is considered available and can be
  321. * set again in that context.
  322. *
  323. * @note API is not thread safe.
  324. *
  325. * @param dev Pointer to the device structure for the driver instance.
  326. * @param chan_id Channel ID.
  327. * @param alarm_cfg Alarm configuration.
  328. *
  329. * @retval 0 If successful.
  330. * @retval -ENOTSUP if request is not supported (device does not support
  331. * interrupts or requested channel).
  332. * @retval -EINVAL if alarm settings are invalid.
  333. * @retval -ETIME if absolute alarm was set too late.
  334. */
  335. __syscall int counter_set_channel_alarm(const struct device *dev,
  336. uint8_t chan_id,
  337. const struct counter_alarm_cfg *alarm_cfg);
  338. static inline int z_impl_counter_set_channel_alarm(const struct device *dev,
  339. uint8_t chan_id,
  340. const struct counter_alarm_cfg *alarm_cfg)
  341. {
  342. const struct counter_driver_api *api =
  343. (struct counter_driver_api *)dev->api;
  344. if (chan_id >= counter_get_num_of_channels(dev)) {
  345. return -ENOTSUP;
  346. }
  347. return api->set_alarm(dev, chan_id, alarm_cfg);
  348. }
  349. /**
  350. * @brief Cancel an alarm on a channel.
  351. *
  352. * @note API is not thread safe.
  353. *
  354. * @param dev Pointer to the device structure for the driver instance.
  355. * @param chan_id Channel ID.
  356. *
  357. * @retval 0 If successful.
  358. * @retval -ENOTSUP if request is not supported or the counter was not started
  359. * yet.
  360. */
  361. __syscall int counter_cancel_channel_alarm(const struct device *dev,
  362. uint8_t chan_id);
  363. static inline int z_impl_counter_cancel_channel_alarm(const struct device *dev,
  364. uint8_t chan_id)
  365. {
  366. const struct counter_driver_api *api =
  367. (struct counter_driver_api *)dev->api;
  368. if (chan_id >= counter_get_num_of_channels(dev)) {
  369. return -ENOTSUP;
  370. }
  371. return api->cancel_alarm(dev, chan_id);
  372. }
  373. /**
  374. * @brief Set counter top value.
  375. *
  376. * Function sets top value and optionally resets the counter to 0 or top value
  377. * depending on counter direction. On turnaround, counter can be reset and
  378. * optional callback is periodically called. Top value can only be changed when
  379. * there is no active channel alarm.
  380. *
  381. * @ref COUNTER_TOP_CFG_DONT_RESET prevents counter reset. When counter is
  382. * running while top value is updated, it is possible that counter progresses
  383. * outside the new top value. In that case, error is returned and optionally
  384. * driver can reset the counter (see @ref COUNTER_TOP_CFG_RESET_WHEN_LATE).
  385. *
  386. * @param dev Pointer to the device structure for the driver instance.
  387. * @param cfg Configuration. Cannot be NULL.
  388. *
  389. * @retval 0 If successful.
  390. * @retval -ENOTSUP if request is not supported (e.g. top value cannot be
  391. * changed or counter cannot/must be reset during top value
  392. update).
  393. * @retval -EBUSY if any alarm is active.
  394. * @retval -ETIME if @ref COUNTER_TOP_CFG_DONT_RESET was set and new top value
  395. * is smaller than current counter value (counter counting up).
  396. */
  397. __syscall int counter_set_top_value(const struct device *dev,
  398. const struct counter_top_cfg *cfg);
  399. static inline int z_impl_counter_set_top_value(const struct device *dev,
  400. const struct counter_top_cfg
  401. *cfg)
  402. {
  403. const struct counter_driver_api *api =
  404. (struct counter_driver_api *)dev->api;
  405. if (cfg->ticks > counter_get_max_top_value(dev)) {
  406. return -EINVAL;
  407. }
  408. return api->set_top_value(dev, cfg);
  409. }
  410. /**
  411. * @brief Function to get pending interrupts
  412. *
  413. * The purpose of this function is to return the interrupt
  414. * status register for the device.
  415. * This is especially useful when waking up from
  416. * low power states to check the wake up source.
  417. *
  418. * @param dev Pointer to the device structure for the driver instance.
  419. *
  420. * @retval 1 if any counter interrupt is pending.
  421. * @retval 0 if no counter interrupt is pending.
  422. */
  423. __syscall int counter_get_pending_int(const struct device *dev);
  424. static inline int z_impl_counter_get_pending_int(const struct device *dev)
  425. {
  426. const struct counter_driver_api *api =
  427. (struct counter_driver_api *)dev->api;
  428. return api->get_pending_int(dev);
  429. }
  430. /**
  431. * @brief Function to retrieve current top value.
  432. *
  433. * @param[in] dev Pointer to the device structure for the driver instance.
  434. *
  435. * @return Top value.
  436. */
  437. __syscall uint32_t counter_get_top_value(const struct device *dev);
  438. static inline uint32_t z_impl_counter_get_top_value(const struct device *dev)
  439. {
  440. const struct counter_driver_api *api =
  441. (struct counter_driver_api *)dev->api;
  442. return api->get_top_value(dev);
  443. }
  444. /**
  445. * @brief Set guard period in counter ticks.
  446. *
  447. * Setting non-zero guard period enables detection of setting absolute alarm
  448. * too late. It limits how far in the future absolute alarm can be set.
  449. *
  450. * Detection of too late setting is vital since if it is not detected alarm
  451. * is delayed by full period of the counter (up to 32 bits). Because of the
  452. * wrapping, it is impossible to distinguish alarm which is short in the past
  453. * from alarm which is targeted to expire after full counter period. In order to
  454. * detect too late setting, longest possible alarm is limited. Absolute value
  455. * cannot exceed: (now + top_value - guard_period) % top_value.
  456. *
  457. * Guard period depends on application and counter frequency. If it is expected
  458. * that absolute alarms setting might be delayed then guard period should
  459. * exceed maximal potential delay. If use case allows, guard period can be set
  460. * very high (e.g. half of the counter top value).
  461. *
  462. * After initialization guard period is set to 0 and late detection is disabled.
  463. *
  464. * @param dev Pointer to the device structure for the driver instance.
  465. * @param ticks Guard period in counter ticks.
  466. * @param flags See @ref COUNTER_GUARD_PERIOD_FLAGS.
  467. *
  468. * @retval 0 if successful.
  469. * @retval -ENOTSUP if function or flags are not supported.
  470. * @retval -EINVAL if ticks value is invalid.
  471. */
  472. __syscall int counter_set_guard_period(const struct device *dev,
  473. uint32_t ticks,
  474. uint32_t flags);
  475. static inline int z_impl_counter_set_guard_period(const struct device *dev,
  476. uint32_t ticks, uint32_t flags)
  477. {
  478. const struct counter_driver_api *api =
  479. (struct counter_driver_api *)dev->api;
  480. if (!api->set_guard_period) {
  481. return -ENOTSUP;
  482. }
  483. return api->set_guard_period(dev, ticks, flags);
  484. }
  485. /**
  486. * @brief Return guard period.
  487. *
  488. * See @ref counter_set_guard_period.
  489. *
  490. * @param dev Pointer to the device structure for the driver instance.
  491. * @param flags See @ref COUNTER_GUARD_PERIOD_FLAGS.
  492. *
  493. * @return Guard period given in counter ticks or 0 if function or flags are
  494. * not supported.
  495. */
  496. __syscall uint32_t counter_get_guard_period(const struct device *dev,
  497. uint32_t flags);
  498. static inline uint32_t z_impl_counter_get_guard_period(const struct device *dev,
  499. uint32_t flags)
  500. {
  501. const struct counter_driver_api *api =
  502. (struct counter_driver_api *)dev->api;
  503. return (api->get_guard_period) ? api->get_guard_period(dev, flags) : 0;
  504. }
  505. #ifdef __cplusplus
  506. }
  507. #endif
  508. /**
  509. * @}
  510. */
  511. #include <syscalls/counter.h>
  512. #endif /* ZEPHYR_INCLUDE_DRIVERS_COUNTER_H_ */