Skip to content

Commit

Permalink
Added schedulers::FifoBusyWait
Browse files Browse the repository at this point in the history
  • Loading branch information
shuhaowu committed Jan 30, 2023
1 parent d67c698 commit 59a115c
Show file tree
Hide file tree
Showing 7 changed files with 110 additions and 12 deletions.
8 changes: 4 additions & 4 deletions examples/simple_example/main.cc
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,12 @@
#include <unistd.h>

// A no-op thread that only serves to do nothing and measure the latency
class CyclicThread : public cactus_rt::CyclicThread<cactus_rt::schedulers::Fifo> {
class CyclicThread : public cactus_rt::CyclicThread<cactus_rt::schedulers::FifoBusyWait> {
public:
CyclicThread(std::vector<size_t> cpu_affinity)
: cactus_rt::CyclicThread<cactus_rt::schedulers::Fifo>("CyclicThread", 1'000'000, /* Period */
cactus_rt::schedulers::Fifo::Config{80 /* Priority */},
cpu_affinity) {}
: cactus_rt::CyclicThread<cactus_rt::schedulers::FifoBusyWait>("CyclicThread", 1'000'000, /* Period */
cactus_rt::schedulers::FifoBusyWait::Config{80 /* Priority */},
cpu_affinity) {}

protected:
bool Loop(int64_t /*now*/) noexcept final {
Expand Down
2 changes: 1 addition & 1 deletion include/cactus_rt/cyclic_thread.h
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ class CyclicThread : public Thread<SchedulerT> {
loop_latency_tracker_.RecordValue(loop_latency);

next_wakeup_time_ = AddTimespecByNs(next_wakeup_time_, period_ns_);
busy_wait_latency = SchedulerT::Sleep(next_wakeup_time_);
busy_wait_latency = SchedulerT::Sleep(next_wakeup_time_, scheduler_config_);

busy_wait_latency_tracker_.RecordValue(busy_wait_latency);
}
Expand Down
3 changes: 2 additions & 1 deletion include/cactus_rt/rt.h
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,8 @@
#include "mutex.h"
#include "schedulers/deadline.h"
#include "schedulers/fifo.h"
#include "schedulers/fifo_busy_wait.h"
#include "schedulers/other.h"
#include "signal_handler.h"
#include "thread.h"
#include "utils.h"
#include "utils.h"
4 changes: 2 additions & 2 deletions include/cactus_rt/schedulers/deadline.h
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ class Deadline {
}
}

inline static double Sleep(const struct timespec& /*next_wakeup_time */) noexcept {
inline static double Sleep(const struct timespec& /*next_wakeup_time*/, const Config& /*config*/) noexcept {
// Ignoring error as man page says "In the Linux implementation, sched_yield() always succeeds."
sched_yield();
return 0.0;
Expand All @@ -72,4 +72,4 @@ inline CyclicThread<schedulers::Deadline>::CyclicThread(const std::string&
}

} // namespace cactus_rt
#endif
#endif
4 changes: 2 additions & 2 deletions include/cactus_rt/schedulers/fifo.h
Original file line number Diff line number Diff line change
Expand Up @@ -32,12 +32,12 @@ class Fifo {
}
}

inline static double Sleep(const struct timespec& next_wakeup_time) noexcept {
inline static double Sleep(const struct timespec& next_wakeup_time, const Config& /*config*/) noexcept {
// TODO: check for errors?
// TODO: can there be remainders?
clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &next_wakeup_time, nullptr);
return 0.0;
}
};
} // namespace cactus_rt::schedulers
#endif
#endif
97 changes: 97 additions & 0 deletions include/cactus_rt/schedulers/fifo_busy_wait.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
#ifndef CACTUS_RT_FIFO_BUSY_WAIT_H_
#define CACTUS_RT_FIFO_BUSY_WAIT_H_

#include <sched.h>
#include <spdlog/spdlog.h>

#include <cerrno>
#include <ctime>

#include "../linux/sched_ext.h"
#include "../utils.h"

