2018-09-29 17:36:42 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_MISC_GUARDED_H
|
|
|
|
#define OPENMW_COMPONENTS_MISC_GUARDED_H
|
|
|
|
|
|
|
|
#include <mutex>
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
namespace Misc
|
|
|
|
{
|
|
|
|
template <class T>
|
|
|
|
class Locked
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Locked(std::mutex& mutex, T& value)
|
|
|
|
: mLock(mutex), mValue(value)
|
|
|
|
{}
|
|
|
|
|
|
|
|
T& get() const
|
|
|
|
{
|
|
|
|
return mValue.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
T* operator ->() const
|
|
|
|
{
|
|
|
|
return std::addressof(get());
|
|
|
|
}
|
|
|
|
|
|
|
|
T& operator *() const
|
|
|
|
{
|
|
|
|
return get();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::unique_lock<std::mutex> mLock;
|
|
|
|
std::reference_wrapper<T> mValue;
|
|
|
|
};
|
|
|
|
|
2018-09-29 19:57:41 +00:00
|
|
|
template <class T>
|
|
|
|
class ScopeGuarded
|
|
|
|
{
|
|
|
|
public:
|
2018-11-01 22:59:27 +00:00
|
|
|
ScopeGuarded()
|
|
|
|
: mMutex()
|
|
|
|
, mValue()
|
|
|
|
{}
|
2018-09-29 19:57:41 +00:00
|
|
|
|
|
|
|
ScopeGuarded(const T& value)
|
2018-11-01 22:59:27 +00:00
|
|
|
: mMutex()
|
|
|
|
, mValue(value)
|
2018-09-29 19:57:41 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
ScopeGuarded(T&& value)
|
2018-11-01 22:59:27 +00:00
|
|
|
: mMutex()
|
|
|
|
, mValue(std::move(value))
|
2018-09-29 19:57:41 +00:00
|
|
|
{}
|
|
|
|
|
2018-09-30 22:33:25 +00:00
|
|
|
template <class ... Args>
|
|
|
|
ScopeGuarded(Args&& ... args)
|
2018-11-01 22:59:27 +00:00
|
|
|
: mMutex()
|
|
|
|
, mValue(std::forward<Args>(args) ...)
|
2018-09-30 22:33:25 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
ScopeGuarded(const ScopeGuarded& other)
|
|
|
|
: mMutex()
|
|
|
|
, mValue(other.lock().get())
|
|
|
|
{}
|
|
|
|
|
|
|
|
ScopeGuarded(ScopeGuarded&& other)
|
|
|
|
: mMutex()
|
|
|
|
, mValue(std::move(other.lock().get()))
|
|
|
|
{}
|
|
|
|
|
2018-09-29 19:57:41 +00:00
|
|
|
Locked<T> lock()
|
|
|
|
{
|
|
|
|
return Locked<T>(mMutex, mValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
Locked<const T> lockConst()
|
|
|
|
{
|
|
|
|
return Locked<const T>(mMutex, mValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::mutex mMutex;
|
|
|
|
T mValue;
|
|
|
|
};
|
|
|
|
|
2018-09-29 17:36:42 +00:00
|
|
|
template <class T>
|
|
|
|
class SharedGuarded
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SharedGuarded()
|
2018-11-01 22:59:27 +00:00
|
|
|
: mMutex(std::make_shared<std::mutex>()), mValue()
|
2018-09-29 17:36:42 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
SharedGuarded(std::shared_ptr<T> value)
|
|
|
|
: mMutex(std::make_shared<std::mutex>()), mValue(std::move(value))
|
|
|
|
{}
|
|
|
|
|
|
|
|
Locked<T> lock() const
|
|
|
|
{
|
|
|
|
return Locked<T>(*mMutex, *mValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
Locked<const T> lockConst() const
|
|
|
|
{
|
|
|
|
return Locked<const T>(*mMutex, *mValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::shared_ptr<std::mutex> mMutex;
|
|
|
|
std::shared_ptr<T> mValue;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|