locks.hpp 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. /*
  2. * Copyright Andrey Semashev 2007 - 2015.
  3. * Distributed under the Boost Software License, Version 1.0.
  4. * (See accompanying file LICENSE_1_0.txt or copy at
  5. * http://www.boost.org/LICENSE_1_0.txt)
  6. */
  7. /*!
  8. * \file locks.hpp
  9. * \author Andrey Semashev
  10. * \date 30.05.2010
  11. *
  12. * \brief This header is the Boost.Log library implementation, see the library documentation
  13. * at http://www.boost.org/doc/libs/release/libs/log/doc/html/index.html.
  14. */
  15. #ifndef BOOST_LOG_DETAIL_LOCKS_HPP_INCLUDED_
  16. #define BOOST_LOG_DETAIL_LOCKS_HPP_INCLUDED_
  17. #include <boost/log/detail/config.hpp>
  18. #include <boost/log/detail/header.hpp>
  19. #ifdef BOOST_HAS_PRAGMA_ONCE
  20. #pragma once
  21. #endif
  22. namespace boost {
  23. #ifndef BOOST_LOG_NO_THREADS
  24. // Forward declaration of Boost.Thread locks. Specified here to avoid including Boost.Thread,
  25. // which would bring in many dependent headers, including a great deal of Boost.DateTime.
  26. template< typename >
  27. class lock_guard;
  28. template< typename >
  29. class shared_lock;
  30. template< typename >
  31. class upgrade_lock;
  32. template< typename >
  33. class unique_lock;
  34. template< typename >
  35. struct is_mutex_type;
  36. #endif // BOOST_LOG_NO_THREADS
  37. BOOST_LOG_OPEN_NAMESPACE
  38. //! An auxiliary pseudo-lock to express no locking requirements in logger features
  39. template< typename MutexT >
  40. class no_lock
  41. {
  42. public:
  43. /*!
  44. * Constructs the pseudo-lock. The mutex is not affected during the construction.
  45. */
  46. explicit no_lock(MutexT&) {}
  47. private:
  48. no_lock(no_lock const&);
  49. no_lock& operator= (no_lock const&);
  50. };
  51. namespace aux {
  52. #ifndef BOOST_LOG_NO_THREADS
  53. //! A trait to detect if the mutex supports exclusive locking
  54. template< typename MutexT >
  55. struct is_exclusively_lockable
  56. {
  57. typedef char true_type;
  58. struct false_type { char t[2]; };
  59. template< typename T >
  60. static true_type check_lockable(T*, void (T::*)() = &T::lock, void (T::*)() = &T::unlock);
  61. static false_type check_lockable(void*);
  62. enum value_t { value = sizeof(check_lockable((MutexT*)NULL)) == sizeof(true_type) };
  63. };
  64. //! A trait to detect if the mutex supports shared locking
  65. template< typename MutexT >
  66. struct is_shared_lockable
  67. {
  68. typedef char true_type;
  69. struct false_type { char t[2]; };
  70. template< typename T >
  71. static true_type check_shared_lockable(T*, void (T::*)() = &T::lock_shared, void (T::*)() = &T::unlock_shared);
  72. static false_type check_shared_lockable(void*);
  73. enum value_t { value = sizeof(check_shared_lockable((MutexT*)NULL)) == sizeof(true_type) };
  74. };
  75. //! An analogue to the minimalistic \c lock_guard template. Defined here to avoid including Boost.Thread.
  76. template< typename MutexT >
  77. struct exclusive_lock_guard
  78. {
  79. explicit exclusive_lock_guard(MutexT& m) : m_Mutex(m)
  80. {
  81. m.lock();
  82. }
  83. ~exclusive_lock_guard()
  84. {
  85. m_Mutex.unlock();
  86. }
  87. private:
  88. exclusive_lock_guard(exclusive_lock_guard const&);
  89. exclusive_lock_guard& operator= (exclusive_lock_guard const&);
  90. private:
  91. MutexT& m_Mutex;
  92. };
  93. //! An analogue to the minimalistic \c lock_guard template that locks \c shared_mutex with shared ownership.
  94. template< typename MutexT >
  95. struct shared_lock_guard
  96. {
  97. explicit shared_lock_guard(MutexT& m) : m_Mutex(m)
  98. {
  99. m.lock_shared();
  100. }
  101. ~shared_lock_guard()
  102. {
  103. m_Mutex.unlock_shared();
  104. }
  105. private:
  106. shared_lock_guard(shared_lock_guard const&);
  107. shared_lock_guard& operator= (shared_lock_guard const&);
  108. private:
  109. MutexT& m_Mutex;
  110. };
  111. //! A deadlock-safe lock type that exclusively locks two mutexes
  112. template< typename MutexT1, typename MutexT2 >
  113. class multiple_unique_lock2
  114. {
  115. public:
  116. multiple_unique_lock2(MutexT1& m1, MutexT2& m2) :
  117. m_p1(&m1),
  118. m_p2(&m2)
  119. {
  120. // Yes, it's not conforming, but it works
  121. // and it doesn't require to #include <functional>
  122. if (static_cast< void* >(m_p1) < static_cast< void* >(m_p2))
  123. {
  124. m_p1->lock();
  125. m_p2->lock();
  126. }
  127. else
  128. {
  129. m_p2->lock();
  130. m_p1->lock();
  131. }
  132. }
  133. ~multiple_unique_lock2()
  134. {
  135. m_p2->unlock();
  136. m_p1->unlock();
  137. }
  138. private:
  139. MutexT1* m_p1;
  140. MutexT2* m_p2;
  141. };
  142. #endif // BOOST_LOG_NO_THREADS
  143. } // namespace aux
  144. BOOST_LOG_CLOSE_NAMESPACE // namespace log
  145. } // namespace boost
  146. #include <boost/log/detail/footer.hpp>
  147. #endif // BOOST_LOG_DETAIL_LOCKS_HPP_INCLUDED_