1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2010 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
30 #include "alloc-util.h"
31 #include "bus-error.h"
33 #include "cgroup-show.h"
34 #include "cgroup-util.h"
37 #include "output-mode.h"
39 #include "path-util.h"
40 #include "unit-name.h"
43 static bool arg_no_pager
= false;
44 static bool arg_kernel_threads
= false;
45 static bool arg_all
= false;
46 static int arg_full
= -1;
47 static char* arg_machine
= NULL
;
49 static void help(void) {
50 printf("%s [OPTIONS...] [CGROUP...]\n\n"
51 "Recursively show control group contents.\n\n"
52 " -h --help Show this help\n"
53 " --version Show package version\n"
54 " --no-pager Do not pipe output into a pager\n"
55 " -a --all Show all groups, including empty\n"
56 " -l --full Do not ellipsize output\n"
57 " -k Include kernel threads in output\n"
58 " -M --machine= Show container\n"
59 , program_invocation_short_name
);
62 static int parse_argv(int argc
, char *argv
[]) {
69 static const struct option options
[] = {
70 { "help", no_argument
, NULL
, 'h' },
71 { "version", no_argument
, NULL
, ARG_VERSION
},
72 { "no-pager", no_argument
, NULL
, ARG_NO_PAGER
},
73 { "all", no_argument
, NULL
, 'a' },
74 { "full", no_argument
, NULL
, 'l' },
75 { "machine", required_argument
, NULL
, 'M' },
84 while ((c
= getopt_long(argc
, argv
, "hkalM:", options
, NULL
)) >= 0)
108 arg_kernel_threads
= true;
112 arg_machine
= optarg
;
119 assert_not_reached("Unhandled option");
125 static int get_cgroup_root(char **ret
) {
126 _cleanup_bus_error_free_ sd_bus_error error
= SD_BUS_ERROR_NULL
;
127 _cleanup_bus_flush_close_unref_ sd_bus
*bus
= NULL
;
128 _cleanup_free_
char *unit
= NULL
, *path
= NULL
;
133 r
= cg_get_root_path(ret
);
135 return log_error_errno(r
, "Failed to get root control group path: %m");
140 m
= strjoina("/run/systemd/machines/", arg_machine
);
141 r
= parse_env_file(m
, NEWLINE
, "SCOPE", &unit
, NULL
);
143 return log_error_errno(r
, "Failed to load machine data: %m");
145 path
= unit_dbus_path_from_name(unit
);
149 r
= bus_connect_transport_systemd(BUS_TRANSPORT_LOCAL
, NULL
, false, &bus
);
151 return log_error_errno(r
, "Failed to create bus connection: %m");
153 r
= sd_bus_get_property_string(
155 "org.freedesktop.systemd1",
157 unit_dbus_interface_from_name(unit
),
162 return log_error_errno(r
, "Failed to query unit control group path: %s", bus_error_message(&error
, r
));
167 static void show_cg_info(const char *controller
, const char *path
) {
169 if (cg_unified() <= 0 && controller
&& !streq(controller
, SYSTEMD_CGROUP_CONTROLLER
))
170 printf("Controller %s; ", controller
);
172 printf("Control group %s:\n", isempty(path
) ? "/" : path
);
176 int main(int argc
, char *argv
[]) {
179 log_parse_environment();
182 r
= parse_argv(argc
, argv
);
187 r
= pager_open(false);
188 if (r
> 0 && arg_full
< 0)
193 arg_all
* OUTPUT_SHOW_ALL
|
194 (arg_full
> 0) * OUTPUT_FULL_WIDTH
;
197 _cleanup_free_
char *root
= NULL
;
200 r
= get_cgroup_root(&root
);
204 for (i
= optind
; i
< argc
; i
++) {
207 if (path_startswith(argv
[i
], "/sys/fs/cgroup")) {
209 printf("Directory %s:\n", argv
[i
]);
212 q
= show_cgroup_by_path(argv
[i
], NULL
, 0, arg_kernel_threads
, output_flags
);
214 _cleanup_free_
char *c
= NULL
, *p
= NULL
, *j
= NULL
;
215 const char *controller
, *path
;
217 r
= cg_split_spec(argv
[i
], &c
, &p
);
219 log_error_errno(r
, "Failed to split argument %s: %m", argv
[i
]);
223 controller
= c
?: SYSTEMD_CGROUP_CONTROLLER
;
225 j
= strjoin(root
, "/", p
, NULL
);
231 path_kill_slashes(j
);
236 show_cg_info(controller
, path
);
238 q
= show_cgroup(controller
, path
, NULL
, 0, arg_kernel_threads
, output_flags
);
249 _cleanup_free_
char *cwd
= NULL
;
251 cwd
= get_current_dir_name();
253 r
= log_error_errno(errno
, "Cannot determine current working directory: %m");
257 if (path_startswith(cwd
, "/sys/fs/cgroup")) {
258 printf("Working directory %s:\n", cwd
);
261 r
= show_cgroup_by_path(cwd
, NULL
, 0, arg_kernel_threads
, output_flags
);
267 _cleanup_free_
char *root
= NULL
;
269 r
= get_cgroup_root(&root
);
273 show_cg_info(SYSTEMD_CGROUP_CONTROLLER
, root
);
276 r
= show_cgroup(SYSTEMD_CGROUP_CONTROLLER
, root
, NULL
, 0, arg_kernel_threads
, output_flags
);
281 log_error_errno(r
, "Failed to list cgroup tree: %m");
286 return r
< 0 ? EXIT_FAILURE
: EXIT_SUCCESS
;