diff options
author | Rouven Czerwinski <rouven@czerwinskis.de> | 2022-05-13 20:30:19 +0200 |
---|---|---|
committer | Kenny Levinsen <kl@kl.wtf> | 2022-05-18 11:20:17 +0200 |
commit | a3a82efbf6b5b3af840c70038b1b599ba29003ac (patch) | |
tree | fe28dbc19271b9dce54f7433a86bab0073fe8ca9 /sway/realtime.c | |
parent | 3f600565e4dd4fb7a45a7b721d518c79f8d41b59 (diff) |
realtime: request SCHED_RR using CAP_SYS_NICE
Try to gain SCHED_RR (round-robin) realtime scheduling privileges before
starting the server. This requires CAP_SYS_NICE on Linux systems.
We additionally register a pthread_atfork callback which resets the
scheduling class back to SCHED_OTHER (the Linux system default).
Due to CAP_SYS_NICE, setting RLIMIT_RTPRIO has no effect on the process
as documented within man 7 sched (from Linux):
Privileged (CAP_SYS_NICE) threads ignore the RLIMIT_RTPRIO limit;
as with older kernels, they can make arbitrary changes to
scheduling policy and priority. See getrlimit(2) for further
information on RLIMIT_RTPRIO
Note that this requires the sway distribution packagers to set the
CAP_SYS_NICE capability on the sway binary.
Supersedes #6992
Diffstat (limited to 'sway/realtime.c')
-rw-r--r-- | sway/realtime.c | 40 |
1 files changed, 40 insertions, 0 deletions
diff --git a/sway/realtime.c b/sway/realtime.c new file mode 100644 index 00000000..11154af0 --- /dev/null +++ b/sway/realtime.c @@ -0,0 +1,40 @@ +#include <sys/resource.h> +#include <sched.h> +#include <unistd.h> +#include <pthread.h> +#include "sway/server.h" +#include "log.h" + +static void child_fork_callback(void) { + struct sched_param param; + + param.sched_priority = 0; + + int ret = pthread_setschedparam(pthread_self(), SCHED_OTHER, ¶m); + if (ret != 0) { + sway_log(SWAY_ERROR, "Failed to reset scheduler policy on fork"); + } +} + +void set_rr_scheduling(void) { + int prio = sched_get_priority_min(SCHED_RR); + int old_policy; + int ret; + struct sched_param param; + + ret = pthread_getschedparam(pthread_self(), &old_policy, ¶m); + if (ret != 0) { + sway_log(SWAY_DEBUG, "Failed to get old scheduling priority"); + return; + } + + param.sched_priority = prio; + + ret = pthread_setschedparam(pthread_self(), SCHED_RR, ¶m); + if (ret != 0) { + sway_log(SWAY_INFO, "Failed to set scheduling priority to %d", prio); + return; + } + + pthread_atfork(NULL, NULL, child_fork_callback); +} |