namespace cactus_rt::schedulers {
/**
* @brief This is a special FIFO scheduler that will busy sleep to maintain a
* jitter-free loop rate at the cost of CPU. This value can be determined from
* cyclictest.
*
* In Madden 2019 ("Challenges Using Linux as a Real-Time Operating System"),
* the author suggested a method of maintaining very strict timing for periodic
* callbacks with the FIFO scheduler. The way it works is as follows:
*
* 1. Before running/programming the application, measure the hardware +
* scheduling latency of the system you're deploying to.
* 2. In the loop, instead of sleeping to the next desired wake up time, sleep
* until the next desired wake up time minus the estimated hardware + scheduling
* latency.
* 3. Once the application wakes up, the code should be slightly ahead of
* schedule. Thus, you busy wait until the original desired wake up time.
*
* This is explained in details at [this * link](https://shuhaowu.com/blog/2022/04-linux-rt-appdev-part4.html#trick-to-deal-with-wake-up-jitter).
* Specifically, one should look at Figure 3 for a detailed schematic on what
* happens.
*/
class FifoBusyWait {
public:
struct Config {
uint32_t priority = 80;

/**
* @brief An estimate of the hardware + scheduling latency. This estimate
* should be greater than the actual hardware + scheduling latency to ensure
* the Loop() function will not be called late.
*/
int32_t scheduling_latency_estimate_ns = 150'000;
};

// This code should really be the same as the Fifo code.
// TODO: figure out a way to reuse that code? Maybe it's not necessary, tho.
inline static void SetThreadScheduling(const Config& config) {
// Self scheduling attributes
sched_attr attr = {};
attr.size = sizeof(attr);
attr.sched_flags = 0;
attr.sched_nice = 0;
attr.sched_priority = config.priority; // Set the scheduler priority
attr.sched_policy = SCHED_FIFO; // Set the scheduler policy

auto ret = sched_setattr(0, &attr, 0);
if (ret < 0) {
SPDLOG_ERROR("unable to sched_setattr: {}", std::strerror(errno));
throw std::runtime_error{"failed to sched_setattr"};
}
}

inline static double Sleep(const struct timespec& next_wakeup_time, const Config& config) noexcept {
// TODO: refactor some of these timespec -> ns int64_t conversion into an utils function
int64_t next_wakeup_time_ns = next_wakeup_time.tv_sec * 1'000'000'000 + next_wakeup_time.tv_nsec;
int64_t now = NowNs();

// In Madden 2019 ("Challenges Using Linux as a Real-Time Operating System"),
// the author suggested that busy wait should occur if the time to next wake
// up is less than 2 x scheduling latency. The reason cited is because if the
// present task goes to sleep, it may take up to 1 x scheduling latency to
// schedule another task, and another 1 x scheduling latency to schedule the
// present task, making it pointless.
//
// Perhaps my understanding is bad, but I'm not sure how to measure the
// above defined scheduling latency. I would just take the max latency number
// from cyclictest instead, which I think is appropriate for just 1x.
if (next_wakeup_time_ns - now >= config.scheduling_latency_estimate_ns) {
struct timespec next_wakeup_time_minus_scheduling_latency = AddTimespecByNs(next_wakeup_time, -config.scheduling_latency_estimate_ns);
clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &next_wakeup_time_minus_scheduling_latency, nullptr);
}

now = NowNs();

while (next_wakeup_time_ns - NowNs() > 0) {
// busy wait
// TODO: add asm nop? Doesn't seem necessary as we're running a bunch of clock_gettime syscalls anyway..
}

return static_cast<double>(NowNs() - now);
}
};
} // namespace cactus_rt::schedulers
#endif
4 changes: 2 additions & 2 deletions include/cactus_rt/schedulers/other.h
Original file line number Diff line number Diff line change
Expand Up @@ -32,12 +32,12 @@ class Other {
}

// Kind of meaningless for SCHED_OTHER because there's no RT guarantees
inline static double Sleep(const struct timespec& next_wakeup_time) noexcept {
inline static double Sleep(const struct timespec& next_wakeup_time, const Config& /*config*/) noexcept {
// TODO: check for errors?
// TODO: can there be remainders?
clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &next_wakeup_time, nullptr);
return 0.0;
}
};
} // namespace cactus_rt::schedulers
#endif
#endif

0 comments on commit 59a115c

Please sign in to comment.