diff options
Diffstat (limited to 'kernel/cgroup/debug.c')
| -rw-r--r-- | kernel/cgroup/debug.c | 357 |
1 files changed, 357 insertions, 0 deletions
diff --git a/kernel/cgroup/debug.c b/kernel/cgroup/debug.c new file mode 100644 index 000000000000..dac46af22782 --- /dev/null +++ b/kernel/cgroup/debug.c | |||
| @@ -0,0 +1,357 @@ | |||
| 1 | /* | ||
| 2 | * Debug controller | ||
| 3 | * | ||
| 4 | * WARNING: This controller is for cgroup core debugging only. | ||
| 5 | * Its interfaces are unstable and subject to changes at any time. | ||
| 6 | */ | ||
| 7 | #include <linux/ctype.h> | ||
| 8 | #include <linux/mm.h> | ||
| 9 | #include <linux/slab.h> | ||
| 10 | |||
| 11 | #include "cgroup-internal.h" | ||
| 12 | |||
| 13 | static struct cgroup_subsys_state * | ||
| 14 | debug_css_alloc(struct cgroup_subsys_state *parent_css) | ||
| 15 | { | ||
| 16 | struct cgroup_subsys_state *css = kzalloc(sizeof(*css), GFP_KERNEL); | ||
| 17 | |||
| 18 | if (!css) | ||
| 19 | return ERR_PTR(-ENOMEM); | ||
| 20 | |||
| 21 | return css; | ||
| 22 | } | ||
| 23 | |||
| 24 | static void debug_css_free(struct cgroup_subsys_state *css) | ||
| 25 | { | ||
| 26 | kfree(css); | ||
| 27 | } | ||
| 28 | |||
| 29 | /* | ||
| 30 | * debug_taskcount_read - return the number of tasks in a cgroup. | ||
| 31 | * @cgrp: the cgroup in question | ||
| 32 | */ | ||
| 33 | static u64 debug_taskcount_read(struct cgroup_subsys_state *css, | ||
| 34 | struct cftype *cft) | ||
| 35 | { | ||
| 36 | return cgroup_task_count(css->cgroup); | ||
| 37 | } | ||
| 38 | |||
| 39 | static int current_css_set_read(struct seq_file *seq, void *v) | ||
| 40 | { | ||
| 41 | struct kernfs_open_file *of = seq->private; | ||
| 42 | struct css_set *cset; | ||
| 43 | struct cgroup_subsys *ss; | ||
| 44 | struct cgroup_subsys_state *css; | ||
| 45 | int i, refcnt; | ||
| 46 | |||
| 47 | if (!cgroup_kn_lock_live(of->kn, false)) | ||
| 48 | return -ENODEV; | ||
| 49 | |||
| 50 | spin_lock_irq(&css_set_lock); | ||
| 51 | rcu_read_lock(); | ||
| 52 | cset = rcu_dereference(current->cgroups); | ||
| 53 | refcnt = refcount_read(&cset->refcount); | ||
| 54 | seq_printf(seq, "css_set %pK %d", cset, refcnt); | ||
| 55 | if (refcnt > cset->nr_tasks) | ||
| 56 | seq_printf(seq, " +%d", refcnt - cset->nr_tasks); | ||
| 57 | seq_puts(seq, "\n"); | ||
| 58 | |||
| 59 | /* | ||
| 60 | * Print the css'es stored in the current css_set. | ||
| 61 | */ | ||
| 62 | for_each_subsys(ss, i) { | ||
| 63 | css = cset->subsys[ss->id]; | ||
| 64 | if (!css) | ||
| 65 | continue; | ||
| 66 | seq_printf(seq, "%2d: %-4s\t- %lx[%d]\n", ss->id, ss->name, | ||
| 67 | (unsigned long)css, css->id); | ||
| 68 | } | ||
| 69 | rcu_read_unlock(); | ||
| 70 | spin_unlock_irq(&css_set_lock); | ||
| 71 | cgroup_kn_unlock(of->kn); | ||
| 72 | return 0; | ||
| 73 | } | ||
| 74 | |||
| 75 | static u64 current_css_set_refcount_read(struct cgroup_subsys_state *css, | ||
| 76 | struct cftype *cft) | ||
| 77 | { | ||
| 78 | u64 count; | ||
| 79 | |||
| 80 | rcu_read_lock(); | ||
| 81 | count = refcount_read(&task_css_set(current)->refcount); | ||
| 82 | rcu_read_unlock(); | ||
| 83 | return count; | ||
| 84 | } | ||
| 85 | |||
| 86 | static int current_css_set_cg_links_read(struct seq_file *seq, void *v) | ||
| 87 | { | ||
| 88 | struct cgrp_cset_link *link; | ||
| 89 | struct css_set *cset; | ||
| 90 | char *name_buf; | ||
| 91 | |||
| 92 | name_buf = kmalloc(NAME_MAX + 1, GFP_KERNEL); | ||
| 93 | if (!name_buf) | ||
| 94 | return -ENOMEM; | ||
| 95 | |||
| 96 | spin_lock_irq(&css_set_lock); | ||
| 97 | rcu_read_lock(); | ||
| 98 | cset = rcu_dereference(current->cgroups); | ||
| 99 | list_for_each_entry(link, &cset->cgrp_links, cgrp_link) { | ||
| 100 | struct cgroup *c = link->cgrp; | ||
| 101 | |||
| 102 | cgroup_name(c, name_buf, NAME_MAX + 1); | ||
| 103 | seq_printf(seq, "Root %d group %s\n", | ||
| 104 | c->root->hierarchy_id, name_buf); | ||
| 105 | } | ||
| 106 | rcu_read_unlock(); | ||
| 107 | spin_unlock_irq(&css_set_lock); | ||
| 108 | kfree(name_buf); | ||
| 109 | return 0; | ||
| 110 | } | ||
| 111 | |||
| 112 | #define MAX_TASKS_SHOWN_PER_CSS 25 | ||
| 113 | static int cgroup_css_links_read(struct seq_file *seq, void *v) | ||
| 114 | { | ||
| 115 | struct cgroup_subsys_state *css = seq_css(seq); | ||
| 116 | struct cgrp_cset_link *link; | ||
| 117 | int dead_cnt = 0, extra_refs = 0; | ||
| 118 | |||
| 119 | spin_lock_irq(&css_set_lock); | ||
| 120 | list_for_each_entry(link, &css->cgroup->cset_links, cset_link) { | ||
| 121 | struct css_set *cset = link->cset; | ||
| 122 | struct task_struct *task; | ||
| 123 | int count = 0; | ||
| 124 | int refcnt = refcount_read(&cset->refcount); | ||
| 125 | |||
| 126 | seq_printf(seq, " %d", refcnt); | ||
| 127 | if (refcnt - cset->nr_tasks > 0) { | ||
| 128 | int extra = refcnt - cset->nr_tasks; | ||
| 129 | |||
| 130 | seq_printf(seq, " +%d", extra); | ||
| 131 | /* | ||
| 132 | * Take out the one additional reference in | ||
| 133 | * init_css_set. | ||
| 134 | */ | ||
| 135 | if (cset == &init_css_set) | ||
| 136 | extra--; | ||
| 137 | extra_refs += extra; | ||
| 138 | } | ||
| 139 | seq_puts(seq, "\n"); | ||
| 140 | |||
| 141 | list_for_each_entry(task, &cset->tasks, cg_list) { | ||
| 142 | if (count++ <= MAX_TASKS_SHOWN_PER_CSS) | ||
| 143 | seq_printf(seq, " task %d\n", | ||
| 144 | task_pid_vnr(task)); | ||
| 145 | } | ||
| 146 | |||
| 147 | list_for_each_entry(task, &cset->mg_tasks, cg_list) { | ||
| 148 | if (count++ <= MAX_TASKS_SHOWN_PER_CSS) | ||
| 149 | seq_printf(seq, " task %d\n", | ||
| 150 | task_pid_vnr(task)); | ||
| 151 | } | ||
| 152 | /* show # of overflowed tasks */ | ||
| 153 | if (count > MAX_TASKS_SHOWN_PER_CSS) | ||
| 154 | seq_printf(seq, " ... (%d)\n", | ||
| 155 | count - MAX_TASKS_SHOWN_PER_CSS); | ||
| 156 | |||
| 157 | if (cset->dead) { | ||
| 158 | seq_puts(seq, " [dead]\n"); | ||
| 159 | dead_cnt++; | ||
| 160 | } | ||
| 161 | |||
| 162 | WARN_ON(count != cset->nr_tasks); | ||
| 163 | } | ||
| 164 | spin_unlock_irq(&css_set_lock); | ||
| 165 | |||
| 166 | if (!dead_cnt && !extra_refs) | ||
| 167 | return 0; | ||
| 168 | |||
| 169 | seq_puts(seq, "\n"); | ||
| 170 | if (extra_refs) | ||
| 171 | seq_printf(seq, "extra references = %d\n", extra_refs); | ||
| 172 | if (dead_cnt) | ||
| 173 | seq_printf(seq, "dead css_sets = %d\n", dead_cnt); | ||
| 174 | |||
| 175 | return 0; | ||
| 176 | } | ||
| 177 | |||
| 178 | static int cgroup_subsys_states_read(struct seq_file *seq, void *v) | ||
| 179 | { | ||
| 180 | struct kernfs_open_file *of = seq->private; | ||
| 181 | struct cgroup *cgrp; | ||
| 182 | struct cgroup_subsys *ss; | ||
| 183 | struct cgroup_subsys_state *css; | ||
| 184 | char pbuf[16]; | ||
| 185 | int i; | ||
| 186 | |||
| 187 | cgrp = cgroup_kn_lock_live(of->kn, false); | ||
| 188 | if (!cgrp) | ||
| 189 | return -ENODEV; | ||
| 190 | |||
| 191 | for_each_subsys(ss, i) { | ||
| 192 | css = rcu_dereference_check(cgrp->subsys[ss->id], true); | ||
| 193 | if (!css) | ||
| 194 | continue; | ||
| 195 | |||
| 196 | pbuf[0] = '\0'; | ||
| 197 | |||
| 198 | /* Show the parent CSS if applicable*/ | ||
| 199 | if (css->parent) | ||
| 200 | snprintf(pbuf, sizeof(pbuf) - 1, " P=%d", | ||
| 201 | css->parent->id); | ||
| 202 | seq_printf(seq, "%2d: %-4s\t- %lx[%d] %d%s\n", ss->id, ss->name, | ||
| 203 | (unsigned long)css, css->id, | ||
| 204 | atomic_read(&css->online_cnt), pbuf); | ||
| 205 | } | ||
| 206 | |||
| 207 | cgroup_kn_unlock(of->kn); | ||
| 208 | return 0; | ||
| 209 | } | ||
| 210 | |||
| 211 | static void cgroup_masks_read_one(struct seq_file *seq, const char *name, | ||
| 212 | u16 mask) | ||
| 213 | { | ||
| 214 | struct cgroup_subsys *ss; | ||
| 215 | int ssid; | ||
| 216 | bool first = true; | ||
| 217 | |||
| 218 | seq_printf(seq, "%-17s: ", name); | ||
| 219 | for_each_subsys(ss, ssid) { | ||
| 220 | if (!(mask & (1 << ssid))) | ||
| 221 | continue; | ||
| 222 | if (!first) | ||
| 223 | seq_puts(seq, ", "); | ||
| 224 | seq_puts(seq, ss->name); | ||
| 225 | first = false; | ||
| 226 | } | ||
| 227 | seq_putc(seq, '\n'); | ||
| 228 | } | ||
| 229 | |||
| 230 | static int cgroup_masks_read(struct seq_file *seq, void *v) | ||
| 231 | { | ||
| 232 | struct kernfs_open_file *of = seq->private; | ||
| 233 | struct cgroup *cgrp; | ||
| 234 | |||
| 235 | cgrp = cgroup_kn_lock_live(of->kn, false); | ||
| 236 | if (!cgrp) | ||
| 237 | return -ENODEV; | ||
| 238 | |||
| 239 | cgroup_masks_read_one(seq, "subtree_control", cgrp->subtree_control); | ||
| 240 | cgroup_masks_read_one(seq, "subtree_ss_mask", cgrp->subtree_ss_mask); | ||
| 241 | |||
| 242 | cgroup_kn_unlock(of->kn); | ||
| 243 | return 0; | ||
| 244 | } | ||
| 245 | |||
| 246 | static u64 releasable_read(struct cgroup_subsys_state *css, struct cftype *cft) | ||
| 247 | { | ||
| 248 | return (!cgroup_is_populated(css->cgroup) && | ||
| 249 | !css_has_online_children(&css->cgroup->self)); | ||
| 250 | } | ||
| 251 | |||
| 252 | static struct cftype debug_legacy_files[] = { | ||
| 253 | { | ||
| 254 | .name = "taskcount", | ||
| 255 | .read_u64 = debug_taskcount_read, | ||
| 256 | }, | ||
| 257 | |||
| 258 | { | ||
| 259 | .name = "current_css_set", | ||
| 260 | .seq_show = current_css_set_read, | ||
| 261 | .flags = CFTYPE_ONLY_ON_ROOT, | ||
| 262 | }, | ||
| 263 | |||
| 264 | { | ||
| 265 | .name = "current_css_set_refcount", | ||
| 266 | .read_u64 = current_css_set_refcount_read, | ||
| 267 | .flags = CFTYPE_ONLY_ON_ROOT, | ||
| 268 | }, | ||
| 269 | |||
| 270 | { | ||
| 271 | .name = "current_css_set_cg_links", | ||
| 272 | .seq_show = current_css_set_cg_links_read, | ||
| 273 | .flags = CFTYPE_ONLY_ON_ROOT, | ||
| 274 | }, | ||
| 275 | |||
| 276 | { | ||
| 277 | .name = "cgroup_css_links", | ||
| 278 | .seq_show = cgroup_css_links_read, | ||
| 279 | }, | ||
| 280 | |||
| 281 | { | ||
| 282 | .name = "cgroup_subsys_states", | ||
| 283 | .seq_show = cgroup_subsys_states_read, | ||
| 284 | }, | ||
| 285 | |||
| 286 | { | ||
| 287 | .name = "cgroup_masks", | ||
| 288 | .seq_show = cgroup_masks_read, | ||
| 289 | }, | ||
| 290 | |||
| 291 | { | ||
| 292 | .name = "releasable", | ||
| 293 | .read_u64 = releasable_read, | ||
| 294 | }, | ||
| 295 | |||
| 296 | { } /* terminate */ | ||
| 297 | }; | ||
| 298 | |||
| 299 | static struct cftype debug_files[] = { | ||
| 300 | { | ||
| 301 | .name = "taskcount", | ||
| 302 | .read_u64 = debug_taskcount_read, | ||
| 303 | }, | ||
| 304 | |||
| 305 | { | ||
| 306 | .name = "current_css_set", | ||
| 307 | .seq_show = current_css_set_read, | ||
| 308 | .flags = CFTYPE_ONLY_ON_ROOT, | ||
| 309 | }, | ||
| 310 | |||
| 311 | { | ||
| 312 | .name = "current_css_set_refcount", | ||
| 313 | .read_u64 = current_css_set_refcount_read, | ||
| 314 | .flags = CFTYPE_ONLY_ON_ROOT, | ||
| 315 | }, | ||
| 316 | |||
| 317 | { | ||
| 318 | .name = "current_css_set_cg_links", | ||
| 319 | .seq_show = current_css_set_cg_links_read, | ||
| 320 | .flags = CFTYPE_ONLY_ON_ROOT, | ||
| 321 | }, | ||
| 322 | |||
| 323 | { | ||
| 324 | .name = "css_links", | ||
| 325 | .seq_show = cgroup_css_links_read, | ||
| 326 | }, | ||
| 327 | |||
| 328 | { | ||
| 329 | .name = "csses", | ||
| 330 | .seq_show = cgroup_subsys_states_read, | ||
| 331 | }, | ||
| 332 | |||
| 333 | { | ||
| 334 | .name = "masks", | ||
| 335 | .seq_show = cgroup_masks_read, | ||
| 336 | }, | ||
| 337 | |||
| 338 | { } /* terminate */ | ||
| 339 | }; | ||
| 340 | |||
| 341 | struct cgroup_subsys debug_cgrp_subsys = { | ||
| 342 | .css_alloc = debug_css_alloc, | ||
| 343 | .css_free = debug_css_free, | ||
| 344 | .legacy_cftypes = debug_legacy_files, | ||
| 345 | }; | ||
| 346 | |||
| 347 | /* | ||
| 348 | * On v2, debug is an implicit controller enabled by "cgroup_debug" boot | ||
| 349 | * parameter. | ||
| 350 | */ | ||
| 351 | static int __init enable_cgroup_debug(char *str) | ||
| 352 | { | ||
| 353 | debug_cgrp_subsys.dfl_cftypes = debug_files; | ||
| 354 | debug_cgrp_subsys.implicit_on_dfl = true; | ||
| 355 | return 1; | ||
| 356 | } | ||
| 357 | __setup("cgroup_debug", enable_cgroup_debug); | ||
