// (C) Copyright 2012 Vicente J. Botet Escriba // Distributed under the Boost Software License, Version 1.0. (See // accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_THREAD_EXTERNALLY_LOCKED_HPP #define BOOST_THREAD_EXTERNALLY_LOCKED_HPP #include #include #include #include #include #include #include #include #include #include #include namespace boost { /** * externally_locked cloaks an object of type T, and actually provides full * access to that object through the get and set member functions, provided you * pass a reference to a strict lock object */ //[externally_locked template class externally_locked; template class externally_locked { //BOOST_CONCEPT_ASSERT(( CopyConstructible )); BOOST_CONCEPT_ASSERT(( BasicLockable )); public: typedef MutexType mutex_type; BOOST_THREAD_COPYABLE_AND_MOVABLE( externally_locked ) /** * Requires: T is a model of CopyConstructible. * Effects: Constructs an externally locked object copying the cloaked type. */ externally_locked(mutex_type& mtx, const T& obj) : obj_(obj), mtx_(&mtx) { } /** * Requires: T is a model of Movable. * Effects: Constructs an externally locked object by moving the cloaked type. */ externally_locked(mutex_type& mtx, BOOST_THREAD_RV_REF(T) obj) : obj_(move(obj)), mtx_(&mtx) { } /** * Requires: T is a model of DefaultConstructible. * Effects: Constructs an externally locked object initializing the cloaked type with the default constructor. */ externally_locked(mutex_type& mtx) // BOOST_NOEXCEPT_IF(BOOST_NOEXCEPT_EXPR(T())) : obj_(), mtx_(&mtx) { } /** * Copy constructor */ externally_locked(externally_locked const& rhs) //BOOST_NOEXCEPT : obj_(rhs.obj_), mtx_(rhs.mtx_) { } /** * Move constructor */ externally_locked(BOOST_THREAD_RV_REF(externally_locked) rhs) //BOOST_NOEXCEPT : obj_(move(rhs.obj_)), mtx_(rhs.mtx_) { } /// assignment externally_locked& operator=(externally_locked const& rhs) //BOOST_NOEXCEPT { obj_=rhs.obj_; mtx_=rhs.mtx_; return *this; } /// move assignment externally_locked& operator=(BOOST_THREAD_RV_REF(externally_locked) rhs) // BOOST_NOEXCEPT { obj_=move(BOOST_THREAD_RV(rhs).obj_); mtx_=rhs.mtx_; return *this; } void swap(externally_locked& rhs) //BOOST_NOEXCEPT_IF(BOOST_NOEXCEPT_EXPR) { swap(obj_, rhs.obj_); swap(mtx_, rhs.mtx_); } /** * Requires: The lk parameter must be locking the associated mtx. * * Returns: The address of the cloaked object.. * * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions */ T& get(strict_lock& lk) { BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return obj_; } const T& get(strict_lock& lk) const { BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return obj_; } template T& get(nested_strict_lock& lk) { BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return obj_; } template const T& get(nested_strict_lock& lk) const { BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return obj_; } /** * Requires: The lk parameter must be locking the associated mtx. * Returns: The address of the cloaked object.. * * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions */ template T& get(Lock& lk) { BOOST_CONCEPT_ASSERT(( StrictLock )); BOOST_STATIC_ASSERT( (is_strict_lock::value)); /*< lk is a strict lock "sur parolle" >*/ BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return obj_; } mutex_type* mutex() const BOOST_NOEXCEPT { return mtx_; } // modifiers void lock() { mtx_->lock(); } void unlock() { mtx_->unlock(); } bool try_lock() { return mtx_->try_lock(); } // todo add time related functions private: T obj_; mutex_type* mtx_; }; //] /** * externally_locked specialization for T& that cloaks an reference to an object of type T, and actually * provides full access to that object through the get and set member functions, provided you * pass a reference to a strict lock object. */ //[externally_locked_ref template class externally_locked { //BOOST_CONCEPT_ASSERT(( CopyConstructible )); BOOST_CONCEPT_ASSERT(( BasicLockable )); public: typedef MutexType mutex_type; BOOST_THREAD_COPYABLE_AND_MOVABLE( externally_locked ) /** * Effects: Constructs an externally locked object storing the cloaked reference object. */ externally_locked(T& obj, mutex_type& mtx) BOOST_NOEXCEPT : obj_(&obj), mtx_(&mtx) { } /// copy constructor externally_locked(externally_locked const& rhs) BOOST_NOEXCEPT : obj_(rhs.obj_), mtx_(rhs.mtx_) { } /// move constructor externally_locked(BOOST_THREAD_RV_REF(externally_locked) rhs) BOOST_NOEXCEPT : obj_(rhs.obj_), mtx_(rhs.mtx_) { } /// assignment externally_locked& operator=(externally_locked const& rhs) BOOST_NOEXCEPT { obj_=rhs.obj_; mtx_=rhs.mtx_; return *this; } /// move assignment externally_locked& operator=(BOOST_THREAD_RV_REF(externally_locked) rhs) BOOST_NOEXCEPT { obj_=rhs.obj_; mtx_=rhs.mtx_; return *this; } void swap(externally_locked& rhs) BOOST_NOEXCEPT { swap(obj_, rhs.obj_); swap(mtx_, rhs.mtx_); } /** * Requires: The lk parameter must be locking the associated mtx. * * Returns: The address of the cloaked object.. * * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions */ T& get(strict_lock const& lk) { BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } const T& get(strict_lock const& lk) const { BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } template T& get(nested_strict_lock const& lk) { BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } template const T& get(nested_strict_lock const& lk) const { BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } /** * Requires: The lk parameter must be locking the associated mtx. * Returns: The address of the cloaked object.. * * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions */ template T& get(Lock const& lk) { BOOST_CONCEPT_ASSERT(( StrictLock )); BOOST_STATIC_ASSERT( (is_strict_lock::value)); /*< lk is a strict lock "sur parolle" >*/ BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } /** * Requires: The lk parameter must be locking the associated mtx. * Returns: The address of the cloaked object.. * * Throws: lock_error if BOOST_THREAD_THROW_IF_PRECONDITION_NOT_SATISFIED is not defined and the lk parameter doesn't satisfy the preconditions */ template T const& get(Lock const& lk) const { BOOST_CONCEPT_ASSERT(( StrictLock )); BOOST_STATIC_ASSERT( (is_strict_lock::value)); /*< lk is a strict lock "sur parolle" >*/ BOOST_STATIC_ASSERT( (is_same::value)); /*< that locks the same type >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } mutex_type* mutex() const BOOST_NOEXCEPT { return mtx_; } void lock() { mtx_->lock(); } void unlock() { mtx_->unlock(); } bool try_lock() { return mtx_->try_lock(); } // todo add time related functions protected: T* obj_; mutex_type* mtx_; }; //] template void swap(externally_locked & lhs, externally_locked & rhs) // BOOST_NOEXCEPT { lhs.swap(rhs); } } #include #endif // header