From 1055a7aa2ea4296b476cb55099bb7d42e24c74b3 Mon Sep 17 00:00:00 2001 From: Florian Fischer <florian.fischer@muhq.space> Date: Mon, 26 Jul 2021 10:44:22 +0200 Subject: [PATCH] [IoContext] wrap CQ locking in if constexpr We don't need to pay the overhead of the atomic operations on each dispatch loop if there is no concurrent access to worker CQs. --- emper/io/IoContext.cpp | 43 +++++++++++++++++++++++------------------- emper/io/IoContext.hpp | 2 ++ 2 files changed, 26 insertions(+), 19 deletions(-) diff --git a/emper/io/IoContext.cpp b/emper/io/IoContext.cpp index b34c5e12..83ab7001 100644 --- a/emper/io/IoContext.cpp +++ b/emper/io/IoContext.cpp @@ -211,24 +211,26 @@ reap_cqes: LOGD("Reaping completions for worker " << std::to_string(worker->getWorkerId())); std::array<struct io_uring_cqe *, CQE_BATCH_COUNT> cqes; - // Someone else is currently reaping completions - if constexpr (callerEnvironment == CallerEnvironment::EMPER) { - if (unlikely(!cq_lock.try_lock())) { - LOGD("worker unsuccessful try_lock"); - return 0; - } - } else { - // We can not reap completions of this IoContext to not race - // with the sleeping worker. - if (waitInflight.load(std::memory_order_acquire)) { - LOGD("Not reaping worker " << std::to_string(worker->getWorkerId()) - << " since this worker is already waiting for its CQEs"); - return 0; - } + if constexpr (needsCqLock) { + // Someone else is currently reaping completions + if constexpr (callerEnvironment == CallerEnvironment::EMPER) { + if (unlikely(!cq_lock.try_lock())) { + LOGD("worker unsuccessful try_lock"); + return 0; + } + } else { + // We can not reap completions of this IoContext to not race + // with the sleeping worker. + if (waitInflight.load(std::memory_order_acquire)) { + LOGD("Not reaping worker " << std::to_string(worker->getWorkerId()) + << " since this worker is already waiting for its CQEs"); + return 0; + } - if (!cq_lock.try_lock_or_increment()) { - LOGD("Global completer unsuccessful try_lock_or_increment"); - return 0; + if (!cq_lock.try_lock_or_increment()) { + LOGD("Global completer unsuccessful try_lock_or_increment"); + return 0; + } } } @@ -245,7 +247,10 @@ reap_cqes: io_uring_cq_advance(&ring, count); - uint32_t globalCompleterAttempts = cq_lock.unlock(); + uint32_t globalCompleterAttempts; + if constexpr (needsCqLock) { + globalCompleterAttempts = cq_lock.unlock(); + } LOGD("got " << count << " cqes from worker " << worker->getWorkerId() << "'s io_uring"); @@ -336,7 +341,7 @@ reap_cqes: } // check if lost wakeup was possible - if constexpr (callerEnvironment == CallerEnvironment::EMPER) { + if constexpr (needsCqLock && callerEnvironment == CallerEnvironment::EMPER) { bool reReap = false; // TODO: How sure are we that this is unlikely? if (unlikely(globalCompleterAttempts > maxRaceFreeCompleterAttempts)) { diff --git a/emper/io/IoContext.hpp b/emper/io/IoContext.hpp index 841246ec..f849bb6f 100644 --- a/emper/io/IoContext.hpp +++ b/emper/io/IoContext.hpp @@ -68,6 +68,8 @@ class IoContext : public Logger<LogSubsystem::IO> { Runtime &runtime; static thread_local IoContext *workerIo; + static constexpr bool needsCqLock = + emper::IO_COMPLETER_BEHAVIOR != emper::IoCompleterBehavior::none; // TryLock protecting the completion queue of ring. ALIGN_TO_CACHE_LINE CqLock cq_lock; struct io_uring ring; -- GitLab