]> git.sesse.net Git - casparcg/blobdiff - common/semaphore.h
Merge pull request #493 from dimitry-ishenko-casparcg/2.1.0
[casparcg] / common / semaphore.h
index 04e36426045c48c9603807a9c7f0a90d1806c9f8..7baccda0e52df188f8e782e135406b8166a8488d 100644 (file)
 #include <boost/thread/mutex.hpp>
 #include <boost/thread/condition_variable.hpp>
 
+#include <map>
+#include <queue>
+#include <functional>
+
 namespace caspar {
 
 template <class N, class Func>
@@ -43,9 +47,10 @@ void repeat_n(N times_to_repeat_block, const Func& func)
  */
 class semaphore : boost::noncopyable
 {
-       mutable boost::mutex mutex_;
-       unsigned int permits_;
-       boost::condition_variable_any permits_available_;
+       mutable boost::mutex                                                                            mutex_;
+       unsigned int                                                                                            permits_;
+       boost::condition_variable_any                                                           permits_available_;
+       std::map<unsigned int, std::queue<std::function<void()>>>       callbacks_per_requested_permits_;
 public:
        /**
         * Constructor.
@@ -66,6 +71,7 @@ public:
 
                ++permits_;
 
+               perform_callback_based_acquire();
                permits_available_.notify_one();
        }
 
@@ -80,6 +86,7 @@ public:
 
                permits_ += permits;
 
+               perform_callback_based_acquire();
                repeat_n(permits, [this] { permits_available_.notify_one(); });
        }
 
@@ -112,11 +119,11 @@ public:
 
                while (true)
                {
-                       auto num_wanted = permits - num_acquired;
-                       auto to_drain = std::min(num_wanted, permits_);
+                       auto num_wanted = permits - num_acquired;
+                       auto to_drain   = std::min(num_wanted, permits_);
 
-                       permits_ -= to_drain;
-                       num_acquired += to_drain;
+                       permits_                -= to_drain;
+                       num_acquired    += to_drain;
 
                        if (num_acquired == permits)
                                break;
@@ -125,6 +132,20 @@ public:
                }
        }
 
+       /**
+       * Acquire a number of permits. Will not block, but instead invoke a callback
+       * when the specified number of permits are available and has been acquired.
+       *
+       * @param permits           The number of permits to acquire.
+       * @param acquired_callback The callback to invoke when acquired.
+       */
+       void acquire(unsigned int permits, std::function<void()> acquired_callback)
+       {
+               boost::unique_lock<boost::mutex> lock(mutex_);
+
+               callbacks_per_requested_permits_[permits].push(std::move(acquired_callback));
+       }
+
        /**
         * Acquire a number of permits. Will block until the given number of
         * permits has been acquired if not enough permits are currently available
@@ -143,11 +164,11 @@ public:
 
                while (true)
                {
-                       auto num_wanted = permits - num_acquired;
-                       auto to_drain = std::min(num_wanted, permits_);
+                       auto num_wanted = permits - num_acquired;
+                       auto to_drain   = std::min(num_wanted, permits_);
 
-                       permits_ -= to_drain;
-                       num_acquired += to_drain;
+                       permits_                -= to_drain;
+                       num_acquired    += to_drain;
 
                        if (num_acquired == permits)
                                break;
@@ -194,6 +215,47 @@ public:
 
                return permits_;
        }
+
+private:
+       void perform_callback_based_acquire()
+       {
+               if (callbacks_per_requested_permits_.empty())
+                       return;
+
+               while (
+                       !callbacks_per_requested_permits_.empty() &&
+                       callbacks_per_requested_permits_.begin()->first <= permits_)
+               {
+                       auto requested_permits_and_callbacks    = callbacks_per_requested_permits_.begin();
+                       auto requested_permits                                  = requested_permits_and_callbacks->first;
+                       auto& callbacks                                                 = requested_permits_and_callbacks->second;
+
+                       if (callbacks.empty())
+                       {
+                               callbacks_per_requested_permits_.erase(requested_permits_and_callbacks);
+                               continue;
+                       }
+
+                       auto& callback                                                  = callbacks.front();
+
+                       permits_ -= requested_permits;
+                       mutex_.unlock();
+
+                       try
+                       {
+                               callback();
+                       }
+                       catch (...)
+                       {
+                       }
+
+                       mutex_.lock();
+                       callbacks.pop();
+
+                       if (callbacks.empty())
+                               callbacks_per_requested_permits_.erase(requested_permits_and_callbacks);
+               }
+       }
 };
 
 /**