aboutsummaryrefslogtreecommitdiff
path: root/sway/realtime.c
diff options
context:
space:
mode:
authorRouven Czerwinski <rouven@czerwinskis.de>2022-05-13 20:30:19 +0200
committerKenny Levinsen <kl@kl.wtf>2022-05-18 11:20:17 +0200
commita3a82efbf6b5b3af840c70038b1b599ba29003ac (patch)
treefe28dbc19271b9dce54f7433a86bab0073fe8ca9 /sway/realtime.c
parent3f600565e4dd4fb7a45a7b721d518c79f8d41b59 (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.c40
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, &param);
+ 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, &param);
+ 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, &param);
+ if (ret != 0) {
+ sway_log(SWAY_INFO, "Failed to set scheduling priority to %d", prio);
+ return;
+ }
+
+ pthread_atfork(NULL, NULL, child_fork_callback);
+}