92.31% Lines (24/26) 88.89% Functions (8/9)
TLA Baseline Branch
Line Hits Code Line Hits Code
1   // 1   //
2   // Copyright (c) 2026 Michael Vandeberg 2   // Copyright (c) 2026 Michael Vandeberg
3   // 3   //
4   // Distributed under the Boost Software License, Version 1.0. (See accompanying 4   // Distributed under the Boost Software License, Version 1.0. (See accompanying
5   // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) 5   // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6   // 6   //
7   // Official repository: https://github.com/cppalliance/corosio 7   // Official repository: https://github.com/cppalliance/corosio
8   // 8   //
9   9  
10   #ifndef BOOST_COROSIO_DETAIL_CONDITIONALLY_ENABLED_MUTEX_HPP 10   #ifndef BOOST_COROSIO_DETAIL_CONDITIONALLY_ENABLED_MUTEX_HPP
11   #define BOOST_COROSIO_DETAIL_CONDITIONALLY_ENABLED_MUTEX_HPP 11   #define BOOST_COROSIO_DETAIL_CONDITIONALLY_ENABLED_MUTEX_HPP
12   12  
13   #include <mutex> 13   #include <mutex>
14   14  
15   namespace boost::corosio::detail { 15   namespace boost::corosio::detail {
16   16  
17   /* Mutex wrapper that becomes a no-op when disabled. 17   /* Mutex wrapper that becomes a no-op when disabled.
18   18  
19   When enabled (the default), lock/unlock delegate to an 19   When enabled (the default), lock/unlock delegate to an
20   underlying std::mutex. When disabled, all operations are 20   underlying std::mutex. When disabled, all operations are
21   no-ops. The enabled flag is fixed after construction. 21   no-ops. The enabled flag is fixed after construction.
22   22  
23   scoped_lock wraps std::unique_lock<std::mutex> internally 23   scoped_lock wraps std::unique_lock<std::mutex> internally
24   so that condvar wait paths (which require the real lock 24   so that condvar wait paths (which require the real lock
25   type) compile and work in multi-threaded mode. 25   type) compile and work in multi-threaded mode.
26   */ 26   */
27   class conditionally_enabled_mutex 27   class conditionally_enabled_mutex
28   { 28   {
29   std::mutex mutex_; 29   std::mutex mutex_;
30   bool enabled_; 30   bool enabled_;
31   31  
32   public: 32   public:
HITCBC 33   11279 explicit conditionally_enabled_mutex(bool enabled = true) noexcept 33   14340 explicit conditionally_enabled_mutex(bool enabled = true) noexcept
HITCBC 34   11279 : enabled_(enabled) 34   14340 : enabled_(enabled)
35   { 35   {
HITCBC 36   11279 } 36   14340 }
37   37  
38   conditionally_enabled_mutex(conditionally_enabled_mutex const&) = delete; 38   conditionally_enabled_mutex(conditionally_enabled_mutex const&) = delete;
39   conditionally_enabled_mutex& operator=(conditionally_enabled_mutex const&) = delete; 39   conditionally_enabled_mutex& operator=(conditionally_enabled_mutex const&) = delete;
40   40  
41   bool enabled() const noexcept 41   bool enabled() const noexcept
42   { 42   {
43   return enabled_; 43   return enabled_;
44   } 44   }
45   45  
HITCBC 46   6918 void set_enabled(bool v) noexcept 46   8960 void set_enabled(bool v) noexcept
47   { 47   {
HITCBC 48   6918 enabled_ = v; 48   8960 enabled_ = v;
HITCBC 49   6918 } 49   8960 }
50   50  
51   // Lockable interface — allows std::lock_guard<conditionally_enabled_mutex> 51   // Lockable interface — allows std::lock_guard<conditionally_enabled_mutex>
HITCBC 52   45606 void lock() { if (enabled_) mutex_.lock(); } 52   58865 void lock() { if (enabled_) mutex_.lock(); }
HITCBC 53   45606 void unlock() { if (enabled_) mutex_.unlock(); } 53   58865 void unlock() { if (enabled_) mutex_.unlock(); }
54   bool try_lock() { return !enabled_ || mutex_.try_lock(); } 54   bool try_lock() { return !enabled_ || mutex_.try_lock(); }
55   55  
56   class scoped_lock 56   class scoped_lock
57   { 57   {
58   std::unique_lock<std::mutex> lock_; 58   std::unique_lock<std::mutex> lock_;
59   bool enabled_; 59   bool enabled_;
60   60  
61   public: 61   public:
HITCBC 62   561254 explicit scoped_lock(conditionally_enabled_mutex& m) 62   482624 explicit scoped_lock(conditionally_enabled_mutex& m)
HITCBC 63   561254 : lock_(m.mutex_, std::defer_lock) 63   482624 : lock_(m.mutex_, std::defer_lock)
HITCBC 64   561254 , enabled_(m.enabled_) 64   482624 , enabled_(m.enabled_)
65   { 65   {
HITCBC 66   561254 if (enabled_) 66   482624 if (enabled_)
HITCBC 67   561238 lock_.lock(); 67   482608 lock_.lock();
HITCBC 68   561254 } 68   482624 }
69   69  
70   scoped_lock(scoped_lock const&) = delete; 70   scoped_lock(scoped_lock const&) = delete;
71   scoped_lock& operator=(scoped_lock const&) = delete; 71   scoped_lock& operator=(scoped_lock const&) = delete;
72   72  
HITCBC 73   534920 void lock() 73   459053 void lock()
74   { 74   {
HITCBC 75   534920 if (enabled_) 75   459053 if (enabled_)
HITCBC 76   534914 lock_.lock(); 76   459047 lock_.lock();
HITCBC 77   534920 } 77   459053 }
78   78  
HITCBC 79   537740 void unlock() 79   461865 void unlock()
80   { 80   {
HITCBC 81   537740 if (enabled_) 81   461865 if (enabled_)
HITCBC 82   537731 lock_.unlock(); 82   461856 lock_.unlock();
HITCBC 83   537740 } 83   461865 }
84   84  
HITCBC 85   538313 bool owns_lock() const noexcept 85   463478 bool owns_lock() const noexcept
86   { 86   {
HITCBC 87   538313 return enabled_ && lock_.owns_lock(); 87   463478 return enabled_ && lock_.owns_lock();
88   } 88   }
89   89  
90   // Access the underlying unique_lock for condvar wait(). 90   // Access the underlying unique_lock for condvar wait().
91   // Only called when locking is enabled. 91   // Only called when locking is enabled.
MISLBC 92   2 std::unique_lock<std::mutex>& underlying() noexcept 92   std::unique_lock<std::mutex>& underlying() noexcept
93   { 93   {
MISLBC 94   2 return lock_; 94   return lock_;
95   } 95   }
96   }; 96   };
97   }; 97   };
98   98  
99   } // namespace boost::corosio::detail 99   } // namespace boost::corosio::detail
100   100  
101   #endif // BOOST_COROSIO_DETAIL_CONDITIONALLY_ENABLED_MUTEX_HPP 101   #endif // BOOST_COROSIO_DETAIL_CONDITIONALLY_ENABLED_MUTEX_HPP