]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/basic/cgroup-util.h
cgtop: hook up new /proc based emulation code for root cgroup memory/cpu stats
[thirdparty/systemd.git] / src / basic / cgroup-util.h
CommitLineData
53e1b683 1/* SPDX-License-Identifier: LGPL-2.1+ */
c2f1db8f 2#pragma once
8c6db833
LP
3
4/***
5 This file is part of systemd.
6
7 Copyright 2010 Lennart Poettering
8
9 systemd is free software; you can redistribute it and/or modify it
5430f7f2
LP
10 under the terms of the GNU Lesser General Public License as published by
11 the Free Software Foundation; either version 2.1 of the License, or
8c6db833
LP
12 (at your option) any later version.
13
14 systemd is distributed in the hope that it will be useful, but
15 WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
5430f7f2 17 Lesser General Public License for more details.
8c6db833 18
5430f7f2 19 You should have received a copy of the GNU Lesser General Public License
8c6db833
LP
20 along with systemd; If not, see <http://www.gnu.org/licenses/>.
21***/
22
35d2e7ec 23#include <dirent.h>
11c3a366
TA
24#include <stdbool.h>
25#include <stdint.h>
71d35b6b 26#include <stdio.h>
f0bef277 27#include <sys/statfs.h>
71d35b6b 28#include <sys/types.h>
c6c18be3 29
f6a6225e 30#include "def.h"
11c3a366
TA
31#include "hashmap.h"
32#include "macro.h"
93cc7779 33#include "set.h"
8c6db833 34
68ac0d05
LP
35#define SYSTEMD_CGROUP_CONTROLLER_LEGACY "name=systemd"
36#define SYSTEMD_CGROUP_CONTROLLER_HYBRID "name=unified"
37#define SYSTEMD_CGROUP_CONTROLLER "_systemd"
38
efdb0237
LP
39/* An enum of well known cgroup controllers */
40typedef enum CGroupController {
41 CGROUP_CONTROLLER_CPU,
00b4a247
LP
42 CGROUP_CONTROLLER_CPUACCT, /* v1 only */
43 CGROUP_CONTROLLER_IO, /* v2 only */
44 CGROUP_CONTROLLER_BLKIO, /* v1 only */
efdb0237 45 CGROUP_CONTROLLER_MEMORY,
00b4a247 46 CGROUP_CONTROLLER_DEVICES, /* v1 only */
03a7b521 47 CGROUP_CONTROLLER_PIDS,
efdb0237
LP
48 _CGROUP_CONTROLLER_MAX,
49 _CGROUP_CONTROLLER_INVALID = -1,
50} CGroupController;
51
52#define CGROUP_CONTROLLER_TO_MASK(c) (1 << (c))
53
4ad49000 54/* A bit mask of well known cgroup controllers */
efdb0237
LP
55typedef enum CGroupMask {
56 CGROUP_MASK_CPU = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_CPU),
57 CGROUP_MASK_CPUACCT = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_CPUACCT),
13c31542 58 CGROUP_MASK_IO = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_IO),
efdb0237
LP
59 CGROUP_MASK_BLKIO = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_BLKIO),
60 CGROUP_MASK_MEMORY = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_MEMORY),
3905f127 61 CGROUP_MASK_DEVICES = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_DEVICES),
03a7b521 62 CGROUP_MASK_PIDS = CGROUP_CONTROLLER_TO_MASK(CGROUP_CONTROLLER_PIDS),
efdb0237
LP
63 _CGROUP_MASK_ALL = CGROUP_CONTROLLER_TO_MASK(_CGROUP_CONTROLLER_MAX) - 1
64} CGroupMask;
4ad49000 65
13c31542
TH
66/* Special values for all weight knobs on unified hierarchy */
67#define CGROUP_WEIGHT_INVALID ((uint64_t) -1)
68#define CGROUP_WEIGHT_MIN UINT64_C(1)
69#define CGROUP_WEIGHT_MAX UINT64_C(10000)
70#define CGROUP_WEIGHT_DEFAULT UINT64_C(100)
71
72#define CGROUP_LIMIT_MIN UINT64_C(0)
73#define CGROUP_LIMIT_MAX ((uint64_t) -1)
74
75static inline bool CGROUP_WEIGHT_IS_OK(uint64_t x) {
76 return
77 x == CGROUP_WEIGHT_INVALID ||
78 (x >= CGROUP_WEIGHT_MIN && x <= CGROUP_WEIGHT_MAX);
79}
80
9be57249
TH
81/* IO limits on unified hierarchy */
82typedef enum CGroupIOLimitType {
83 CGROUP_IO_RBPS_MAX,
84 CGROUP_IO_WBPS_MAX,
ac06a0cf
TH
85 CGROUP_IO_RIOPS_MAX,
86 CGROUP_IO_WIOPS_MAX,
9be57249
TH
87
88 _CGROUP_IO_LIMIT_TYPE_MAX,
89 _CGROUP_IO_LIMIT_TYPE_INVALID = -1
90} CGroupIOLimitType;
91
92extern const uint64_t cgroup_io_limit_defaults[_CGROUP_IO_LIMIT_TYPE_MAX];
93
94const char* cgroup_io_limit_type_to_string(CGroupIOLimitType t) _const_;
95CGroupIOLimitType cgroup_io_limit_type_from_string(const char *s) _pure_;
96
d53d9474
LP
97/* Special values for the cpu.shares attribute */
98#define CGROUP_CPU_SHARES_INVALID ((uint64_t) -1)
99#define CGROUP_CPU_SHARES_MIN UINT64_C(2)
100#define CGROUP_CPU_SHARES_MAX UINT64_C(262144)
101#define CGROUP_CPU_SHARES_DEFAULT UINT64_C(1024)
102
103static inline bool CGROUP_CPU_SHARES_IS_OK(uint64_t x) {
104 return
105 x == CGROUP_CPU_SHARES_INVALID ||
106 (x >= CGROUP_CPU_SHARES_MIN && x <= CGROUP_CPU_SHARES_MAX);
107}
108
109/* Special values for the blkio.weight attribute */
110#define CGROUP_BLKIO_WEIGHT_INVALID ((uint64_t) -1)
111#define CGROUP_BLKIO_WEIGHT_MIN UINT64_C(10)
112#define CGROUP_BLKIO_WEIGHT_MAX UINT64_C(1000)
113#define CGROUP_BLKIO_WEIGHT_DEFAULT UINT64_C(500)
114
115static inline bool CGROUP_BLKIO_WEIGHT_IS_OK(uint64_t x) {
116 return
117 x == CGROUP_BLKIO_WEIGHT_INVALID ||
118 (x >= CGROUP_BLKIO_WEIGHT_MIN && x <= CGROUP_BLKIO_WEIGHT_MAX);
119}
120
f5058264
TH
121/* Default resource limits */
122#define DEFAULT_TASKS_MAX_PERCENTAGE 15U /* 15% of PIDs, 4915 on default settings */
123#define DEFAULT_USER_TASKS_MAX_PERCENTAGE 33U /* 33% of PIDs, 10813 on default settings */
124
5da38d07
TH
125typedef enum CGroupUnified {
126 CGROUP_UNIFIED_UNKNOWN = -1,
127 CGROUP_UNIFIED_NONE = 0, /* Both systemd and controllers on legacy */
128 CGROUP_UNIFIED_SYSTEMD = 1, /* Only systemd on unified */
129 CGROUP_UNIFIED_ALL = 2, /* Both systemd and controllers on unified */
130} CGroupUnified;
131
5954c074
LP
132/*
133 * General rules:
134 *
135 * We accept named hierarchies in the syntax "foo" and "name=foo".
136 *
137 * We expect that named hierarchies do not conflict in name with a
138 * kernel hierarchy, modulo the "name=" prefix.
139 *
140 * We always generate "normalized" controller names, i.e. without the
141 * "name=" prefix.
142 *
143 * We require absolute cgroup paths. When returning, we will always
144 * generate paths with multiple adjacent / removed.
145 */
146
c6c18be3 147int cg_enumerate_processes(const char *controller, const char *path, FILE **_f);
c6c18be3 148int cg_read_pid(FILE *f, pid_t *_pid);
ab2c3861
TH
149int cg_read_event(const char *controller, const char *path, const char *event,
150 char **val);
c6c18be3 151
35d2e7ec
LP
152int cg_enumerate_subgroups(const char *controller, const char *path, DIR **_d);
153int cg_read_subgroup(DIR *d, char **fn);
154
1d98fef1
LP
155typedef enum CGroupFlags {
156 CGROUP_SIGCONT = 1,
157 CGROUP_IGNORE_SELF = 2,
158 CGROUP_REMOVE = 4,
159} CGroupFlags;
8c6db833 160
1d98fef1
LP
161typedef void (*cg_kill_log_func_t)(pid_t pid, int sig, void *userdata);
162
163int cg_kill(const char *controller, const char *path, int sig, CGroupFlags flags, Set *s, cg_kill_log_func_t kill_log, void *userdata);
164int cg_kill_recursive(const char *controller, const char *path, int sig, CGroupFlags flags, Set *s, cg_kill_log_func_t kill_log, void *userdata);
165
166int cg_migrate(const char *cfrom, const char *pfrom, const char *cto, const char *pto, CGroupFlags flags);
167int cg_migrate_recursive(const char *cfrom, const char *pfrom, const char *cto, const char *pto, CGroupFlags flags);
168int cg_migrate_recursive_fallback(const char *cfrom, const char *pfrom, const char *cto, const char *pto, CGroupFlags flags);
35d2e7ec
LP
169
170int cg_split_spec(const char *spec, char **controller, char **path);
7027ff61 171int cg_mangle_path(const char *path, char **result);
8c6db833
LP
172
173int cg_get_path(const char *controller, const char *path, const char *suffix, char **fs);
3474ae3c 174int cg_get_path_and_check(const char *controller, const char *path, const char *suffix, char **fs);
7027ff61
LP
175
176int cg_pid_get_path(const char *controller, pid_t pid, char **path);
8c6db833
LP
177
178int cg_trim(const char *controller, const char *path, bool delete_root);
35d2e7ec 179
4ad49000 180int cg_rmdir(const char *controller, const char *path);
8c6db833 181
4ad49000 182int cg_create(const char *controller, const char *path);
8c6db833 183int cg_attach(const char *controller, const char *path, pid_t pid);
13b84ec7 184int cg_attach_fallback(const char *controller, const char *path, pid_t pid);
8c6db833
LP
185int cg_create_and_attach(const char *controller, const char *path, pid_t pid);
186
4ad49000 187int cg_set_attribute(const char *controller, const char *path, const char *attribute, const char *value);
934277fe 188int cg_get_attribute(const char *controller, const char *path, const char *attribute, char **ret);
66ebf6c0 189int cg_get_keyed_attribute(const char *controller, const char *path, const char *attribute, const char **keys, char **values);
4ad49000 190
62b9bb26 191int cg_set_access(const char *controller, const char *path, uid_t uid, gid_t gid);
8c6db833 192
4b58153d
LP
193int cg_set_xattr(const char *controller, const char *path, const char *name, const void *value, size_t size, int flags);
194int cg_get_xattr(const char *controller, const char *path, const char *name, void *value, size_t size);
195
8c6db833 196int cg_install_release_agent(const char *controller, const char *agent);
ad929bcc 197int cg_uninstall_release_agent(const char *controller);
8c6db833 198
6f883237
LP
199int cg_is_empty(const char *controller, const char *path);
200int cg_is_empty_recursive(const char *controller, const char *path);
8c6db833 201
7027ff61 202int cg_get_root_path(char **path);
6c03089c 203
7027ff61 204int cg_path_get_session(const char *path, char **session);
ae018d9b 205int cg_path_get_owner_uid(const char *path, uid_t *uid);
6c03089c
LP
206int cg_path_get_unit(const char *path, char **unit);
207int cg_path_get_user_unit(const char *path, char **unit);
7027ff61 208int cg_path_get_machine_name(const char *path, char **machine);
1021b21b 209int cg_path_get_slice(const char *path, char **slice);
329ac4bc 210int cg_path_get_user_slice(const char *path, char **slice);
7027ff61 211
751bc6ac 212int cg_shift_path(const char *cgroup, const char *cached_root, const char **shifted);
e9174f29 213int cg_pid_get_path_shifted(pid_t pid, const char *cached_root, char **cgroup);
6c03089c 214
7027ff61 215int cg_pid_get_session(pid_t pid, char **session);
ae018d9b 216int cg_pid_get_owner_uid(pid_t pid, uid_t *uid);
ba1261bc 217int cg_pid_get_unit(pid_t pid, char **unit);
ef1673d1 218int cg_pid_get_user_unit(pid_t pid, char **unit);
7027ff61 219int cg_pid_get_machine_name(pid_t pid, char **machine);
1021b21b 220int cg_pid_get_slice(pid_t pid, char **slice);
329ac4bc 221int cg_pid_get_user_slice(pid_t pid, char **slice);
1f73f0f1 222
7027ff61 223int cg_path_decode_unit(const char *cgroup, char **unit);
96cde13a 224
ae018d9b 225char *cg_escape(const char *p);
44a6b1b6 226char *cg_unescape(const char *p) _pure_;
78edb35a 227
185a0874 228bool cg_controller_is_valid(const char *p);
a016b922
LP
229
230int cg_slice_to_path(const char *unit, char **ret);
4ad49000 231
efdb0237 232typedef const char* (*cg_migrate_callback_t)(CGroupMask mask, void *userdata);
03b90d4b 233
efdb0237
LP
234int cg_create_everywhere(CGroupMask supported, CGroupMask mask, const char *path);
235int cg_attach_everywhere(CGroupMask supported, const char *path, pid_t pid, cg_migrate_callback_t callback, void *userdata);
236int cg_attach_many_everywhere(CGroupMask supported, const char *path, Set* pids, cg_migrate_callback_t callback, void *userdata);
237int cg_migrate_everywhere(CGroupMask supported, const char *from, const char *to, cg_migrate_callback_t callback, void *userdata);
238int cg_trim_everywhere(CGroupMask supported, const char *path, bool delete_root);
239int cg_enable_everywhere(CGroupMask supported, CGroupMask mask, const char *p);
4ad49000 240
efdb0237 241int cg_mask_supported(CGroupMask *ret);
aae7e17f
FB
242int cg_mask_from_string(const char *s, CGroupMask *ret);
243int cg_mask_to_string(CGroupMask mask, char **ret);
b12afc8c 244
6925a0de 245int cg_kernel_controllers(Set **controllers);
efdb0237 246
3228995c
CB
247bool cg_ns_supported(void);
248
b4cccbc1
LP
249int cg_all_unified(void);
250int cg_hybrid_unified(void);
c22800e4 251int cg_unified_controller(const char *controller);
415fc41c 252int cg_unified_flush(void);
efdb0237
LP
253
254bool cg_is_unified_wanted(void);
255bool cg_is_legacy_wanted(void);
a4464b95 256bool cg_is_hybrid_wanted(void);
efdb0237
LP
257
258const char* cgroup_controller_to_string(CGroupController c) _const_;
259CGroupController cgroup_controller_from_string(const char *s) _pure_;
d53d9474 260
13c31542 261int cg_weight_parse(const char *s, uint64_t *ret);
d53d9474
LP
262int cg_cpu_shares_parse(const char *s, uint64_t *ret);
263int cg_blkio_weight_parse(const char *s, uint64_t *ret);
f0bef277
EV
264
265bool is_cgroup_fs(const struct statfs *s);
266bool fd_is_cgroup_fs(int fd);