thread_timer.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. /*
  2. * Copyright (c) 2018 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Kernel thread timer support
  9. *
  10. * This module provides thread timer support.
  11. */
  12. #include <kernel.h>
  13. #include <kernel_structs.h>
  14. #include <thread_timer.h>
  15. #include <string.h>
  16. #undef THREAD_TIMER_DEBUG
  17. //#define CONFIG_THREAD_TIMER_DEBUG
  18. //#define THREAD_TIMER_DEBUG
  19. #ifdef THREAD_TIMER_DEBUG
  20. #include <sys/printk.h>
  21. #define TT_DEBUG(fmt, ...) printk("[%d] thread %p: " fmt, k_uptime_get_32(), \
  22. (void *)_current, ##__VA_ARGS__)
  23. #else
  24. #define TT_DEBUG(fmt, ...)
  25. #endif
  26. #define compare_time(a, b) ((int)((u32_t)(a) - (u32_t)(b)))
  27. #ifdef CONFIG_THREAD_TIMER_DEBUG
  28. static void _dump_thread_timer(struct thread_timer *ttimer)
  29. {
  30. printk("timer %p, prev: %p, next: %p\n",
  31. ttimer, ttimer->node.prev, ttimer->node.next);
  32. printk("\tthread: %p, period %d ms, delay %d ms\n",
  33. _current, ttimer->period, ttimer->duration);
  34. printk("\texpiry_time: %u, expiry_fn: %p, arg %p\n\n",
  35. ttimer->expiry_time,
  36. ttimer->expiry_fn, ttimer->expiry_fn_arg);
  37. }
  38. void _dump_thread_timer_q(void)
  39. {
  40. sys_dlist_t *thread_timer_q = &_current->thread_timer_q;
  41. struct thread_timer *ttimer;
  42. printk("thread: %p, thread_timer_q: %p, head: %p, tail: %p\n",
  43. _current, thread_timer_q, thread_timer_q->head,
  44. thread_timer_q->tail);
  45. SYS_DLIST_FOR_EACH_CONTAINER(thread_timer_q, ttimer, node) {
  46. _dump_thread_timer(ttimer);
  47. }
  48. }
  49. #endif
  50. static void _thread_timer_remove(struct k_thread *thread, struct thread_timer *ttimer)
  51. {
  52. struct thread_timer *in_q, *next;
  53. SYS_DLIST_FOR_EACH_CONTAINER_SAFE(&thread->thread_timer_q, in_q, next, node) {
  54. if (ttimer == in_q) {
  55. sys_dlist_remove(&in_q->node);
  56. return;
  57. }
  58. }
  59. }
  60. static void _thread_timer_insert(struct k_thread *thread, struct thread_timer *ttimer,
  61. u32_t expiry_time)
  62. {
  63. sys_dlist_t *thread_timer_q = &thread->thread_timer_q;
  64. struct thread_timer *in_q;
  65. SYS_DLIST_FOR_EACH_CONTAINER(thread_timer_q, in_q, node) {
  66. if (compare_time(ttimer->expiry_time, in_q->expiry_time) < 0) {
  67. sys_dlist_insert(&in_q->node, &ttimer->node);
  68. return;
  69. }
  70. }
  71. sys_dlist_append(thread_timer_q, &ttimer->node);
  72. }
  73. void thread_timer_init(struct thread_timer *ttimer, thread_timer_expiry_t expiry_fn,
  74. void *expiry_fn_arg)
  75. {
  76. __ASSERT(ttimer != NULL, "");
  77. if (!ttimer) {
  78. return;
  79. }
  80. TT_DEBUG("timer %p: init func %p arg 0x%x\n", ttimer,
  81. ttimer->expiry_fn, ttimer->expiry_fn_arg);
  82. /* remove thread timer if already submited */
  83. _thread_timer_remove(_current, ttimer);
  84. memset(ttimer, 0, sizeof(struct thread_timer));
  85. ttimer->expiry_fn = expiry_fn;
  86. ttimer->expiry_fn_arg = expiry_fn_arg;
  87. sys_dlist_init(&ttimer->node);
  88. }
  89. void thread_timer_start(struct thread_timer *ttimer, s32_t duration, s32_t period)
  90. {
  91. u32_t cur_time;
  92. if (!ttimer) {
  93. return;
  94. }
  95. __ASSERT((ttimer != NULL) && (ttimer->expiry_fn != NULL), "");
  96. _thread_timer_remove(_current, ttimer);
  97. cur_time = k_uptime_get_32();
  98. ttimer->expiry_time = cur_time + duration;
  99. ttimer->period = period;
  100. ttimer->duration = duration;
  101. TT_DEBUG("timer %p: start duration %d period %d, expiry_time %d\n",
  102. ttimer, duration, period, ttimer->expiry_time);
  103. _thread_timer_insert(_current, ttimer, ttimer->expiry_time);
  104. }
  105. void thread_timer_stop(struct thread_timer *ttimer)
  106. {
  107. __ASSERT(ttimer != NULL, "");
  108. TT_DEBUG("timer %p: stop\n", ttimer);
  109. _thread_timer_remove(_current, ttimer);
  110. }
  111. bool thread_timer_is_running(struct thread_timer *ttimer)
  112. {
  113. struct thread_timer *in_q;
  114. __ASSERT(ttimer != NULL, "");
  115. SYS_DLIST_FOR_EACH_CONTAINER(&_current->thread_timer_q, in_q, node) {
  116. if (ttimer == in_q)
  117. return true;
  118. }
  119. return false;
  120. }
  121. int thread_timer_next_timeout(void)
  122. {
  123. struct thread_timer *ttimer;
  124. int timeout;
  125. ttimer = SYS_DLIST_PEEK_HEAD_CONTAINER(&_current->thread_timer_q, ttimer, node);
  126. if (ttimer) {
  127. timeout = (int)((u32_t)ttimer->expiry_time - k_uptime_get_32());
  128. return (timeout < 0) ? 0 : timeout;
  129. }
  130. return -1;
  131. }
  132. void thread_timer_handle_expired(void)
  133. {
  134. struct thread_timer *ttimer, *next;
  135. u32_t cur_time;
  136. cur_time = k_uptime_get_32();
  137. SYS_DLIST_FOR_EACH_CONTAINER_SAFE(&_current->thread_timer_q, ttimer, next, node) {
  138. if (compare_time(ttimer->expiry_time, cur_time) > 0) {
  139. /* no expired thread timer */
  140. return;
  141. }
  142. /* remove this expiry thread timer */
  143. sys_dlist_remove(&ttimer->node);
  144. if (ttimer->expiry_fn) {
  145. /* resubmit this thread timer if it is a period timer */
  146. if (ttimer->period != 0) {
  147. thread_timer_start(ttimer, ttimer->period,
  148. ttimer->period);
  149. }
  150. TT_DEBUG("timer %p: call %p\n", ttimer, ttimer->expiry_fn);
  151. /* invoke thread timer expiry function */
  152. ttimer->expiry_fn(ttimer, ttimer->expiry_fn_arg);
  153. }
  154. }
  155. }