timer.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. /*
  2. * Waitable timers management
  3. *
  4. * Copyright (C) 1999 Alexandre Julliard
  5. *
  6. * This library is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 2.1 of the License, or (at your option) any later version.
  10. *
  11. * This library is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public
  17. * License along with this library; if not, write to the Free Software
  18. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  19. */
  20. #include "config.h"
  21. #include "wine/port.h"
  22. #include <assert.h>
  23. #include <stdio.h>
  24. #include <stdlib.h>
  25. #include <sys/time.h>
  26. #include <sys/types.h>
  27. #include "windef.h"
  28. #include "file.h"
  29. #include "handle.h"
  30. #include "request.h"
  31. struct timer
  32. {
  33. struct object obj; /* object header */
  34. int manual; /* manual reset */
  35. int signaled; /* current signaled state */
  36. int period; /* timer period in ms */
  37. struct timeval when; /* next expiration */
  38. struct timeout_user *timeout; /* timeout user */
  39. struct thread *thread; /* thread that set the APC function */
  40. void *callback; /* callback APC function */
  41. void *arg; /* callback argument */
  42. };
  43. static void timer_dump( struct object *obj, int verbose );
  44. static int timer_signaled( struct object *obj, struct thread *thread );
  45. static int timer_satisfied( struct object *obj, struct thread *thread );
  46. static void timer_destroy( struct object *obj );
  47. static const struct object_ops timer_ops =
  48. {
  49. sizeof(struct timer), /* size */
  50. timer_dump, /* dump */
  51. add_queue, /* add_queue */
  52. remove_queue, /* remove_queue */
  53. timer_signaled, /* signaled */
  54. timer_satisfied, /* satisfied */
  55. no_get_fd, /* get_fd */
  56. timer_destroy /* destroy */
  57. };
  58. /* create a timer object */
  59. static struct timer *create_timer( const WCHAR *name, size_t len, int manual )
  60. {
  61. struct timer *timer;
  62. if ((timer = create_named_object( sync_namespace, &timer_ops, name, len )))
  63. {
  64. if (get_error() != STATUS_OBJECT_NAME_COLLISION)
  65. {
  66. /* initialize it if it didn't already exist */
  67. timer->manual = manual;
  68. timer->signaled = 0;
  69. timer->when.tv_sec = 0;
  70. timer->when.tv_usec = 0;
  71. timer->period = 0;
  72. timer->timeout = NULL;
  73. timer->thread = NULL;
  74. }
  75. }
  76. return timer;
  77. }
  78. /* callback on timer expiration */
  79. static void timer_callback( void *private )
  80. {
  81. struct timer *timer = (struct timer *)private;
  82. /* queue an APC */
  83. if (timer->thread)
  84. {
  85. if (!thread_queue_apc( timer->thread, &timer->obj, timer->callback, APC_TIMER, 0,
  86. (void *)timer->when.tv_sec, (void *)timer->when.tv_usec, timer->arg))
  87. {
  88. release_object( timer->thread );
  89. timer->thread = NULL;
  90. }
  91. }
  92. if (timer->period) /* schedule the next expiration */
  93. {
  94. add_timeout( &timer->when, timer->period );
  95. timer->timeout = add_timeout_user( &timer->when, timer_callback, timer );
  96. }
  97. else timer->timeout = NULL;
  98. /* wake up waiters */
  99. timer->signaled = 1;
  100. wake_up( &timer->obj, 0 );
  101. }
  102. /* cancel a running timer */
  103. static int cancel_timer( struct timer *timer )
  104. {
  105. int signaled = timer->signaled;
  106. if (timer->timeout)
  107. {
  108. remove_timeout_user( timer->timeout );
  109. timer->timeout = NULL;
  110. }
  111. if (timer->thread)
  112. {
  113. thread_cancel_apc( timer->thread, &timer->obj, 0 );
  114. release_object( timer->thread );
  115. timer->thread = NULL;
  116. }
  117. return signaled;
  118. }
  119. /* set the timer expiration and period */
  120. static int set_timer( struct timer *timer, const abs_time_t *expire, int period,
  121. void *callback, void *arg )
  122. {
  123. int signaled = cancel_timer( timer );
  124. if (timer->manual)
  125. {
  126. period = 0; /* period doesn't make any sense for a manual timer */
  127. timer->signaled = 0;
  128. }
  129. if (!expire->sec && !expire->usec)
  130. {
  131. /* special case: use now + period as first expiration */
  132. gettimeofday( &timer->when, 0 );
  133. add_timeout( &timer->when, period );
  134. }
  135. else
  136. {
  137. timer->when.tv_sec = expire->sec;
  138. timer->when.tv_usec = expire->usec;
  139. }
  140. timer->period = period;
  141. timer->callback = callback;
  142. timer->arg = arg;
  143. if (callback) timer->thread = (struct thread *)grab_object( current );
  144. timer->timeout = add_timeout_user( &timer->when, timer_callback, timer );
  145. return signaled;
  146. }
  147. static void timer_dump( struct object *obj, int verbose )
  148. {
  149. struct timer *timer = (struct timer *)obj;
  150. assert( obj->ops == &timer_ops );
  151. fprintf( stderr, "Timer manual=%d when=%ld.%06ld period=%d ",
  152. timer->manual, timer->when.tv_sec, timer->when.tv_usec, timer->period );
  153. dump_object_name( &timer->obj );
  154. fputc( '\n', stderr );
  155. }
  156. static int timer_signaled( struct object *obj, struct thread *thread )
  157. {
  158. struct timer *timer = (struct timer *)obj;
  159. assert( obj->ops == &timer_ops );
  160. return timer->signaled;
  161. }
  162. static int timer_satisfied( struct object *obj, struct thread *thread )
  163. {
  164. struct timer *timer = (struct timer *)obj;
  165. assert( obj->ops == &timer_ops );
  166. if (!timer->manual) timer->signaled = 0;
  167. return 0;
  168. }
  169. static void timer_destroy( struct object *obj )
  170. {
  171. struct timer *timer = (struct timer *)obj;
  172. assert( obj->ops == &timer_ops );
  173. if (timer->timeout) remove_timeout_user( timer->timeout );
  174. if (timer->thread) release_object( timer->thread );
  175. }
  176. /* create a timer */
  177. DECL_HANDLER(create_timer)
  178. {
  179. struct timer *timer;
  180. reply->handle = 0;
  181. if ((timer = create_timer( get_req_data(), get_req_data_size(), req->manual )))
  182. {
  183. reply->handle = alloc_handle( current->process, timer, req->access, req->inherit );
  184. release_object( timer );
  185. }
  186. }
  187. /* open a handle to a timer */
  188. DECL_HANDLER(open_timer)
  189. {
  190. reply->handle = open_object( sync_namespace, get_req_data(), get_req_data_size(),
  191. &timer_ops, req->access, req->inherit );
  192. }
  193. /* set a waitable timer */
  194. DECL_HANDLER(set_timer)
  195. {
  196. struct timer *timer;
  197. if ((timer = (struct timer *)get_handle_obj( current->process, req->handle,
  198. TIMER_MODIFY_STATE, &timer_ops )))
  199. {
  200. reply->signaled = set_timer( timer, &req->expire, req->period, req->callback, req->arg );
  201. release_object( timer );
  202. }
  203. }
  204. /* cancel a waitable timer */
  205. DECL_HANDLER(cancel_timer)
  206. {
  207. struct timer *timer;
  208. if ((timer = (struct timer *)get_handle_obj( current->process, req->handle,
  209. TIMER_MODIFY_STATE, &timer_ops )))
  210. {
  211. reply->signaled = cancel_timer( timer );
  212. release_object( timer );
  213. }
  214. }
  215. /* Get information on a waitable timer */
  216. DECL_HANDLER(get_timer_info)
  217. {
  218. struct timer *timer;
  219. if ((timer = (struct timer *)get_handle_obj( current->process, req->handle,
  220. TIMER_QUERY_STATE, &timer_ops )))
  221. {
  222. reply->when.sec = timer->when.tv_sec;
  223. reply->when.usec = timer->when.tv_usec;
  224. reply->signaled = timer->signaled;
  225. release_object( timer );
  226. }
  227. }