diff --git a/emper/strategies/laws/LawsDispatcher.cpp b/emper/strategies/laws/LawsDispatcher.cpp index 698b7d27aeba3d9ba93e13e1de0c8b6182bd07f2..8044fb404423b1f5bb0725c2d7d6f0eddc60ff73 100644 --- a/emper/strategies/laws/LawsDispatcher.cpp +++ b/emper/strategies/laws/LawsDispatcher.cpp @@ -2,14 +2,13 @@ // Copyright © 2020-2021 Florian Schmaus #include "LawsDispatcher.hpp" -#include <atomic> // for atomic, memory_order_relaxed - #include "Common.hpp" // for DIE_MSG #include "Emper.hpp" #include "LawsStrategy.hpp" // for LawsStrategy, LawsStrategy::FiberSource #include "NextFiberResult.hpp" #include "Runtime.hpp" #include "emper-common.h" +#include "strategies/laws/LawsWorkerStats.hpp" void LawsDispatcher::recycle(Fiber* fiber) { // If the ref count has not reached zero yet, do not recycle the @@ -37,16 +36,16 @@ void LawsDispatcher::dispatchLoop() { auto fiberSource = static_cast<LawsStrategy::FiberSource>(next.metadata); switch (fiberSource) { case LawsStrategy::FiberSource::fromPriority: - lawsStrategy.dispatchedFiberFromPriority.fetch_add(1, std::memory_order_relaxed); + LawsStrategy::stats.dispatchedFibersFromPriority++; break; case LawsStrategy::FiberSource::local: - lawsStrategy.dispatchedFiberFromLocal.fetch_add(1, std::memory_order_relaxed); + LawsStrategy::stats.dispatchedFibersFromLocal++; break; case LawsStrategy::FiberSource::stolen: - lawsStrategy.dispatchedFiberStolen.fetch_add(1, std::memory_order_relaxed); + LawsStrategy::stats.dispatchedFibersStolen++; break; case LawsStrategy::FiberSource::anywhereQueue: - lawsStrategy.dispatchedFiberFromAnywhere.fetch_add(1, std::memory_order_relaxed); + LawsStrategy::stats.dispatchedFibersFromAnywhereQueue++; break; default: DIE_MSG("Unknown fiber source: " << next.metadata); diff --git a/emper/strategies/laws/LawsDispatcher.hpp b/emper/strategies/laws/LawsDispatcher.hpp index e18f4eada5ca92530cc5745f4956b79130c245d5..d10b856c71b85b1f518532fc767e0eeced8e8365 100644 --- a/emper/strategies/laws/LawsDispatcher.hpp +++ b/emper/strategies/laws/LawsDispatcher.hpp @@ -3,29 +3,16 @@ #pragma once #include "Dispatcher.hpp" -#include "emper-common.h" // IWYU pragma: keep -#include "emper-config.h" // IWYU pragma: keep class Fiber; -class LawsStrategy; class Runtime; class LawsDispatcher : public Dispatcher { private: -#pragma GCC diagnostic push -#pragma GCC diagnostic ignored "-Wattributes" - LawsStrategy& lawsStrategy -#ifndef EMPER_STATS - ATTR_UNUSED -#endif - ; -#pragma GCC diagnostic pop - void recycle(Fiber* fiber) override; public: - LawsDispatcher(Runtime& runtime, LawsStrategy& lawsStrategy) - : Dispatcher(runtime), lawsStrategy(lawsStrategy) {} + LawsDispatcher(Runtime& runtime) : Dispatcher(runtime) {} void dispatchLoop() override; }; diff --git a/emper/strategies/laws/LawsScheduler.cpp b/emper/strategies/laws/LawsScheduler.cpp index 04647b975241352e64ab93b60c331cdb10d44d4b..6ea84c23fef24c9f9ad52be0b17303c2718039df 100644 --- a/emper/strategies/laws/LawsScheduler.cpp +++ b/emper/strategies/laws/LawsScheduler.cpp @@ -2,7 +2,6 @@ // Copyright © 2020-2021 Florian Schmaus #include "LawsScheduler.hpp" -#include <atomic> // for atomic, memory_order_relaxed #include <cstdint> #include "Emper.hpp" @@ -10,11 +9,11 @@ #include "NextFiberResult.hpp" #include "Runtime.hpp" #include "emper-common.h" +#include "strategies/laws/LawsWorkerStats.hpp" thread_local LawsScheduler::LawsMpscQueue LawsScheduler::priorityQueue; -LawsScheduler::LawsScheduler(Runtime& runtime, LawsStrategy& lawsStrategy) - : AbstractWorkStealingScheduler(runtime), lawsStrategy(lawsStrategy) { +LawsScheduler::LawsScheduler(Runtime& runtime) : AbstractWorkStealingScheduler(runtime) { const workerid_t workerCount = runtime.getWorkerCount(); priorityQueues = new LawsScheduler::LawsMpscQueue*[workerCount]; @@ -38,9 +37,8 @@ void LawsScheduler::scheduleInternal(Fiber& fiber) { // We found a fiber to schedule on a remote prority queue. increaseRefCount(fiber); priorityQueues[affinity]->enqueue(&fiber); - if constexpr (emper::STATS) { - lawsStrategy.scheduledFibersToRemotePriority.fetch_add(1, std::memory_order_relaxed); - } + + emper::statsIncr(LawsStrategy::stats.scheduledFibersToPriority); } scheduleViaWorkStealing: diff --git a/emper/strategies/laws/LawsScheduler.hpp b/emper/strategies/laws/LawsScheduler.hpp index a1edcaf995b5e1358ddd794a05f24091c021ff06..d95a4bf06146caaca0af83b35bfd0183108d4019 100644 --- a/emper/strategies/laws/LawsScheduler.hpp +++ b/emper/strategies/laws/LawsScheduler.hpp @@ -3,12 +3,9 @@ #pragma once #include "Fiber.hpp" -#include "emper-common.h" // IWYU pragma: keep -#include "emper-config.h" // IWYU pragma: keep #include "lib/adt/MpscQueue.hpp" #include "strategies/AbstractWorkStealingScheduler.hpp" -class LawsStrategy; class Runtime; struct NextFiberResult; @@ -20,20 +17,11 @@ class LawsScheduler : public AbstractWorkStealingScheduler { static thread_local LawsMpscQueue priorityQueue; -#pragma GCC diagnostic push -#pragma GCC diagnostic ignored "-Wattributes" - LawsStrategy& lawsStrategy -#ifndef EMPER_STATS - ATTR_UNUSED -#endif - ; -#pragma GCC diagnostic pop - protected: void scheduleInternal(Fiber& fiber) override; public: - LawsScheduler(Runtime& runtime, LawsStrategy& lawsStrategy); + LawsScheduler(Runtime& runtime); auto nextFiber() -> NextFiberResult override; }; diff --git a/emper/strategies/laws/LawsStrategy.cpp b/emper/strategies/laws/LawsStrategy.cpp index a12a5ad074ad09dd39fd6011bcf787d8823cb176..57116344cd124bab6f6a48d604cc7f97aacc6683 100644 --- a/emper/strategies/laws/LawsStrategy.cpp +++ b/emper/strategies/laws/LawsStrategy.cpp @@ -2,19 +2,20 @@ // Copyright © 2020-2021 Florian Schmaus #include "LawsStrategy.hpp" +#include <functional> + #include "strategies/laws/LawsDispatcher.hpp" // for LawsDispatcher #include "strategies/laws/LawsScheduler.hpp" // for LawsScheduler #include "strategies/laws/LawsStrategyStats.hpp" // for LawsStrategyStats +#include "strategies/laws/LawsWorkerStats.hpp" + +thread_local LawsWorkerStats LawsStrategy::stats; LawsStrategy::LawsStrategy(Runtime& runtime) : AbstractWorkStealingStrategy(runtime), - scheduler(runtime, *this), - dispatcher(runtime, *this), - scheduledFibersToRemotePriority(0), - dispatchedFiberFromPriority(0), - dispatchedFiberFromLocal(0), - dispatchedFiberStolen(0), - dispatchedFiberFromAnywhere(0) {} + scheduler(runtime), + dispatcher(runtime), + allWorkerStats([] { return &stats; }, runtime) {} auto LawsStrategy::getScheduler() -> LawsScheduler& { return scheduler; } diff --git a/emper/strategies/laws/LawsStrategy.hpp b/emper/strategies/laws/LawsStrategy.hpp index d06448689b0664c3d2d30c606ae2d1f5cc07ba20..113427d3df2d735d1e717315715fde2c1a6cbb6d 100644 --- a/emper/strategies/laws/LawsStrategy.hpp +++ b/emper/strategies/laws/LawsStrategy.hpp @@ -2,10 +2,10 @@ // Copyright © 2020-2021 Florian Schmaus #pragma once -#include <atomic> #include <cstdint> #include <memory> +#include "WorkerLocalData.hpp" #include "strategies/AbstractWorkStealingScheduler.hpp" #include "strategies/AbstractWorkStealingStrategy.hpp" #include "strategies/laws/LawsDispatcher.hpp" @@ -15,6 +15,7 @@ class LawsStrategyFactory; class LawsStrategyStats; class Runtime; class RuntimeStrategyStats; +struct LawsWorkerStats; class LawsStrategy : public AbstractWorkStealingStrategy { private: @@ -29,12 +30,9 @@ class LawsStrategy : public AbstractWorkStealingStrategy { LawsScheduler scheduler; LawsDispatcher dispatcher; - // TODO: Align those all to cache line! - std::atomic<std::uint64_t> scheduledFibersToRemotePriority; - std::atomic<std::uint64_t> dispatchedFiberFromPriority; - std::atomic<std::uint64_t> dispatchedFiberFromLocal; - std::atomic<std::uint64_t> dispatchedFiberStolen; - std::atomic<std::uint64_t> dispatchedFiberFromAnywhere; + static thread_local LawsWorkerStats stats; + + WorkerLocalData<LawsWorkerStats> allWorkerStats; LawsStrategy(Runtime& runtime); diff --git a/emper/strategies/laws/LawsStrategyStats.cpp b/emper/strategies/laws/LawsStrategyStats.cpp index 966c84c37f01f69b33379af3c1f020a2283ff276..451d112606b1043ad7d5774a7a04ccec21f7ed93 100644 --- a/emper/strategies/laws/LawsStrategyStats.cpp +++ b/emper/strategies/laws/LawsStrategyStats.cpp @@ -2,45 +2,31 @@ // Copyright © 2020-2021 Florian Schmaus #include "LawsStrategyStats.hpp" -#include <atomic> #include <iostream> +#include <string> #include "LawsStrategy.hpp" +#include "WorkerLocalData.hpp" LawsStrategyStats::LawsStrategyStats(LawsStrategy& lawsStrategy) : AbstractWorkStealingStats(lawsStrategy), - scheduledFibersToRemotePriority(lawsStrategy.scheduledFibersToRemotePriority), - dispatchedFiberFromPriority(lawsStrategy.dispatchedFiberFromPriority), - dispatchedFiberFromLocal(lawsStrategy.dispatchedFiberFromLocal), - dispatchedFiberStolen(lawsStrategy.dispatchedFiberStolen), - dispatchedFiberFromAnywhere(lawsStrategy.dispatchedFiberFromAnywhere) {} - -auto LawsStrategyStats::getScheduledFibersToRemotePriority() const -> uint64_t { - return scheduledFibersToRemotePriority; -} - -auto LawsStrategyStats::getDispatchedFiberFromPriority() const -> uint64_t { - return dispatchedFiberFromPriority; -} - -auto LawsStrategyStats::getDispatchedFiberFromLocal() const -> uint64_t { - return dispatchedFiberFromLocal; -} - -auto LawsStrategyStats::getDispatchedFiberStolen() const -> uint64_t { - return dispatchedFiberStolen; -} - -auto LawsStrategyStats::getDispatchedFiberFromAnywhere() const -> uint64_t { - return dispatchedFiberFromAnywhere; + workerStats(lawsStrategy.allWorkerStats.getSnapshot()) { + for (auto& workerStat : workerStats) { + comulatedWorkerStats += workerStat; + } } void LawsStrategyStats::print() { - // TODO: Print also the stats from AbstractWorkStealingStrategy. - std::cout << "LawsStrategyStats" - << " scheduledFibersToRemotePriority:" << scheduledFibersToRemotePriority - << " dispatchedFiberFromPriority:" << dispatchedFiberFromPriority - << " dispatchedFiberFromLocal:" << dispatchedFiberFromLocal - << " dispatchedFiberStolen:" << dispatchedFiberStolen - << " dispatchedFiberFromAnywhere:" << dispatchedFiberFromAnywhere << std::endl; + AbstractWorkStealingStats::print(); + + std::cout << "total-scheduled-fibers-to-priority: " + << std::to_string(comulatedWorkerStats.scheduledFibersToPriority) << std::endl + << "total-dispatched-fibers-from-priority: " + << std::to_string(comulatedWorkerStats.dispatchedFibersFromPriority) << std::endl + << "total-dispatched-fibers-from-local: " + << std::to_string(comulatedWorkerStats.dispatchedFibersFromLocal) << std::endl + << "total-dispatched-fibers-stolen: " + << std::to_string(comulatedWorkerStats.dispatchedFibersStolen) << std::endl + << "total-dispatched-fibers-from-anywhere-queue: " + << std::to_string(comulatedWorkerStats.dispatchedFibersFromAnywhereQueue) << std::endl; } diff --git a/emper/strategies/laws/LawsStrategyStats.hpp b/emper/strategies/laws/LawsStrategyStats.hpp index 2d4cff89c5d6021867009fcb595014bae4002461..35c30d9feb9297a5229efd1e3ed40ac069f2704a 100644 --- a/emper/strategies/laws/LawsStrategyStats.hpp +++ b/emper/strategies/laws/LawsStrategyStats.hpp @@ -2,28 +2,19 @@ // Copyright © 2020-2021 Florian Schmaus #pragma once -#include <cstdint> +#include <vector> #include "strategies/AbstractWorkStealingStats.hpp" +#include "strategies/laws/LawsWorkerStats.hpp" class LawsStrategy; class LawsStrategyStats : public AbstractWorkStealingStats { - private: - const uint64_t scheduledFibersToRemotePriority; - const uint64_t dispatchedFiberFromPriority; - const uint64_t dispatchedFiberFromLocal; - const uint64_t dispatchedFiberStolen; - const uint64_t dispatchedFiberFromAnywhere; - public: - LawsStrategyStats(LawsStrategy& lawsStrategy); + std::vector<LawsWorkerStats> workerStats; + LawsWorkerStats comulatedWorkerStats; - [[nodiscard]] auto getScheduledFibersToRemotePriority() const -> uint64_t; - [[nodiscard]] auto getDispatchedFiberFromPriority() const -> uint64_t; - [[nodiscard]] auto getDispatchedFiberFromLocal() const -> uint64_t; - [[nodiscard]] auto getDispatchedFiberStolen() const -> uint64_t; - [[nodiscard]] auto getDispatchedFiberFromAnywhere() const -> uint64_t; + LawsStrategyStats(LawsStrategy& lawsStrategy); void print() override; }; diff --git a/emper/strategies/laws/LawsWorkerStats.hpp b/emper/strategies/laws/LawsWorkerStats.hpp new file mode 100644 index 0000000000000000000000000000000000000000..f60b211d1dbd96dda5319e7f2047a5c96d690e22 --- /dev/null +++ b/emper/strategies/laws/LawsWorkerStats.hpp @@ -0,0 +1,22 @@ +// SPDX-License-Identifier: LGPL-3.0-or-later +// Copyright © 2020-2021 Florian Schmaus +#pragma once + +#include <cstdint> + +struct LawsWorkerStats { + uint64_t scheduledFibersToPriority = 0; + uint64_t dispatchedFibersFromPriority = 0; + uint64_t dispatchedFibersFromLocal = 0; + uint64_t dispatchedFibersStolen = 0; + uint64_t dispatchedFibersFromAnywhereQueue = 0; + + auto operator+=(const LawsWorkerStats& other) -> LawsWorkerStats& { + scheduledFibersToPriority += other.scheduledFibersToPriority; + dispatchedFibersFromPriority += other.dispatchedFibersFromPriority; + dispatchedFibersFromLocal += other.dispatchedFibersFromLocal; + dispatchedFibersStolen += other.dispatchedFibersStolen; + dispatchedFibersFromAnywhereQueue += other.dispatchedFibersFromAnywhereQueue; + return *this; + } +};