blob: e56271d308e6f53f5a3dfe406654be6842e0b2c9 [file] [log] [blame]
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00001//===------------------------- mutex.cpp ----------------------------------===//
2//
Howard Hinnantf5256e12010-05-11 21:36:01 +00003// The LLVM Compiler Infrastructure
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00004//
Howard Hinnantb64f8b02010-11-16 22:09:02 +00005// This file is dual licensed under the MIT and the University of Illinois Open
6// Source Licenses. See LICENSE.TXT for details.
Howard Hinnantbc8d3f92010-05-11 19:42:16 +00007//
8//===----------------------------------------------------------------------===//
9
Howard Hinnant499c61f2012-07-21 16:13:09 +000010#define _LIBCPP_BUILDING_MUTEX
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000011#include "mutex"
12#include "limits"
13#include "system_error"
14#include "cassert"
15
16_LIBCPP_BEGIN_NAMESPACE_STD
Jonathan Roelofs8d86b2e2014-09-05 19:45:05 +000017#ifndef _LIBCPP_HAS_NO_THREADS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000018
19const defer_lock_t defer_lock = {};
20const try_to_lock_t try_to_lock = {};
21const adopt_lock_t adopt_lock = {};
22
23mutex::~mutex()
24{
Howard Hinnantec3773c2011-12-01 20:21:04 +000025 pthread_mutex_destroy(&__m_);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000026}
27
28void
29mutex::lock()
30{
31 int ec = pthread_mutex_lock(&__m_);
32 if (ec)
33 __throw_system_error(ec, "mutex lock failed");
34}
35
36bool
Howard Hinnant499c61f2012-07-21 16:13:09 +000037mutex::try_lock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000038{
39 return pthread_mutex_trylock(&__m_) == 0;
40}
41
42void
Howard Hinnant499c61f2012-07-21 16:13:09 +000043mutex::unlock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000044{
45 int ec = pthread_mutex_unlock(&__m_);
Howard Hinnantfc910cb2013-09-21 21:26:37 +000046 (void)ec;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000047 assert(ec == 0);
48}
49
50// recursive_mutex
51
52recursive_mutex::recursive_mutex()
53{
54 pthread_mutexattr_t attr;
55 int ec = pthread_mutexattr_init(&attr);
56 if (ec)
57 goto fail;
58 ec = pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
59 if (ec)
60 {
61 pthread_mutexattr_destroy(&attr);
62 goto fail;
63 }
64 ec = pthread_mutex_init(&__m_, &attr);
65 if (ec)
66 {
67 pthread_mutexattr_destroy(&attr);
68 goto fail;
69 }
70 ec = pthread_mutexattr_destroy(&attr);
71 if (ec)
72 {
73 pthread_mutex_destroy(&__m_);
74 goto fail;
75 }
76 return;
77fail:
78 __throw_system_error(ec, "recursive_mutex constructor failed");
79}
80
81recursive_mutex::~recursive_mutex()
82{
83 int e = pthread_mutex_destroy(&__m_);
Howard Hinnantfc910cb2013-09-21 21:26:37 +000084 (void)e;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000085 assert(e == 0);
86}
87
88void
89recursive_mutex::lock()
90{
91 int ec = pthread_mutex_lock(&__m_);
92 if (ec)
93 __throw_system_error(ec, "recursive_mutex lock failed");
94}
95
96void
Howard Hinnant499c61f2012-07-21 16:13:09 +000097recursive_mutex::unlock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +000098{
99 int e = pthread_mutex_unlock(&__m_);
Howard Hinnantfc910cb2013-09-21 21:26:37 +0000100 (void)e;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000101 assert(e == 0);
102}
103
104bool
Howard Hinnant499c61f2012-07-21 16:13:09 +0000105recursive_mutex::try_lock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000106{
107 return pthread_mutex_trylock(&__m_) == 0;
108}
109
110// timed_mutex
111
112timed_mutex::timed_mutex()
113 : __locked_(false)
114{
115}
116
117timed_mutex::~timed_mutex()
118{
119 lock_guard<mutex> _(__m_);
120}
121
122void
123timed_mutex::lock()
124{
125 unique_lock<mutex> lk(__m_);
126 while (__locked_)
127 __cv_.wait(lk);
128 __locked_ = true;
129}
130
131bool
Howard Hinnant499c61f2012-07-21 16:13:09 +0000132timed_mutex::try_lock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000133{
134 unique_lock<mutex> lk(__m_, try_to_lock);
135 if (lk.owns_lock() && !__locked_)
136 {
137 __locked_ = true;
138 return true;
139 }
140 return false;
141}
142
143void
Howard Hinnant499c61f2012-07-21 16:13:09 +0000144timed_mutex::unlock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000145{
146 lock_guard<mutex> _(__m_);
147 __locked_ = false;
148 __cv_.notify_one();
149}
150
151// recursive_timed_mutex
152
153recursive_timed_mutex::recursive_timed_mutex()
154 : __count_(0),
Howard Hinnantadff4892010-05-24 17:49:41 +0000155 __id_(0)
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000156{
157}
158
159recursive_timed_mutex::~recursive_timed_mutex()
160{
161 lock_guard<mutex> _(__m_);
162}
163
164void
165recursive_timed_mutex::lock()
166{
167 pthread_t id = pthread_self();
168 unique_lock<mutex> lk(__m_);
169 if (pthread_equal(id, __id_))
170 {
171 if (__count_ == numeric_limits<size_t>::max())
172 __throw_system_error(EAGAIN, "recursive_timed_mutex lock limit reached");
173 ++__count_;
174 return;
175 }
176 while (__count_ != 0)
177 __cv_.wait(lk);
178 __count_ = 1;
179 __id_ = id;
180}
181
182bool
Howard Hinnant499c61f2012-07-21 16:13:09 +0000183recursive_timed_mutex::try_lock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000184{
185 pthread_t id = pthread_self();
186 unique_lock<mutex> lk(__m_, try_to_lock);
187 if (lk.owns_lock() && (__count_ == 0 || pthread_equal(id, __id_)))
188 {
189 if (__count_ == numeric_limits<size_t>::max())
190 return false;
191 ++__count_;
192 __id_ = id;
193 return true;
194 }
195 return false;
196}
197
198void
Howard Hinnant499c61f2012-07-21 16:13:09 +0000199recursive_timed_mutex::unlock() _NOEXCEPT
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000200{
201 unique_lock<mutex> lk(__m_);
202 if (--__count_ == 0)
203 {
Howard Hinnantadff4892010-05-24 17:49:41 +0000204 __id_ = 0;
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000205 lk.unlock();
206 __cv_.notify_one();
207 }
208}
209
Jonathan Roelofs8d86b2e2014-09-05 19:45:05 +0000210#endif // !_LIBCPP_HAS_NO_THREADS
211
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000212// If dispatch_once_f ever handles C++ exceptions, and if one can get to it
213// without illegal macros (unexpected macros not beginning with _UpperCase or
214// __lowercase), and if it stops spinning waiting threads, then call_once should
215// call into dispatch_once_f instead of here. Relevant radar this code needs to
216// keep in sync with: 7741191.
217
Jonathan Roelofs8d86b2e2014-09-05 19:45:05 +0000218#ifndef _LIBCPP_HAS_NO_THREADS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000219static pthread_mutex_t mut = PTHREAD_MUTEX_INITIALIZER;
220static pthread_cond_t cv = PTHREAD_COND_INITIALIZER;
Jonathan Roelofs8d86b2e2014-09-05 19:45:05 +0000221#endif
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000222
223void
224__call_once(volatile unsigned long& flag, void* arg, void(*func)(void*))
225{
Jonathan Roelofs8d86b2e2014-09-05 19:45:05 +0000226#if defined(_LIBCPP_HAS_NO_THREADS)
227 if (flag == 0)
228 {
229#ifndef _LIBCPP_NO_EXCEPTIONS
230 try
231 {
232#endif // _LIBCPP_NO_EXCEPTIONS
233 flag = 1;
234 func(arg);
235 flag = ~0ul;
236#ifndef _LIBCPP_NO_EXCEPTIONS
237 }
238 catch (...)
239 {
240 flag = 0ul;
241 throw;
242 }
243#endif // _LIBCPP_NO_EXCEPTIONS
244 }
245#else // !_LIBCPP_HAS_NO_THREADS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000246 pthread_mutex_lock(&mut);
247 while (flag == 1)
248 pthread_cond_wait(&cv, &mut);
249 if (flag == 0)
250 {
Howard Hinnantd4444702010-08-11 17:04:31 +0000251#ifndef _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000252 try
253 {
Howard Hinnant16e6e1d2010-08-22 00:03:27 +0000254#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000255 flag = 1;
256 pthread_mutex_unlock(&mut);
257 func(arg);
258 pthread_mutex_lock(&mut);
259 flag = ~0ul;
260 pthread_mutex_unlock(&mut);
261 pthread_cond_broadcast(&cv);
Howard Hinnantd4444702010-08-11 17:04:31 +0000262#ifndef _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000263 }
264 catch (...)
265 {
266 pthread_mutex_lock(&mut);
267 flag = 0ul;
268 pthread_mutex_unlock(&mut);
Howard Hinnant21aefc32010-06-03 16:42:57 +0000269 pthread_cond_broadcast(&cv);
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000270 throw;
271 }
Howard Hinnant16e6e1d2010-08-22 00:03:27 +0000272#endif // _LIBCPP_NO_EXCEPTIONS
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000273 }
274 else
275 pthread_mutex_unlock(&mut);
Jonathan Roelofs8d86b2e2014-09-05 19:45:05 +0000276#endif // !_LIBCPP_HAS_NO_THREADS
277
Howard Hinnantbc8d3f92010-05-11 19:42:16 +0000278}
279
280_LIBCPP_END_NAMESPACE_STD