timer.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <errno.h>
  8. #include <string.h>
  9. #include <sys/printk.h>
  10. #include <posix/time.h>
  11. #define ACTIVE 1
  12. #define NOT_ACTIVE 0
  13. static void zephyr_timer_wrapper(struct k_timer *ztimer);
  14. struct timer_obj {
  15. struct k_timer ztimer;
  16. void (*sigev_notify_function)(sigval val);
  17. sigval val;
  18. struct timespec interval; /* Reload value */
  19. uint32_t reload; /* Reload value in ms */
  20. uint32_t status;
  21. };
  22. K_MEM_SLAB_DEFINE(posix_timer_slab, sizeof(struct timer_obj),
  23. CONFIG_MAX_TIMER_COUNT, 4);
  24. static void zephyr_timer_wrapper(struct k_timer *ztimer)
  25. {
  26. struct timer_obj *timer;
  27. timer = (struct timer_obj *)ztimer;
  28. if (timer->reload == 0U) {
  29. timer->status = NOT_ACTIVE;
  30. }
  31. (timer->sigev_notify_function)(timer->val);
  32. }
  33. /**
  34. * @brief Create a per-process timer.
  35. *
  36. * This API does not accept SIGEV_THREAD as valid signal event notification
  37. * type.
  38. *
  39. * See IEEE 1003.1
  40. */
  41. int timer_create(clockid_t clockid, struct sigevent *evp, timer_t *timerid)
  42. {
  43. struct timer_obj *timer;
  44. if (clockid != CLOCK_MONOTONIC || evp == NULL ||
  45. (evp->sigev_notify != SIGEV_NONE &&
  46. evp->sigev_notify != SIGEV_SIGNAL)) {
  47. errno = EINVAL;
  48. return -1;
  49. }
  50. if (k_mem_slab_alloc(&posix_timer_slab, (void **)&timer, K_MSEC(100)) == 0) {
  51. (void)memset(timer, 0, sizeof(struct timer_obj));
  52. } else {
  53. errno = ENOMEM;
  54. return -1;
  55. }
  56. timer->sigev_notify_function = evp->sigev_notify_function;
  57. timer->val = evp->sigev_value;
  58. timer->interval.tv_sec = 0;
  59. timer->interval.tv_nsec = 0;
  60. timer->reload = 0U;
  61. timer->status = NOT_ACTIVE;
  62. if (evp->sigev_notify == SIGEV_NONE) {
  63. k_timer_init(&timer->ztimer, NULL, NULL);
  64. } else {
  65. k_timer_init(&timer->ztimer, zephyr_timer_wrapper, NULL);
  66. }
  67. *timerid = (timer_t)timer;
  68. return 0;
  69. }
  70. /**
  71. * @brief Get amount of time left for expiration on a per-process timer.
  72. *
  73. * See IEEE 1003.1
  74. */
  75. int timer_gettime(timer_t timerid, struct itimerspec *its)
  76. {
  77. struct timer_obj *timer = (struct timer_obj *)timerid;
  78. int32_t remaining, leftover;
  79. int64_t nsecs, secs;
  80. if (timer == NULL) {
  81. errno = EINVAL;
  82. return -1;
  83. }
  84. if (timer->status == ACTIVE) {
  85. remaining = k_timer_remaining_get(&timer->ztimer);
  86. secs = remaining / MSEC_PER_SEC;
  87. leftover = remaining - (secs * MSEC_PER_SEC);
  88. nsecs = (int64_t)leftover * NSEC_PER_MSEC;
  89. its->it_value.tv_sec = (int32_t) secs;
  90. its->it_value.tv_nsec = (int32_t) nsecs;
  91. } else {
  92. /* Timer is disarmed */
  93. its->it_value.tv_sec = 0;
  94. its->it_value.tv_nsec = 0;
  95. }
  96. /* The interval last set by timer_settime() */
  97. its->it_interval = timer->interval;
  98. return 0;
  99. }
  100. /**
  101. * @brief Sets expiration time of per-process timer.
  102. *
  103. * See IEEE 1003.1
  104. */
  105. int timer_settime(timer_t timerid, int flags, const struct itimerspec *value,
  106. struct itimerspec *ovalue)
  107. {
  108. struct timer_obj *timer = (struct timer_obj *) timerid;
  109. uint32_t duration, current;
  110. if (timer == NULL ||
  111. value->it_interval.tv_nsec < 0 ||
  112. value->it_interval.tv_nsec >= NSEC_PER_SEC ||
  113. value->it_value.tv_nsec < 0 ||
  114. value->it_value.tv_nsec >= NSEC_PER_SEC) {
  115. errno = EINVAL;
  116. return -1;
  117. }
  118. /* Save time to expire and old reload value. */
  119. if (ovalue != NULL) {
  120. timer_gettime(timerid, ovalue);
  121. }
  122. /* Stop the timer if the value is 0 */
  123. if ((value->it_value.tv_sec == 0) && (value->it_value.tv_nsec == 0)) {
  124. if (timer->status == ACTIVE) {
  125. k_timer_stop(&timer->ztimer);
  126. }
  127. timer->status = NOT_ACTIVE;
  128. return 0;
  129. }
  130. /* Calculate timer period */
  131. timer->reload = _ts_to_ms(&value->it_interval);
  132. timer->interval.tv_sec = value->it_interval.tv_sec;
  133. timer->interval.tv_nsec = value->it_interval.tv_nsec;
  134. /* Calcaulte timer duration */
  135. duration = _ts_to_ms(&(value->it_value));
  136. if ((flags & TIMER_ABSTIME) != 0) {
  137. current = k_timer_remaining_get(&timer->ztimer);
  138. if (current >= duration) {
  139. duration = 0U;
  140. } else {
  141. duration -= current;
  142. }
  143. }
  144. if (timer->status == ACTIVE) {
  145. k_timer_stop(&timer->ztimer);
  146. }
  147. timer->status = ACTIVE;
  148. k_timer_start(&timer->ztimer, K_MSEC(duration), K_MSEC(timer->reload));
  149. return 0;
  150. }
  151. /**
  152. * @brief Delete a per-process timer.
  153. *
  154. * See IEEE 1003.1
  155. */
  156. int timer_delete(timer_t timerid)
  157. {
  158. struct timer_obj *timer = (struct timer_obj *) timerid;
  159. if (timer == NULL) {
  160. errno = EINVAL;
  161. return -1;
  162. }
  163. if (timer->status == ACTIVE) {
  164. timer->status = NOT_ACTIVE;
  165. k_timer_stop(&timer->ztimer);
  166. }
  167. k_mem_slab_free(&posix_timer_slab, (void *) &timer);
  168. return 0;
  169. }