lockable_traits.hpp 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. // Distributed under the Boost Software License, Version 1.0. (See
  2. // accompanying file LICENSE_1_0.txt or copy at
  3. // http://www.boost.org/LICENSE_1_0.txt)
  4. // (C) Copyright 2007 Anthony Williams
  5. // (C) Copyright 2011-2012 Vicente J. Botet Escriba
  6. #ifndef BOOST_THREAD_LOCKABLE_TRAITS_HPP
  7. #define BOOST_THREAD_LOCKABLE_TRAITS_HPP
  8. #include <boost/thread/detail/config.hpp>
  9. #include <boost/assert.hpp>
  10. #include <boost/detail/workaround.hpp>
  11. #include <boost/type_traits/is_class.hpp>
  12. #include <boost/config/abi_prefix.hpp>
  13. // todo make use of integral_constant, true_type and false_type
  14. namespace boost
  15. {
  16. namespace sync
  17. {
  18. #if defined(BOOST_NO_SFINAE) || \
  19. BOOST_WORKAROUND(__IBMCPP__, BOOST_TESTED_AT(600)) || \
  20. BOOST_WORKAROUND(__SUNPRO_CC, BOOST_TESTED_AT(0x590))
  21. #if ! defined BOOST_THREAD_NO_AUTO_DETECT_MUTEX_TYPES
  22. #define BOOST_THREAD_NO_AUTO_DETECT_MUTEX_TYPES
  23. #endif
  24. #endif
  25. #ifndef BOOST_THREAD_NO_AUTO_DETECT_MUTEX_TYPES
  26. namespace detail
  27. {
  28. #define BOOST_THREAD_DEFINE_HAS_MEMBER_CALLED(member_name) \
  29. template<typename T, bool=boost::is_class<T>::value> \
  30. struct has_member_called_##member_name \
  31. { \
  32. BOOST_STATIC_CONSTANT(bool, value=false); \
  33. }; \
  34. \
  35. template<typename T> \
  36. struct has_member_called_##member_name<T,true> \
  37. { \
  38. typedef char true_type; \
  39. struct false_type \
  40. { \
  41. true_type dummy[2]; \
  42. }; \
  43. \
  44. struct fallback { int member_name; }; \
  45. struct derived: \
  46. T, fallback \
  47. { \
  48. derived(); \
  49. }; \
  50. \
  51. template<int fallback::*> struct tester; \
  52. \
  53. template<typename U> \
  54. static false_type has_member(tester<&U::member_name>*); \
  55. template<typename U> \
  56. static true_type has_member(...); \
  57. \
  58. BOOST_STATIC_CONSTANT( \
  59. bool, value=sizeof(has_member<derived>(0))==sizeof(true_type)); \
  60. }
  61. BOOST_THREAD_DEFINE_HAS_MEMBER_CALLED(lock)
  62. ; BOOST_THREAD_DEFINE_HAS_MEMBER_CALLED(unlock);
  63. BOOST_THREAD_DEFINE_HAS_MEMBER_CALLED(try_lock);
  64. template<typename T,bool=has_member_called_lock<T>::value >
  65. struct has_member_lock
  66. {
  67. BOOST_STATIC_CONSTANT(bool, value=false);
  68. };
  69. template<typename T>
  70. struct has_member_lock<T,true>
  71. {
  72. typedef char true_type;
  73. struct false_type
  74. {
  75. true_type dummy[2];
  76. };
  77. template<typename U,typename V>
  78. static true_type has_member(V (U::*)());
  79. template<typename U>
  80. static false_type has_member(U);
  81. BOOST_STATIC_CONSTANT(
  82. bool,value=sizeof(has_member_lock<T>::has_member(&T::lock))==sizeof(true_type));
  83. };
  84. template<typename T,bool=has_member_called_unlock<T>::value >
  85. struct has_member_unlock
  86. {
  87. BOOST_STATIC_CONSTANT(bool, value=false);
  88. };
  89. template<typename T>
  90. struct has_member_unlock<T,true>
  91. {
  92. typedef char true_type;
  93. struct false_type
  94. {
  95. true_type dummy[2];
  96. };
  97. template<typename U,typename V>
  98. static true_type has_member(V (U::*)());
  99. template<typename U>
  100. static false_type has_member(U);
  101. BOOST_STATIC_CONSTANT(
  102. bool,value=sizeof(has_member_unlock<T>::has_member(&T::unlock))==sizeof(true_type));
  103. };
  104. template<typename T,bool=has_member_called_try_lock<T>::value >
  105. struct has_member_try_lock
  106. {
  107. BOOST_STATIC_CONSTANT(bool, value=false);
  108. };
  109. template<typename T>
  110. struct has_member_try_lock<T,true>
  111. {
  112. typedef char true_type;
  113. struct false_type
  114. {
  115. true_type dummy[2];
  116. };
  117. template<typename U>
  118. static true_type has_member(bool (U::*)());
  119. template<typename U>
  120. static false_type has_member(U);
  121. BOOST_STATIC_CONSTANT(
  122. bool,value=sizeof(has_member_try_lock<T>::has_member(&T::try_lock))==sizeof(true_type));
  123. };
  124. }
  125. template<typename T>
  126. struct is_basic_lockable
  127. {
  128. BOOST_STATIC_CONSTANT(bool, value = detail::has_member_lock<T>::value &&
  129. detail::has_member_unlock<T>::value);
  130. };
  131. template<typename T>
  132. struct is_lockable
  133. {
  134. BOOST_STATIC_CONSTANT(bool, value =
  135. is_basic_lockable<T>::value &&
  136. detail::has_member_try_lock<T>::value);
  137. };
  138. #else
  139. template<typename T>
  140. struct is_basic_lockable
  141. {
  142. BOOST_STATIC_CONSTANT(bool, value = false);
  143. };
  144. template<typename T>
  145. struct is_lockable
  146. {
  147. BOOST_STATIC_CONSTANT(bool, value = false);
  148. };
  149. #endif
  150. template<typename T>
  151. struct is_recursive_mutex_sur_parole
  152. {
  153. BOOST_STATIC_CONSTANT(bool, value = false);
  154. };
  155. template<typename T>
  156. struct is_recursive_mutex_sur_parolle : is_recursive_mutex_sur_parole<T>
  157. {
  158. };
  159. template<typename T>
  160. struct is_recursive_basic_lockable
  161. {
  162. BOOST_STATIC_CONSTANT(bool, value = is_basic_lockable<T>::value &&
  163. is_recursive_mutex_sur_parolle<T>::value);
  164. };
  165. template<typename T>
  166. struct is_recursive_lockable
  167. {
  168. BOOST_STATIC_CONSTANT(bool, value = is_lockable<T>::value &&
  169. is_recursive_mutex_sur_parolle<T>::value);
  170. };
  171. }
  172. template<typename T>
  173. struct is_mutex_type
  174. {
  175. BOOST_STATIC_CONSTANT(bool, value = sync::is_lockable<T>::value);
  176. };
  177. }
  178. #include <boost/config/abi_suffix.hpp>
  179. #endif