Added fiber-compatible semaphore.
[folly.git] / folly / fibers / Semaphore.cpp
diff --git a/folly/fibers/Semaphore.cpp b/folly/fibers/Semaphore.cpp
new file mode 100644 (file)
index 0000000..40efae3
--- /dev/null
@@ -0,0 +1,94 @@
+/*
+ * Copyright 2016 Facebook, Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *   http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+#include "Semaphore.h"
+
+namespace folly {
+namespace fibers {
+
+bool Semaphore::signalSlow() {
+  // If we signalled a release, notify the waitlist
+  SYNCHRONIZED(waitList_) {
+    auto testVal = tokens_.load(std::memory_order_acquire);
+    if (testVal != 0) {
+      return false;
+    }
+
+    if (waitList_.empty()) {
+      // If the waitlist is now empty, ensure the token count increments
+      // No need for CAS here as we will always be under the mutex
+      CHECK(tokens_.compare_exchange_strong(
+          testVal, testVal + 1, std::memory_order_relaxed));
+    } else {
+      // trigger waiter if there is one
+      waitList_.front()->post();
+      waitList_.pop();
+    }
+  } // SYNCHRONIZED(waitList_)
+  return true;
+}
+
+void Semaphore::signal() {
+  auto oldVal = tokens_.load(std::memory_order_acquire);
+  do {
+    if (oldVal == 0) {
+      if (signalSlow()) {
+        break;
+      }
+    }
+  } while (!tokens_.compare_exchange_weak(
+      oldVal,
+      oldVal + 1,
+      std::memory_order_release,
+      std::memory_order_acquire));
+}
+
+bool Semaphore::waitSlow() {
+  // Slow path, create a baton and acquire a mutex to update the wait list
+  folly::fibers::Baton waitBaton;
+
+  SYNCHRONIZED(waitList_) {
+    auto testVal = tokens_.load(std::memory_order_acquire);
+    if (testVal != 0) {
+      return false;
+    }
+    // prepare baton and add to queue
+    waitList_.push(&waitBaton);
+  }
+  // If we managed to create a baton, wait on it
+  // This has to be done here so the mutex has been released
+  waitBaton.wait();
+  return true;
+}
+
+void Semaphore::wait() {
+  auto oldVal = tokens_.load(std::memory_order_acquire);
+  do {
+    if (oldVal == 0) {
+      // If waitSlow fails it is because the token is non-zero by the time
+      // the lock is taken, so we can just continue round the loop
+      if (waitSlow()) {
+        break;
+      }
+    }
+  } while (!tokens_.compare_exchange_weak(
+      oldVal,
+      oldVal - 1,
+      std::memory_order_release,
+      std::memory_order_acquire));
+}
+
+} // namespace fibers
+} // namespace folly