futures/Promise.h \
futures/QueuedImmediateExecutor.h \
futures/ScheduledExecutor.h \
+ futures/SharedPromise.h \
+ futures/SharedPromise-inl.h \
futures/Timekeeper.h \
futures/Try-inl.h \
futures/Try.h \
}
template <class T>
-void Promise<T>::setTry(Try<T> t) {
+void Promise<T>::setTry(Try<T>&& t) {
throwIfFulfilled();
core_->setResult(std::move(t));
}
template <class M>
void setValue(M&& value);
- void setTry(Try<T> t);
+ void setTry(Try<T>&& t);
/** Fulfill this Promise with the result of a function that takes no
arguments and returns something implicitly convertible to T.
--- /dev/null
+/*
+ * Copyright 2015 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.
+ */
+
+#pragma once
+
+namespace folly {
+
+template <class T>
+SharedPromise<T>::SharedPromise(SharedPromise<T>&& other) noexcept {
+ *this = std::move(other);
+}
+
+template <class T>
+SharedPromise<T>& SharedPromise<T>::operator=(
+ SharedPromise<T>&& other) noexcept {
+ if (this == &other) {
+ return *this;
+ }
+
+ // std::lock will perform deadlock avoidance, in case
+ // Thread A: p1 = std::move(p2)
+ // Thread B: p2 = std::move(p1)
+ // race each other
+ std::lock(mutex_, other.mutex_);
+ std::lock_guard<std::mutex> g1(mutex_, std::adopt_lock);
+ std::lock_guard<std::mutex> g2(other.mutex_, std::adopt_lock);
+
+ std::swap(size_, other.size_);
+ std::swap(hasValue_, other.hasValue_);
+ std::swap(try_, other.try_);
+ std::swap(promises_, other.promises_);
+
+ return *this;
+}
+
+template <class T>
+size_t SharedPromise<T>::size() {
+ std::lock_guard<std::mutex> g(mutex_);
+ return size_;
+}
+
+template <class T>
+Future<T> SharedPromise<T>::getFuture() {
+ std::lock_guard<std::mutex> g(mutex_);
+ size_++;
+ if (hasValue_) {
+ return makeFuture<T>(Try<T>(try_));
+ } else {
+ promises_.emplace_back();
+ return promises_.back().getFuture();
+ }
+}
+
+template <class T>
+template <class E>
+typename std::enable_if<std::is_base_of<std::exception, E>::value>::type
+SharedPromise<T>::setException(E const& e) {
+ setTry(Try<T>(e));
+}
+
+template <class T>
+void SharedPromise<T>::setException(std::exception_ptr const& ep) {
+ setTry(Try<T>(ep));
+}
+
+template <class T>
+void SharedPromise<T>::setException(exception_wrapper ew) {
+ setTry(Try<T>(std::move(ew)));
+}
+
+template <class T>
+void SharedPromise<T>::setInterruptHandler(
+ std::function<void(exception_wrapper const&)> fn) {
+ std::lock_guard<std::mutex> g(mutex_);
+ if (hasValue_) {
+ return;
+ }
+ for (auto& p : promises_) {
+ p.setInterruptHandler(fn);
+ }
+}
+
+template <class T>
+template <class M>
+void SharedPromise<T>::setValue(M&& v) {
+ setTry(Try<T>(std::forward<M>(v)));
+}
+
+template <class T>
+template <class F>
+void SharedPromise<T>::setWith(F&& func) {
+ setTry(makeTryFunction(std::forward<F>(func)));
+}
+
+template <class T>
+void SharedPromise<T>::setTry(Try<T>&& t) {
+ std::vector<Promise<T>> promises;
+
+ {
+ std::lock_guard<std::mutex> g(mutex_);
+ if (hasValue_) {
+ throw PromiseAlreadySatisfied();
+ }
+ hasValue_ = true;
+ try_ = std::move(t);
+ promises.swap(promises_);
+ }
+
+ for (auto& p : promises) {
+ p.setTry(Try<T>(try_));
+ }
+}
+
+}
--- /dev/null
+/*
+ * Copyright 2015 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.
+ */
+
+#pragma once
+
+#include <folly/futures/Promise.h>
+
+namespace folly {
+
+/*
+ * SharedPromise provides the same interface as Promise, but you can extract
+ * multiple Futures from it, i.e. you can call getFuture() as many times as
+ * you'd like. When the SharedPromise is fulfilled, all of the Futures will be
+ * called back. Calls to getFuture() after the SharedPromise is fulfilled return
+ * a completed Future. If you find yourself constructing collections of Promises
+ * and fulfilling them simultaneously with the same value, consider this
+ * utility instead. Likewise, if you find yourself in need of setting multiple
+ * callbacks on the same Future (which is indefinitely unsupported), consider
+ * refactoring to use SharedPromise to "split" the Future.
+ */
+template <class T>
+class SharedPromise {
+public:
+ SharedPromise() = default;
+ ~SharedPromise() = default;
+
+ // not copyable
+ SharedPromise(SharedPromise const&) = delete;
+ SharedPromise& operator=(SharedPromise const&) = delete;
+
+ // movable
+ SharedPromise(SharedPromise<T>&&) noexcept;
+ SharedPromise& operator=(SharedPromise<T>&&) noexcept;
+
+ /** Return a Future tied to the shared core state. This can be called only
+ once, thereafter Future already retrieved exception will be raised. */
+ Future<T> getFuture();
+
+ /** Return the number of Futures associated with this SharedPromise */
+ size_t size();
+
+ /** Fulfill the SharedPromise with an exception_wrapper */
+ void setException(exception_wrapper ew);
+
+ /** Fulfill the SharedPromise with an exception_ptr, e.g.
+ try {
+ ...
+ } catch (...) {
+ p.setException(std::current_exception());
+ }
+ */
+ void setException(std::exception_ptr const&) DEPRECATED;
+
+ /** Fulfill the SharedPromise with an exception type E, which can be passed to
+ std::make_exception_ptr(). Useful for originating exceptions. If you
+ caught an exception the exception_wrapper form is more appropriate.
+ */
+ template <class E>
+ typename std::enable_if<std::is_base_of<std::exception, E>::value>::type
+ setException(E const&);
+
+ /// Set an interrupt handler to handle interrupts. See the documentation for
+ /// Future::raise(). Your handler can do whatever it wants, but if you
+ /// bother to set one then you probably will want to fulfill the SharedPromise with
+ /// an exception (or special value) indicating how the interrupt was
+ /// handled.
+ void setInterruptHandler(std::function<void(exception_wrapper const&)>);
+
+ /// Fulfill this SharedPromise<void>
+ template <class B = T>
+ typename std::enable_if<std::is_void<B>::value, void>::type
+ setValue() {
+ set(Try<T>());
+ }
+
+ /// Sugar to fulfill this SharedPromise<Unit>
+ template <class B = T>
+ typename std::enable_if<std::is_same<Unit, B>::value, void>::type
+ setValue() {
+ set(Try<T>(T()));
+ }
+
+ /** Set the value (use perfect forwarding for both move and copy) */
+ template <class M>
+ void setValue(M&& value);
+
+ void setTry(Try<T>&& t);
+
+ /** Fulfill this SharedPromise with the result of a function that takes no
+ arguments and returns something implicitly convertible to T.
+ Captures exceptions. e.g.
+
+ p.setWith([] { do something that may throw; return a T; });
+ */
+ template <class F>
+ void setWith(F&& func);
+
+private:
+ std::mutex mutex_;
+ size_t size_{0};
+ bool hasValue_{false};
+ Try<T> try_;
+ std::vector<Promise<T>> promises_;
+};
+
+}
+
+#include <folly/futures/Future.h>
+#include <folly/futures/SharedPromise-inl.h>
namespace folly {
template <class T>
-Try<T>::Try(Try<T>&& t) : contains_(t.contains_) {
+Try<T>::Try(Try<T>&& t) noexcept : contains_(t.contains_) {
if (contains_ == Contains::VALUE) {
new (&value_)T(std::move(t.value_));
} else if (contains_ == Contains::EXCEPTION) {
}
template <class T>
-Try<T>& Try<T>::operator=(Try<T>&& t) {
+Try<T>& Try<T>::operator=(Try<T>&& t) noexcept {
+ if (this == &t) {
+ return *this;
+ }
+
this->~Try();
contains_ = t.contains_;
if (contains_ == Contains::VALUE) {
return *this;
}
+template <class T>
+Try<T>::Try(const Try<T>& t) {
+ static_assert(
+ std::is_copy_constructible<T>::value,
+ "T must be copyable for Try<T> to be copyable");
+ contains_ = t.contains_;
+ if (contains_ == Contains::VALUE) {
+ new (&value_)T(t.value_);
+ } else if (contains_ == Contains::EXCEPTION) {
+ new (&e_)std::unique_ptr<exception_wrapper>();
+ e_ = folly::make_unique<exception_wrapper>(*(t.e_));
+ }
+}
+
+template <class T>
+Try<T>& Try<T>::operator=(const Try<T>& t) {
+ static_assert(
+ std::is_copy_constructible<T>::value,
+ "T must be copyable for Try<T> to be copyable");
+ this->~Try();
+ contains_ = t.contains_;
+ if (contains_ == Contains::VALUE) {
+ new (&value_)T(t.value_);
+ } else if (contains_ == Contains::EXCEPTION) {
+ new (&e_)std::unique_ptr<exception_wrapper>();
+ e_ = folly::make_unique<exception_wrapper>(*(t.e_));
+ }
+ return *this;
+}
+
template <class T>
Try<T>::~Try() {
if (contains_ == Contains::VALUE) {
}
// Move constructor
- Try(Try<T>&& t);
+ Try(Try<T>&& t) noexcept;
// Move assigner
- Try& operator=(Try<T>&& t);
+ Try& operator=(Try<T>&& t) noexcept;
- // Non-copyable
- Try(const Try<T>& t) = delete;
- // Non-copyable
- Try& operator=(const Try<T>& t) = delete;
+ // Copy constructor
+ Try(const Try& t);
+ // Copy assigner
+ Try& operator=(const Try& t);
~Try();
--- /dev/null
+/*
+ * Copyright 2015 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 <folly/futures/SharedPromise.h>
+#include <gtest/gtest.h>
+
+using namespace folly;
+
+TEST(SharedPromise, SetGet) {
+ SharedPromise<int> p;
+ p.setValue(1);
+ auto f1 = p.getFuture();
+ auto f2 = p.getFuture();
+ EXPECT_EQ(1, f1.value());
+ EXPECT_EQ(1, f2.value());
+}
+TEST(SharedPromise, GetSet) {
+ SharedPromise<int> p;
+ auto f1 = p.getFuture();
+ auto f2 = p.getFuture();
+ p.setValue(1);
+ EXPECT_EQ(1, f1.value());
+ EXPECT_EQ(1, f2.value());
+}
+
+TEST(SharedPromise, GetSetGet) {
+ SharedPromise<int> p;
+ auto f1 = p.getFuture();
+ p.setValue(1);
+ auto f2 = p.getFuture();
+ EXPECT_EQ(1, f1.value());
+ EXPECT_EQ(1, f2.value());
+}
+
+TEST(SharedPromise, Reset) {
+ SharedPromise<int> p;
+
+ auto f1 = p.getFuture();
+ p.setValue(1);
+ EXPECT_EQ(1, f1.value());
+
+ p = SharedPromise<int>();
+ auto f2 = p.getFuture();
+ EXPECT_FALSE(f2.isReady());
+ p.setValue(2);
+ EXPECT_EQ(2, f2.value());
+}
+
+TEST(SharedPromise, GetMoveSet) {
+ SharedPromise<int> p;
+ auto f = p.getFuture();
+ auto p2 = std::move(p);
+ p2.setValue(1);
+ EXPECT_EQ(1, f.value());
+}
+
+TEST(SharedPromise, SetMoveGet) {
+ SharedPromise<int> p;
+ p.setValue(1);
+ auto p2 = std::move(p);
+ auto f = p2.getFuture();
+ EXPECT_EQ(1, f.value());
+}
+
+TEST(SharedPromise, MoveSetGet) {
+ SharedPromise<int> p;
+ auto p2 = std::move(p);
+ p2.setValue(1);
+ auto f = p2.getFuture();
+ EXPECT_EQ(1, f.value());
+}
+
+TEST(SharedPromise, MoveGetSet) {
+ SharedPromise<int> p;
+ auto p2 = std::move(p);
+ auto f = p2.getFuture();
+ p2.setValue(1);
+ EXPECT_EQ(1, f.value());
+}
+
+TEST(SharedPromise, MoveMove) {
+ SharedPromise<std::shared_ptr<int>> p;
+ auto f1 = p.getFuture();
+ auto f2 = p.getFuture();
+ auto p2 = std::move(p);
+ p = std::move(p2);
+ p.setValue(std::make_shared<int>(1));
+}
using namespace folly;
+// Make sure we can copy Trys for copyable types
+TEST(Try, copy) {
+ Try<int> t;
+ auto t2 = t;
+}
+
+// But don't choke on move-only types
+TEST(Try, moveOnly) {
+ Try<std::unique_ptr<int>> t;
+ std::vector<Try<std::unique_ptr<int>>> v;
+ v.reserve(10);
+}
+
TEST(Try, makeTryFunction) {
auto func = []() {
return folly::make_unique<int>(1);
getContext()->fireWrite(std::move(sends_))
.then([promises](Try<void> t) mutable {
for (auto& p : *promises) {
- p.setTry(t);
+ p.setTry(Try<void>(t));
}
});
}