mirror of
https://github.com/boostorg/thread.git
synced 2026-01-27 07:22:11 +00:00
Thread: Added externally_locked
[SVN r81087]
This commit is contained in:
320
include/boost/thread/externally_locked.hpp
Normal file
320
include/boost/thread/externally_locked.hpp
Normal file
@@ -0,0 +1,320 @@
|
||||
// (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 <boost/thread/detail/config.hpp>
|
||||
|
||||
#include <boost/thread/exceptions.hpp>
|
||||
#include <boost/thread/lock_concepts.hpp>
|
||||
#include <boost/thread/lock_traits.hpp>
|
||||
#include <boost/thread/lockable_concepts.hpp>
|
||||
#include <boost/thread/strict_lock.hpp>
|
||||
|
||||
#include <boost/static_assert.hpp>
|
||||
#include <boost/type_traits/is_same.hpp>
|
||||
#include <boost/throw_exception.hpp>
|
||||
#include <boost/utility/swap.hpp>
|
||||
|
||||
#include <boost/config/abi_prefix.hpp>
|
||||
|
||||
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 <typename T, typename MutexType = boost::mutex>
|
||||
class externally_locked
|
||||
{
|
||||
//BOOST_CONCEPT_ASSERT(( CopyConstructible<T> ));
|
||||
BOOST_CONCEPT_ASSERT(( BasicLockable<MutexType> ));
|
||||
|
||||
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 copying 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_EXTERNALLY_LOCKED_DONT_CHECK_SAME is not defined and the lk parameter doesn't satisfy the preconditions
|
||||
*/
|
||||
T& get(strict_lock<mutex_type>& lk)
|
||||
{
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
return obj_;
|
||||
}
|
||||
|
||||
const T& get(strict_lock<mutex_type>& lk) const
|
||||
{
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
return obj_;
|
||||
}
|
||||
|
||||
template <class Lock>
|
||||
T& get(nested_strict_lock<Lock>& lk)
|
||||
{
|
||||
BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
return obj_;
|
||||
}
|
||||
|
||||
template <class Lock>
|
||||
const T& get(nested_strict_lock<Lock>& lk) const
|
||||
{
|
||||
BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
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_EXTERNALLY_LOCKED_DONT_CHECK_SAME is not defined and the lk parameter doesn't satisfy the preconditions
|
||||
*/
|
||||
template <class Lock>
|
||||
T& get(Lock& lk)
|
||||
{
|
||||
BOOST_CONCEPT_ASSERT(( StrictLock<Lock> ));
|
||||
BOOST_STATIC_ASSERT( (is_strict_lock<Lock>::value)); /*< lk is a strict lock "sur parolle" >*/
|
||||
BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_OWNERSHIP /*< define BOOST_THREAD_EXTERNALLY_LOCKED_NO_CHECK_OWNERSHIP if you don't want to check lock ownership >*/
|
||||
if (!lk.own_lock()) throw lock_error(); /*< run time check throw if no locked >*/
|
||||
#endif
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME
|
||||
if (lk.mutex()!=mtx_) throw lock_error();
|
||||
#endif
|
||||
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<T&,M> 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 <typename T, typename MutexType>
|
||||
class externally_locked<T&, MutexType>
|
||||
{
|
||||
//BOOST_CONCEPT_ASSERT(( CopyConstructible<T> ));
|
||||
BOOST_CONCEPT_ASSERT(( BasicLockable<MutexType> ));
|
||||
|
||||
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_EXTERNALLY_LOCKED_DONT_CHECK_SAME is not defined and the lk parameter doesn't satisfy the preconditions
|
||||
*/
|
||||
T& get(strict_lock<mutex_type>& lk)
|
||||
{
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
return *obj_;
|
||||
}
|
||||
|
||||
const T& get(strict_lock<mutex_type>& lk) const
|
||||
{
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
return *obj_;
|
||||
}
|
||||
|
||||
template <class Lock>
|
||||
T& get(nested_strict_lock<Lock>& lk)
|
||||
{
|
||||
BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (lk.mutex()!=mtx_) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
return *obj_;
|
||||
}
|
||||
|
||||
template <class Lock>
|
||||
const T& get(nested_strict_lock<Lock>& lk) const
|
||||
{
|
||||
BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME /*< define BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME if you don't want to check lk check the same mtx >*/
|
||||
if (!lk.is_locking(&mtx_)) throw lock_error(); /*< run time check throw if not locks the same >*/
|
||||
#endif
|
||||
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_EXTERNALLY_LOCKED_DONT_CHECK_SAME is not defined and the lk parameter doesn't satisfy the preconditions
|
||||
*/
|
||||
template <class Lock>
|
||||
T& get(Lock& lk)
|
||||
{
|
||||
BOOST_CONCEPT_ASSERT(( StrictLock<Lock> ));
|
||||
BOOST_STATIC_ASSERT( (is_strict_lock<Lock>::value)); /*< lk is a strict lock "sur parolle" >*/
|
||||
BOOST_STATIC_ASSERT( (is_same<mutex_type, typename Lock::mutex_type>::value)); /*< that locks the same type >*/
|
||||
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_OWNERSHIP /*< define BOOST_THREAD_EXTERNALLY_LOCKED_NO_CHECK_OWNERSHIP if you don't want to check lock ownership >*/
|
||||
if (!lk.own_lock()) throw lock_error(); /*< run time check throw if no locked >*/
|
||||
#endif
|
||||
#ifndef BOOST_THREAD_EXTERNALLY_LOCKED_DONT_CHECK_SAME
|
||||
if (lk.mutex()!=mtx_) throw lock_error();
|
||||
#endif
|
||||
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 <typename T, typename MutexType>
|
||||
void swap(externally_locked<T, MutexType> & lhs, externally_locked<T, MutexType> & rhs)
|
||||
{
|
||||
lhs.swap(rhs);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
#include <boost/config/abi_suffix.hpp>
|
||||
|
||||
#endif // header
|
||||
Reference in New Issue
Block a user