mutex.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. /*
  2. * Server-side mutex management
  3. *
  4. * Copyright (C) 1998 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 "windef.h"
  26. #include "handle.h"
  27. #include "thread.h"
  28. #include "request.h"
  29. struct mutex
  30. {
  31. struct object obj; /* object header */
  32. struct thread *owner; /* mutex owner */
  33. unsigned int count; /* recursion count */
  34. int abandoned; /* has it been abandoned? */
  35. struct mutex *next;
  36. struct mutex *prev;
  37. };
  38. static void mutex_dump( struct object *obj, int verbose );
  39. static int mutex_signaled( struct object *obj, struct thread *thread );
  40. static int mutex_satisfied( struct object *obj, struct thread *thread );
  41. static void mutex_destroy( struct object *obj );
  42. static const struct object_ops mutex_ops =
  43. {
  44. sizeof(struct mutex), /* size */
  45. mutex_dump, /* dump */
  46. add_queue, /* add_queue */
  47. remove_queue, /* remove_queue */
  48. mutex_signaled, /* signaled */
  49. mutex_satisfied, /* satisfied */
  50. no_get_fd, /* get_fd */
  51. mutex_destroy /* destroy */
  52. };
  53. static struct mutex *create_mutex( const WCHAR *name, size_t len, int owned )
  54. {
  55. struct mutex *mutex;
  56. if ((mutex = create_named_object( sync_namespace, &mutex_ops, name, len )))
  57. {
  58. if (get_error() != STATUS_OBJECT_NAME_COLLISION)
  59. {
  60. /* initialize it if it didn't already exist */
  61. mutex->count = 0;
  62. mutex->owner = NULL;
  63. mutex->abandoned = 0;
  64. mutex->next = mutex->prev = NULL;
  65. if (owned) mutex_satisfied( &mutex->obj, current );
  66. }
  67. }
  68. return mutex;
  69. }
  70. /* release a mutex once the recursion count is 0 */
  71. static void do_release( struct mutex *mutex )
  72. {
  73. assert( !mutex->count );
  74. /* remove the mutex from the thread list of owned mutexes */
  75. if (mutex->next) mutex->next->prev = mutex->prev;
  76. if (mutex->prev) mutex->prev->next = mutex->next;
  77. else mutex->owner->mutex = mutex->next;
  78. mutex->owner = NULL;
  79. mutex->next = mutex->prev = NULL;
  80. wake_up( &mutex->obj, 0 );
  81. }
  82. void abandon_mutexes( struct thread *thread )
  83. {
  84. while (thread->mutex)
  85. {
  86. struct mutex *mutex = thread->mutex;
  87. assert( mutex->owner == thread );
  88. mutex->count = 0;
  89. mutex->abandoned = 1;
  90. do_release( mutex );
  91. }
  92. }
  93. static void mutex_dump( struct object *obj, int verbose )
  94. {
  95. struct mutex *mutex = (struct mutex *)obj;
  96. assert( obj->ops == &mutex_ops );
  97. fprintf( stderr, "Mutex count=%u owner=%p ", mutex->count, mutex->owner );
  98. dump_object_name( &mutex->obj );
  99. fputc( '\n', stderr );
  100. }
  101. static int mutex_signaled( struct object *obj, struct thread *thread )
  102. {
  103. struct mutex *mutex = (struct mutex *)obj;
  104. assert( obj->ops == &mutex_ops );
  105. return (!mutex->count || (mutex->owner == thread));
  106. }
  107. static int mutex_satisfied( struct object *obj, struct thread *thread )
  108. {
  109. struct mutex *mutex = (struct mutex *)obj;
  110. assert( obj->ops == &mutex_ops );
  111. assert( !mutex->count || (mutex->owner == thread) );
  112. if (!mutex->count++) /* FIXME: avoid wrap-around */
  113. {
  114. assert( !mutex->owner );
  115. mutex->owner = thread;
  116. mutex->prev = NULL;
  117. if ((mutex->next = thread->mutex)) mutex->next->prev = mutex;
  118. thread->mutex = mutex;
  119. }
  120. if (!mutex->abandoned) return 0;
  121. mutex->abandoned = 0;
  122. return 1;
  123. }
  124. static void mutex_destroy( struct object *obj )
  125. {
  126. struct mutex *mutex = (struct mutex *)obj;
  127. assert( obj->ops == &mutex_ops );
  128. if (!mutex->count) return;
  129. mutex->count = 0;
  130. do_release( mutex );
  131. }
  132. /* create a mutex */
  133. DECL_HANDLER(create_mutex)
  134. {
  135. struct mutex *mutex;
  136. reply->handle = 0;
  137. if ((mutex = create_mutex( get_req_data(), get_req_data_size(), req->owned )))
  138. {
  139. reply->handle = alloc_handle( current->process, mutex, req->access, req->inherit );
  140. release_object( mutex );
  141. }
  142. }
  143. /* open a handle to a mutex */
  144. DECL_HANDLER(open_mutex)
  145. {
  146. reply->handle = open_object( sync_namespace, get_req_data(), get_req_data_size(),
  147. &mutex_ops, req->access, req->inherit );
  148. }
  149. /* release a mutex */
  150. DECL_HANDLER(release_mutex)
  151. {
  152. struct mutex *mutex;
  153. if ((mutex = (struct mutex *)get_handle_obj( current->process, req->handle,
  154. MUTEX_MODIFY_STATE, &mutex_ops )))
  155. {
  156. if (!mutex->count || (mutex->owner != current)) set_error( STATUS_MUTANT_NOT_OWNED );
  157. else
  158. {
  159. reply->prev_count = mutex->count;
  160. if (!--mutex->count) do_release( mutex );
  161. }
  162. release_object( mutex );
  163. }
  164. }