blob: a8701b55f8cb515d981fcfad15332c0bcf22cc31 [file] [log] [blame]
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +00001/* vi: set sw=4 ts=4: */
2/*
3 * chrt - manipulate real-time attributes of a process
Bernhard Reutner-Fischer6c4dade2008-09-25 12:13:34 +00004 * Copyright (c) 2006-2007 Bernhard Reutner-Fischer
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +00005 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02006 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +00007 */
Denys Vlasenkofb4da162016-11-22 23:14:24 +01008//config:config CHRT
Denys Vlasenkob097a842018-12-28 03:20:17 +01009//config: bool "chrt (4.7 kb)"
Denys Vlasenkofb4da162016-11-22 23:14:24 +010010//config: default y
11//config: help
Denys Vlasenkoa4972312018-01-25 09:34:23 +010012//config: Manipulate real-time attributes of a process.
Denys Vlasenko72089cf2017-07-21 09:50:55 +020013//config: This requires sched_{g,s}etparam support in your libc.
Pere Orga5bc8c002011-04-11 03:29:49 +020014
Denys Vlasenko5c527dc2017-08-04 19:55:01 +020015//applet:IF_CHRT(APPLET_NOEXEC(chrt, chrt, BB_DIR_USR_BIN, BB_SUID_DROP, chrt))
Denys Vlasenkof88e3bf2016-11-22 23:54:17 +010016
17//kbuild:lib-$(CONFIG_CHRT) += chrt.o
18
Pere Orga5bc8c002011-04-11 03:29:49 +020019//usage:#define chrt_trivial_usage
Denys Vlasenkoa2f18d92020-12-18 04:12:51 +010020//usage: "-m | -p [PRIO] PID | [-rfobi] PRIO PROG ARGS"
Pere Orga5bc8c002011-04-11 03:29:49 +020021//usage:#define chrt_full_usage "\n\n"
22//usage: "Change scheduling priority and class for a process\n"
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +020023//usage: "\n -m Show min/max priorities"
Pere Orga5bc8c002011-04-11 03:29:49 +020024//usage: "\n -p Operate on PID"
25//usage: "\n -r Set SCHED_RR class"
26//usage: "\n -f Set SCHED_FIFO class"
27//usage: "\n -o Set SCHED_OTHER class"
Povilas Kanapickas1616fb82018-01-13 23:26:39 +020028//usage: "\n -b Set SCHED_BATCH class"
Povilas Kanapickas0f6e4352018-01-13 23:27:37 +020029//usage: "\n -i Set SCHED_IDLE class"
Pere Orga5bc8c002011-04-11 03:29:49 +020030//usage:
31//usage:#define chrt_example_usage
32//usage: "$ chrt -r 4 sleep 900; x=$!\n"
33//usage: "$ chrt -f -p 3 $x\n"
34//usage: "You need CAP_SYS_NICE privileges to set scheduling attributes of a process"
35
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000036#include <sched.h>
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000037#include "libbb.h"
Denys Vlasenko66ad9be2018-01-23 16:01:13 +010038#ifndef SCHED_IDLE
39# define SCHED_IDLE 5
40#endif
Denys Vlasenko922f6f52010-07-01 16:42:27 +020041
Christian Eggers89f063b2020-11-15 20:25:09 +010042//musl has no __MUSL__ or similar define to check for,
43//but its <sys/types.h> has these lines:
44// #define __NEED_fsblkcnt_t
45// #define __NEED_fsfilcnt_t
46#if defined(__linux__) && defined(__NEED_fsblkcnt_t) && defined(__NEED_fsfilcnt_t)
47# define LIBC_IS_MUSL 1
48# include <sys/syscall.h>
49#else
50# define LIBC_IS_MUSL 0
51#endif
52
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +020053static const char *policy_name(int pol)
54{
55 if (pol > 6)
56 return utoa(pol);
57 return nth_string(
58 "OTHER" "\0" /* 0:SCHED_OTHER */
59 "FIFO" "\0" /* 1:SCHED_FIFO */
60 "RR" "\0" /* 2:SCHED_RR */
61 "BATCH" "\0" /* 3:SCHED_BATCH */
62 "ISO" "\0" /* 4:SCHED_ISO */
63 "IDLE" "\0" /* 5:SCHED_IDLE */
64 "DEADLINE", /* 6:SCHED_DEADLINE */
65 pol
66 );
67}
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000068
Denis Vlasenkoac678ec2007-04-16 22:32:04 +000069static void show_min_max(int pol)
70{
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +020071 const char *fmt = "SCHED_%s min/max priority\t: %u/%u\n";
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000072 int max, min;
Denys Vlasenko922f6f52010-07-01 16:42:27 +020073
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000074 max = sched_get_priority_max(pol);
75 min = sched_get_priority_min(pol);
Denys Vlasenko922f6f52010-07-01 16:42:27 +020076 if ((max|min) < 0)
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +020077 fmt = "SCHED_%s not supported\n";
78 printf(fmt, policy_name(pol), min, max);
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000079}
80
81#define OPT_m (1<<0)
82#define OPT_p (1<<1)
83#define OPT_r (1<<2)
84#define OPT_f (1<<3)
85#define OPT_o (1<<4)
Povilas Kanapickas1616fb82018-01-13 23:26:39 +020086#define OPT_b (1<<5)
Povilas Kanapickas0f6e4352018-01-13 23:27:37 +020087#define OPT_i (1<<6)
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000088
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000089int chrt_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenkoa60f84e2008-07-05 09:18:54 +000090int chrt_main(int argc UNUSED_PARAM, char **argv)
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +000091{
92 pid_t pid = 0;
93 unsigned opt;
94 struct sched_param sp;
Denis Vlasenkob44c7902008-03-17 09:29:43 +000095 char *pid_str;
96 char *priority = priority; /* for compiler */
97 const char *current_new;
98 int policy = SCHED_RR;
Christian Eggers89f063b2020-11-15 20:25:09 +010099 int ret;
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000100
Denys Vlasenko66ad9be2018-01-23 16:01:13 +0100101 opt = getopt32(argv, "^"
102 "+" "mprfobi"
103 "\0"
104 /* only one policy accepted: */
105 "r--fobi:f--robi:o--rfbi:b--rfoi:i--rfob"
106 );
Denys Vlasenko1393fc12010-12-19 04:07:50 +0100107 if (opt & OPT_m) { /* print min/max and exit */
Denys Vlasenko66ad9be2018-01-23 16:01:13 +0100108 show_min_max(SCHED_OTHER);
Denys Vlasenko1393fc12010-12-19 04:07:50 +0100109 show_min_max(SCHED_FIFO);
110 show_min_max(SCHED_RR);
Povilas Kanapickas1616fb82018-01-13 23:26:39 +0200111 show_min_max(SCHED_BATCH);
Povilas Kanapickas0f6e4352018-01-13 23:27:37 +0200112 show_min_max(SCHED_IDLE);
Denys Vlasenko1393fc12010-12-19 04:07:50 +0100113 fflush_stdout_and_exit(EXIT_SUCCESS);
114 }
Denys Vlasenko66ad9be2018-01-23 16:01:13 +0100115 //if (opt & OPT_r)
116 // policy = SCHED_RR; - default, already set
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000117 if (opt & OPT_f)
118 policy = SCHED_FIFO;
119 if (opt & OPT_o)
120 policy = SCHED_OTHER;
Povilas Kanapickas1616fb82018-01-13 23:26:39 +0200121 if (opt & OPT_b)
122 policy = SCHED_BATCH;
Povilas Kanapickas0f6e4352018-01-13 23:27:37 +0200123 if (opt & OPT_i)
124 policy = SCHED_IDLE;
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000125
Denis Vlasenko42cc3042008-03-24 02:05:58 +0000126 argv += optind;
Denys Vlasenko1393fc12010-12-19 04:07:50 +0100127 if (!argv[0])
128 bb_show_usage();
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000129 if (opt & OPT_p) {
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000130 pid_str = *argv++;
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200131 if (*argv) { /* "-p PRIO PID [...]" */
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000132 priority = pid_str;
133 pid_str = *argv;
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000134 }
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200135 /* else "-p PID", and *argv == NULL */
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000136 pid = xatoul_range(pid_str, 1, ((unsigned)(pid_t)ULONG_MAX) >> 1);
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000137 } else {
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000138 priority = *argv++;
139 if (!*argv)
140 bb_show_usage();
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000141 }
142
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000143 current_new = "current\0new";
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000144 if (opt & OPT_p) {
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000145 int pol;
146 print_rt_info:
Christian Eggers89f063b2020-11-15 20:25:09 +0100147#if LIBC_IS_MUSL
148 /* musl libc returns ENOSYS for its sched_getscheduler library
149 * function, because the sched_getscheduler Linux kernel system call
150 * does not conform to Posix; so we use the system call directly
151 */
152 pol = syscall(SYS_sched_getscheduler, pid);
153#else
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000154 pol = sched_getscheduler(pid);
Christian Eggers89f063b2020-11-15 20:25:09 +0100155#endif
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000156 if (pol < 0)
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200157 bb_perror_msg_and_die("can't %cet pid %u's policy", 'g', (int)pid);
Denys Vlasenko43d09e72019-04-05 16:59:07 +0200158#ifdef SCHED_RESET_ON_FORK
159 /* "Since Linux 2.6.32, the SCHED_RESET_ON_FORK flag
160 * can be ORed in policy when calling sched_setscheduler().
161 * As a result of including this flag, children created by
162 * fork(2) do not inherit privileged scheduling policies"
163 *
164 * This bit is also returned by sched_getscheduler()!
165 * (TODO: do we want to show it?)
166 */
167 pol &= ~SCHED_RESET_ON_FORK;
168#endif
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200169 printf("pid %u's %s scheduling policy: SCHED_%s\n",
170 pid, current_new, policy_name(pol)
171 );
Christian Eggers89f063b2020-11-15 20:25:09 +0100172#if LIBC_IS_MUSL
173 ret = syscall(SYS_sched_getparam, pid, &sp);
174#else
175 ret = sched_getparam(pid, &sp);
176#endif
177 if (ret)
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200178 bb_perror_msg_and_die("can't get pid %u's attributes", (int)pid);
179 printf("pid %u's %s scheduling priority: %d\n",
180 (int)pid, current_new, sp.sched_priority
181 );
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000182 if (!*argv) {
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200183 /* Either it was just "-p PID",
184 * or it was "-p PRIO PID" and we came here
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000185 * for the second time (see goto below) */
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000186 return EXIT_SUCCESS;
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000187 }
188 *argv = NULL;
189 current_new += 8;
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000190 }
191
Povilas Kanapickas1616fb82018-01-13 23:26:39 +0200192 sp.sched_priority = xstrtou_range(priority, 0,
Denys Vlasenko66ad9be2018-01-23 16:01:13 +0100193 sched_get_priority_min(policy), sched_get_priority_max(policy)
194 );
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000195
Christian Eggers89f063b2020-11-15 20:25:09 +0100196#if LIBC_IS_MUSL
197 ret = syscall(SYS_sched_setscheduler, pid, policy, &sp);
198#else
199 ret = sched_setscheduler(pid, policy, &sp);
200#endif
201 if (ret)
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200202 bb_perror_msg_and_die("can't %cet pid %u's policy", 's', (int)pid);
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000203
Denys Vlasenkoae5ca6f2019-04-05 12:03:48 +0200204 if (!argv[0]) /* "-p PRIO PID [...]" */
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000205 goto print_rt_info;
Denis Vlasenkob44c7902008-03-17 09:29:43 +0000206
Pascal Bellard21e8e8d2010-07-04 00:57:03 +0200207 BB_EXECVP_or_die(argv);
Bernhard Reutner-Fischer71bc71a2007-03-09 16:56:38 +0000208}