From a49d453b5fe195b61c5c159268f403b20a24224c Mon Sep 17 00:00:00 2001 From: Sahas Panda Date: Tue, 28 May 2024 15:02:04 -0400 Subject: [PATCH] feat(libsinsp): Add wrapper for read/write lock `mutex.h` already has a nice abstraction that allows exclusive access to an object. This change extends that paradigm to allow similar simple function calls that allow read/write locking primitives using a `shared_mutex`. Signed-off-by: Sahas Panda --- userspace/libsinsp/mutex.h | 168 ++++++++++++++++++++++++++++++++++++- 1 file changed, 166 insertions(+), 2 deletions(-) diff --git a/userspace/libsinsp/mutex.h b/userspace/libsinsp/mutex.h index d561f305ec..4cdaf75217 100644 --- a/userspace/libsinsp/mutex.h +++ b/userspace/libsinsp/mutex.h @@ -19,7 +19,9 @@ limitations under the License. #pragma once #include +#include #include +#include namespace libsinsp { template @@ -92,7 +94,7 @@ class ConstMutexGuard { // a writable guard can be demoted to a read-only one, but *not* the other way around ConstMutexGuard(MutexGuard &&rhs) noexcept : m_lock(std::move(rhs.m_lock)), - m_inner(rhs.m_inner) // NOLINT(google-explicit-constructor) + m_inner(rhs.m_inner) // NOLINT(google-explicit-constructor) {} const T *operator->() const @@ -118,6 +120,83 @@ class ConstMutexGuard { const T *m_inner; }; +/** + * \brief A wrapper to allow synchronized write access to a value owned by a SharedMutex, + * while allowing concurrent read access to the value. + * + * @tparam T type of the value protected by the mutex + * + * It works by simply holding a `std::unique_lock` object that keeps the shared_mutex + * exclusively locked while it exists, and unlocks it upon destruction + */ +template class SharedMutexGuard +{ +public: + SharedMutexGuard(std::unique_lock wlock, T *inner): + m_write_lock(std::move(wlock)), m_inner(inner) + { + } + + // we cannot copy a SharedMutexGuard, only move + SharedMutexGuard(SharedMutexGuard &rhs) = delete; + SharedMutexGuard &operator=(SharedMutexGuard &rhs) = delete; + SharedMutexGuard(SharedMutexGuard &&rhs) noexcept: + m_write_lock(std::move(rhs.m_write_lock)), m_inner(rhs.m_inner) + { + } + + T *operator->() { return m_inner; } + + T &operator*() { return *m_inner; } + + /** + * Validate that the guarded object exists. + */ + bool valid() const { return m_inner != nullptr; } + +private: + std::unique_lock m_write_lock; + T *m_inner; +}; + +/** + * \brief A wrapper to allow synchronized const read access to a value owned by a SharedMutex + * + * @tparam T type of the value protected by the mutex + * + * It works by simply holding a `std::shared_lock` object that keeps the shared_mutex + * read locked while it exists, and unlocks it upon destruction + */ +template class ConstSharedMutexGuard +{ +public: + ConstSharedMutexGuard(std::shared_lock rlock, const T *inner): + m_read_lock(std::move(rlock)), m_inner(inner) + { + } + + // we cannot copy a ConstSharedMutexGuard, only move + ConstSharedMutexGuard(ConstSharedMutexGuard &rhs) = delete; + ConstSharedMutexGuard &operator=(ConstSharedMutexGuard &rhs) = delete; + ConstSharedMutexGuard(ConstSharedMutexGuard &&rhs) noexcept: + m_read_lock(std::move(rhs.m_read_lock)), m_inner(rhs.m_inner) + { + } + + const T *operator->() const { return m_inner; } + + const T &operator*() const { return *m_inner; } + + /** + * Validate that the guarded object exists. + */ + bool valid() const { return m_inner != nullptr; } + +private: + std::shared_lock m_read_lock = {}; + const T *m_inner; +}; + /** * \brief Wrap a value of type T, enforcing synchronized access * @@ -178,4 +257,89 @@ class Mutex { mutable std::mutex m_lock; T m_inner; }; -} + +/** + * \brief Wrap a value of type T, enforcing synchronized access while allowing for simultaneous readers + * + * @tparam T type of the wrapped value + * + * The class owns a value of type T and a shared_mutex. The only way to access the T inside + * is via the read_lock() and write_lock() methods, which return guard objecta that unlock the mutex + * once they falls out of scope + * + * To protect an object with a shared_mutex, declare a variable of type `SharedMutex`, e.g. + * + * SharedMutex> m_locked_map; + * + * Then, to exclusively access the variable for writes, call .write_lock() on the SharedMutex object: + * + * SharedMutexGuard> locked = m_locked_map.write_lock(); + * + * Now you can call the inner object's methods directly on the guard object, + * which behaves like a smart pointer to the inner object: + * + * size_t num_elts = locked->size(); + * + * For concurrent read access to the inner object, use .read_lock() call. + * + * ConstSharedMutexGuard> locked = m_locked_map.read_lock(); + * + */ +template class SharedMutex +{ +public: + using time_type = decltype(sinsp_utils::get_current_time_ns()); + SharedMutex() = default; + + SharedMutex(T inner): m_inner(std::move(inner)) {} + + /** + * \brief Lock the mutex, allowing read access to the stored object + * + * The returned guard object allows access to the protected data + * via operator * or -> and ensures the lock is held as long as + * the guard object exists + * + * `ConstSharedMutexGuard` only allows read-only access to the protected object + */ + ConstSharedMutexGuard read_lock() const + { + auto start_ns = sinsp_utils::get_current_time_ns(); + auto mux_guard = ConstSharedMutexGuard(std::shared_lock(m_shared_lock), &m_inner); + auto end_ns = sinsp_utils::get_current_time_ns(); + m_read_lock_wait_time += (end_ns - start_ns); + m_read_lock_wait_count++; + return mux_guard; + } + + /** + * \brief Lock the mutex, allowing exclusive write access to the stored object + * + * The returned guard object allows access to the protected data + * via operator * or -> and ensures the lock is held as long as + * the guard object exists + */ + SharedMutexGuard write_lock() + { + auto start_ns = sinsp_utils::get_current_time_ns(); + auto mux_guard = SharedMutexGuard(std::unique_lock(m_shared_lock), &m_inner); + auto end_ns = sinsp_utils::get_current_time_ns(); + m_write_lock_wait_time += (end_ns - start_ns); + m_write_lock_wait_count++; + return mux_guard; + } + + time_type get_avg_read_lock_wait_time() const { return (m_read_lock_wait_time / m_read_lock_wait_count); } + + time_type get_avg_write_lock_wait_time() const { return (m_write_lock_wait_time / m_write_lock_wait_count); } + +private: + mutable std::shared_mutex m_shared_lock; + T m_inner; + mutable time_type m_read_lock_wait_time = 0; + mutable time_type m_write_lock_wait_time = 0; + mutable time_type m_read_lock_wait_count = 0; + mutable time_type m_write_lock_wait_count = 0; +}; + +} // namespace libsinsp