From 82b660f5009ff1893612f66e60dc16f12f837d6c Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Mon, 23 Sep 2024 10:21:37 +0200 Subject: [PATCH 01/25] first skeleton of PeriodicThread implementation --- .../BipedalLocomotion/System/PeriodicThread.h | 44 +++++ src/System/src/PeriodicThread.cpp | 175 ++++++++++++++++++ 2 files changed, 219 insertions(+) create mode 100644 src/System/include/BipedalLocomotion/System/PeriodicThread.h create mode 100644 src/System/src/PeriodicThread.cpp diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h new file mode 100644 index 0000000000..22c68b57ca --- /dev/null +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -0,0 +1,44 @@ +#include +#include + +namespace BipedalLocomotion +{ +namespace System +{ +/** + * @brief This class implements a periodic thread. The user has to inherit from this class and + * implement the virtual methods. + */ + +class PeriodicThread +{ +public: + // Default constructor + PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), + int maximumNumberOfAcceptedDeadlineMiss = -1, + int priority = 0, + int policy = SCHED_OTHER); + + // Virtual destructor + virtual ~PeriodicThread() = default; + + /** + * @brief This method is called at each iteration of the thread + */ + virtual void run() = 0; + /** + * @brief This method is called at the beginning of the thread + */ + virtual void threadInit() = 0; + /** + * @brief This method is called at the end of the thread + */ + virtual void stop() = 0; + +private: + // private implementation + class Impl; + std::unique_ptr m_impl; +}; +} // namespace System +} // namespace BipedalLocomotion \ No newline at end of file diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp new file mode 100644 index 0000000000..a9e31cc0ac --- /dev/null +++ b/src/System/src/PeriodicThread.cpp @@ -0,0 +1,175 @@ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +namespace BipedalLocomotion +{ +namespace System +{ + +class PeriodicThread::Impl +{ + // maximum number of accepted deadline misses + int m_maximumNumberOfAcceptedDeadlineMiss; + // deadline miss + int m_deadlineMiss; + // scheuling policy + int m_policy; + // priority + int m_priority; + // period + std::chrono::nanoseconds m_period; + // owner + std::unique_ptr m_owner; + // thread + std::thread m_thread; + // is running + bool m_isRunning = false; + // stop flag + static std::atomic _askToStop; + +public: + Impl(std::unique_ptr owner, + std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), + int maximumNumberOfAcceptedDeadlineMiss = -1, + int priority = 0, + int policy = SCHED_OTHER) + : m_owner(std::move(owner)) + , m_period(period) + , m_thread() + , m_deadlineMiss(0) + , m_maximumNumberOfAcceptedDeadlineMiss(maximumNumberOfAcceptedDeadlineMiss) + , m_priority(priority) + , m_policy(policy) + , m_isRunning(false) + { + // set the signal handler for ctrl+c + std::signal(SIGINT, signalHandler); + }; + +private: + static void signalHandler(int sig) + { + _askToStop = true; + } + + void step() + { + // get the current time + auto now = std::chrono::high_resolution_clock::now(); + // get the next wake up time + auto nextWakeUpTime = now + m_period; + + // run user defined function + m_owner->run(); + + // check if the deadline is missed + if (nextWakeUpTime < std::chrono::high_resolution_clock::now()) + { + m_deadlineMiss++; + if (m_maximumNumberOfAcceptedDeadlineMiss > 0) + { + if (m_deadlineMiss > m_maximumNumberOfAcceptedDeadlineMiss) + { + // we have to close the runner + m_isRunning = false; + return; + } + } + } + + // yield the CPU + std::this_thread::yield(); + + // wait until the next deadline + std::this_thread::sleep_until(nextWakeUpTime); + }; + + void run() + { + while ((m_isRunning) || !(_askToStop)) + { + step(); + } + stop(); + }; + + void start() + { + m_isRunning = true; + + // lambda wrapper for the thread function + auto threadFunctionLambda = [this]() { this->threadFunction(); }; + + m_thread = std::thread(threadFunctionLambda); + }; + + void setPolicy() + { + // get the current thread native handle + pthread_t nativeHandle = pthread_self(); + // get the current thread parameters + sched_param params; + params.sched_priority = m_priority; + // set the new policy + pthread_setschedparam(nativeHandle, m_policy, ¶ms); + }; + + void threadInit() + { + m_owner->threadInit(); + } + + void threadFunction() + { + threadInit(); + setPolicy(); + run(); + }; + + void stop() + { + m_owner->stop(); + }; + + // destructor + ~Impl() + { + if (m_thread.joinable()) + { + m_thread.join(); + } + }; +}; + +// static member initialization +std::atomic PeriodicThread::Impl::_askToStop = false; + +PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, + int maximumNumberOfAcceptedDeadlineMiss, + int priority, + int policy) +{ + m_impl = std::make_unique(std::unique_ptr(this), + period, + maximumNumberOfAcceptedDeadlineMiss, + priority, + policy); +} + +} // namespace System +} // namespace BipedalLocomotion + +// main function +int main() +{ + + return 0; +} \ No newline at end of file From 1089c7e15c122807ae8953db0e8e69886ea98708 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Mon, 23 Sep 2024 10:28:24 +0200 Subject: [PATCH 02/25] switch to Bipedal clock --- src/System/src/PeriodicThread.cpp | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index a9e31cc0ac..4006dfe30e 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -1,4 +1,6 @@ +#include #include + #include #include #include @@ -63,7 +65,7 @@ class PeriodicThread::Impl void step() { // get the current time - auto now = std::chrono::high_resolution_clock::now(); + auto now = BipedalLocomotion::clock().now(); // get the next wake up time auto nextWakeUpTime = now + m_period; @@ -71,7 +73,7 @@ class PeriodicThread::Impl m_owner->run(); // check if the deadline is missed - if (nextWakeUpTime < std::chrono::high_resolution_clock::now()) + if (nextWakeUpTime < BipedalLocomotion::clock().now()) { m_deadlineMiss++; if (m_maximumNumberOfAcceptedDeadlineMiss > 0) @@ -86,10 +88,10 @@ class PeriodicThread::Impl } // yield the CPU - std::this_thread::yield(); + BipedalLocomotion::clock().yield(); // wait until the next deadline - std::this_thread::sleep_until(nextWakeUpTime); + BipedalLocomotion::clock().sleepUntil(nextWakeUpTime); }; void run() From 7d5f66e4df61ed881f21e5769d3c0389b31a7560 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Mon, 23 Sep 2024 17:42:26 +0200 Subject: [PATCH 03/25] use raw pointers to avoid circular dependency --- src/System/CMakeLists.txt | 3 +- .../BipedalLocomotion/System/PeriodicThread.h | 49 +++- src/System/src/PeriodicThread.cpp | 214 +++++++++++++----- 3 files changed, 202 insertions(+), 64 deletions(-) diff --git a/src/System/CMakeLists.txt b/src/System/CMakeLists.txt index dd30345c6f..a71f15117a 100644 --- a/src/System/CMakeLists.txt +++ b/src/System/CMakeLists.txt @@ -19,9 +19,10 @@ if(FRAMEWORK_COMPILE_System) ${H_PREFIX}/QuitHandler.h ${H_PREFIX}/Barrier.h ${H_PREFIX}/TimeProfiler.h ${H_PREFIX}/WeightProvider.h ${H_PREFIX}/ConstantWeightProvider.h + ${H_PREFIX}/PeriodicThread.h SOURCES src/VariablesHandler.cpp src/LinearTask.cpp src/StdClock.cpp src/Clock.cpp src/QuitHandler.cpp src/Barrier.cpp - src/ConstantWeightProvider.cpp src/TimeProfiler.cpp + src/ConstantWeightProvider.cpp src/TimeProfiler.cpp src/PeriodicThread.cpp PUBLIC_LINK_LIBRARIES BipedalLocomotion::ParametersHandler Eigen3::Eigen SUBDIRECTORIES tests YarpImplementation RosImplementation ) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 22c68b57ca..96e7f53d35 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -1,3 +1,13 @@ +/** + * @file PeriodicThread.h + * @authors Lorenzo Moretti + * @copyright 2024 Istituto Italiano di Tecnologia (IIT). This software may be modified and + * distributed under the terms of the BSD-3-Clause license. + */ + +#ifndef BIPEDAL_LOCOMOTION_SYSTEM_PERIODIC_THREAD_H +#define BIPEDAL_LOCOMOTION_SYSTEM_PERIODIC_THREAD_H + #include #include @@ -19,26 +29,45 @@ class PeriodicThread int priority = 0, int policy = SCHED_OTHER); - // Virtual destructor - virtual ~PeriodicThread() = default; + // Destructor + ~PeriodicThread(); /** - * @brief This method is called at each iteration of the thread + * @brief This method is called at each iteration of the thread. + * Override this method to implement the thread itself. + * @return true if the thread has to continue, false otherwise. */ - virtual void run() = 0; + virtual bool run() = 0; + /** - * @brief This method is called at the beginning of the thread + * @brief This method is called at the beginning of the thread. + * @return true if the initialization was successful, false otherwise. */ - virtual void threadInit() = 0; + virtual bool threadInit(); + /** - * @brief This method is called at the end of the thread + * @brief Start the thread + * @return true if the thread was correctly started, false otherwise. */ - virtual void stop() = 0; + bool start(); + + /** + * @brief Call this method to stop the thread. + */ + void stop(); + + /** + * @brief Check if the thread is running. + * @return true if the thread is running, false otherwise. + */ + bool isRunning(); private: // private implementation class Impl; - std::unique_ptr m_impl; + Impl* m_impl; }; } // namespace System -} // namespace BipedalLocomotion \ No newline at end of file +} // namespace BipedalLocomotion + +#endif // BIPEDAL_LOCOMOTION_SYSTEM_PERIODIC_THREAD_H \ No newline at end of file diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 4006dfe30e..6b04f75549 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -1,6 +1,8 @@ #include #include +#include +#include #include #include #include @@ -16,6 +18,9 @@ namespace BipedalLocomotion namespace System { +/** + * @brief PeriodicThread implementation + */ class PeriodicThread::Impl { // maximum number of accepted deadline misses @@ -29,21 +34,33 @@ class PeriodicThread::Impl // period std::chrono::nanoseconds m_period; // owner - std::unique_ptr m_owner; + PeriodicThread* m_owner; // thread std::thread m_thread; // is running - bool m_isRunning = false; - // stop flag - static std::atomic _askToStop; + std::atomic m_isRunning = false; + // ask to stop + std::atomic m_askToStop = false; + // force to stop flag + static std::atomic _forceToStop; public: - Impl(std::unique_ptr owner, + /** + * @brief Constructor + * @param owner owner of the thread + * @param period period of the thread + * @param maximumNumberOfAcceptedDeadlineMiss maximum number of accepted deadline misses + * @param priority priority of the thread + * @param policy policy of the thread + * For example, in Linux,you can set the policy to SCHED_FIFO and priority to 80 in order to + * improve the real time performances of the thread. + */ + Impl(PeriodicThread* owner, std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), int maximumNumberOfAcceptedDeadlineMiss = -1, int priority = 0, int policy = SCHED_OTHER) - : m_owner(std::move(owner)) + : m_owner(owner) , m_period(period) , m_thread() , m_deadlineMiss(0) @@ -56,12 +73,76 @@ class PeriodicThread::Impl std::signal(SIGINT, signalHandler); }; + /** + * @brief Destructor, it joins the thread. + */ + ~Impl() + { + if (m_thread.joinable()) + { + m_thread.join(); + } + }; + + /** + * @brief Start the thread + * @return true if the thread was correctly started, false otherwise. + */ + bool start() + { + if (m_isRunning) + { + // thread is already running + BipedalLocomotion::log()->error("[PeriodicThread::start] The thread is already " + "running."); + return false; + } else + { + m_isRunning = true; + } + + // lambda wrapper for the thread function + auto threadFunctionLambda = [this]() { this->threadFunction(); }; + + m_thread = std::thread(threadFunctionLambda); + + return m_thread.joinable(); + }; + + /** + * @brief Stop the thread + */ + void stop() + { + m_askToStop = true; + }; + + /** + * @brief Check if the thread is running + * @return true if the thread is running, false otherwise. + */ + bool isRunning() + { + return m_isRunning; + }; + private: + /** + * @brief Signal handler. It is used to stop the thread when ctrl+c is pressed. + * @param sig signal + */ static void signalHandler(int sig) { - _askToStop = true; + if (sig == SIGINT) + { + _forceToStop = true; + } } + /** + * @brief Step function. It is called at each iteration of the thread. Advances the thread by + * one step. It runs the user defined function and oversees the thread timing and resources. + */ void step() { // get the current time @@ -70,10 +151,21 @@ class PeriodicThread::Impl auto nextWakeUpTime = now + m_period; // run user defined function - m_owner->run(); + if (!m_owner->run()) + { + m_isRunning = false; + return; + } + + // check if the thread has to stop + if (m_askToStop) + { + m_isRunning = false; + return; + } // check if the deadline is missed - if (nextWakeUpTime < BipedalLocomotion::clock().now()) + if (BipedalLocomotion::clock().now() > nextWakeUpTime) { m_deadlineMiss++; if (m_maximumNumberOfAcceptedDeadlineMiss > 0) @@ -94,84 +186,100 @@ class PeriodicThread::Impl BipedalLocomotion::clock().sleepUntil(nextWakeUpTime); }; + /** + * @brief Run function. It is the main function of the thread. + */ void run() { - while ((m_isRunning) || !(_askToStop)) + while ((m_isRunning) || !(_forceToStop)) { step(); } - stop(); }; - void start() - { - m_isRunning = true; - - // lambda wrapper for the thread function - auto threadFunctionLambda = [this]() { this->threadFunction(); }; - - m_thread = std::thread(threadFunctionLambda); - }; - - void setPolicy() + /** + * @brief Set the policy of the thread. + * @return true if the policy was correctly set, false otherwise. + */ + bool setPolicy() { +#ifdef __linux__ // get the current thread native handle pthread_t nativeHandle = pthread_self(); // get the current thread parameters sched_param params; params.sched_priority = m_priority; // set the new policy - pthread_setschedparam(nativeHandle, m_policy, ¶ms); + return (pthread_setschedparam(nativeHandle, m_policy, ¶ms) == 0); +#else + return true; +#endif }; - void threadInit() + /** + * @brief Initialize the thread. + * @return true if the thread was correctly initialized, false otherwise. + */ + bool threadInit() { - m_owner->threadInit(); + return m_owner->threadInit(); } + /** + * @brief Thread function. It is the function passed to std::thread(). + */ void threadFunction() { - threadInit(); - setPolicy(); - run(); - }; - - void stop() - { - m_owner->stop(); - }; + constexpr auto logPrefix = "[PeriodicThread::threadFunction]"; - // destructor - ~Impl() - { - if (m_thread.joinable()) + if (!threadInit()) { - m_thread.join(); + BipedalLocomotion::log()->error("{} Failed to initialize the thread", logPrefix); + return; + } + if (!setPolicy()) + { + BipedalLocomotion::log()->error("{} Failed to set the policy", logPrefix); + return; } + run(); }; -}; + +}; // class Impl // static member initialization -std::atomic PeriodicThread::Impl::_askToStop = false; +std::atomic PeriodicThread::Impl::_forceToStop = false; PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, int maximumNumberOfAcceptedDeadlineMiss, int priority, int policy) + : m_impl(new Impl(this, period, maximumNumberOfAcceptedDeadlineMiss, priority, policy)){}; + +PeriodicThread::~PeriodicThread() { - m_impl = std::make_unique(std::unique_ptr(this), - period, - maximumNumberOfAcceptedDeadlineMiss, - priority, - policy); -} + delete m_impl; +}; -} // namespace System -} // namespace BipedalLocomotion +bool PeriodicThread::threadInit() +{ + return true; +}; -// main function -int main() +void PeriodicThread::stop() { + m_impl->stop(); +}; + +bool PeriodicThread::start() +{ + return m_impl->start(); +} - return 0; -} \ No newline at end of file +bool PeriodicThread::isRunning() +{ + return m_impl->isRunning(); +} + +} // namespace System +} // namespace BipedalLocomotion \ No newline at end of file From 80c1c397c412130427be2b645992d34629a6a0c4 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Mon, 23 Sep 2024 18:33:38 +0200 Subject: [PATCH 04/25] reduce smart pointers where possible --- .../BipedalLocomotion/System/PeriodicThread.h | 2 +- src/System/src/PeriodicThread.cpp | 12 +++++++----- 2 files changed, 8 insertions(+), 6 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 96e7f53d35..ef946f02e5 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -65,7 +65,7 @@ class PeriodicThread private: // private implementation class Impl; - Impl* m_impl; + std::unique_ptr m_impl; }; } // namespace System } // namespace BipedalLocomotion diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 6b04f75549..649d0ae5bf 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -2,7 +2,6 @@ #include #include -#include #include #include #include @@ -254,13 +253,16 @@ PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, int maximumNumberOfAcceptedDeadlineMiss, int priority, int policy) - : m_impl(new Impl(this, period, maximumNumberOfAcceptedDeadlineMiss, priority, policy)){}; - -PeriodicThread::~PeriodicThread() { - delete m_impl; + m_impl = std::make_unique(this, + period, + maximumNumberOfAcceptedDeadlineMiss, + priority, + policy); }; +PeriodicThread::~PeriodicThread(){}; + bool PeriodicThread::threadInit() { return true; From eb597b492d066aeb31c26893ffa69db6506e8f02 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Thu, 26 Sep 2024 09:53:45 +0200 Subject: [PATCH 05/25] switch from pure virtual to virtual run memebr function It is not clear why with the pure virtual run function, things do not work. It seems that the pure virtual function is being called, rather than the user override one. This causes a segmentation fault. --- examples/CMakeLists.txt | 1 + examples/PeriodicThread/CMakeLists.txt | 5 +++ examples/PeriodicThread/main.cpp | 41 +++++++++++++++++++ .../BipedalLocomotion/System/PeriodicThread.h | 13 +++--- src/System/src/PeriodicThread.cpp | 23 +++++++++-- 5 files changed, 74 insertions(+), 9 deletions(-) create mode 100644 examples/PeriodicThread/CMakeLists.txt create mode 100644 examples/PeriodicThread/main.cpp diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index a45244e810..90550ac318 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -1 +1,2 @@ add_subdirectory(UnicycleTrajectoryGenerator) +add_subdirectory(PeriodicThread) diff --git a/examples/PeriodicThread/CMakeLists.txt b/examples/PeriodicThread/CMakeLists.txt new file mode 100644 index 0000000000..0fca178f5b --- /dev/null +++ b/examples/PeriodicThread/CMakeLists.txt @@ -0,0 +1,5 @@ +add_executable(PeriodicThreadExample main.cpp) + +target_link_libraries(PeriodicThreadExample +BipedalLocomotion::System +BipedalLocomotion::TextLogging) \ No newline at end of file diff --git a/examples/PeriodicThread/main.cpp b/examples/PeriodicThread/main.cpp new file mode 100644 index 0000000000..21f33a6609 --- /dev/null +++ b/examples/PeriodicThread/main.cpp @@ -0,0 +1,41 @@ +#include +#include +#include +#include +#include + +class Thread : public BipedalLocomotion::System::PeriodicThread +{ +public: + Thread(); + bool run() override; + + bool threadInit() override; +}; + +bool Thread::run() +{ + + BipedalLocomotion::clock().sleepFor(std::chrono::milliseconds(500)); + BipedalLocomotion::log()->info("[Thread::run] Thread is running."); + + return true; +} + +Thread::Thread() + : BipedalLocomotion::System::PeriodicThread(std::chrono::milliseconds(1000)){}; + +bool Thread::threadInit() +{ + BipedalLocomotion::log()->info("[Thread::threadInit] Thread is initialized."); + return true; +} + +int main() +{ + // Thread thread; + auto thread = Thread(); + std::cerr << "Thread class pointer id: " << &thread << std::endl; + thread.start(); + return EXIT_SUCCESS; +} diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index ef946f02e5..9b7f902d53 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -24,20 +24,21 @@ class PeriodicThread { public: // Default constructor - PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), - int maximumNumberOfAcceptedDeadlineMiss = -1, - int priority = 0, - int policy = SCHED_OTHER); + explicit PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), + int maximumNumberOfAcceptedDeadlineMiss = -1, + int priority = 0, + int policy = SCHED_OTHER); // Destructor - ~PeriodicThread(); + virtual ~PeriodicThread(); /** * @brief This method is called at each iteration of the thread. * Override this method to implement the thread itself. * @return true if the thread has to continue, false otherwise. */ - virtual bool run() = 0; + // virtual bool run() = 0; + virtual bool run(); /** * @brief This method is called at the beginning of the thread. diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 649d0ae5bf..5bcd496737 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -6,11 +6,13 @@ #include #include #include +#include #include #include #include #include #include +#include namespace BipedalLocomotion { @@ -136,6 +138,9 @@ class PeriodicThread::Impl { _forceToStop = true; } + + BipedalLocomotion::log()->error("stopping the thread, _forcetostop: {}", + _forceToStop.load()); } /** @@ -149,12 +154,16 @@ class PeriodicThread::Impl // get the next wake up time auto nextWakeUpTime = now + m_period; + std::cerr << " m_owner id " << m_owner << std::endl; + // run user defined function + BipedalLocomotion::log()->debug("about to call the owner run function"); if (!m_owner->run()) { m_isRunning = false; return; } + BipedalLocomotion::log()->debug("the owner run function has been called"); // check if the thread has to stop if (m_askToStop) @@ -190,10 +199,13 @@ class PeriodicThread::Impl */ void run() { - while ((m_isRunning) || !(_forceToStop)) + std::cerr << " m_impl id " << this << std::endl; + while ((m_isRunning) && !(_forceToStop.load())) { - step(); + this->step(); } + + BipedalLocomotion::log()->error("closing thread"); }; /** @@ -241,7 +253,7 @@ class PeriodicThread::Impl BipedalLocomotion::log()->error("{} Failed to set the policy", logPrefix); return; } - run(); + this->run(); }; }; // class Impl @@ -263,6 +275,11 @@ PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, PeriodicThread::~PeriodicThread(){}; +bool PeriodicThread::run() +{ + return true; +}; + bool PeriodicThread::threadInit() { return true; From 2a47b1195cc18f98d38a150013cfa02c6ab8d1cc Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Thu, 26 Sep 2024 18:09:58 +0200 Subject: [PATCH 06/25] remove use of pimpl pattern --- examples/PeriodicThread/main.cpp | 10 +- .../BipedalLocomotion/System/PeriodicThread.h | 75 +++- src/System/src/PeriodicThread.cpp | 380 ++++++------------ 3 files changed, 193 insertions(+), 272 deletions(-) diff --git a/examples/PeriodicThread/main.cpp b/examples/PeriodicThread/main.cpp index 21f33a6609..abe5ab9a27 100644 --- a/examples/PeriodicThread/main.cpp +++ b/examples/PeriodicThread/main.cpp @@ -15,10 +15,7 @@ class Thread : public BipedalLocomotion::System::PeriodicThread bool Thread::run() { - - BipedalLocomotion::clock().sleepFor(std::chrono::milliseconds(500)); BipedalLocomotion::log()->info("[Thread::run] Thread is running."); - return true; } @@ -37,5 +34,12 @@ int main() auto thread = Thread(); std::cerr << "Thread class pointer id: " << &thread << std::endl; thread.start(); + + while (thread.isRunning()) + { + std::this_thread::sleep_for(std::chrono::milliseconds(10000)); + thread.stop(); + } + BipedalLocomotion::log()->info("[main] Thread about to exit."); return EXIT_SUCCESS; } diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 9b7f902d53..65d35297af 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -8,8 +8,9 @@ #ifndef BIPEDAL_LOCOMOTION_SYSTEM_PERIODIC_THREAD_H #define BIPEDAL_LOCOMOTION_SYSTEM_PERIODIC_THREAD_H +#include #include -#include +#include namespace BipedalLocomotion { @@ -24,20 +25,43 @@ class PeriodicThread { public: // Default constructor - explicit PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), - int maximumNumberOfAcceptedDeadlineMiss = -1, - int priority = 0, - int policy = SCHED_OTHER); + PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), + int maximumNumberOfAcceptedDeadlineMiss = -1, + int priority = 0, + int policy = SCHED_OTHER); // Destructor virtual ~PeriodicThread(); + // Copy constructor + PeriodicThread(const PeriodicThread&) = delete; + + // Copy assignment operator + PeriodicThread& operator=(const PeriodicThread&) = delete; + + /** + * @brief Start the thread + * @return true if the thread was correctly started, false otherwise. + */ + bool start(); + + /** + * @brief Call this method to stop the thread. + */ + void stop(); + + /** + * @brief Check if the thread is running. + * @return true if the thread is running, false otherwise. + */ + bool isRunning(); + +protected: /** * @brief This method is called at each iteration of the thread. - * Override this method to implement the thread itself. + * Override this method to implement the task to be performed from the thread. * @return true if the thread has to continue, false otherwise. */ - // virtual bool run() = 0; virtual bool run(); /** @@ -46,27 +70,40 @@ class PeriodicThread */ virtual bool threadInit(); +private: /** - * @brief Start the thread - * @return true if the thread was correctly started, false otherwise. + * @brief run the periodic thread. */ - bool start(); + void threadFunction(); /** - * @brief Call this method to stop the thread. + * @brief Advance the thread of one step, calling the user defined run function once. */ - void stop(); + void advance(); /** - * @brief Check if the thread is running. - * @return true if the thread is running, false otherwise. + * @brief Set the policy of the thread. + * @return true if the policy was correctly set, false otherwise. */ - bool isRunning(); + bool setPolicy(); -private: - // private implementation - class Impl; - std::unique_ptr m_impl; + std::chrono::nanoseconds m_period = std::chrono::nanoseconds(100000); /**< Period of the thread. + */ + + int m_maximumNumberOfAcceptedDeadlineMiss = -1; /**< Maximum number of accepted deadline miss. + */ + + int m_deadlineMiss = 0; /**< Number of deadline miss. */ + + int m_priority = 0; /**< Priority of the thread. */ + + int m_policy = SCHED_OTHER; /**< Policy of the thread. */ + + std::atomic m_isRunning = false; /**< Flag to check if the thread is running. */ + + std::atomic m_askToStop = false; /**< Flag to ask to stop the thread. */ + + std::thread m_thread; /**< Thread object. */ }; } // namespace System } // namespace BipedalLocomotion diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 5bcd496737..d809199dfd 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -1,282 +1,92 @@ -#include -#include -#include - #include #include #include -#include -#include -#include #include #include -#include -#include -#include + +#include +#include +#include namespace BipedalLocomotion { namespace System { -/** - * @brief PeriodicThread implementation - */ -class PeriodicThread::Impl -{ - // maximum number of accepted deadline misses - int m_maximumNumberOfAcceptedDeadlineMiss; - // deadline miss - int m_deadlineMiss; - // scheuling policy - int m_policy; - // priority - int m_priority; - // period - std::chrono::nanoseconds m_period; - // owner - PeriodicThread* m_owner; - // thread - std::thread m_thread; - // is running - std::atomic m_isRunning = false; - // ask to stop - std::atomic m_askToStop = false; - // force to stop flag - static std::atomic _forceToStop; - -public: - /** - * @brief Constructor - * @param owner owner of the thread - * @param period period of the thread - * @param maximumNumberOfAcceptedDeadlineMiss maximum number of accepted deadline misses - * @param priority priority of the thread - * @param policy policy of the thread - * For example, in Linux,you can set the policy to SCHED_FIFO and priority to 80 in order to - * improve the real time performances of the thread. - */ - Impl(PeriodicThread* owner, - std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), - int maximumNumberOfAcceptedDeadlineMiss = -1, - int priority = 0, - int policy = SCHED_OTHER) - : m_owner(owner) - , m_period(period) - , m_thread() - , m_deadlineMiss(0) - , m_maximumNumberOfAcceptedDeadlineMiss(maximumNumberOfAcceptedDeadlineMiss) - , m_priority(priority) - , m_policy(policy) - , m_isRunning(false) - { - // set the signal handler for ctrl+c - std::signal(SIGINT, signalHandler); - }; - - /** - * @brief Destructor, it joins the thread. - */ - ~Impl() - { - if (m_thread.joinable()) - { - m_thread.join(); - } - }; +PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, + int maximumNumberOfAcceptedDeadlineMiss, + int priority, + int policy) + : m_period(period) + , m_maximumNumberOfAcceptedDeadlineMiss(maximumNumberOfAcceptedDeadlineMiss) + , m_priority(priority) + , m_policy(policy) + , m_deadlineMiss(0){}; - /** - * @brief Start the thread - * @return true if the thread was correctly started, false otherwise. - */ - bool start() +PeriodicThread::~PeriodicThread() +{ + // join the thread + if (m_thread.joinable()) { - if (m_isRunning) - { - // thread is already running - BipedalLocomotion::log()->error("[PeriodicThread::start] The thread is already " - "running."); - return false; - } else - { - m_isRunning = true; - } - - // lambda wrapper for the thread function - auto threadFunctionLambda = [this]() { this->threadFunction(); }; + m_thread.join(); + } +}; - m_thread = std::thread(threadFunctionLambda); +void PeriodicThread::threadFunction() +{ + constexpr auto logPrefix = "[PeriodicThread::threadFunction]"; - return m_thread.joinable(); - }; + BipedalLocomotion::log()->error("{} about to initialize the thread", logPrefix); - /** - * @brief Stop the thread - */ - void stop() + // thread initialization + if (!threadInit()) { - m_askToStop = true; - }; - - /** - * @brief Check if the thread is running - * @return true if the thread is running, false otherwise. - */ - bool isRunning() - { - return m_isRunning; - }; - -private: - /** - * @brief Signal handler. It is used to stop the thread when ctrl+c is pressed. - * @param sig signal - */ - static void signalHandler(int sig) - { - if (sig == SIGINT) - { - _forceToStop = true; - } - - BipedalLocomotion::log()->error("stopping the thread, _forcetostop: {}", - _forceToStop.load()); + BipedalLocomotion::log()->error("{} Failed to initialize the thread", logPrefix); + return; } - /** - * @brief Step function. It is called at each iteration of the thread. Advances the thread by - * one step. It runs the user defined function and oversees the thread timing and resources. - */ - void step() - { - // get the current time - auto now = BipedalLocomotion::clock().now(); - // get the next wake up time - auto nextWakeUpTime = now + m_period; - - std::cerr << " m_owner id " << m_owner << std::endl; - - // run user defined function - BipedalLocomotion::log()->debug("about to call the owner run function"); - if (!m_owner->run()) - { - m_isRunning = false; - return; - } - BipedalLocomotion::log()->debug("the owner run function has been called"); - - // check if the thread has to stop - if (m_askToStop) - { - m_isRunning = false; - return; - } + BipedalLocomotion::log()->error("{} thread initialized", logPrefix); - // check if the deadline is missed - if (BipedalLocomotion::clock().now() > nextWakeUpTime) - { - m_deadlineMiss++; - if (m_maximumNumberOfAcceptedDeadlineMiss > 0) - { - if (m_deadlineMiss > m_maximumNumberOfAcceptedDeadlineMiss) - { - // we have to close the runner - m_isRunning = false; - return; - } - } - } - - // yield the CPU - BipedalLocomotion::clock().yield(); - - // wait until the next deadline - BipedalLocomotion::clock().sleepUntil(nextWakeUpTime); - }; - - /** - * @brief Run function. It is the main function of the thread. - */ - void run() + // set the policy + if (!setPolicy()) { - std::cerr << " m_impl id " << this << std::endl; - while ((m_isRunning) && !(_forceToStop.load())) - { - this->step(); - } + BipedalLocomotion::log()->error("{} Failed to set the policy", logPrefix); + return; + } - BipedalLocomotion::log()->error("closing thread"); - }; + BipedalLocomotion::log()->error("{} policy set", logPrefix); - /** - * @brief Set the policy of the thread. - * @return true if the policy was correctly set, false otherwise. - */ - bool setPolicy() + // run loop + BipedalLocomotion::log()->error("{} advance... {}", logPrefix, m_isRunning.load()); + while (m_isRunning.load()) { -#ifdef __linux__ - // get the current thread native handle - pthread_t nativeHandle = pthread_self(); - // get the current thread parameters - sched_param params; - params.sched_priority = m_priority; - // set the new policy - return (pthread_setschedparam(nativeHandle, m_policy, ¶ms) == 0); -#else - return true; -#endif - }; - - /** - * @brief Initialize the thread. - * @return true if the thread was correctly initialized, false otherwise. - */ - bool threadInit() - { - return m_owner->threadInit(); + this->advance(); } - /** - * @brief Thread function. It is the function passed to std::thread(). - */ - void threadFunction() - { - constexpr auto logPrefix = "[PeriodicThread::threadFunction]"; + m_isRunning.store(false); + BipedalLocomotion::log()->error("{} Completing... {}", logPrefix, m_isRunning.load()); - if (!threadInit()) - { - BipedalLocomotion::log()->error("{} Failed to initialize the thread", logPrefix); - return; - } - if (!setPolicy()) - { - BipedalLocomotion::log()->error("{} Failed to set the policy", logPrefix); - return; - } - this->run(); - }; - -}; // class Impl - -// static member initialization -std::atomic PeriodicThread::Impl::_forceToStop = false; + return; +}; -PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, - int maximumNumberOfAcceptedDeadlineMiss, - int priority, - int policy) +bool PeriodicThread::setPolicy() { - m_impl = std::make_unique(this, - period, - maximumNumberOfAcceptedDeadlineMiss, - priority, - policy); +#ifdef __linux__ + // get the current thread native handle + pthread_t nativeHandle = pthread_self(); + // get the current thread parameters + sched_param params; + params.sched_priority = m_priority; + // set the new policy + return (pthread_setschedparam(nativeHandle, m_policy, ¶ms) == 0); +#else + return true; +#endif }; -PeriodicThread::~PeriodicThread(){}; - bool PeriodicThread::run() { + BipedalLocomotion::log()->info("[PeriodicThread::run] Base Class."); return true; }; @@ -287,18 +97,88 @@ bool PeriodicThread::threadInit() void PeriodicThread::stop() { - m_impl->stop(); + m_askToStop.store(true); }; bool PeriodicThread::start() { - return m_impl->start(); -} + BipedalLocomotion::log()->info("[PeriodicThread::start] Thread is already running: {}", + m_isRunning.load()); + + if (m_isRunning.load()) + { + // thread is already running + BipedalLocomotion::log()->error("[PeriodicThread::start] The thread is already " + "running."); + return false; + } else + { + m_isRunning.store(true); + } + + BipedalLocomotion::log()->info("[PeriodicThread::start] Thread is running: {}", + m_isRunning.load()); + + // lambda wrapper for the thread function + auto threadFunctionLambda = [this]() { this->threadFunction(); }; + + m_thread = std::thread(threadFunctionLambda); + + BipedalLocomotion::log()->info("[PeriodicThread::start] Thread runs"); + + return m_thread.joinable(); +}; bool PeriodicThread::isRunning() { - return m_impl->isRunning(); + return m_isRunning.load(); } +void PeriodicThread::advance() +{ + // get the current time + auto now = BipedalLocomotion::clock().now(); + // get the next wake up time + auto nextWakeUpTime = now + m_period; + + // run user overridden function + if (!run()) + { + BipedalLocomotion::log()->info("[PeriodicThread::advance] Run failed"); + m_isRunning.store(false); + return; + } + + // check if the thread has to stop + if (m_askToStop.load()) + { + BipedalLocomotion::log()->info("[PeriodicThread::advance] Asked to stop"); + m_isRunning.store(false); + return; + } + + // check if the deadline is missed + if (BipedalLocomotion::clock().now() > nextWakeUpTime) + { + m_deadlineMiss++; + if (m_maximumNumberOfAcceptedDeadlineMiss > 0) + { + if (m_deadlineMiss > m_maximumNumberOfAcceptedDeadlineMiss) + { + // we have to close the runner + BipedalLocomotion::log()->info("[PeriodicThread::advance] Deadline missed"); + m_isRunning.store(false); + return; + } + } + } + + // yield the CPU + BipedalLocomotion::clock().yield(); + + // wait until the next deadline + BipedalLocomotion::clock().sleepUntil(nextWakeUpTime); +}; + } // namespace System } // namespace BipedalLocomotion \ No newline at end of file From aa3399f6901d24783b896c6122fe16fdbefa2fa4 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Thu, 26 Sep 2024 18:21:27 +0200 Subject: [PATCH 07/25] make run method pure virtual and prints clean up --- examples/PeriodicThread/main.cpp | 4 +-- .../BipedalLocomotion/System/PeriodicThread.h | 2 +- src/System/src/PeriodicThread.cpp | 28 ------------------- 3 files changed, 2 insertions(+), 32 deletions(-) diff --git a/examples/PeriodicThread/main.cpp b/examples/PeriodicThread/main.cpp index abe5ab9a27..fe95c88ddf 100644 --- a/examples/PeriodicThread/main.cpp +++ b/examples/PeriodicThread/main.cpp @@ -2,7 +2,6 @@ #include #include #include -#include class Thread : public BipedalLocomotion::System::PeriodicThread { @@ -32,14 +31,13 @@ int main() { // Thread thread; auto thread = Thread(); - std::cerr << "Thread class pointer id: " << &thread << std::endl; thread.start(); while (thread.isRunning()) { std::this_thread::sleep_for(std::chrono::milliseconds(10000)); thread.stop(); + BipedalLocomotion::log()->info("[main] Thread is asked to stop."); } - BipedalLocomotion::log()->info("[main] Thread about to exit."); return EXIT_SUCCESS; } diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 65d35297af..3fe0a1b6c8 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -62,7 +62,7 @@ class PeriodicThread * Override this method to implement the task to be performed from the thread. * @return true if the thread has to continue, false otherwise. */ - virtual bool run(); + virtual bool run() = 0; /** * @brief This method is called at the beginning of the thread. diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index d809199dfd..32c08419b1 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -36,35 +36,25 @@ void PeriodicThread::threadFunction() { constexpr auto logPrefix = "[PeriodicThread::threadFunction]"; - BipedalLocomotion::log()->error("{} about to initialize the thread", logPrefix); - // thread initialization if (!threadInit()) { - BipedalLocomotion::log()->error("{} Failed to initialize the thread", logPrefix); return; } - BipedalLocomotion::log()->error("{} thread initialized", logPrefix); - // set the policy if (!setPolicy()) { - BipedalLocomotion::log()->error("{} Failed to set the policy", logPrefix); return; } - BipedalLocomotion::log()->error("{} policy set", logPrefix); - // run loop - BipedalLocomotion::log()->error("{} advance... {}", logPrefix, m_isRunning.load()); while (m_isRunning.load()) { this->advance(); } m_isRunning.store(false); - BipedalLocomotion::log()->error("{} Completing... {}", logPrefix, m_isRunning.load()); return; }; @@ -84,12 +74,6 @@ bool PeriodicThread::setPolicy() #endif }; -bool PeriodicThread::run() -{ - BipedalLocomotion::log()->info("[PeriodicThread::run] Base Class."); - return true; -}; - bool PeriodicThread::threadInit() { return true; @@ -102,30 +86,21 @@ void PeriodicThread::stop() bool PeriodicThread::start() { - BipedalLocomotion::log()->info("[PeriodicThread::start] Thread is already running: {}", - m_isRunning.load()); if (m_isRunning.load()) { // thread is already running - BipedalLocomotion::log()->error("[PeriodicThread::start] The thread is already " - "running."); return false; } else { m_isRunning.store(true); } - BipedalLocomotion::log()->info("[PeriodicThread::start] Thread is running: {}", - m_isRunning.load()); - // lambda wrapper for the thread function auto threadFunctionLambda = [this]() { this->threadFunction(); }; m_thread = std::thread(threadFunctionLambda); - BipedalLocomotion::log()->info("[PeriodicThread::start] Thread runs"); - return m_thread.joinable(); }; @@ -144,7 +119,6 @@ void PeriodicThread::advance() // run user overridden function if (!run()) { - BipedalLocomotion::log()->info("[PeriodicThread::advance] Run failed"); m_isRunning.store(false); return; } @@ -152,7 +126,6 @@ void PeriodicThread::advance() // check if the thread has to stop if (m_askToStop.load()) { - BipedalLocomotion::log()->info("[PeriodicThread::advance] Asked to stop"); m_isRunning.store(false); return; } @@ -166,7 +139,6 @@ void PeriodicThread::advance() if (m_deadlineMiss > m_maximumNumberOfAcceptedDeadlineMiss) { // we have to close the runner - BipedalLocomotion::log()->info("[PeriodicThread::advance] Deadline missed"); m_isRunning.store(false); return; } From 4c9f97f6802503c709ccf1769ccce67efbee9afd Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Mon, 30 Sep 2024 11:57:04 +0200 Subject: [PATCH 08/25] add sedPeriod member function --- .../BipedalLocomotion/System/PeriodicThread.h | 7 ++++++ src/System/src/PeriodicThread.cpp | 22 +++++++++++++++++++ 2 files changed, 29 insertions(+) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 3fe0a1b6c8..7f7d9ee492 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -56,6 +56,13 @@ class PeriodicThread */ bool isRunning(); + /** + * @brief Set the period of the thread. + * @param period period of the thread. + * @return true if the period was correctly set, false otherwise. + */ + bool setPeriod(std::chrono::nanoseconds period); + protected: /** * @brief This method is called at each iteration of the thread. diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 32c08419b1..e352ed586b 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -74,6 +74,18 @@ bool PeriodicThread::setPolicy() #endif }; +bool PeriodicThread::setPeriod(std::chrono::nanoseconds period) +{ + if (m_isRunning.load()) + { + BipedalLocomotion::log()->error("[PeriodicThread::setPeriod] The thread is running. The " + "period cannot be changed."); + return false; + } + m_period = period; + return true; +} + bool PeriodicThread::threadInit() { return true; @@ -81,6 +93,16 @@ bool PeriodicThread::threadInit() void PeriodicThread::stop() { + if (!m_isRunning.load()) + { + // thread is not running + return; + } + if (m_askToStop.load()) + { + // thread is already asked to stop + return; + } m_askToStop.store(true); }; From 3cd8eec201fbc261012be378461765c40fbab4ca Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 1 Oct 2024 17:17:25 +0200 Subject: [PATCH 09/25] add synchronization feature through blf barrier --- examples/PeriodicThread/main.cpp | 33 +++++++++++---- .../BipedalLocomotion/System/PeriodicThread.h | 26 ++++++++++-- src/System/src/PeriodicThread.cpp | 42 +++++++++++++++---- 3 files changed, 81 insertions(+), 20 deletions(-) diff --git a/examples/PeriodicThread/main.cpp b/examples/PeriodicThread/main.cpp index fe95c88ddf..27b9fa9da5 100644 --- a/examples/PeriodicThread/main.cpp +++ b/examples/PeriodicThread/main.cpp @@ -1,12 +1,16 @@ +#include +#include + +#include #include #include #include -#include class Thread : public BipedalLocomotion::System::PeriodicThread { public: Thread(); + ~Thread(); bool run() override; bool threadInit() override; @@ -21,6 +25,11 @@ bool Thread::run() Thread::Thread() : BipedalLocomotion::System::PeriodicThread(std::chrono::milliseconds(1000)){}; +Thread::~Thread() +{ + BipedalLocomotion::log()->info("[Thread::~Thread] Thread is destroyed."); +}; + bool Thread::threadInit() { BipedalLocomotion::log()->info("[Thread::threadInit] Thread is initialized."); @@ -29,15 +38,25 @@ bool Thread::threadInit() int main() { + + auto barrier = BipedalLocomotion::System::Barrier::create(2); + // Thread thread; - auto thread = Thread(); - thread.start(); + auto thread1 = Thread(); + thread1.start(barrier); + + BipedalLocomotion::clock().sleepFor(std::chrono::milliseconds(2000)); + + auto thread2 = Thread(); + thread2.setPeriod(std::chrono::seconds(2)); + thread2.start(barrier); - while (thread.isRunning()) + while (thread1.isRunning()) { - std::this_thread::sleep_for(std::chrono::milliseconds(10000)); - thread.stop(); - BipedalLocomotion::log()->info("[main] Thread is asked to stop."); + std::this_thread::sleep_for(std::chrono::milliseconds(4000)); + thread1.stop(); + BipedalLocomotion::log()->info("[main] Thread 1 is asked to stop."); } + BipedalLocomotion::log()->info("[main] About to exit the application."); return EXIT_SUCCESS; } diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 7f7d9ee492..6ce56fc5c7 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -10,7 +10,12 @@ #include #include +#include +#include #include +#include + +#include namespace BipedalLocomotion { @@ -40,10 +45,12 @@ class PeriodicThread PeriodicThread& operator=(const PeriodicThread&) = delete; /** - * @brief Start the thread + * @brief Start the thread. + * @param barrier barrier to synchronize the thread. If nullptr, the thread will start + * immediately, without waiting for other threads to reach the barrier. * @return true if the thread was correctly started, false otherwise. */ - bool start(); + bool start(std::shared_ptr barrier = nullptr); /** * @brief Call this method to stop the thread. @@ -88,16 +95,23 @@ class PeriodicThread */ void advance(); + /** + * @brief Synchronize the thread. + */ + void synchronize(); + /** * @brief Set the policy of the thread. * @return true if the policy was correctly set, false otherwise. */ bool setPolicy(); - std::chrono::nanoseconds m_period = std::chrono::nanoseconds(100000); /**< Period of the thread. + std::chrono::nanoseconds m_period = std::chrono::nanoseconds(100000); /**< Period of the + * thread. */ - int m_maximumNumberOfAcceptedDeadlineMiss = -1; /**< Maximum number of accepted deadline miss. + int m_maximumNumberOfAcceptedDeadlineMiss = -1; /**< Maximum number of accepted deadline + * miss. */ int m_deadlineMiss = 0; /**< Number of deadline miss. */ @@ -111,6 +125,10 @@ class PeriodicThread std::atomic m_askToStop = false; /**< Flag to ask to stop the thread. */ std::thread m_thread; /**< Thread object. */ + + std::shared_ptr m_barrier; /**< Barrier to synchronize the + * thread. + */ }; } // namespace System } // namespace BipedalLocomotion diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index e352ed586b..27acc2f09c 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -1,6 +1,8 @@ #include #include #include +#include +#include #include #include @@ -25,11 +27,24 @@ PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, PeriodicThread::~PeriodicThread() { + // stop the thread, if it is running + if (m_isRunning.load()) + { + stop(); + } + // join the thread if (m_thread.joinable()) { m_thread.join(); + m_thread = std::thread(); + BipedalLocomotion::log()->info("[PeriodicThread::~PeriodicThread] The thread joined."); } + + // + std::cerr << "[PeriodicThread::~PeriodicThread] The thread object is " + "destroyed." + << std::endl; }; void PeriodicThread::threadFunction() @@ -48,8 +63,11 @@ void PeriodicThread::threadFunction() return; } + // synchronize the thread + synchronize(); + // run loop - while (m_isRunning.load()) + while (m_isRunning.load() && !m_askToStop.load()) { this->advance(); } @@ -91,6 +109,16 @@ bool PeriodicThread::threadInit() return true; }; +void PeriodicThread::synchronize() +{ + if (!(m_barrier == nullptr)) + { + BipedalLocomotion::log()->debug("[PeriodicThread::synchronize] This thread is waiting for " + "the other threads."); + m_barrier->wait(); + } +}; + void PeriodicThread::stop() { if (!m_isRunning.load()) @@ -106,7 +134,7 @@ void PeriodicThread::stop() m_askToStop.store(true); }; -bool PeriodicThread::start() +bool PeriodicThread::start(std::shared_ptr barrier) { if (m_isRunning.load()) @@ -118,6 +146,9 @@ bool PeriodicThread::start() m_isRunning.store(true); } + // store the barrier + m_barrier = barrier; + // lambda wrapper for the thread function auto threadFunctionLambda = [this]() { this->threadFunction(); }; @@ -145,13 +176,6 @@ void PeriodicThread::advance() return; } - // check if the thread has to stop - if (m_askToStop.load()) - { - m_isRunning.store(false); - return; - } - // check if the deadline is missed if (BipedalLocomotion::clock().now() > nextWakeUpTime) { From 531d179adc30cc5afd010fc85b6e09196e0fc970 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Wed, 2 Oct 2024 17:10:41 +0200 Subject: [PATCH 10/25] add unit tests for periodic thread --- src/System/tests/CMakeLists.txt | 5 ++ src/System/tests/PeriodicThreadTest.cpp | 112 ++++++++++++++++++++++++ 2 files changed, 117 insertions(+) create mode 100644 src/System/tests/PeriodicThreadTest.cpp diff --git a/src/System/tests/CMakeLists.txt b/src/System/tests/CMakeLists.txt index 7cdcbdd15e..96969487e1 100644 --- a/src/System/tests/CMakeLists.txt +++ b/src/System/tests/CMakeLists.txt @@ -11,3 +11,8 @@ add_bipedal_test( NAME AdvanceableRunner SOURCES AdvanceableRunnerTest.cpp LINKS BipedalLocomotion::System BipedalLocomotion::TextLogging BipedalLocomotion::ParametersHandler) + +add_bipedal_test( + NAME PeriodicThread + SOURCES PeriodicThreadTest.cpp + LINKS BipedalLocomotion::System BipedalLocomotion::TextLogging) diff --git a/src/System/tests/PeriodicThreadTest.cpp b/src/System/tests/PeriodicThreadTest.cpp new file mode 100644 index 0000000000..ab6b71139f --- /dev/null +++ b/src/System/tests/PeriodicThreadTest.cpp @@ -0,0 +1,112 @@ +/** + * @file PeriodicThreadTest.cpp + * @authors Lorenzo Moretti + * @copyright This software may be modified and distributed under the terms of the GNU Lesser + * General Public License v2.1 or any later version. + */ + +#include +#include +#include + +// Catch2 +#include + +#include +#include +#include + +using namespace BipedalLocomotion::System; +using namespace std::chrono_literals; + +class Thread : public PeriodicThread +{ +protected: + bool threadInit() override + { + m_counter.store(0); + return true; + } + bool run() override + { + m_counter++; + return true; + } + +private: + std::atomic m_counter{0}; + +public: + int getCounter() + { + return m_counter.load(); + } +}; + +TEST_CASE("Test Periodic Thread", "[PeriodicThread]") +{ + using namespace std::chrono_literals; + + BipedalLocomotion::System::ClockBuilder::setFactory( + std::make_shared()); + + auto period = 50ms; + + // create + auto thread = Thread(); + + // set the period + REQUIRE(thread.setPeriod(period)); + + // set the policy + REQUIRE(thread.setPolicy(SCHED_OTHER, 0)); + + // start the thread + REQUIRE(thread.start()); + + BipedalLocomotion::clock().sleepFor(10 * period); + + // stop the thread + thread.stop(); + BipedalLocomotion::clock().sleepFor(period); + + // check if the thread is stopped + REQUIRE(!thread.isRunning()); +} + +TEST_CASE("Test Periodic Thread", "[PeriodicThreadSynchronization]") +{ + using namespace std::chrono_literals; + + auto period = 100ms; + + auto barrier = BipedalLocomotion::System::Barrier::create(2); + + // create two threads + auto thread1 = Thread(); + thread1.setPeriod(period); + + auto thread2 = Thread(); + thread2.setPeriod(period); + + // start thread 1 + REQUIRE(thread1.start(barrier)); + BipedalLocomotion::clock().sleepFor(5 * period); + + // check that the thread 1 is waiting for thread 2 + // (i.e. threadInit has been called, but run has not been called yet) + REQUIRE(thread1.isRunning()); + REQUIRE(!thread2.isRunning()); + REQUIRE(thread1.getCounter() == 0); + + // start thread 2 + REQUIRE(thread2.start(barrier)); + BipedalLocomotion::clock().sleepFor(5 * period); + + // check that the thread 2 is running + REQUIRE(thread2.isRunning()); + + // stop the threads + thread1.stop(); + thread2.stop(); +} \ No newline at end of file From 7e96f0f918faa9c0bdb75995af702d3d0ef89c43 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Wed, 2 Oct 2024 18:11:17 +0200 Subject: [PATCH 11/25] add state machine and missing member function --- .../BipedalLocomotion/System/PeriodicThread.h | 41 +++++++-- src/System/src/PeriodicThread.cpp | 86 +++++++++++++------ 2 files changed, 97 insertions(+), 30 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 6ce56fc5c7..2a93c38bc7 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -21,11 +21,24 @@ namespace BipedalLocomotion { namespace System { + +/** + * @brief Enum class which defines the state of the periodic thread state machine. + */ +enum class PeriodicThreadState +{ + INACTIVE, /**< The thread is not yet active. */ + STARTED, /**< The thread has been started. */ + INITIALIZED, /**< The thread has been initialized.*/ + RUNNING, /**< The thread is running. */ + IDLE, /**< The thread is idling. */ + STOPPED /**< The thread is stopped. */ +}; + /** * @brief This class implements a periodic thread. The user has to inherit from this class and * implement the virtual methods. */ - class PeriodicThread { public: @@ -57,6 +70,16 @@ class PeriodicThread */ void stop(); + /** + * @brief Suspend the thread. + */ + bool suspend(); + + /** + * @brief Resume the thread. + */ + bool resume(); + /** * @brief Check if the thread is running. * @return true if the thread is running, false otherwise. @@ -70,6 +93,14 @@ class PeriodicThread */ bool setPeriod(std::chrono::nanoseconds period); + /** + * @brief Set the policy of the thread. + * @param policy policy of the thread. + * @param priority priority of the thread. + * @return true if the policy was correctly set, false otherwise. + */ + bool setPolicy(int policy, int priority = 0); + protected: /** * @brief This method is called at each iteration of the thread. @@ -120,12 +151,12 @@ class PeriodicThread int m_policy = SCHED_OTHER; /**< Policy of the thread. */ - std::atomic m_isRunning = false; /**< Flag to check if the thread is running. */ - - std::atomic m_askToStop = false; /**< Flag to ask to stop the thread. */ - std::thread m_thread; /**< Thread object. */ + std::atomic m_state = PeriodicThreadState::INACTIVE; /**< State of the + * thread. + */ + std::shared_ptr m_barrier; /**< Barrier to synchronize the * thread. */ diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 27acc2f09c..de751f9db8 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -23,12 +23,13 @@ PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, , m_maximumNumberOfAcceptedDeadlineMiss(maximumNumberOfAcceptedDeadlineMiss) , m_priority(priority) , m_policy(policy) - , m_deadlineMiss(0){}; + , m_deadlineMiss(0) + , m_state(PeriodicThreadState::INACTIVE){}; PeriodicThread::~PeriodicThread() { - // stop the thread, if it is running - if (m_isRunning.load()) + // stop the thread, if it is not already stopped + if (m_state.load() != PeriodicThreadState::STOPPED) { stop(); } @@ -63,16 +64,20 @@ void PeriodicThread::threadFunction() return; } + m_state.store(PeriodicThreadState::INITIALIZED); + // synchronize the thread synchronize(); // run loop - while (m_isRunning.load() && !m_askToStop.load()) + m_state.store(PeriodicThreadState::RUNNING); + + while (m_state == PeriodicThreadState::RUNNING || m_state == PeriodicThreadState::IDLE) { this->advance(); } - m_isRunning.store(false); + m_state.store(PeriodicThreadState::STOPPED); return; }; @@ -92,12 +97,25 @@ bool PeriodicThread::setPolicy() #endif }; +bool PeriodicThread::setPolicy(int priority, int policy) +{ + if (m_state.load() != PeriodicThreadState::INACTIVE) + { + BipedalLocomotion::log()->error("[PeriodicThread::setPolicy] The thread has already " + "started. The policy and priority cannot be changed."); + return false; + } + m_priority = priority; + m_policy = policy; + return true; +}; + bool PeriodicThread::setPeriod(std::chrono::nanoseconds period) { - if (m_isRunning.load()) + if (m_state.load() != PeriodicThreadState::INACTIVE) { - BipedalLocomotion::log()->error("[PeriodicThread::setPeriod] The thread is running. The " - "period cannot be changed."); + BipedalLocomotion::log()->error("[PeriodicThread::setPeriod] The thread has already " + "started. The period cannot be changed."); return false; } m_period = period; @@ -121,29 +139,43 @@ void PeriodicThread::synchronize() void PeriodicThread::stop() { - if (!m_isRunning.load()) + // stop the thread only if it is running or idling + if ((m_state.load() == PeriodicThreadState::RUNNING + || m_state.load() == PeriodicThreadState::IDLE)) { - // thread is not running - return; + m_state.store(PeriodicThreadState::STOPPED); } - if (m_askToStop.load()) +}; + +bool PeriodicThread::suspend() +{ + if (m_state.load() == PeriodicThreadState::RUNNING) { - // thread is already asked to stop - return; + m_state.store(PeriodicThreadState::IDLE); + return true; } - m_askToStop.store(true); + return false; }; -bool PeriodicThread::start(std::shared_ptr barrier) +bool PeriodicThread::resume() { + if (m_state.load() == PeriodicThreadState::IDLE) + { + m_state.store(PeriodicThreadState::RUNNING); + return true; + } + return false; +}; - if (m_isRunning.load()) +bool PeriodicThread::start(std::shared_ptr barrier) +{ + // only an inactive thread can be started + if (m_state.load() != PeriodicThreadState::INACTIVE) { - // thread is already running return false; } else { - m_isRunning.store(true); + m_state.store(PeriodicThreadState::STARTED); } // store the barrier @@ -159,7 +191,7 @@ bool PeriodicThread::start(std::shared_ptr b bool PeriodicThread::isRunning() { - return m_isRunning.load(); + return (m_state.load() == PeriodicThreadState::RUNNING); } void PeriodicThread::advance() @@ -169,11 +201,15 @@ void PeriodicThread::advance() // get the next wake up time auto nextWakeUpTime = now + m_period; - // run user overridden function - if (!run()) + // run user overridden function, when not idling + if (m_state.load() != PeriodicThreadState::IDLE) { - m_isRunning.store(false); - return; + if (!run()) + { + // an error occurred, stop the thread + m_state.store(PeriodicThreadState::STOPPED); + return; + } } // check if the deadline is missed @@ -185,7 +221,7 @@ void PeriodicThread::advance() if (m_deadlineMiss > m_maximumNumberOfAcceptedDeadlineMiss) { // we have to close the runner - m_isRunning.store(false); + m_state.store(PeriodicThreadState::STOPPED); return; } } From 1316986325f0c5fe0ddcfa27bfbbce9b108c564a Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Thu, 3 Oct 2024 11:37:52 +0200 Subject: [PATCH 12/25] improve state machine and update unit tests --- .../BipedalLocomotion/System/PeriodicThread.h | 13 ++++-- src/System/src/PeriodicThread.cpp | 42 ++++++++++--------- src/System/tests/PeriodicThreadTest.cpp | 25 ++++++++--- 3 files changed, 53 insertions(+), 27 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 2a93c38bc7..37b27d00c4 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -30,7 +30,7 @@ enum class PeriodicThreadState INACTIVE, /**< The thread is not yet active. */ STARTED, /**< The thread has been started. */ INITIALIZED, /**< The thread has been initialized.*/ - RUNNING, /**< The thread is running. */ + EXECUTING, /**< The thread is executing the task. */ IDLE, /**< The thread is idling. */ STOPPED /**< The thread is stopped. */ }; @@ -81,11 +81,18 @@ class PeriodicThread bool resume(); /** - * @brief Check if the thread is running. + * @brief Check if the thread is running. This means that the thread has started and not yet + * stopped. * @return true if the thread is running, false otherwise. */ bool isRunning(); + /** + * @brief Check if the thread is initialized. + * @return true if the thread is initialized, false otherwise. + */ + bool isInitialized(); + /** * @brief Set the period of the thread. * @param period period of the thread. @@ -107,7 +114,7 @@ class PeriodicThread * Override this method to implement the task to be performed from the thread. * @return true if the thread has to continue, false otherwise. */ - virtual bool run() = 0; + virtual bool run(); /** * @brief This method is called at the beginning of the thread. diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index de751f9db8..2714594e51 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -2,7 +2,6 @@ #include #include #include -#include #include #include @@ -28,24 +27,15 @@ PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, PeriodicThread::~PeriodicThread() { - // stop the thread, if it is not already stopped - if (m_state.load() != PeriodicThreadState::STOPPED) - { - stop(); - } + // force stop of the thread + m_state.store(PeriodicThreadState::STOPPED); // join the thread if (m_thread.joinable()) { m_thread.join(); m_thread = std::thread(); - BipedalLocomotion::log()->info("[PeriodicThread::~PeriodicThread] The thread joined."); } - - // - std::cerr << "[PeriodicThread::~PeriodicThread] The thread object is " - "destroyed." - << std::endl; }; void PeriodicThread::threadFunction() @@ -70,9 +60,9 @@ void PeriodicThread::threadFunction() synchronize(); // run loop - m_state.store(PeriodicThreadState::RUNNING); + m_state.store(PeriodicThreadState::EXECUTING); - while (m_state == PeriodicThreadState::RUNNING || m_state == PeriodicThreadState::IDLE) + while (m_state == PeriodicThreadState::EXECUTING || m_state == PeriodicThreadState::IDLE) { this->advance(); } @@ -127,6 +117,11 @@ bool PeriodicThread::threadInit() return true; }; +bool PeriodicThread::run() +{ + return false; +}; + void PeriodicThread::synchronize() { if (!(m_barrier == nullptr)) @@ -139,19 +134,21 @@ void PeriodicThread::synchronize() void PeriodicThread::stop() { - // stop the thread only if it is running or idling - if ((m_state.load() == PeriodicThreadState::RUNNING + // stop the thread only if it is executing or idling + if ((m_state.load() == PeriodicThreadState::EXECUTING || m_state.load() == PeriodicThreadState::IDLE)) { m_state.store(PeriodicThreadState::STOPPED); + BipedalLocomotion::log()->debug("[PeriodicThread::stop] Thread is stopped."); } }; bool PeriodicThread::suspend() { - if (m_state.load() == PeriodicThreadState::RUNNING) + if (m_state.load() == PeriodicThreadState::EXECUTING) { m_state.store(PeriodicThreadState::IDLE); + BipedalLocomotion::log()->debug("[PeriodicThread::suspend] Thread is suspended."); return true; } return false; @@ -161,7 +158,8 @@ bool PeriodicThread::resume() { if (m_state.load() == PeriodicThreadState::IDLE) { - m_state.store(PeriodicThreadState::RUNNING); + m_state.store(PeriodicThreadState::EXECUTING); + BipedalLocomotion::log()->debug("[PeriodicThread::resume] Thread is resumed."); return true; } return false; @@ -191,7 +189,13 @@ bool PeriodicThread::start(std::shared_ptr b bool PeriodicThread::isRunning() { - return (m_state.load() == PeriodicThreadState::RUNNING); + return ((m_state.load() != PeriodicThreadState::INACTIVE) + && (m_state.load() != PeriodicThreadState::STOPPED)); +} + +bool PeriodicThread::isInitialized() +{ + return (m_state.load() == PeriodicThreadState::INITIALIZED); } void PeriodicThread::advance() diff --git a/src/System/tests/PeriodicThreadTest.cpp b/src/System/tests/PeriodicThreadTest.cpp index ab6b71139f..ee3cf7aa29 100644 --- a/src/System/tests/PeriodicThreadTest.cpp +++ b/src/System/tests/PeriodicThreadTest.cpp @@ -64,7 +64,23 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThread]") // start the thread REQUIRE(thread.start()); - BipedalLocomotion::clock().sleepFor(10 * period); + BipedalLocomotion::clock().sleepFor(period); + + // check if the thread is running + REQUIRE(thread.isRunning()); + + // suspend the thread + REQUIRE(thread.suspend()); + BipedalLocomotion::clock().sleepFor(period); + + // check if the thread is suspended + int counter = thread.getCounter(); + BipedalLocomotion::clock().sleepFor(period); + REQUIRE(counter == thread.getCounter()); + + // resume the thread + REQUIRE(thread.resume()); + BipedalLocomotion::clock().sleepFor(period); // stop the thread thread.stop(); @@ -91,17 +107,16 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThreadSynchronization]") // start thread 1 REQUIRE(thread1.start(barrier)); - BipedalLocomotion::clock().sleepFor(5 * period); + BipedalLocomotion::clock().sleepFor(2 * period); // check that the thread 1 is waiting for thread 2 // (i.e. threadInit has been called, but run has not been called yet) - REQUIRE(thread1.isRunning()); + REQUIRE(thread1.isInitialized()); REQUIRE(!thread2.isRunning()); - REQUIRE(thread1.getCounter() == 0); // start thread 2 REQUIRE(thread2.start(barrier)); - BipedalLocomotion::clock().sleepFor(5 * period); + BipedalLocomotion::clock().sleepFor(period); // check that the thread 2 is running REQUIRE(thread2.isRunning()); From aa53266cd41d2e254e790c587b1269150abe6a30 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Thu, 3 Oct 2024 11:43:01 +0200 Subject: [PATCH 13/25] add more unit tests --- src/System/tests/PeriodicThreadTest.cpp | 30 +++++++++++++++++++++---- 1 file changed, 26 insertions(+), 4 deletions(-) diff --git a/src/System/tests/PeriodicThreadTest.cpp b/src/System/tests/PeriodicThreadTest.cpp index ee3cf7aa29..2de09130d3 100644 --- a/src/System/tests/PeriodicThreadTest.cpp +++ b/src/System/tests/PeriodicThreadTest.cpp @@ -45,8 +45,6 @@ class Thread : public PeriodicThread TEST_CASE("Test Periodic Thread", "[PeriodicThread]") { - using namespace std::chrono_literals; - BipedalLocomotion::System::ClockBuilder::setFactory( std::make_shared()); @@ -92,8 +90,6 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThread]") TEST_CASE("Test Periodic Thread", "[PeriodicThreadSynchronization]") { - using namespace std::chrono_literals; - auto period = 100ms; auto barrier = BipedalLocomotion::System::Barrier::create(2); @@ -124,4 +120,30 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThreadSynchronization]") // stop the threads thread1.stop(); thread2.stop(); +} + +TEST_CASE("Test Periodic Thread", "[PeriodicThreadNotAllowed]") +{ + + auto period = 50ms; + + // create + auto thread = Thread(); + + // start the thread + REQUIRE(thread.start()); + BipedalLocomotion::clock().sleepFor(period); + + // try to set the period + REQUIRE(!thread.setPeriod(period)); + + // try to set the policy + REQUIRE(!thread.setPolicy(SCHED_OTHER, 0)); + + // stop the thread + thread.stop(); + BipedalLocomotion::clock().sleepFor(period); + + // try to resume the thread + REQUIRE(!thread.resume()); } \ No newline at end of file From bf2966902ae817f4a6e47d95b89d516e71d9ade5 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Fri, 4 Oct 2024 09:43:02 +0200 Subject: [PATCH 14/25] add early wake up option --- .../BipedalLocomotion/System/PeriodicThread.h | 14 +++++++++- src/System/src/PeriodicThread.cpp | 27 ++++++++++++++++--- 2 files changed, 36 insertions(+), 5 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 37b27d00c4..9b97c8e724 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -46,7 +46,8 @@ class PeriodicThread PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), int maximumNumberOfAcceptedDeadlineMiss = -1, int priority = 0, - int policy = SCHED_OTHER); + int policy = SCHED_OTHER, + bool earlyWakeUp = false); // Destructor virtual ~PeriodicThread(); @@ -154,6 +155,10 @@ class PeriodicThread int m_deadlineMiss = 0; /**< Number of deadline miss. */ + std::chrono::nanoseconds m_wakeUpTime = std::chrono::nanoseconds(0); /**< Wake up time of the + * thread. + */ + int m_priority = 0; /**< Priority of the thread. */ int m_policy = SCHED_OTHER; /**< Policy of the thread. */ @@ -167,6 +172,13 @@ class PeriodicThread std::shared_ptr m_barrier; /**< Barrier to synchronize the * thread. */ + const std::chrono::nanoseconds m_schedulerLatency + = std::chrono::microseconds(100); /**< Scheduler latency when waking up a thread. Indeed, it + varies depending on the OS. For now we fix it to a + constant value. Note that in real-time OS it might be + smaller. */ + bool m_earlyWakeUp = false; /**< If true, the thread will be awaken before and busy wait until + the actual wake up time. */ }; } // namespace System } // namespace BipedalLocomotion diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 2714594e51..67246ea21c 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -17,12 +17,15 @@ namespace System PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, int maximumNumberOfAcceptedDeadlineMiss, int priority, - int policy) + int policy, + bool earlyWakeUp) : m_period(period) , m_maximumNumberOfAcceptedDeadlineMiss(maximumNumberOfAcceptedDeadlineMiss) , m_priority(priority) , m_policy(policy) , m_deadlineMiss(0) + , m_wakeUpTime(std::chrono::nanoseconds(0)) + , m_earlyWakeUp(earlyWakeUp) , m_state(PeriodicThreadState::INACTIVE){}; PeriodicThread::~PeriodicThread() @@ -202,8 +205,18 @@ void PeriodicThread::advance() { // get the current time auto now = BipedalLocomotion::clock().now(); + + // busy wait until wake up time + if (m_earlyWakeUp) + { + while (now < m_wakeUpTime) + { + now = BipedalLocomotion::clock().now(); + } + } + // get the next wake up time - auto nextWakeUpTime = now + m_period; + m_wakeUpTime = now + m_period; // run user overridden function, when not idling if (m_state.load() != PeriodicThreadState::IDLE) @@ -217,7 +230,7 @@ void PeriodicThread::advance() } // check if the deadline is missed - if (BipedalLocomotion::clock().now() > nextWakeUpTime) + if (BipedalLocomotion::clock().now() > m_wakeUpTime) { m_deadlineMiss++; if (m_maximumNumberOfAcceptedDeadlineMiss > 0) @@ -235,7 +248,13 @@ void PeriodicThread::advance() BipedalLocomotion::clock().yield(); // wait until the next deadline - BipedalLocomotion::clock().sleepUntil(nextWakeUpTime); + if (m_earlyWakeUp) + { + BipedalLocomotion::clock().sleepUntil(m_wakeUpTime - m_schedulerLatency); + } else + { + BipedalLocomotion::clock().sleepUntil(m_wakeUpTime); + } }; } // namespace System From e52512e459cb69aaa6e9c596fb6734cd9d2939f9 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Fri, 4 Oct 2024 10:46:15 +0200 Subject: [PATCH 15/25] add some setters and update unit tests --- .../BipedalLocomotion/System/PeriodicThread.h | 22 ++++++++++-- src/System/src/PeriodicThread.cpp | 34 +++++++++++++++++-- 2 files changed, 52 insertions(+), 4 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 9b97c8e724..bcd92bc56f 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -109,6 +109,25 @@ class PeriodicThread */ bool setPolicy(int policy, int priority = 0); + /** + * @brief Set the maximum number of accepted deadline miss. + * @param maximumNumberOfAcceptedDeadlineMiss maximum number of accepted deadline miss. + */ + bool setMaximumNumberOfAcceptedDeadlineMiss(int maximumNumberOfAcceptedDeadlineMiss); + + /** + * @brief Get the number of deadline miss. + * @return number of deadline miss. + */ + int getNumberOfDeadlineMiss(); + + /** + * @brief Enable the early wake up. The thread will be awaken before and busy wait until the + * actual wake up time. + * @return true if the early wake up was correctly set, false otherwise. + */ + bool enableEarlyWakeUp(); + protected: /** * @brief This method is called at each iteration of the thread. @@ -153,12 +172,11 @@ class PeriodicThread * miss. */ - int m_deadlineMiss = 0; /**< Number of deadline miss. */ + std::atomic m_deadlineMiss = 0; /**< Number of deadline miss. */ std::chrono::nanoseconds m_wakeUpTime = std::chrono::nanoseconds(0); /**< Wake up time of the * thread. */ - int m_priority = 0; /**< Priority of the thread. */ int m_policy = SCHED_OTHER; /**< Policy of the thread. */ diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 67246ea21c..d400dd2e36 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -115,6 +115,36 @@ bool PeriodicThread::setPeriod(std::chrono::nanoseconds period) return true; } +bool PeriodicThread::setMaximumNumberOfAcceptedDeadlineMiss(int maximumNumberOfAcceptedDeadlineMiss) +{ + if (m_state.load() != PeriodicThreadState::INACTIVE) + { + BipedalLocomotion::log()->error("[PeriodicThread::setMaximumNumberOfAcceptedDeadlineMiss] " + "The thread has already started. The maximum number of " + "accepted deadline miss cannot be changed."); + return false; + } + m_maximumNumberOfAcceptedDeadlineMiss = maximumNumberOfAcceptedDeadlineMiss; + return true; +} + +int PeriodicThread::getNumberOfDeadlineMiss() +{ + return m_deadlineMiss.load(); +} + +bool PeriodicThread::enableEarlyWakeUp() +{ + if (m_state.load() != PeriodicThreadState::INACTIVE) + { + BipedalLocomotion::log()->error("[PeriodicThread::enableEarlyWakeUp] The thread has " + "already started. The early wake up cannot be changed."); + return false; + } + m_earlyWakeUp = true; + return true; +} + bool PeriodicThread::threadInit() { return true; @@ -232,10 +262,10 @@ void PeriodicThread::advance() // check if the deadline is missed if (BipedalLocomotion::clock().now() > m_wakeUpTime) { - m_deadlineMiss++; + m_deadlineMiss.fetch_add(1); // increment the number of deadline miss if (m_maximumNumberOfAcceptedDeadlineMiss > 0) { - if (m_deadlineMiss > m_maximumNumberOfAcceptedDeadlineMiss) + if (m_deadlineMiss.load() > m_maximumNumberOfAcceptedDeadlineMiss) { // we have to close the runner m_state.store(PeriodicThreadState::STOPPED); From 2461257505c99fc21a5286042c6479294574b413 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Fri, 4 Oct 2024 11:45:44 +0200 Subject: [PATCH 16/25] add some unit tests --- src/System/tests/PeriodicThreadTest.cpp | 57 +++++++++++++++++++++++++ 1 file changed, 57 insertions(+) diff --git a/src/System/tests/PeriodicThreadTest.cpp b/src/System/tests/PeriodicThreadTest.cpp index 2de09130d3..5f29a0b157 100644 --- a/src/System/tests/PeriodicThreadTest.cpp +++ b/src/System/tests/PeriodicThreadTest.cpp @@ -43,6 +43,29 @@ class Thread : public PeriodicThread } }; +class ThreadWithDeadlineMiss : public PeriodicThread +{ +protected: + bool threadInit() override + { + return true; + } + bool run() override + { + BipedalLocomotion::clock().sleepFor(waitTime); + return true; + } + +private: + std::chrono::nanoseconds waitTime; + +public: + void setWaitTime(std::chrono::nanoseconds time) + { + waitTime = time; + } +}; + TEST_CASE("Test Periodic Thread", "[PeriodicThread]") { BipedalLocomotion::System::ClockBuilder::setFactory( @@ -140,10 +163,44 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThreadNotAllowed]") // try to set the policy REQUIRE(!thread.setPolicy(SCHED_OTHER, 0)); + // try to set the maximum number of accepted deadline miss + REQUIRE(!thread.setMaximumNumberOfAcceptedDeadlineMiss(0)); + + // try to enable early wake up + REQUIRE(!thread.enableEarlyWakeUp()); + // stop the thread thread.stop(); BipedalLocomotion::clock().sleepFor(period); // try to resume the thread REQUIRE(!thread.resume()); +} + +TEST_CASE("Test Periodic Thread", "[PeriodicThreadDeadlineMiss]") +{ + auto period = 50ms; + + // create + auto thread = ThreadWithDeadlineMiss(); + + // set the period + REQUIRE(thread.setPeriod(period)); + + // set the wait time in the run function to trigger a deadline miss + thread.setWaitTime(2 * period); + + // set the maximum number of accepted deadline miss + REQUIRE(thread.setMaximumNumberOfAcceptedDeadlineMiss(2)); + + // start the thread + REQUIRE(thread.start()); + + BipedalLocomotion::clock().sleepFor(10 * period); + + // check if the thread is stopped + REQUIRE(!thread.isRunning()); + + // check the number of deadline miss + REQUIRE(thread.getNumberOfDeadlineMiss() == 3); } \ No newline at end of file From 1441c8536be97991d8487e1ab923aadc939d2749 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Fri, 4 Oct 2024 13:11:30 +0200 Subject: [PATCH 17/25] thread policy and priority only in Linux --- .../BipedalLocomotion/System/PeriodicThread.h | 13 ++++++++++-- src/System/src/PeriodicThread.cpp | 21 +++++++++++++++++++ src/System/tests/PeriodicThreadTest.cpp | 4 ++++ 3 files changed, 36 insertions(+), 2 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index bcd92bc56f..da24647b00 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -11,9 +11,7 @@ #include #include #include -#include #include -#include #include @@ -42,12 +40,19 @@ enum class PeriodicThreadState class PeriodicThread { public: +#ifdef __linux__ // Default constructor PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), int maximumNumberOfAcceptedDeadlineMiss = -1, int priority = 0, int policy = SCHED_OTHER, bool earlyWakeUp = false); +#else + // Default constructor + PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), + int maximumNumberOfAcceptedDeadlineMiss = -1, + bool earlyWakeUp = false); +#endif // Destructor virtual ~PeriodicThread(); @@ -101,6 +106,7 @@ class PeriodicThread */ bool setPeriod(std::chrono::nanoseconds period); +#ifdef __linux__ /** * @brief Set the policy of the thread. * @param policy policy of the thread. @@ -108,6 +114,7 @@ class PeriodicThread * @return true if the policy was correctly set, false otherwise. */ bool setPolicy(int policy, int priority = 0); +#endif /** * @brief Set the maximum number of accepted deadline miss. @@ -177,9 +184,11 @@ class PeriodicThread std::chrono::nanoseconds m_wakeUpTime = std::chrono::nanoseconds(0); /**< Wake up time of the * thread. */ +#ifdef __linux__ int m_priority = 0; /**< Priority of the thread. */ int m_policy = SCHED_OTHER; /**< Policy of the thread. */ +#endif std::thread m_thread; /**< Thread object. */ diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index d400dd2e36..7edf1a112e 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -2,8 +2,11 @@ #include #include #include + +#ifdef __linux__ #include #include +#endif #include #include @@ -14,6 +17,8 @@ namespace BipedalLocomotion namespace System { +#ifdef __linux__ + PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, int maximumNumberOfAcceptedDeadlineMiss, int priority, @@ -28,6 +33,20 @@ PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, , m_earlyWakeUp(earlyWakeUp) , m_state(PeriodicThreadState::INACTIVE){}; +#else + +PeriodicThread::PeriodicThread(std::chrono::nanoseconds period, + int maximumNumberOfAcceptedDeadlineMiss, + bool earlyWakeUp) + : m_period(period) + , m_maximumNumberOfAcceptedDeadlineMiss(maximumNumberOfAcceptedDeadlineMiss) + , m_deadlineMiss(0) + , m_wakeUpTime(std::chrono::nanoseconds(0)) + , m_earlyWakeUp(earlyWakeUp) + , m_state(PeriodicThreadState::INACTIVE){}; + +#endif + PeriodicThread::~PeriodicThread() { // force stop of the thread @@ -90,6 +109,7 @@ bool PeriodicThread::setPolicy() #endif }; +#ifdef __linux__ bool PeriodicThread::setPolicy(int priority, int policy) { if (m_state.load() != PeriodicThreadState::INACTIVE) @@ -102,6 +122,7 @@ bool PeriodicThread::setPolicy(int priority, int policy) m_policy = policy; return true; }; +#endif bool PeriodicThread::setPeriod(std::chrono::nanoseconds period) { diff --git a/src/System/tests/PeriodicThreadTest.cpp b/src/System/tests/PeriodicThreadTest.cpp index 5f29a0b157..5206cb8e45 100644 --- a/src/System/tests/PeriodicThreadTest.cpp +++ b/src/System/tests/PeriodicThreadTest.cpp @@ -79,8 +79,10 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThread]") // set the period REQUIRE(thread.setPeriod(period)); +#ifdef __linux__ // set the policy REQUIRE(thread.setPolicy(SCHED_OTHER, 0)); +#endif // start the thread REQUIRE(thread.start()); @@ -160,8 +162,10 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThreadNotAllowed]") // try to set the period REQUIRE(!thread.setPeriod(period)); +#ifdef __linux__ // try to set the policy REQUIRE(!thread.setPolicy(SCHED_OTHER, 0)); +#endif // try to set the maximum number of accepted deadline miss REQUIRE(!thread.setMaximumNumberOfAcceptedDeadlineMiss(0)); From b92d1ca48063b3ce48170f9f429a688df7233464 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Mon, 7 Oct 2024 10:16:38 +0200 Subject: [PATCH 18/25] YarpLoggerDevice inherits from blf PeriodicThread --- .../BipedalLocomotion/YarpRobotLoggerDevice.h | 11 ++--- .../src/YarpRobotLoggerDevice.cpp | 46 +++++++------------ 2 files changed, 20 insertions(+), 37 deletions(-) diff --git a/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h b/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h index 63f09af1db..9913852261 100644 --- a/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h +++ b/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h @@ -21,13 +21,13 @@ #include #include #include -#include #include #include #include #include +#include #include #include #include @@ -37,20 +37,17 @@ namespace BipedalLocomotion class YarpRobotLoggerDevice : public yarp::dev::DeviceDriver, public yarp::dev::IMultipleWrapper, - public yarp::os::PeriodicThread + public BipedalLocomotion::System::PeriodicThread { public: - YarpRobotLoggerDevice(double period, - yarp::os::ShouldUseSystemClock useSystemClock - = yarp::os::ShouldUseSystemClock::No); - YarpRobotLoggerDevice(); + YarpRobotLoggerDevice(double period = 0.01); ~YarpRobotLoggerDevice(); virtual bool open(yarp::os::Searchable& config) final; virtual bool close() final; virtual bool attachAll(const yarp::dev::PolyDriverList& poly) final; virtual bool detachAll() final; - virtual void run() final; + virtual bool run() final; private: std::chrono::nanoseconds m_previousTimestamp; diff --git a/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp b/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp index ad60885224..14e2918e5e 100644 --- a/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp +++ b/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp @@ -93,23 +93,9 @@ void YarpRobotLoggerDevice::VectorsCollectionSignal::disconnect() } } -YarpRobotLoggerDevice::YarpRobotLoggerDevice(double period, - yarp::os::ShouldUseSystemClock useSystemClock) - : yarp::os::PeriodicThread(period, useSystemClock) -{ - // Use the yarp clock in blf - BipedalLocomotion::System::ClockBuilder::setFactory( - std::make_shared()); - - // the logging message are streamed using yarp - BipedalLocomotion::TextLogging::LoggerBuilder::setFactory( - std::make_shared()); - - m_sendDataRT = false; -} - -YarpRobotLoggerDevice::YarpRobotLoggerDevice() - : yarp::os::PeriodicThread(0.01, yarp::os::ShouldUseSystemClock::No) +YarpRobotLoggerDevice::YarpRobotLoggerDevice(double period) + : BipedalLocomotion::System::PeriodicThread( + std::chrono::nanoseconds(static_cast(period * 1e9))) { // Use the yarp clock in blf BipedalLocomotion::System::ClockBuilder::setFactory( @@ -149,7 +135,7 @@ bool YarpRobotLoggerDevice::open(yarp::os::Searchable& config) double devicePeriod{0.01}; if (params->getParameter("sampling_period_in_s", devicePeriod)) { - this->setPeriod(devicePeriod); + this->setPeriod(std::chrono::nanoseconds(static_cast(devicePeriod))); } if (!params->getParameter("text_logging_subnames", m_textLoggingSubnames)) @@ -1189,22 +1175,21 @@ void YarpRobotLoggerDevice::lookForExogenousSignals() continue; } - log()->info("[YarpRobotLoggerDevice::lookForExogenousSignals] Attempt to get the " - "metadata for the vectors collection signal named: {}", + log()->info("[YarpRobotLoggerDevice::lookForExogenousSignals] Attempt to get " + "the metadata for the vectors collection signal named: {}", name); if (!signal.client.getMetadata(signal.metadata)) { - log()->warn("[YarpRobotLoggerDevice::lookForExogenousSignals] Unable to get " - "the metadata for the signal named: {}. The exogenous signal will " - "not contain the metadata.", + log()->warn("[YarpRobotLoggerDevice::lookForExogenousSignals] Unable to " + "get the metadata for the signal named: {}. The exogenous " + "signal will not contain the metadata.", name); } } } signal.connected = connectionDone; - } }; @@ -1429,7 +1414,7 @@ void YarpRobotLoggerDevice::recordVideo(const std::string& cameraName, VideoWrit } } -void YarpRobotLoggerDevice::run() +bool YarpRobotLoggerDevice::run() { auto logData = [this](const std::string& name, const auto& data, const double time) { m_bufferManager.push_back(data, time, name); @@ -1453,7 +1438,7 @@ void YarpRobotLoggerDevice::run() std::chrono::duration(m_previousTimestamp), std::chrono::duration(t), std::chrono::duration(t - m_previousTimestamp)); - return; + return true; } } @@ -1718,12 +1703,11 @@ void YarpRobotLoggerDevice::run() yarp::os::Bottle* b = m_textLoggingPort.read(false); if (b != nullptr) { - msg = BipedalLocomotion::TextLoggingEntry::deserializeMessage(*b, - std::to_string(time)); + msg = BipedalLocomotion::TextLoggingEntry::deserializeMessage(*b, std::to_string(time)); if (msg.isValid) { signalFullName = msg.portSystem + "::" + msg.portPrefix + "::" + msg.processName - + "::p" + msg.processPID; + + "::p" + msg.processPID; // matlab does not support the character - as a key of a struct findAndReplaceAll(signalFullName, "-", "_"); @@ -1736,7 +1720,7 @@ void YarpRobotLoggerDevice::run() m_bufferManager.addChannel({signalFullName, {1, 1}}); m_textLogsStoredInManager.insert(signalFullName); } - //Not using logData here because we don't want to stream the data to RT + // Not using logData here because we don't want to stream the data to RT m_bufferManager.push_back(msg, time, signalFullName); } bufferportSize = m_textLoggingPort.getPendingReads(); @@ -1753,6 +1737,8 @@ void YarpRobotLoggerDevice::run() m_previousTimestamp = t; m_firstRun = false; + + return true; } bool YarpRobotLoggerDevice::saveCallback(const std::string& fileName, From f0d9d83e2b0c11f269a8b87c9e04bed15a1c1b74 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 8 Oct 2024 09:55:01 +0200 Subject: [PATCH 19/25] add check of successfull initialization of the thread --- .../BipedalLocomotion/System/PeriodicThread.h | 11 ++++ src/System/src/PeriodicThread.cpp | 63 +++++++++++++++++-- 2 files changed, 70 insertions(+), 4 deletions(-) diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index da24647b00..2dbd934b75 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -10,7 +10,9 @@ #include #include +#include #include +#include #include #include @@ -206,6 +208,15 @@ class PeriodicThread smaller. */ bool m_earlyWakeUp = false; /**< If true, the thread will be awaken before and busy wait until the actual wake up time. */ + + std::condition_variable m_cv; /**< Condition variable to check for + * initialization. + */ + std::atomic m_ready = false; /**< Flag to signal that the inizialization task are + completed. */ + std::atomic m_initializationSuccessful = false; /**< Flag to signal the result of + initialization */ + std::mutex m_cv_mtx; /**< Mutex to protect the condition variable. */ }; } // namespace System } // namespace BipedalLocomotion diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 7edf1a112e..1d89c4d63e 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -2,6 +2,7 @@ #include #include #include +#include #ifdef __linux__ #include @@ -64,15 +65,36 @@ void PeriodicThread::threadFunction() { constexpr auto logPrefix = "[PeriodicThread::threadFunction]"; + m_initializationSuccessful.store(true); + // thread initialization if (!threadInit()) { - return; + log()->error("{} The thread initialization failed.", logPrefix); + m_initializationSuccessful.store(false); } // set the policy if (!setPolicy()) { + log()->error("{} The policy and priority cannot be set.", logPrefix); + m_initializationSuccessful.store(false); + } + + // Notify that initialization is completed (successful or not) + { + // https://en.cppreference.com/w/cpp/thread/condition_variable: + // Even if the shared variable is atomic, it must be modified while owning the mutex to + // correctly publish the modification to the waiting thread. + std::lock_guard lock(m_cv_mtx); + m_ready.store(true); + } + m_cv.notify_one(); + + // cannot continue if the initialization was not successful + if (!m_initializationSuccessful.load()) + { + m_state.store(PeriodicThreadState::STOPPED); return; } @@ -96,14 +118,34 @@ void PeriodicThread::threadFunction() bool PeriodicThread::setPolicy() { + constexpr auto logPrefix = "[PeriodicThread::setPolicy]"; + #ifdef __linux__ // get the current thread native handle pthread_t nativeHandle = pthread_self(); // get the current thread parameters sched_param params; params.sched_priority = m_priority; - // set the new policy - return (pthread_setschedparam(nativeHandle, m_policy, ¶ms) == 0); + // Set the scheduling policy to SCHED_FIFO and priority + int ret = pthread_setschedparam(nativeHandle, SCHED_FIFO, ¶ms); + if (ret != 0) + { + log()->error("{} Failed to set scheduling policy, with error: {}", logPrefix, ret); + if (ret == EPERM) + { + log()->error("{} The calling thread does not have the appropriate privileges to set " + "the requested scheduling policy and parameters. Try to run the " + "YarpLoggerDevice with 'sudo -E'.", + logPrefix); + } + return false; + } else + { + log()->info("{} Scheduling policy set to SCHED_FIFO with priority {}", + logPrefix, + params.sched_priority); + return true; + } #else return true; #endif @@ -221,6 +263,9 @@ bool PeriodicThread::resume() bool PeriodicThread::start(std::shared_ptr barrier) { + + std::unique_lock lock(m_cv_mtx); // lock the mutex for the condition variable + // only an inactive thread can be started if (m_state.load() != PeriodicThreadState::INACTIVE) { @@ -236,9 +281,19 @@ bool PeriodicThread::start(std::shared_ptr b // lambda wrapper for the thread function auto threadFunctionLambda = [this]() { this->threadFunction(); }; + // start the thread m_thread = std::thread(threadFunctionLambda); - return m_thread.joinable(); + // check if the thread is joinable + if (!m_thread.joinable()) + { + return false; + } + + // check if initialization was successful and thread started + m_cv.wait(lock, [this] { return m_ready.load(); }); + + return m_initializationSuccessful.load(); }; bool PeriodicThread::isRunning() From 7d98305923fe2c930a5e822c87bfe766e2518e76 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 8 Oct 2024 10:14:37 +0200 Subject: [PATCH 20/25] fix regression --- src/System/src/PeriodicThread.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 1d89c4d63e..799f6b356b 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -127,7 +127,7 @@ bool PeriodicThread::setPolicy() sched_param params; params.sched_priority = m_priority; // Set the scheduling policy to SCHED_FIFO and priority - int ret = pthread_setschedparam(nativeHandle, SCHED_FIFO, ¶ms); + int ret = pthread_setschedparam(nativeHandle, m_policy, ¶ms); if (ret != 0) { log()->error("{} Failed to set scheduling policy, with error: {}", logPrefix, ret); @@ -141,9 +141,9 @@ bool PeriodicThread::setPolicy() return false; } else { - log()->info("{} Scheduling policy set to SCHED_FIFO with priority {}", - logPrefix, - params.sched_priority); + log()->debug("{} Scheduling policy set to SCHED_FIFO with priority {}", + logPrefix, + params.sched_priority); return true; } #else From 0cafd21c39a1f296e195e0379a29e72abffbfe1a Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 8 Oct 2024 10:36:21 +0200 Subject: [PATCH 21/25] fix order of input arguments of a member function --- src/System/src/PeriodicThread.cpp | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 799f6b356b..02b7d70637 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -141,8 +141,9 @@ bool PeriodicThread::setPolicy() return false; } else { - log()->debug("{} Scheduling policy set to SCHED_FIFO with priority {}", + log()->debug("{} Scheduling policy set to {} with priority {}", logPrefix, + m_policy, params.sched_priority); return true; } @@ -152,7 +153,7 @@ bool PeriodicThread::setPolicy() }; #ifdef __linux__ -bool PeriodicThread::setPolicy(int priority, int policy) +bool PeriodicThread::setPolicy(int policy, int priority) { if (m_state.load() != PeriodicThreadState::INACTIVE) { From 68247630143cb10b5928e44886db22b38c6a8e1b Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 8 Oct 2024 10:41:19 +0200 Subject: [PATCH 22/25] add real time strategies to yarp logger device --- .../BipedalLocomotion/YarpRobotLoggerDevice.h | 9 +++ .../src/YarpRobotLoggerDevice.cpp | 75 +++++++++++++++++++ 2 files changed, 84 insertions(+) diff --git a/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h b/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h index 9913852261..d762df46f6 100644 --- a/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h +++ b/devices/YarpRobotLoggerDevice/include/BipedalLocomotion/YarpRobotLoggerDevice.h @@ -174,6 +174,15 @@ class YarpRobotLoggerDevice : public yarp::dev::DeviceDriver, std::mutex m_bufferManagerMutex; robometry::BufferManager m_bufferManager; + enum class RealTimeSchedulingStrategy + { + None, + EarlyWakeUp, + FIFO, + EarlyWakeUpAndFIFO, + }; + RealTimeSchedulingStrategy m_RealTimeSchedulingStrategy{RealTimeSchedulingStrategy::None}; + void lookForNewLogs(); void lookForExogenousSignals(); diff --git a/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp b/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp index 14e2918e5e..f453f0957b 100644 --- a/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp +++ b/devices/YarpRobotLoggerDevice/src/YarpRobotLoggerDevice.cpp @@ -15,6 +15,9 @@ #include #include #include +#ifdef __linux__ +#include +#endif #include #include @@ -138,6 +141,78 @@ bool YarpRobotLoggerDevice::open(yarp::os::Searchable& config) this->setPeriod(std::chrono::nanoseconds(static_cast(devicePeriod))); } + std::string realTimeSchedulingStrategy{"none"}; + if (!params->getParameter("real_time_scheduling_strategy", realTimeSchedulingStrategy)) + { + log()->info("{} The 'real_time_scheduling_strategy' parameter is not found. " + "YarpLoggerDevice will run without any real time strategy.", + logPrefix); + } + if (realTimeSchedulingStrategy == "none") + { + m_RealTimeSchedulingStrategy = RealTimeSchedulingStrategy::None; + } else if (realTimeSchedulingStrategy == "early_wakeup") + { + m_RealTimeSchedulingStrategy = RealTimeSchedulingStrategy::EarlyWakeUp; + } else if (realTimeSchedulingStrategy == "fifo") + { + m_RealTimeSchedulingStrategy = RealTimeSchedulingStrategy::FIFO; + } else if (realTimeSchedulingStrategy == "early_wakeup_and_fifo") + { + m_RealTimeSchedulingStrategy = RealTimeSchedulingStrategy::EarlyWakeUpAndFIFO; + } else + { + log()->error("{} The 'real_time_scheduling_strategy' parameter is not valid. Available " + "options are 'none', 'early_wakeup', 'fifo', 'early_wakeup_and_fifo'.", + logPrefix); + return false; + } + + switch (m_RealTimeSchedulingStrategy) + { + + case RealTimeSchedulingStrategy::None: + break; + + case RealTimeSchedulingStrategy::EarlyWakeUp: + if (!this->enableEarlyWakeUp()) + { + log()->error("{} Failed to enable the early wake up.", logPrefix); + return false; + } + break; +#ifdef __linux__ + case RealTimeSchedulingStrategy::FIFO: + if (!this->setPolicy(SCHED_FIFO, 80)) + { + log()->error("{} Failed to set the policy to SCHED_FIFO.", logPrefix); + return false; + } + log()->info("{} The FIFO scheduling policy is set.", logPrefix); + break; + case RealTimeSchedulingStrategy::EarlyWakeUpAndFIFO: + if (!this->enableEarlyWakeUp()) + { + log()->error("{} Failed to enable the early wake up.", logPrefix); + return false; + } + if (!this->setPolicy(SCHED_FIFO, 80)) + { + log()->error("{} Failed to set the policy to SCHED_FIFO.", logPrefix); + return false; + } + break; +#else + case RealTimeSchedulingStrategy::FIFO: + log()->error("{} The FIFO scheduling policy is not supported on this OS.", logPrefix); + return false; + case RealTimeSchedulingStrategy::EarlyWakeUpAndFIFO: + log()->error("{} The EarlyWakeUpAndFIFO scheduling policy is not supported on this OS.", + logPrefix); + return false; +#endif + } + if (!params->getParameter("text_logging_subnames", m_textLoggingSubnames)) { log()->info("{} Unable to get the 'text_logging_subnames' parameter for the telemetry. All " From 758728eba682f6f1a909f49b7250569e94ed8db9 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 8 Oct 2024 11:10:08 +0200 Subject: [PATCH 23/25] update YarpLoggerDevice readme --- devices/YarpRobotLoggerDevice/README.md | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/devices/YarpRobotLoggerDevice/README.md b/devices/YarpRobotLoggerDevice/README.md index 3b6fe796f4..c3ff265767 100644 --- a/devices/YarpRobotLoggerDevice/README.md +++ b/devices/YarpRobotLoggerDevice/README.md @@ -138,3 +138,28 @@ robot-log-visualizer Then, you can open the mat file generated by the logger and explore the logged data as in the following video: [robot-log-visualizer.webm](https://github.com/ami-iit/robot-log-visualizer/assets/16744101/3fd5c516-da17-4efa-b83b-392b5ce1383b) + +## How to reduce latency +If you are experiencing some latency in logged data, you can try to enable different real time scheduling strategies through the `real_time_scheduling_strategy` in the [yarp-robot-logger.xml](https://github.com/ami-iit/bipedal-locomotion-framework/blob/master/devices/YarpRobotLoggerDevice/app/robots/ergoCubSN000/yarp-robot-logger.xml). For example: + +```xml + + + ergocub + 0.001 + early_wakeup_and_fifo +``` + +The available strategies are `none` (which is the default and means no strategy will be applied), `early_wakeup`, `fifo`, and `early_wakeup_and_fifo`. +The `early_wakeup` strategy makes the YarpLoggerDevice run thread wake up earlier and then busy wait until it is time to resume. +The `fifo` strategy increases the YarpLoggerDevice run thread priority and changes its scheduling policy to SCHED-FIFO. +The `early_wakeup_and_fifo` combines the `early_wakeup` and `fifo` strategies. + +Note that the `fifo` and `early_wakeup_and_fifo` strategies are only available for Linux. Moreover, you should run the `YarpLoggerDevice` with elevated privileges when deploying them. +For example: + +```console +sudo -E /yarprobotinterface --config launch-yarp-robot-logger.xml +``` + +with `` being the directory containing `yarprobotinterface`, that you can determine with the terminal command `which yarprobotinterface`. \ No newline at end of file From 5958329e3f5a87df74638f3d87aa4bd620856f68 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Tue, 8 Oct 2024 12:20:38 +0200 Subject: [PATCH 24/25] update changelog --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0d25be00ac..ec02876298 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,7 @@ All notable changes to this project are documented in this file. - Implement low-pass filter for estimated friction torques in `JointTorqueControlDevice` (https://github.com/ami-iit/bipedal-locomotion-framework/pull/892) - Add `blf-motor-current-tracking.py` application (https://github.com/ami-iit/bipedal-locomotion-framework/pull/894) - Add the possibility to initialize the base position and the feet pose in the `unicycleTrajectoryGenerator` (https://github.com/ami-iit/bipedal-locomotion-framework/pull/887) +- Add a `Periodic Thread Class` and add options to reduce `YarpLoggerDevice` latency (https://github.com/ami-iit/bipedal-locomotion-framework/pull/889) ### Changed From 82b60abdbef2544d7d10c91863eca1345b016a52 Mon Sep 17 00:00:00 2001 From: Lorenzo Date: Wed, 9 Oct 2024 15:16:24 +0200 Subject: [PATCH 25/25] add some getters and do some polishing --- examples/PeriodicThread/main.cpp | 27 ++++-- .../BipedalLocomotion/System/PeriodicThread.h | 94 +++++++++++++------ src/System/src/PeriodicThread.cpp | 57 +++++++---- src/System/tests/PeriodicThreadTest.cpp | 3 + 4 files changed, 124 insertions(+), 57 deletions(-) diff --git a/examples/PeriodicThread/main.cpp b/examples/PeriodicThread/main.cpp index 27b9fa9da5..292124ed14 100644 --- a/examples/PeriodicThread/main.cpp +++ b/examples/PeriodicThread/main.cpp @@ -1,5 +1,6 @@ #include #include +#include #include #include @@ -9,7 +10,8 @@ class Thread : public BipedalLocomotion::System::PeriodicThread { public: - Thread(); + std::string name = "Thread"; + Thread(std::string name); ~Thread(); bool run() override; @@ -18,21 +20,22 @@ class Thread : public BipedalLocomotion::System::PeriodicThread bool Thread::run() { - BipedalLocomotion::log()->info("[Thread::run] Thread is running."); + BipedalLocomotion::log()->info("[Thread::run] {} is running.", name); return true; } -Thread::Thread() - : BipedalLocomotion::System::PeriodicThread(std::chrono::milliseconds(1000)){}; +Thread::Thread(std::string name) + : name(name) + , BipedalLocomotion::System::PeriodicThread(std::chrono::milliseconds(1000)){}; Thread::~Thread() { - BipedalLocomotion::log()->info("[Thread::~Thread] Thread is destroyed."); + BipedalLocomotion::log()->info("[Thread::~Thread] {} is destroyed.", name); }; bool Thread::threadInit() { - BipedalLocomotion::log()->info("[Thread::threadInit] Thread is initialized."); + BipedalLocomotion::log()->info("[Thread::threadInit] {} is initialized.", name); return true; } @@ -42,20 +45,26 @@ int main() auto barrier = BipedalLocomotion::System::Barrier::create(2); // Thread thread; - auto thread1 = Thread(); + auto thread1 = Thread("Thread 1"); thread1.start(barrier); BipedalLocomotion::clock().sleepFor(std::chrono::milliseconds(2000)); - auto thread2 = Thread(); + auto thread2 = Thread("Thread 2"); thread2.setPeriod(std::chrono::seconds(2)); thread2.start(barrier); - while (thread1.isRunning()) + while (thread1.isRunning() || thread2.isRunning()) { std::this_thread::sleep_for(std::chrono::milliseconds(4000)); thread1.stop(); BipedalLocomotion::log()->info("[main] Thread 1 is asked to stop."); + + if (!thread1.isRunning()) + { + thread2.stop(); + BipedalLocomotion::log()->info("[main] Thread 2 is asked to stop."); + } } BipedalLocomotion::log()->info("[main] About to exit the application."); return EXIT_SUCCESS; diff --git a/src/System/include/BipedalLocomotion/System/PeriodicThread.h b/src/System/include/BipedalLocomotion/System/PeriodicThread.h index 2dbd934b75..7538f75110 100644 --- a/src/System/include/BipedalLocomotion/System/PeriodicThread.h +++ b/src/System/include/BipedalLocomotion/System/PeriodicThread.h @@ -36,14 +36,21 @@ enum class PeriodicThreadState }; /** - * @brief This class implements a periodic thread. The user has to inherit from this class and - * implement the virtual methods. + * @brief The PeriodicThread class is designed to implement a periodic thread that performs a + * specific task at regular intervals. The task is defined by overriding the virtual run method. The + * class provides various methods to control the thread's lifecycle, including starting, stopping, + * suspending, and resuming the thread. It also allows configuring thread-specific parameters like + * priority and scheduling policy (on Linux systems). Additionally, it supports synchronization with + * other threads using barriers and can handle early wake-up scenarios to minimize latency. + * Finally, the class provides a mechanism to monitor the number of deadline misses, which can be + * useful for real-time applications. */ class PeriodicThread { public: #ifdef __linux__ // Default constructor + PeriodicThread(std::chrono::nanoseconds period = std::chrono::nanoseconds(100000), int maximumNumberOfAcceptedDeadlineMiss = -1, int priority = 0, @@ -68,7 +75,9 @@ class PeriodicThread /** * @brief Start the thread. * @param barrier barrier to synchronize the thread. If nullptr, the thread will start - * immediately, without waiting for other threads to reach the barrier. + * immediately, without waiting for other threads to reach the barrier. Instead, if for example + * the barrier is creted as std::make_shared(2), the thread + * will wait for another thread to reach the barrier before starting. * @return true if the thread was correctly started, false otherwise. */ bool start(std::shared_ptr barrier = nullptr); @@ -93,13 +102,13 @@ class PeriodicThread * stopped. * @return true if the thread is running, false otherwise. */ - bool isRunning(); + bool isRunning() const; /** * @brief Check if the thread is initialized. * @return true if the thread is initialized, false otherwise. */ - bool isInitialized(); + bool isInitialized() const; /** * @brief Set the period of the thread. @@ -108,14 +117,28 @@ class PeriodicThread */ bool setPeriod(std::chrono::nanoseconds period); + /** + * @brief Get the period of the thread. + * @return period of the thread. + */ + std::chrono::nanoseconds getPeriod() const; + #ifdef __linux__ /** - * @brief Set the policy of the thread. + * @brief Set the policy and priority of the thread before starting it. When starting, the + * thread will try to use this policy and priority. * @param policy policy of the thread. * @param priority priority of the thread. - * @return true if the policy was correctly set, false otherwise. + * @return true if the policy and priority are correctly set, false otherwise. */ bool setPolicy(int policy, int priority = 0); + + /** + * @brief Get the policy and prority of the thread. + * @return policy of the thread. + */ + void getPolicy(int& policy, int& priority) const; + #endif /** @@ -124,11 +147,17 @@ class PeriodicThread */ bool setMaximumNumberOfAcceptedDeadlineMiss(int maximumNumberOfAcceptedDeadlineMiss); + /** + * @brief Get the maximum number of accepted deadline miss. + * @return maximum number of accepted deadline miss. + */ + int getMaximumNumberOfAcceptedDeadlineMiss() const; + /** * @brief Get the number of deadline miss. * @return number of deadline miss. */ - int getNumberOfDeadlineMiss(); + int getNumberOfDeadlineMiss() const; /** * @brief Enable the early wake up. The thread will be awaken before and busy wait until the @@ -137,6 +166,12 @@ class PeriodicThread */ bool enableEarlyWakeUp(); + /** + * @brief Check if the early wake up is enabled. + * @return true if the early wake up is enabled, false otherwise. + */ + bool isEarlyWakeUpEnabled() const; + protected: /** * @brief This method is called at each iteration of the thread. @@ -153,33 +188,34 @@ class PeriodicThread private: /** - * @brief run the periodic thread. + * @brief This is the function being executed by the std::thread. */ void threadFunction(); /** - * @brief Advance the thread of one step, calling the user defined run function once. + * @brief Advance the thread of one step, calling the virtual run function once. */ void advance(); /** - * @brief Synchronize the thread. + * @brief Synchronize the thread with other threads with a barrier. */ void synchronize(); /** - * @brief Set the policy of the thread. - * @return true if the policy was correctly set, false otherwise. + * @brief Set the policy and priority of the thread. + * @return true if the policy and priority were correctly set, false otherwise. */ bool setPolicy(); - std::chrono::nanoseconds m_period = std::chrono::nanoseconds(100000); /**< Period of the - * thread. - */ + std::atomic m_period = std::chrono::nanoseconds(100000); /**< Period + * of the + * thread. + */ - int m_maximumNumberOfAcceptedDeadlineMiss = -1; /**< Maximum number of accepted deadline - * miss. - */ + std::atomic m_maximumNumberOfAcceptedDeadlineMiss = -1; /**< Maximum number of accepted + * deadline miss. + */ std::atomic m_deadlineMiss = 0; /**< Number of deadline miss. */ @@ -187,9 +223,9 @@ class PeriodicThread * thread. */ #ifdef __linux__ - int m_priority = 0; /**< Priority of the thread. */ + std::atomic m_priority = 0; /**< Priority of the thread. */ - int m_policy = SCHED_OTHER; /**< Policy of the thread. */ + std::atomic m_policy = SCHED_OTHER; /**< Policy of the thread. */ #endif std::thread m_thread; /**< Thread object. */ @@ -206,17 +242,15 @@ class PeriodicThread varies depending on the OS. For now we fix it to a constant value. Note that in real-time OS it might be smaller. */ - bool m_earlyWakeUp = false; /**< If true, the thread will be awaken before and busy wait until - the actual wake up time. */ - - std::condition_variable m_cv; /**< Condition variable to check for - * initialization. - */ - std::atomic m_ready = false; /**< Flag to signal that the inizialization task are - completed. */ + std::atomic m_earlyWakeUp = false; /**< If true, the thread will be awaken before and busy + wait until the actual wake up time. */ + + std::condition_variable m_cv; /**< Condition variable to check for initialization.*/ + std::mutex m_cvMutex; /**< Mutex to protect the condition variable. */ + std::atomic m_ready = false; /**< Flag to signal that the inizialization tasks of the + thread are completed. */ std::atomic m_initializationSuccessful = false; /**< Flag to signal the result of initialization */ - std::mutex m_cv_mtx; /**< Mutex to protect the condition variable. */ }; } // namespace System } // namespace BipedalLocomotion diff --git a/src/System/src/PeriodicThread.cpp b/src/System/src/PeriodicThread.cpp index 02b7d70637..3507c5cc2d 100644 --- a/src/System/src/PeriodicThread.cpp +++ b/src/System/src/PeriodicThread.cpp @@ -86,7 +86,7 @@ void PeriodicThread::threadFunction() // https://en.cppreference.com/w/cpp/thread/condition_variable: // Even if the shared variable is atomic, it must be modified while owning the mutex to // correctly publish the modification to the waiting thread. - std::lock_guard lock(m_cv_mtx); + std::lock_guard lock(m_cvMutex); m_ready.store(true); } m_cv.notify_one(); @@ -125,9 +125,9 @@ bool PeriodicThread::setPolicy() pthread_t nativeHandle = pthread_self(); // get the current thread parameters sched_param params; - params.sched_priority = m_priority; + params.sched_priority = m_priority.load(); // Set the scheduling policy to SCHED_FIFO and priority - int ret = pthread_setschedparam(nativeHandle, m_policy, ¶ms); + int ret = pthread_setschedparam(nativeHandle, m_policy.load(), ¶ms); if (ret != 0) { log()->error("{} Failed to set scheduling policy, with error: {}", logPrefix, ret); @@ -135,7 +135,7 @@ bool PeriodicThread::setPolicy() { log()->error("{} The calling thread does not have the appropriate privileges to set " "the requested scheduling policy and parameters. Try to run the " - "YarpLoggerDevice with 'sudo -E'.", + "application with 'sudo -E'.", logPrefix); } return false; @@ -143,7 +143,7 @@ bool PeriodicThread::setPolicy() { log()->debug("{} Scheduling policy set to {} with priority {}", logPrefix, - m_policy, + m_policy.load(), params.sched_priority); return true; } @@ -161,10 +161,16 @@ bool PeriodicThread::setPolicy(int policy, int priority) "started. The policy and priority cannot be changed."); return false; } - m_priority = priority; - m_policy = policy; + m_policy.store(policy); + m_priority.store(priority); return true; }; + +void PeriodicThread::getPolicy(int& policy, int& priority) const +{ + policy = m_policy.load(); + priority = m_priority.load(); +}; #endif bool PeriodicThread::setPeriod(std::chrono::nanoseconds period) @@ -179,6 +185,11 @@ bool PeriodicThread::setPeriod(std::chrono::nanoseconds period) return true; } +std::chrono::nanoseconds PeriodicThread::getPeriod() const +{ + return m_period.load(); +} + bool PeriodicThread::setMaximumNumberOfAcceptedDeadlineMiss(int maximumNumberOfAcceptedDeadlineMiss) { if (m_state.load() != PeriodicThreadState::INACTIVE) @@ -188,11 +199,16 @@ bool PeriodicThread::setMaximumNumberOfAcceptedDeadlineMiss(int maximumNumberOfA "accepted deadline miss cannot be changed."); return false; } - m_maximumNumberOfAcceptedDeadlineMiss = maximumNumberOfAcceptedDeadlineMiss; + m_maximumNumberOfAcceptedDeadlineMiss.store(maximumNumberOfAcceptedDeadlineMiss); return true; } -int PeriodicThread::getNumberOfDeadlineMiss() +int PeriodicThread::getMaximumNumberOfAcceptedDeadlineMiss() const +{ + return m_maximumNumberOfAcceptedDeadlineMiss.load(); +} + +int PeriodicThread::getNumberOfDeadlineMiss() const { return m_deadlineMiss.load(); } @@ -205,7 +221,7 @@ bool PeriodicThread::enableEarlyWakeUp() "already started. The early wake up cannot be changed."); return false; } - m_earlyWakeUp = true; + m_earlyWakeUp.store(true); return true; } @@ -265,7 +281,7 @@ bool PeriodicThread::resume() bool PeriodicThread::start(std::shared_ptr barrier) { - std::unique_lock lock(m_cv_mtx); // lock the mutex for the condition variable + std::unique_lock lock(m_cvMutex); // lock the mutex for the condition variable // only an inactive thread can be started if (m_state.load() != PeriodicThreadState::INACTIVE) @@ -297,24 +313,29 @@ bool PeriodicThread::start(std::shared_ptr b return m_initializationSuccessful.load(); }; -bool PeriodicThread::isRunning() +bool PeriodicThread::isRunning() const { return ((m_state.load() != PeriodicThreadState::INACTIVE) && (m_state.load() != PeriodicThreadState::STOPPED)); } -bool PeriodicThread::isInitialized() +bool PeriodicThread::isInitialized() const { return (m_state.load() == PeriodicThreadState::INITIALIZED); } +bool PeriodicThread::isEarlyWakeUpEnabled() const +{ + return (m_earlyWakeUp.load()); +} + void PeriodicThread::advance() { // get the current time auto now = BipedalLocomotion::clock().now(); // busy wait until wake up time - if (m_earlyWakeUp) + if (isEarlyWakeUpEnabled()) { while (now < m_wakeUpTime) { @@ -323,7 +344,7 @@ void PeriodicThread::advance() } // get the next wake up time - m_wakeUpTime = now + m_period; + m_wakeUpTime = now + m_period.load(); // run user overridden function, when not idling if (m_state.load() != PeriodicThreadState::IDLE) @@ -340,9 +361,9 @@ void PeriodicThread::advance() if (BipedalLocomotion::clock().now() > m_wakeUpTime) { m_deadlineMiss.fetch_add(1); // increment the number of deadline miss - if (m_maximumNumberOfAcceptedDeadlineMiss > 0) + if (m_maximumNumberOfAcceptedDeadlineMiss.load() > 0) { - if (m_deadlineMiss.load() > m_maximumNumberOfAcceptedDeadlineMiss) + if (m_deadlineMiss.load() > m_maximumNumberOfAcceptedDeadlineMiss.load()) { // we have to close the runner m_state.store(PeriodicThreadState::STOPPED); @@ -355,7 +376,7 @@ void PeriodicThread::advance() BipedalLocomotion::clock().yield(); // wait until the next deadline - if (m_earlyWakeUp) + if (isEarlyWakeUpEnabled()) { BipedalLocomotion::clock().sleepUntil(m_wakeUpTime - m_schedulerLatency); } else diff --git a/src/System/tests/PeriodicThreadTest.cpp b/src/System/tests/PeriodicThreadTest.cpp index 5206cb8e45..52e3f9b390 100644 --- a/src/System/tests/PeriodicThreadTest.cpp +++ b/src/System/tests/PeriodicThreadTest.cpp @@ -173,6 +173,9 @@ TEST_CASE("Test Periodic Thread", "[PeriodicThreadNotAllowed]") // try to enable early wake up REQUIRE(!thread.enableEarlyWakeUp()); + // check that early wake up has not enabled + REQUIRE(!thread.isEarlyWakeUpEnabled()); + // stop the thread thread.stop(); BipedalLocomotion::clock().sleepFor(period);