mirror of
https://github.com/systemd/systemd.git
synced 2025-01-10 05:18:17 +03:00
cgtop: recursively count cgroup member tasks
When showing the number of tasks in a cgroup, recursively count tasks in child cgroups and include them in the number. This ensures that the number of tasks is cummulative the same way as memory, cpu and IO resources are. Old behaviour can be restored by passing the new --recursive=no switch.
This commit is contained in:
parent
41ba8b6e69
commit
3cb5beea0c
@ -176,6 +176,18 @@
|
|||||||
the count.</para></listitem>
|
the count.</para></listitem>
|
||||||
</varlistentry>
|
</varlistentry>
|
||||||
|
|
||||||
|
<varlistentry>
|
||||||
|
<term><option>--recursive=</option></term>
|
||||||
|
|
||||||
|
<listitem><para>Controls whether the number of tasks shown for
|
||||||
|
a control group shall include all tasks that are contained in
|
||||||
|
any of the child control groups as well. Takes a boolean
|
||||||
|
argument, defaults to <literal>yes</literal>. If enabled the
|
||||||
|
tasks in child control groups are included, if disabled only
|
||||||
|
the tasks in the control group itself are
|
||||||
|
counted.</para></listitem>
|
||||||
|
</varlistentry>
|
||||||
|
|
||||||
<varlistentry>
|
<varlistentry>
|
||||||
<term><option>-n</option></term>
|
<term><option>-n</option></term>
|
||||||
<term><option>--iterations=</option></term>
|
<term><option>--iterations=</option></term>
|
||||||
@ -212,7 +224,6 @@
|
|||||||
|
|
||||||
</refsect1>
|
</refsect1>
|
||||||
|
|
||||||
|
|
||||||
<refsect1>
|
<refsect1>
|
||||||
<title>Keys</title>
|
<title>Keys</title>
|
||||||
|
|
||||||
|
@ -66,6 +66,7 @@ static bool arg_batch = false;
|
|||||||
static bool arg_raw = false;
|
static bool arg_raw = false;
|
||||||
static usec_t arg_delay = 1*USEC_PER_SEC;
|
static usec_t arg_delay = 1*USEC_PER_SEC;
|
||||||
static bool arg_kernel_threads = false;
|
static bool arg_kernel_threads = false;
|
||||||
|
static bool arg_recursive = true;
|
||||||
|
|
||||||
static enum {
|
static enum {
|
||||||
ORDER_PATH,
|
ORDER_PATH,
|
||||||
@ -109,7 +110,14 @@ static const char *maybe_format_bytes(char *buf, size_t l, bool is_valid, off_t
|
|||||||
return format_bytes(buf, l, t);
|
return format_bytes(buf, l, t);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int process(const char *controller, const char *path, Hashmap *a, Hashmap *b, unsigned iteration) {
|
static int process(
|
||||||
|
const char *controller,
|
||||||
|
const char *path,
|
||||||
|
Hashmap *a,
|
||||||
|
Hashmap *b,
|
||||||
|
unsigned iteration,
|
||||||
|
Group **ret) {
|
||||||
|
|
||||||
Group *g;
|
Group *g;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
@ -303,6 +311,9 @@ static int process(const char *controller, const char *path, Hashmap *a, Hashmap
|
|||||||
g->io_iteration = iteration;
|
g->io_iteration = iteration;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ret)
|
||||||
|
*ret = g;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -312,9 +323,11 @@ static int refresh_one(
|
|||||||
Hashmap *a,
|
Hashmap *a,
|
||||||
Hashmap *b,
|
Hashmap *b,
|
||||||
unsigned iteration,
|
unsigned iteration,
|
||||||
unsigned depth) {
|
unsigned depth,
|
||||||
|
Group **ret) {
|
||||||
|
|
||||||
_cleanup_closedir_ DIR *d = NULL;
|
_cleanup_closedir_ DIR *d = NULL;
|
||||||
|
Group *ours;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
assert(controller);
|
assert(controller);
|
||||||
@ -324,7 +337,7 @@ static int refresh_one(
|
|||||||
if (depth > arg_depth)
|
if (depth > arg_depth)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
r = process(controller, path, a, b, iteration);
|
r = process(controller, path, a, b, iteration, &ours);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -336,10 +349,13 @@ static int refresh_one(
|
|||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
_cleanup_free_ char *fn = NULL, *p = NULL;
|
_cleanup_free_ char *fn = NULL, *p = NULL;
|
||||||
|
Group *child = NULL;
|
||||||
|
|
||||||
r = cg_read_subgroup(d, &fn);
|
r = cg_read_subgroup(d, &fn);
|
||||||
if (r <= 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
if (r == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
p = strjoin(path, "/", fn, NULL);
|
p = strjoin(path, "/", fn, NULL);
|
||||||
if (!p)
|
if (!p)
|
||||||
@ -347,12 +363,30 @@ static int refresh_one(
|
|||||||
|
|
||||||
path_kill_slashes(p);
|
path_kill_slashes(p);
|
||||||
|
|
||||||
r = refresh_one(controller, p, a, b, iteration, depth + 1);
|
r = refresh_one(controller, p, a, b, iteration, depth + 1, &child);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
|
if (arg_recursive &&
|
||||||
|
child &&
|
||||||
|
child->n_tasks_valid &&
|
||||||
|
streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {
|
||||||
|
|
||||||
|
/* Recursively sum up processes */
|
||||||
|
|
||||||
|
if (ours->n_tasks_valid)
|
||||||
|
ours->n_tasks += child->n_tasks;
|
||||||
|
else {
|
||||||
|
ours->n_tasks = child->n_tasks;
|
||||||
|
ours->n_tasks_valid = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return r;
|
if (ret)
|
||||||
|
*ret = ours;
|
||||||
|
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int refresh(const char *root, Hashmap *a, Hashmap *b, unsigned iteration) {
|
static int refresh(const char *root, Hashmap *a, Hashmap *b, unsigned iteration) {
|
||||||
@ -360,16 +394,16 @@ static int refresh(const char *root, Hashmap *a, Hashmap *b, unsigned iteration)
|
|||||||
|
|
||||||
assert(a);
|
assert(a);
|
||||||
|
|
||||||
r = refresh_one(SYSTEMD_CGROUP_CONTROLLER, root, a, b, iteration, 0);
|
r = refresh_one(SYSTEMD_CGROUP_CONTROLLER, root, a, b, iteration, 0, NULL);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
r = refresh_one("cpuacct", root, a, b, iteration, 0);
|
r = refresh_one("cpuacct", root, a, b, iteration, 0, NULL);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
r = refresh_one("memory", root, a, b, iteration, 0);
|
r = refresh_one("memory", root, a, b, iteration, 0, NULL);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
r = refresh_one("blkio", root, a, b, iteration, 0);
|
r = refresh_one("blkio", root, a, b, iteration, 0, NULL);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -379,11 +413,11 @@ static int refresh(const char *root, Hashmap *a, Hashmap *b, unsigned iteration)
|
|||||||
static int group_compare(const void*a, const void *b) {
|
static int group_compare(const void*a, const void *b) {
|
||||||
const Group *x = *(Group**)a, *y = *(Group**)b;
|
const Group *x = *(Group**)a, *y = *(Group**)b;
|
||||||
|
|
||||||
if (arg_order != ORDER_TASKS) {
|
if (arg_order != ORDER_TASKS || arg_recursive) {
|
||||||
/* Let's make sure that the parent is always before
|
/* Let's make sure that the parent is always before
|
||||||
* the child. Except when ordering by tasks, since
|
* the child. Except when ordering by tasks and
|
||||||
* that is actually not accumulative for all
|
* recursive summing is off, since that is actually
|
||||||
* children. */
|
* not accumulative for all children. */
|
||||||
|
|
||||||
if (path_startswith(y->path, x->path))
|
if (path_startswith(y->path, x->path))
|
||||||
return -1;
|
return -1;
|
||||||
@ -573,6 +607,7 @@ static void help(void) {
|
|||||||
" --cpu=percentage Show CPU usage as percentage (default)\n"
|
" --cpu=percentage Show CPU usage as percentage (default)\n"
|
||||||
" --cpu=time Show CPU usage as time\n"
|
" --cpu=time Show CPU usage as time\n"
|
||||||
" -k Include kernel threads in task count\n"
|
" -k Include kernel threads in task count\n"
|
||||||
|
" --recursive=BOOL Sum up task count recursively\n"
|
||||||
" -d --delay=DELAY Delay between updates\n"
|
" -d --delay=DELAY Delay between updates\n"
|
||||||
" -n --iterations=N Run for N iterations before exiting\n"
|
" -n --iterations=N Run for N iterations before exiting\n"
|
||||||
" -b --batch Run in batch mode, accepting no input\n"
|
" -b --batch Run in batch mode, accepting no input\n"
|
||||||
@ -587,23 +622,24 @@ static int parse_argv(int argc, char *argv[]) {
|
|||||||
ARG_DEPTH,
|
ARG_DEPTH,
|
||||||
ARG_CPU_TYPE,
|
ARG_CPU_TYPE,
|
||||||
ARG_ORDER,
|
ARG_ORDER,
|
||||||
|
ARG_RECURSIVE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct option options[] = {
|
static const struct option options[] = {
|
||||||
{ "help", no_argument, NULL, 'h' },
|
{ "help", no_argument, NULL, 'h' },
|
||||||
{ "version", no_argument, NULL, ARG_VERSION },
|
{ "version", no_argument, NULL, ARG_VERSION },
|
||||||
{ "delay", required_argument, NULL, 'd' },
|
{ "delay", required_argument, NULL, 'd' },
|
||||||
{ "iterations", required_argument, NULL, 'n' },
|
{ "iterations", required_argument, NULL, 'n' },
|
||||||
{ "batch", no_argument, NULL, 'b' },
|
{ "batch", no_argument, NULL, 'b' },
|
||||||
{ "raw", no_argument, NULL, 'r' },
|
{ "raw", no_argument, NULL, 'r' },
|
||||||
{ "depth", required_argument, NULL, ARG_DEPTH },
|
{ "depth", required_argument, NULL, ARG_DEPTH },
|
||||||
{ "cpu", optional_argument, NULL, ARG_CPU_TYPE },
|
{ "cpu", optional_argument, NULL, ARG_CPU_TYPE },
|
||||||
{ "order", required_argument, NULL, ARG_ORDER },
|
{ "order", required_argument, NULL, ARG_ORDER },
|
||||||
|
{ "recursive", required_argument, NULL, ARG_RECURSIVE },
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
||||||
int c;
|
int c, r;
|
||||||
int r;
|
|
||||||
|
|
||||||
assert(argc >= 1);
|
assert(argc >= 1);
|
||||||
assert(argv);
|
assert(argv);
|
||||||
@ -712,6 +748,16 @@ static int parse_argv(int argc, char *argv[]) {
|
|||||||
arg_kernel_threads = true;
|
arg_kernel_threads = true;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case ARG_RECURSIVE:
|
||||||
|
r = parse_boolean(optarg);
|
||||||
|
if (r < 0) {
|
||||||
|
log_error("Failed to parse --recursive= argument: %s", optarg);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
arg_recursive = r;
|
||||||
|
break;
|
||||||
|
|
||||||
case '?':
|
case '?':
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user