uClibc_mutex.h 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. /* Copyright (C) 2006 Manuel Novoa III <mjn3@codepoet.org>
  2. *
  3. * GNU Library General Public License (LGPL) version 2 or later.
  4. *
  5. * Dedicated to Toni. See uClibc/DEDICATION.mjn3 for details.
  6. */
  7. #ifndef _UCLIBC_MUTEX_H
  8. #define _UCLIBC_MUTEX_H
  9. #include <features.h>
  10. #ifdef __UCLIBC_HAS_THREADS__
  11. #include <pthread.h>
  12. #include <bits/libc-lock.h>
  13. #include <bits/uClibc_pthread.h>
  14. #define __uclibc_maybe_call(FUNC, ARGS) \
  15. (__extension__ ({ \
  16. __typeof (FUNC) *_fn = (FUNC); \
  17. if (_fn != NULL) { (*_fn) ARGS; } \
  18. }))
  19. #define __UCLIBC_MUTEX_TYPE pthread_mutex_t
  20. #define __UCLIBC_MUTEX(M) pthread_mutex_t M
  21. #define __UCLIBC_MUTEX_INIT(M,I) pthread_mutex_t M = I
  22. #define __UCLIBC_MUTEX_STATIC(M,I) static pthread_mutex_t M = I
  23. #define __UCLIBC_MUTEX_EXTERN(M) extern pthread_mutex_t M
  24. #define __UCLIBC_MUTEX_INIT_VAR(M) \
  25. __uclibc_maybe_call(__pthread_mutex_init,(&(M),NULL))
  26. #define __UCLIBC_MUTEX_LOCK_CANCEL_UNSAFE(M) \
  27. __uclibc_maybe_call(__pthread_mutex_lock,(&(M)))
  28. #define __UCLIBC_MUTEX_UNLOCK_CANCEL_UNSAFE(M) \
  29. __uclibc_maybe_call(__pthread_mutex_unlock,(&(M)))
  30. #define __UCLIBC_MUTEX_TRYLOCK_CANCEL_UNSAFE(M) \
  31. __uclibc_maybe_call(__pthread_mutex_trylock,(&(M)))
  32. #define __UCLIBC_MUTEX_CONDITIONAL_LOCK(M,C) \
  33. do { \
  34. struct _pthread_cleanup_buffer __infunc_pthread_cleanup_buffer; \
  35. int __infunc_need_locking = ((C) && (__pthread_mutex_lock != NULL)); \
  36. if (__infunc_need_locking) { \
  37. _pthread_cleanup_push_defer(&__infunc_pthread_cleanup_buffer, \
  38. (void (*) (void *))__pthread_mutex_unlock, \
  39. &(M)); \
  40. __pthread_mutex_lock(&(M)); \
  41. } \
  42. ((void)0)
  43. #define __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M,C) \
  44. if (__infunc_need_locking) { \
  45. _pthread_cleanup_pop_restore(&__infunc_pthread_cleanup_buffer,1); \
  46. } \
  47. } while (0)
  48. #define __UCLIBC_MUTEX_AUTO_LOCK_VAR(A) int A
  49. #define __UCLIBC_MUTEX_AUTO_LOCK(M,A,V) \
  50. __UCLIBC_MUTEX_CONDITIONAL_LOCK(M,((A=(V)) == 0))
  51. #define __UCLIBC_MUTEX_AUTO_UNLOCK(M,A) \
  52. __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M,(A == 0))
  53. #define __UCLIBC_MUTEX_LOCK(M) \
  54. __UCLIBC_MUTEX_CONDITIONAL_LOCK(M, 1)
  55. #define __UCLIBC_MUTEX_UNLOCK(M) \
  56. __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M, 1)
  57. #ifdef __USE_STDIO_FUTEXES__
  58. #include <bits/stdio-lock.h>
  59. #define __UCLIBC_IO_MUTEX(M) _IO_lock_t M
  60. #define __UCLIBC_IO_MUTEX_LOCK(M) _IO_lock_lock(M)
  61. #define __UCLIBC_IO_MUTEX_UNLOCK(M) _IO_lock_unlock(M)
  62. #define __UCLIBC_IO_MUTEX_TRYLOCK(M) _IO_lock_trylock(M)
  63. #define __UCLIBC_IO_MUTEX_INIT(M) _IO_lock_t M = _IO_lock_initializer
  64. #define __UCLIBC_IO_MUTEX_EXTERN(M) extern _IO_lock_t M
  65. #define __UCLIBC_IO_MUTEX_CONDITIONAL_LOCK(M,C) \
  66. if (C) { \
  67. _IO_lock_lock(M); \
  68. }
  69. #define __UCLIBC_IO_MUTEX_CONDITIONAL_UNLOCK(M,C) \
  70. if (C) { \
  71. _IO_lock_unlock(M); \
  72. }
  73. #define __UCLIBC_IO_MUTEX_AUTO_LOCK(M,A,V) \
  74. __UCLIBC_IO_MUTEX_CONDITIONAL_LOCK(M,((A=(V))) == 0)
  75. #define __UCLIBC_IO_MUTEX_AUTO_UNLOCK(M,A) \
  76. __UCLIBC_IO_MUTEX_CONDITIONAL_UNLOCK(M,((A) == 0))
  77. #define __UCLIBC_IO_MUTEX_LOCK_CANCEL_UNSAFE(M) _IO_lock_lock(M)
  78. #define __UCLIBC_IO_MUTEX_UNLOCK_CANCEL_UNSAFE(M) _IO_lock_unlock(M)
  79. #else /* of __USE_STDIO_FUTEXES__ */
  80. #define __UCLIBC_IO_MUTEX(M) __UCLIBC_MUTEX(M)
  81. #define __UCLIBC_IO_MUTEX_LOCK(M) __UCLIBC_MUTEX_CONDITIONAL_LOCK(M, 1)
  82. #define __UCLIBC_IO_MUTEX_UNLOCK(M) __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M, 1)
  83. #define __UCLIBC_IO_MUTEX_TRYLOCK(M) __UCLIBC_MUTEX_TRYLOCK_CANCEL_UNSAFE(M)
  84. #define __UCLIBC_IO_MUTEX_INIT(M) __UCLIBC_MUTEX_INIT(M, PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP)
  85. #define __UCLIBC_IO_MUTEX_EXTERN(M) __UCLIBC_MUTEX_EXTERN(M)
  86. #define __UCLIBC_IO_MUTEX_AUTO_LOCK(M,A,V) __UCLIBC_MUTEX_AUTO_LOCK(M,A,V)
  87. #define __UCLIBC_IO_MUTEX_AUTO_UNLOCK(M,A) __UCLIBC_MUTEX_AUTO_UNLOCK(M,A)
  88. #define __UCLIBC_IO_MUTEX_LOCK_CANCEL_UNSAFE(M) __UCLIBC_MUTEX_LOCK_CANCEL_UNSAFE(M)
  89. #define __UCLIBC_IO_MUTEX_UNLOCK_CANCEL_UNSAFE(M) __UCLIBC_MUTEX_UNLOCK_CANCEL_UNSAFE(M)
  90. #define __UCLIBC_IO_MUTEX_CONDITIONAL_LOCK(M,C) __UCLIBC_MUTEX_CONDITIONAL_LOCK(M, 1)
  91. #define __UCLIBC_IO_MUTEX_CONDITIONAL_UNLOCK(M,C) __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M, 1)
  92. #endif /* of __USE_STDIO_FUTEXES__ */
  93. #else /* of __UCLIBC_HAS_THREADS__ */
  94. #define __UCLIBC_MUTEX(M) void *__UCLIBC_MUTEX_DUMMY_ ## M
  95. #define __UCLIBC_MUTEX_INIT(M,I) extern void *__UCLIBC_MUTEX_DUMMY_ ## M
  96. #define __UCLIBC_MUTEX_STATIC(M,I) extern void *__UCLIBC_MUTEX_DUMMY_ ## M
  97. #define __UCLIBC_MUTEX_EXTERN(M) extern void *__UCLIBC_MUTEX_DUMMY_ ## M
  98. #define __UCLIBC_MUTEX_LOCK_CANCEL_UNSAFE(M) ((void)0)
  99. #define __UCLIBC_MUTEX_UNLOCK_CANCEL_UNSAFE(M) ((void)0)
  100. #define __UCLIBC_MUTEX_TRYLOCK_CANCEL_UNSAFE(M) (0) /* Always succeed? */
  101. #define __UCLIBC_MUTEX_CONDITIONAL_LOCK(M,C) ((void)0)
  102. #define __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M,C) ((void)0)
  103. #define __UCLIBC_MUTEX_AUTO_LOCK_VAR(A) ((void)0)
  104. #define __UCLIBC_MUTEX_AUTO_LOCK(M,A,V) ((void)0)
  105. #define __UCLIBC_MUTEX_AUTO_UNLOCK(M,A) ((void)0)
  106. #define __UCLIBC_MUTEX_LOCK(M) ((void)0)
  107. #define __UCLIBC_MUTEX_UNLOCK(M) ((void)0)
  108. #define __UCLIBC_IO_MUTEX(M) __UCLIBC_MUTEX(M)
  109. #define __UCLIBC_IO_MUTEX_LOCK(M) __UCLIBC_MUTEX_CONDITIONAL_LOCK(M, 1)
  110. #define __UCLIBC_IO_MUTEX_UNLOCK(M) __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M, 1)
  111. #define __UCLIBC_IO_MUTEX_TRYLOCK(M) __UCLIBC_MUTEX_TRYLOCK_CANCEL_UNSAFE(M)
  112. #define __UCLIBC_IO_MUTEX_INIT(M) __UCLIBC_MUTEX_INIT(M, PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP)
  113. #define __UCLIBC_IO_MUTEX_EXTERN(M) __UCLIBC_MUTEX_EXTERN(M)
  114. #define __UCLIBC_IO_MUTEX_AUTO_LOCK(M,A,V) __UCLIBC_MUTEX_AUTO_LOCK(M,A,V)
  115. #define __UCLIBC_IO_MUTEX_AUTO_UNLOCK(M,A) __UCLIBC_MUTEX_AUTO_UNLOCK(M,A)
  116. #define __UCLIBC_IO_MUTEX_LOCK_CANCEL_UNSAFE(M) __UCLIBC_MUTEX_LOCK_CANCEL_UNSAFE(M)
  117. #define __UCLIBC_IO_MUTEX_UNLOCK_CANCEL_UNSAFE(M) __UCLIBC_MUTEX_UNLOCK_CANCEL_UNSAFE(M)
  118. #define __UCLIBC_IO_MUTEX_CONDITIONAL_LOCK(M,C) __UCLIBC_MUTEX_CONDITIONAL_LOCK(M, 1)
  119. #define __UCLIBC_IO_MUTEX_CONDITIONAL_UNLOCK(M,C) __UCLIBC_MUTEX_CONDITIONAL_UNLOCK(M, 1)
  120. #endif /* of __UCLIBC_HAS_THREADS__ */
  121. #define __UCLIBC_IO_MUTEX_TRYLOCK_CANCEL_UNSAFE(M) \
  122. __UCLIBC_IO_MUTEX_TRYLOCK(M)
  123. #endif /* _UCLIBC_MUTEX_H */