Chris@16
|
1 //
|
Chris@16
|
2 // detail/scoped_lock.hpp
|
Chris@16
|
3 // ~~~~~~~~~~~~~~~~~~~~~~
|
Chris@16
|
4 //
|
Chris@101
|
5 // Copyright (c) 2003-2015 Christopher M. Kohlhoff (chris at kohlhoff dot com)
|
Chris@16
|
6 //
|
Chris@16
|
7 // Distributed under the Boost Software License, Version 1.0. (See accompanying
|
Chris@16
|
8 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
9 //
|
Chris@16
|
10
|
Chris@16
|
11 #ifndef BOOST_ASIO_DETAIL_SCOPED_LOCK_HPP
|
Chris@16
|
12 #define BOOST_ASIO_DETAIL_SCOPED_LOCK_HPP
|
Chris@16
|
13
|
Chris@16
|
14 #if defined(_MSC_VER) && (_MSC_VER >= 1200)
|
Chris@16
|
15 # pragma once
|
Chris@16
|
16 #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
|
Chris@16
|
17
|
Chris@16
|
18 #include <boost/asio/detail/noncopyable.hpp>
|
Chris@16
|
19
|
Chris@16
|
20 #include <boost/asio/detail/push_options.hpp>
|
Chris@16
|
21
|
Chris@16
|
22 namespace boost {
|
Chris@16
|
23 namespace asio {
|
Chris@16
|
24 namespace detail {
|
Chris@16
|
25
|
Chris@16
|
26 // Helper class to lock and unlock a mutex automatically.
|
Chris@16
|
27 template <typename Mutex>
|
Chris@16
|
28 class scoped_lock
|
Chris@16
|
29 : private noncopyable
|
Chris@16
|
30 {
|
Chris@16
|
31 public:
|
Chris@16
|
32 // Tag type used to distinguish constructors.
|
Chris@16
|
33 enum adopt_lock_t { adopt_lock };
|
Chris@16
|
34
|
Chris@16
|
35 // Constructor adopts a lock that is already held.
|
Chris@16
|
36 scoped_lock(Mutex& m, adopt_lock_t)
|
Chris@16
|
37 : mutex_(m),
|
Chris@16
|
38 locked_(true)
|
Chris@16
|
39 {
|
Chris@16
|
40 }
|
Chris@16
|
41
|
Chris@16
|
42 // Constructor acquires the lock.
|
Chris@16
|
43 explicit scoped_lock(Mutex& m)
|
Chris@16
|
44 : mutex_(m)
|
Chris@16
|
45 {
|
Chris@16
|
46 mutex_.lock();
|
Chris@16
|
47 locked_ = true;
|
Chris@16
|
48 }
|
Chris@16
|
49
|
Chris@16
|
50 // Destructor releases the lock.
|
Chris@16
|
51 ~scoped_lock()
|
Chris@16
|
52 {
|
Chris@16
|
53 if (locked_)
|
Chris@16
|
54 mutex_.unlock();
|
Chris@16
|
55 }
|
Chris@16
|
56
|
Chris@16
|
57 // Explicitly acquire the lock.
|
Chris@16
|
58 void lock()
|
Chris@16
|
59 {
|
Chris@16
|
60 if (!locked_)
|
Chris@16
|
61 {
|
Chris@16
|
62 mutex_.lock();
|
Chris@16
|
63 locked_ = true;
|
Chris@16
|
64 }
|
Chris@16
|
65 }
|
Chris@16
|
66
|
Chris@16
|
67 // Explicitly release the lock.
|
Chris@16
|
68 void unlock()
|
Chris@16
|
69 {
|
Chris@16
|
70 if (locked_)
|
Chris@16
|
71 {
|
Chris@16
|
72 mutex_.unlock();
|
Chris@16
|
73 locked_ = false;
|
Chris@16
|
74 }
|
Chris@16
|
75 }
|
Chris@16
|
76
|
Chris@16
|
77 // Test whether the lock is held.
|
Chris@16
|
78 bool locked() const
|
Chris@16
|
79 {
|
Chris@16
|
80 return locked_;
|
Chris@16
|
81 }
|
Chris@16
|
82
|
Chris@16
|
83 // Get the underlying mutex.
|
Chris@16
|
84 Mutex& mutex()
|
Chris@16
|
85 {
|
Chris@16
|
86 return mutex_;
|
Chris@16
|
87 }
|
Chris@16
|
88
|
Chris@16
|
89 private:
|
Chris@16
|
90 // The underlying mutex.
|
Chris@16
|
91 Mutex& mutex_;
|
Chris@16
|
92
|
Chris@16
|
93 // Whether the mutex is currently locked or unlocked.
|
Chris@16
|
94 bool locked_;
|
Chris@16
|
95 };
|
Chris@16
|
96
|
Chris@16
|
97 } // namespace detail
|
Chris@16
|
98 } // namespace asio
|
Chris@16
|
99 } // namespace boost
|
Chris@16
|
100
|
Chris@16
|
101 #include <boost/asio/detail/pop_options.hpp>
|
Chris@16
|
102
|
Chris@16
|
103 #endif // BOOST_ASIO_DETAIL_SCOPED_LOCK_HPP
|