From a780df5e7f0e5fe2a5c4acc13685a13d9b918d4f Mon Sep 17 00:00:00 2001 From: Florian Fischer <florian.fl.fischer@fau.de> Date: Sat, 6 Feb 2021 14:21:26 +0100 Subject: [PATCH] [IO] add FutureError and throw it when wait is called on a Future with callback The wait() call on a Future with callback currently will never return therefore calling wait() is disallowed. --- emper/io/Future.cpp | 4 ++++ emper/io/Future.hpp | 12 ++++++++++++ 2 files changed, 16 insertions(+) diff --git a/emper/io/Future.cpp b/emper/io/Future.cpp index 8d59a3d6..50870da1 100644 --- a/emper/io/Future.cpp +++ b/emper/io/Future.cpp @@ -44,6 +44,10 @@ template PartialCompletableFuture::CompletionType PartialCompletableFuture::tryComplete<CallerEnvironment::ANYWHERE>(int32_t res); auto Future::wait() -> int32_t { + if (unlikely(callback)) { + throw FutureError("Futures with registered callback must not be awaited"); + } + LOGD("Waiting on " << this); sem.wait(); diff --git a/emper/io/Future.hpp b/emper/io/Future.hpp index f21de80b..91f157a1 100644 --- a/emper/io/Future.hpp +++ b/emper/io/Future.hpp @@ -11,6 +11,8 @@ #include <cstdlib> // for abort #include <functional> #include <ostream> // for operator<<, ostream, basic_ost... +#include <stdexcept> +#include <string> #include "BinaryPrivateSemaphore.hpp" // for BPS #include "CallerEnvironment.hpp" // for CallerEnvironment, ANYWHERE @@ -25,6 +27,11 @@ struct io_uring_sqe; namespace emper::io { class Stats; +class FutureError : public std::logic_error { + friend class Future; + FutureError(const std::string& what) : std::logic_error(what) {} +}; + /* * @brief Future representing an IO request which can be awaited */ @@ -115,6 +122,11 @@ class Future : public Logger<LogSubsystem::IO> { : op(op), fd(fd), buf(buf), len(len), offsetOrFlags(offsetOrFlags){}; public: + // Clang-tidy warns about the exception possibly thrown by + // ~Future -> cancel -> wait + // But this exception will never be thrown because wait() throws an exception + // only if callback is set and if callback is set ~Future does not call cancel. + // NOLINTNEXTLINE(bugprone-exception-escape) virtual ~Future() { if (isForgotten() || callback) { return; -- GitLab