<term><varname>CPUSchedulingPolicy=</varname></term>
<listitem><para>Sets the CPU scheduling policy for executed processes. Takes one of <option>other</option>,
- <option>batch</option>, <option>idle</option>, <option>fifo</option> or <option>rr</option>. See
+ <option>batch</option>, <option>idle</option>, <option>fifo</option>, <option>rr</option> or <option>ext</option>. See
<citerefentry project='man-pages'><refentrytitle>sched_setscheduler</refentrytitle><manvolnum>2</manvolnum></citerefentry> for
details.</para></listitem>
</varlistentry>
}
bool sched_policy_is_valid(int i) {
- return IN_SET(i, SCHED_OTHER, SCHED_BATCH, SCHED_IDLE, SCHED_FIFO, SCHED_RR);
+ return IN_SET(i, SCHED_OTHER, SCHED_BATCH, SCHED_IDLE, SCHED_FIFO, SCHED_RR, SCHED_EXT);
}
bool sched_priority_is_valid(int i) {
return i >= 0 && i <= sched_get_priority_max(SCHED_RR);
}
+bool sched_policy_supported(int policy) {
+ return sched_get_priority_min(policy) >= 0;
+}
+
+/* Wrappers around sched_get_priority_{min,max}() that gracefully handles missing SCHED_EXT support in the kernel */
+int sched_get_priority_min_safe(int policy) {
+ int r;
+
+ r = sched_get_priority_min(policy);
+ if (r >= 0)
+ return r;
+
+ /* Fallback priority */
+ return 0;
+}
+
+int sched_get_priority_max_safe(int policy) {
+ int r;
+
+ r = sched_get_priority_max(policy);
+ if (r >= 0)
+ return r;
+
+ return 0;
+}
+
/* The cached PID, possible values:
*
* == UNSET [0] → cache not initialized yet
static const char* const sched_policy_table[] = {
[SCHED_OTHER] = "other",
[SCHED_BATCH] = "batch",
- [SCHED_IDLE] = "idle",
- [SCHED_FIFO] = "fifo",
- [SCHED_RR] = "rr",
+ [SCHED_IDLE] = "idle",
+ [SCHED_FIFO] = "fifo",
+ [SCHED_EXT] = "ext",
+ [SCHED_RR] = "rr",
};
DEFINE_STRING_TABLE_LOOKUP_WITH_FALLBACK(sched_policy, int, INT_MAX);
bool nice_is_valid(int n) _const_;
bool sched_policy_is_valid(int i) _const_;
+bool sched_policy_supported(int i);
bool sched_priority_is_valid(int i) _const_;
+int sched_get_priority_min_safe(int i);
+int sched_get_priority_max_safe(int i);
#define PID_AUTOMATIC ((pid_t) INT_MIN) /* special value indicating "acquire pid from connection peer" */
return r;
c->cpu_sched_policy = q;
- c->cpu_sched_priority = CLAMP(c->cpu_sched_priority, sched_get_priority_min(q), sched_get_priority_max(q));
+ c->cpu_sched_priority = CLAMP(c->cpu_sched_priority, sched_get_priority_min_safe(q), sched_get_priority_max_safe(q));
c->cpu_sched_set = true;
unit_write_settingf(u, flags, name, "CPUSchedulingPolicy=%s", s);
r = sched_setattr(/* pid= */ 0, &attr, /* flags= */ 0);
if (r < 0) {
- *exit_status = EXIT_SETSCHEDULER;
- return log_error_errno(errno, "Failed to set up CPU scheduling: %m");
+ if (errno != EINVAL || sched_policy_supported(attr.sched_policy)) {
+ *exit_status = EXIT_SETSCHEDULER;
+ return log_error_errno(errno, "Failed to set up CPU scheduling: %m");
+ }
+
+ log_warning_errno(errno, "CPU scheduling policy %u is not supported, ignoring: %m", attr.sched_policy);
}
}
return 0;
}
+ if (!sched_policy_supported(x))
+ log_syntax(unit, LOG_WARNING, filename, line, x, "Unsupported CPU scheduling policy: %s", rvalue);
+
c->cpu_sched_policy = x;
/* Moving to or from real-time policy? We need to adjust the priority */
- c->cpu_sched_priority = CLAMP(c->cpu_sched_priority, sched_get_priority_min(x), sched_get_priority_max(x));
+ c->cpu_sched_priority = CLAMP(c->cpu_sched_priority, sched_get_priority_min_safe(x), sched_get_priority_max_safe(x));
c->cpu_sched_set = true;
return 0;
int missing_sched_setattr(pid_t pid, struct sched_attr *attr, unsigned flags);
# define sched_setattr missing_sched_setattr
#endif
+
+/* f0e1a0643a59bf1f922fa209cec86a170b784f3f (6.12),
+ * defined in sched.h in glibc since glibc-2.41. */
+#ifndef SCHED_EXT
+# define SCHED_EXT 7
+#else
+static_assert(SCHED_EXT == 7, "");
+#endif
int main(int argc, char *argv[]) {
_cleanup_(rm_rf_physical_and_freep) char *runtime_dir = NULL;
_cleanup_(manager_freep) Manager *m = NULL;
- Unit *idle_ok, *idle_bad, *rr_ok, *rr_bad, *rr_sched;
+ Unit *idle_ok, *idle_bad, *rr_ok, *rr_bad, *rr_sched, *ext_ok;
Service *ser;
int r;
assert_se(ser->exec_context.cpu_sched_policy == SCHED_RR);
assert_se(ser->exec_context.cpu_sched_priority == 99);
+ /* load ext ok */
+ assert_se(manager_load_startable_unit_or_warn(m, "sched_ext_ok.service", NULL, &ext_ok) >= 0);
+ ser = SERVICE(ext_ok);
+ assert_se(ser->exec_context.cpu_sched_policy == SCHED_EXT);
+ assert_se(ser->exec_context.cpu_sched_priority == 0);
+
return EXIT_SUCCESS;
}
--- /dev/null
+# SPDX-License-Identifier: LGPL-2.1-or-later
+[Unit]
+Description=Sched ext with prio 0
+
+[Service]
+ExecStart=true
+CPUSchedulingPriority=0
+CPUSchedulingPolicy=ext