mirror-chatterino2/src/common/UniqueAccess.hpp
pajlada 115d198434
Make AccessGuard use a shared_mutex instead (#2702)
This allows `accessConst` to use a shared lock instead of a unique lock
2021-05-01 15:19:41 +00:00

100 lines
1.9 KiB
C++

#pragma once
#include <mutex>
#include <shared_mutex>
#include <type_traits>
namespace chatterino {
template <typename T, typename LockType = std::unique_lock<std::shared_mutex>>
class AccessGuard
{
public:
AccessGuard(T &element, std::shared_mutex &mutex)
: element_(&element)
, lock_(mutex)
{
}
AccessGuard(AccessGuard<T, LockType> &&other)
: element_(other.element_)
, lock_(std::move(other.lock_))
{
}
AccessGuard<T, LockType> &operator=(AccessGuard<T, LockType> &&other)
{
this->element_ = other.element_;
this->lock_ = std::move(other.lock_);
return *this;
}
T *operator->() const
{
return this->element_;
}
T &operator*() const
{
return *this->element_;
}
private:
T *element_{};
LockType lock_;
};
template <typename T>
using SharedAccessGuard =
AccessGuard<const T, std::shared_lock<std::shared_mutex>>;
template <typename T>
class UniqueAccess
{
public:
UniqueAccess()
: element_(T())
{
}
UniqueAccess(const T &element)
: element_(element)
{
}
UniqueAccess(T &&element)
: element_(element)
{
}
UniqueAccess<T> &operator=(const T &element)
{
this->element_ = element;
return *this;
}
UniqueAccess<T> &operator=(T &&element)
{
this->element_ = element;
return *this;
}
AccessGuard<T> access() const
{
return AccessGuard<T>(this->element_, this->mutex_);
}
template <typename X = T,
typename = std::enable_if_t<!std::is_const<X>::value>>
SharedAccessGuard<const X> accessConst() const
{
return SharedAccessGuard<const T>(this->element_, this->mutex_);
}
private:
mutable T element_;
mutable std::shared_mutex mutex_;
};
} // namespace chatterino