/* * Copyright (c) 2018 Jaroslav Jindrak * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * - The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef LIBCPP_MUTEX #define LIBCPP_MUTEX #include #include #include namespace std { /** * 20.4.1.2.1, class mutex: */ class mutex { public: constexpr mutex() noexcept : mtx_{} { aux::threading::mutex::init(mtx_); } ~mutex(); mutex(const mutex&) = delete; mutex& operator=(const mutex&) = delete; void lock(); bool try_lock(); void unlock(); using native_handle_type = aux::mutex_t*; native_handle_type native_handle(); private: aux::mutex_t mtx_; }; /** * 30.4.1.2.2, class recursive_mutex: */ class recursive_mutex { public: constexpr recursive_mutex() noexcept : mtx_{}, lock_level_{}, owner_{} { aux::threading::mutex::init(mtx_); } ~recursive_mutex(); recursive_mutex(const recursive_mutex&) = delete; recursive_mutex& operator=(const recursive_mutex&) = delete; void lock(); bool try_lock() noexcept; void unlock(); using native_handle_type = aux::mutex_t*; native_handle_type native_handle(); private: aux::mutex_t mtx_; size_t lock_level_; thread::id owner_; }; /** * 30.4.1.3.1, class timed_mutex: */ class timed_mutex { public: timed_mutex() noexcept; ~timed_mutex(); timed_mutex(const timed_mutex&) = delete; timed_mutex& operator=(const timed_mutex&) = delete; void lock(); bool try_lock(); void unlock(); template bool try_lock_for(const chrono::duration& rel_time) { auto time = aux::threading::time::convert(rel_time); return aux::threading::mutex::try_lock_for(time); } template bool try_lock_until(const chrono::time_point& abs_time) { auto dur = (abs_time - Clock::now()); auto time = aux::threading::time::convert(dur); return aux::threading::mutex::try_lock_for(time); } using native_handle_type = aux::mutex_t*; native_handle_type native_handle(); private: aux::mutex_t mtx_; }; /** * 30.4.1.3.2, class recursive_timed_mutex: */ class recursive_timed_mutex { public: recursive_timed_mutex() noexcept : mtx_{}, lock_level_{}, owner_{} { aux::threading::mutex::init(mtx_); } ~recursive_timed_mutex(); recursive_timed_mutex(const recursive_timed_mutex&) = delete; recursive_timed_mutex& operator=(const recursive_timed_mutex&) = delete; void lock(); bool try_lock(); void unlock(); template bool try_lock_for(const chrono::duration& rel_time) { if (owner_ == this_thread::get_id()) return true; auto time = aux::threading::time::convert(rel_time); auto ret = aux::threading::mutex::try_lock_for(time); if (ret) ++lock_level_; return ret; } template bool try_lock_until(const chrono::time_point& abs_time) { if (owner_ == this_thread::get_id()) return true; auto dur = (abs_time - Clock::now()); auto time = aux::threading::time::convert(dur); auto ret = aux::threading::mutex::try_lock_for(time); if (ret) ++lock_level_; return ret; } using native_handle_type = aux::mutex_t*; native_handle_type native_handle(); private: aux::mutex_t mtx_; size_t lock_level_; thread::id owner_; }; struct defer_lock_t { /* DUMMY BODY */ }; struct try_to_lock_t { /* DUMMY BODY */ }; struct adopt_lock_t { /* DUMMY BODY */ }; constexpr defer_lock_t defer_lock { /* DUMMY BODY */ }; constexpr try_to_lock_t try_to_lock { /* DUMMY BODY */ }; constexpr adopt_lock_t adopt_lock { /* DUMMY BODY */ }; /** * 30.4.2.1, class template lock_guard: */ template class lock_guard { public: using mutex_type = Mutex; explicit lock_guard(mutex_type& mtx) : mtx_{mtx} { mtx.lock(); } lock_guard(mutex_type& mtx, adopt_lock_t) : mtx_{mtx} { /* DUMMY BODY */ } ~lock_guard() { mtx_.unlock(); } lock_guard(const lock_guard&) = delete; lock_guard& operator=(const lock_guard&) = delete; private: mutex_type& mtx_; }; template class unique_lock; template void swap(unique_lock& lhs, unique_lock& rhs) noexcept; template int try_lock(L1& l1, L2& l2, L3&... ls); template void lock(L1& l1, L2& l2, L3&... ls); struct once_flag { constexpr once_flag() noexcept; once_flag(const once_flag&) = delete; once_flag& operator=(const once_flag&) = delete; }; template void call_once(once_flag& flag, Callable&& func, Args&&... args); } #endif