Path: blob/main/contrib/llvm-project/libcxx/include/__mutex/unique_lock.h
35262 views
//===----------------------------------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#ifndef _LIBCPP___MUTEX_UNIQUE_LOCK_H9#define _LIBCPP___MUTEX_UNIQUE_LOCK_H1011#include <__chrono/duration.h>12#include <__chrono/time_point.h>13#include <__config>14#include <__memory/addressof.h>15#include <__mutex/tag_types.h>16#include <__system_error/system_error.h>17#include <__utility/swap.h>18#include <cerrno>1920#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)21# pragma GCC system_header22#endif2324#ifndef _LIBCPP_HAS_NO_THREADS2526_LIBCPP_BEGIN_NAMESPACE_STD2728template <class _Mutex>29class _LIBCPP_TEMPLATE_VIS unique_lock {30public:31typedef _Mutex mutex_type;3233private:34mutex_type* __m_;35bool __owns_;3637public:38_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock() _NOEXCEPT : __m_(nullptr), __owns_(false) {}39_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI explicit unique_lock(mutex_type& __m)40: __m_(std::addressof(__m)), __owns_(true) {41__m_->lock();42}4344_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock(mutex_type& __m, defer_lock_t) _NOEXCEPT45: __m_(std::addressof(__m)),46__owns_(false) {}4748_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock(mutex_type& __m, try_to_lock_t)49: __m_(std::addressof(__m)), __owns_(__m.try_lock()) {}5051_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock(mutex_type& __m, adopt_lock_t)52: __m_(std::addressof(__m)), __owns_(true) {}5354template <class _Clock, class _Duration>55_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock(mutex_type& __m, const chrono::time_point<_Clock, _Duration>& __t)56: __m_(std::addressof(__m)), __owns_(__m.try_lock_until(__t)) {}5758template <class _Rep, class _Period>59_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock(mutex_type& __m, const chrono::duration<_Rep, _Period>& __d)60: __m_(std::addressof(__m)), __owns_(__m.try_lock_for(__d)) {}6162_LIBCPP_HIDE_FROM_ABI ~unique_lock() {63if (__owns_)64__m_->unlock();65}6667unique_lock(unique_lock const&) = delete;68unique_lock& operator=(unique_lock const&) = delete;6970_LIBCPP_NODISCARD _LIBCPP_HIDE_FROM_ABI unique_lock(unique_lock&& __u) _NOEXCEPT71: __m_(__u.__m_),72__owns_(__u.__owns_) {73__u.__m_ = nullptr;74__u.__owns_ = false;75}7677_LIBCPP_HIDE_FROM_ABI unique_lock& operator=(unique_lock&& __u) _NOEXCEPT {78if (__owns_)79__m_->unlock();8081__m_ = __u.__m_;82__owns_ = __u.__owns_;83__u.__m_ = nullptr;84__u.__owns_ = false;85return *this;86}8788void lock();89bool try_lock();9091template <class _Rep, class _Period>92bool try_lock_for(const chrono::duration<_Rep, _Period>& __d);9394template <class _Clock, class _Duration>95bool try_lock_until(const chrono::time_point<_Clock, _Duration>& __t);9697void unlock();9899_LIBCPP_HIDE_FROM_ABI void swap(unique_lock& __u) _NOEXCEPT {100std::swap(__m_, __u.__m_);101std::swap(__owns_, __u.__owns_);102}103104_LIBCPP_HIDE_FROM_ABI mutex_type* release() _NOEXCEPT {105mutex_type* __m = __m_;106__m_ = nullptr;107__owns_ = false;108return __m;109}110111_LIBCPP_HIDE_FROM_ABI bool owns_lock() const _NOEXCEPT { return __owns_; }112_LIBCPP_HIDE_FROM_ABI explicit operator bool() const _NOEXCEPT { return __owns_; }113_LIBCPP_HIDE_FROM_ABI mutex_type* mutex() const _NOEXCEPT { return __m_; }114};115_LIBCPP_CTAD_SUPPORTED_FOR_TYPE(unique_lock);116117template <class _Mutex>118void unique_lock<_Mutex>::lock() {119if (__m_ == nullptr)120__throw_system_error(EPERM, "unique_lock::lock: references null mutex");121if (__owns_)122__throw_system_error(EDEADLK, "unique_lock::lock: already locked");123__m_->lock();124__owns_ = true;125}126127template <class _Mutex>128bool unique_lock<_Mutex>::try_lock() {129if (__m_ == nullptr)130__throw_system_error(EPERM, "unique_lock::try_lock: references null mutex");131if (__owns_)132__throw_system_error(EDEADLK, "unique_lock::try_lock: already locked");133__owns_ = __m_->try_lock();134return __owns_;135}136137template <class _Mutex>138template <class _Rep, class _Period>139bool unique_lock<_Mutex>::try_lock_for(const chrono::duration<_Rep, _Period>& __d) {140if (__m_ == nullptr)141__throw_system_error(EPERM, "unique_lock::try_lock_for: references null mutex");142if (__owns_)143__throw_system_error(EDEADLK, "unique_lock::try_lock_for: already locked");144__owns_ = __m_->try_lock_for(__d);145return __owns_;146}147148template <class _Mutex>149template <class _Clock, class _Duration>150bool unique_lock<_Mutex>::try_lock_until(const chrono::time_point<_Clock, _Duration>& __t) {151if (__m_ == nullptr)152__throw_system_error(EPERM, "unique_lock::try_lock_until: references null mutex");153if (__owns_)154__throw_system_error(EDEADLK, "unique_lock::try_lock_until: already locked");155__owns_ = __m_->try_lock_until(__t);156return __owns_;157}158159template <class _Mutex>160void unique_lock<_Mutex>::unlock() {161if (!__owns_)162__throw_system_error(EPERM, "unique_lock::unlock: not locked");163__m_->unlock();164__owns_ = false;165}166167template <class _Mutex>168inline _LIBCPP_HIDE_FROM_ABI void swap(unique_lock<_Mutex>& __x, unique_lock<_Mutex>& __y) _NOEXCEPT {169__x.swap(__y);170}171172_LIBCPP_END_NAMESPACE_STD173174#endif // _LIBCPP_HAS_NO_THREADS175176#endif // _LIBCPP___MUTEX_UNIQUE_LOCK_H177178179