uClibc_mutex.h 6.4 KB

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