From 9b311838f4bf5fd9037949cc5910d8c66c26fcaa Mon Sep 17 00:00:00 2001 From: Andrii Grynenko Date: Mon, 16 May 2016 13:13:24 -0700 Subject: [PATCH] Make await exception safe Summary: This fixes fibers::await to correctly handle exception thrown by user-passed lambda. Await still always waits for the promise to be fulfilled (if the promise was not moved out, it will be destroyed and thus auto-fulfilled with "promise not fulfilled" exception). However if user-passed lambda throws, promise result is ignored (even if exception) and exception thrown by lambda is re-thrown. Reviewed By: pavlo-fb Differential Revision: D3303393 fbshipit-source-id: c4ba01fde0e156cc88e5d07aaf763e3abe121d11 --- folly/experimental/fibers/FiberManager-inl.h | 15 +++++++++-- folly/experimental/fibers/test/FibersTest.cpp | 25 +++++++++++++++++++ 2 files changed, 38 insertions(+), 2 deletions(-) diff --git a/folly/experimental/fibers/FiberManager-inl.h b/folly/experimental/fibers/FiberManager-inl.h index 697f14ba..5d0d5a90 100644 --- a/folly/experimental/fibers/FiberManager-inl.h +++ b/folly/experimental/fibers/FiberManager-inl.h @@ -547,12 +547,23 @@ typename FirstArgOf::type::value_type inline await(F&& func) { typedef typename FirstArgOf::type::value_type Result; folly::Try result; + std::exception_ptr funcException; Baton baton; - baton.wait([&func, &result, &baton]() mutable { - func(Promise(result, baton)); + baton.wait([&func, &result, &baton, &funcException]() mutable { + try { + func(Promise(result, baton)); + } catch (...) { + // Save the exception, but still wait for baton to be posted by user code + // or promise destructor. + funcException = std::current_exception(); + } }); + if (UNLIKELY(funcException != nullptr)) { + std::rethrow_exception(funcException); + } + return folly::moveFromTry(result); } } diff --git a/folly/experimental/fibers/test/FibersTest.cpp b/folly/experimental/fibers/test/FibersTest.cpp index c5b14f18..f4db1f6f 100644 --- a/folly/experimental/fibers/test/FibersTest.cpp +++ b/folly/experimental/fibers/test/FibersTest.cpp @@ -319,6 +319,31 @@ TEST(FiberManager, addTasksNoncopyable) { loopController.loop(std::move(loopFunc)); } +TEST(FiberManager, awaitThrow) { + folly::EventBase evb; + struct ExpectedException {}; + getFiberManager(evb) + .addTaskFuture([&] { + EXPECT_THROW( + await([](Promise p) { + p.setValue(42); + throw ExpectedException(); + }), + ExpectedException + ); + + EXPECT_THROW( + await([&](Promise p) { + evb.runInEventBaseThread([p = std::move(p)]() mutable { + p.setValue(42); + }); + throw ExpectedException(); + }), + ExpectedException); + }) + .waitVia(&evb); +} + TEST(FiberManager, addTasksThrow) { std::vector> pendingFibers; bool taskAdded = false; -- 2.34.1