Commit 62a493ca authored by Kenton Varda's avatar Kenton Varda

Add `ExternalMutexGuarded<T>`.

This class lets you hold a temporary value that is only safe to access under lock, while you release the lock to do some other work.
parent de0ddc0e
...@@ -221,5 +221,96 @@ TEST(Mutex, LazyException) { ...@@ -221,5 +221,96 @@ TEST(Mutex, LazyException) {
#endif #endif
} }
class OnlyTouchUnderLock {
public:
OnlyTouchUnderLock(): ptr(nullptr) {}
OnlyTouchUnderLock(MutexGuarded<uint>& ref): ptr(&ref) {
ptr->getAlreadyLockedExclusive()++;
}
OnlyTouchUnderLock(OnlyTouchUnderLock&& other): ptr(other.ptr) {
other.ptr = nullptr;
if (ptr) {
// Just verify it's locked. Don't increment because different compilers may or may not
// elide moves.
ptr->getAlreadyLockedExclusive();
}
}
OnlyTouchUnderLock& operator=(OnlyTouchUnderLock&& other) {
if (ptr) {
ptr->getAlreadyLockedExclusive()++;
}
ptr = other.ptr;
other.ptr = nullptr;
if (ptr) {
// Just verify it's locked. Don't increment because different compilers may or may not
// elide moves.
ptr->getAlreadyLockedExclusive();
}
return *this;
}
~OnlyTouchUnderLock() noexcept(false) {
if (ptr != nullptr) {
ptr->getAlreadyLockedExclusive()++;
}
}
void frob() {
ptr->getAlreadyLockedExclusive()++;
}
private:
MutexGuarded<uint>* ptr;
};
KJ_TEST("ExternalMutexGuarded<T> destroy after release") {
MutexGuarded<uint> guarded(0);
{
ExternalMutexGuarded<OnlyTouchUnderLock> ext;
{
auto lock = guarded.lockExclusive();
ext.set(lock, guarded);
KJ_EXPECT(*lock == 1, *lock);
ext.get(lock).frob();
KJ_EXPECT(*lock == 2, *lock);
}
{
auto lock = guarded.lockExclusive();
auto released = ext.release(lock);
KJ_EXPECT(*lock == 2, *lock);
released.frob();
KJ_EXPECT(*lock == 3, *lock);
}
}
{
auto lock = guarded.lockExclusive();
KJ_EXPECT(*lock == 4, *lock);
}
}
KJ_TEST("ExternalMutexGuarded<T> destroy without release") {
MutexGuarded<uint> guarded(0);
{
ExternalMutexGuarded<OnlyTouchUnderLock> ext;
{
auto lock = guarded.lockExclusive();
ext.set(lock, guarded);
KJ_EXPECT(*lock == 1);
ext.get(lock).frob();
KJ_EXPECT(*lock == 2);
}
}
{
auto lock = guarded.lockExclusive();
KJ_EXPECT(*lock == 3);
}
}
} // namespace } // namespace
} // namespace kj } // namespace kj
...@@ -218,6 +218,8 @@ private: ...@@ -218,6 +218,8 @@ private:
template <typename U> template <typename U>
friend class MutexGuarded; friend class MutexGuarded;
template <typename U>
friend class ExternalMutexGuarded;
}; };
template <typename T> template <typename T>
...@@ -308,6 +310,81 @@ class MutexGuarded<const T> { ...@@ -308,6 +310,81 @@ class MutexGuarded<const T> {
static_assert(sizeof(T) < 0, "MutexGuarded's type cannot be const."); static_assert(sizeof(T) < 0, "MutexGuarded's type cannot be const.");
}; };
template <typename T>
class ExternalMutexGuarded {
// Holds a value that can only be manipulated while some other mutex is locked.
//
// The ExternalMutexGuarded<T> lives *outside* the scope of any lock on the mutex, but ensures
// that the value it holds can only be accessed under lock by forcing the caller to present a
// lock before accessing the value.
//
// Additionally, ExternalMutexGuarded<T>'s destructor will take an exclusive lock on the mutex
// while destroying the held value, unless the value has been release()ed before hand.
//
// The type T must have the following properties (which probably all movable types satisfy):
// - T is movable.
// - Immediately after any of the following has happened, T's destructor is effectively a no-op
// (hence certainly not requiring locks):
// - The value has been default-constructed.
// - The value has been initialized by-move from a default-constructed T.
// - The value has been moved away.
// - If ExternalMutexGuarded<T> is ever moved, then T must have a move constructor and move
// assignment operator that do not follow any pointers, therefore do not need to take a lock.
public:
ExternalMutexGuarded() = default;
~ExternalMutexGuarded() noexcept(false) {
if (mutex != nullptr) {
mutex->lock(_::Mutex::EXCLUSIVE);
KJ_DEFER(mutex->unlock(_::Mutex::EXCLUSIVE));
value = T();
}
}
ExternalMutexGuarded(ExternalMutexGuarded&& other)
: mutex(other.mutex), value(kj::mv(other.value)) {
other.mutex = nullptr;
}
ExternalMutexGuarded& operator=(ExternalMutexGuarded&& other) {
mutex = other.mutex;
value = kj::mv(other.value);
other.mutex = nullptr;
return *this;
}
template <typename U>
void set(Locked<U>& lock, T&& newValue) {
KJ_IREQUIRE(mutex == nullptr);
mutex = lock.mutex;
value = kj::mv(newValue);
}
template <typename U>
T& get(Locked<U>& lock) {
KJ_IREQUIRE(lock.mutex == mutex);
return value;
}
template <typename U>
const T& get(Locked<const U>& lock) const {
KJ_IREQUIRE(lock.mutex == mutex);
return value;
}
template <typename U>
T release(Locked<U>& lock) {
// Release (move away) the value. This allows the destructor to skip locking the mutex.
KJ_IREQUIRE(lock.mutex == mutex);
T result = kj::mv(value);
mutex = nullptr;
return result;
}
private:
_::Mutex* mutex = nullptr;
T value;
};
template <typename T> template <typename T>
class Lazy { class Lazy {
// A lazily-initialized value. // A lazily-initialized value.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment