// (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 { //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. */ BOOST_CONSTEXPR 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. */ BOOST_CONSTEXPR 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) : obj_(), mtx_(&mtx) { } /** * Move constructor */ externally_locked(BOOST_THREAD_RV_REF(externally_locked) rhs) : obj_(move(rhs.obj_)), mtx_(rhs.mtx_) { rhs.mtx_=0; } /** * 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(), lock_error() ); /*< run time check throw if no locked >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return obj_; } mutex_type* mutex() { 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_MOVABLE_ONLY( externally_locked ) /** * Effects: Constructs an externally locked object storing the cloaked reference object. */ externally_locked(T& obj, mutex_type& mtx) : obj_(&obj), mtx_(&mtx) { } /// move constructor externally_locked(BOOST_THREAD_RV_REF(externally_locked) rhs) : obj_(rhs.obj_), mtx_(rhs.mtx_) { rhs.obj_=0; rhs.mtx_=0; } void swap(externally_locked& rhs) { 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(), lock_error() ); /*< run time check throw if no locked >*/ 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(), lock_error() ); /*< run time check throw if no locked >*/ BOOST_THREAD_ASSERT_PRECONDITION( lk.owns_lock(mtx_), lock_error() ); /*< run time check throw if not locks the same >*/ return *obj_; } mutex_type* mutex() { 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) { lhs.swap(rhs); } } #include #endif // header