--- /dev/null
+/*
+ * Copyright (C) 2022 Jérémie Galarneau <jeremie.galarneau@efficios.com>
+ *
+ * SPDX-License-Identifier: LGPL-2.1-only
+ *
+ */
+
+#ifndef LTTNG_PTHREAD_LOCK_H
+#define LTTNG_PTHREAD_LOCK_H
+
+#include <common/exception.hpp>
+
+#include <pthread.h>
+#include <mutex>
+
+namespace lttng {
+namespace pthread {
+
+namespace details {
+/*
+ * Class wrapping pthread mutexes and satisfying the Mutex named requirements, except
+ * for the "Default Constructible" requirement. The class is not default-constructible since the
+ * intention is to ease the transition of existing C-code using pthread mutexes to idiomatic C++.
+ *
+ * New code should use std::mutex.
+ */
+class mutex {
+public:
+ mutex(pthread_mutex_t& mutex_p) : _mutex{mutex_p}
+ {
+ }
+
+ /* "Not copyable" and "not moveable" Mutex requirements. */
+ mutex(mutex const &) = delete;
+ mutex &operator=(mutex const &) = delete;
+
+ void lock()
+ {
+ if (pthread_mutex_lock(&_mutex) != 0) {
+ LTTNG_THROW_POSIX("Failed to lock mutex", errno);
+ }
+ }
+
+ bool try_lock()
+ {
+ const auto ret = pthread_mutex_trylock(&_mutex);
+
+ if (ret == 0) {
+ return true;
+ } else if (errno == EBUSY || errno == EAGAIN) {
+ return false;
+ } else {
+ LTTNG_THROW_POSIX("Failed to try to lock mutex", errno);
+ }
+ }
+
+ void unlock()
+ {
+ if (pthread_mutex_unlock(&_mutex) != 0) {
+ LTTNG_THROW_POSIX("Failed to unlock mutex", errno);
+ }
+ }
+
+private:
+ pthread_mutex_t& _mutex;
+};
+} /* namespace details */
+
+/*
+ * Provides the basic concept of std::lock_guard for posix mutexes.
+ *
+ * `lock` is held for the duration of lock_guard's lifetime.
+ */
+class lock_guard {
+public:
+ lock_guard(pthread_mutex_t& mutex) : _mutex{mutex}, _guard(_mutex)
+ {
+ }
+
+ lock_guard(const lock_guard &) = delete;
+
+private:
+ details::mutex _mutex;
+ std::lock_guard<details::mutex> _guard;
+};
+
+} /* namespace pthread */
+} /* namespace lttng */
+
+#endif /* LTTNG_PTHREAD_LOCK_H */
--- /dev/null
+/*
+ * Copyright (C) 2022 Jérémie Galarneau <jeremie.galarneau@efficios.com>
+ *
+ * SPDX-License-Identifier: LGPL-2.1-only
+ *
+ */
+
+#ifndef LTTNG_URCU_H
+#define LTTNG_URCU_H
+
+#define _LGPL_SOURCE
+#include <urcu.h>
+#include <mutex>
+
+namespace lttng {
+namespace urcu {
+
+namespace details {
+/*
+ * Wrapper around an urcu read lock which satisfies the 'Mutex' named
+ * requirements of C++11. Satisfying those requirements facilitates the use of
+ * standard concurrency support library facilities.
+ *
+ * read_lock is under the details namespace since it is unlikely to be used
+ * directly by exception-safe code. See read_lock_guard.
+ */
+class read_lock {
+public:
+ read_lock() = default;
+
+ /* "Not copyable" and "not moveable" Mutex requirements. */
+ read_lock(read_lock const &) = delete;
+ read_lock &operator=(read_lock const &) = delete;
+
+ void lock()
+ {
+ rcu_read_lock();
+ }
+
+ bool try_lock()
+ {
+ lock();
+ return true;
+ }
+
+ void unlock()
+ {
+ rcu_read_unlock();
+ }
+};
+} /* namespace details */
+
+/*
+ * Provides the basic concept of std::lock_guard for rcu reader locks.
+ *
+ * The RCU reader lock is held for the duration of lock_guard's lifetime.
+ */
+class read_lock_guard {
+public:
+ read_lock_guard() : _guard(_lock)
+ {
+ }
+
+ read_lock_guard(const read_lock_guard &) = delete;
+
+private:
+ details::read_lock _lock;
+ std::lock_guard<details::read_lock> _guard;
+};
+
+} /* namespace urcu */
+} /* namespace lttng */
+
+#endif /* LTTNG_URCU_H */