123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848 |
- /* SPDX-License-Identifier: GPL-2.0 */
- #include <stdbool.h>
- #include <linux/limits.h>
- #include <sys/ptrace.h>
- #include <sys/types.h>
- #include <sys/mman.h>
- #include <unistd.h>
- #include <stdio.h>
- #include <errno.h>
- #include <stdlib.h>
- #include <string.h>
- #include <sys/wait.h>
- #include "../kselftest.h"
- #include "cgroup_util.h"
- #define DEBUG
- #ifdef DEBUG
- #define debug(args...) fprintf(stderr, args)
- #else
- #define debug(args...)
- #endif
- /*
- * Check if the cgroup is frozen by looking at the cgroup.events::frozen value.
- */
- static int cg_check_frozen(const char *cgroup, bool frozen)
- {
- if (frozen) {
- if (cg_read_strstr(cgroup, "cgroup.events", "frozen 1") != 0) {
- debug("Cgroup %s isn't frozen\n", cgroup);
- return -1;
- }
- } else {
- /*
- * Check the cgroup.events::frozen value.
- */
- if (cg_read_strstr(cgroup, "cgroup.events", "frozen 0") != 0) {
- debug("Cgroup %s is frozen\n", cgroup);
- return -1;
- }
- }
- return 0;
- }
- /*
- * Freeze the given cgroup.
- */
- static int cg_freeze_nowait(const char *cgroup, bool freeze)
- {
- return cg_write(cgroup, "cgroup.freeze", freeze ? "1" : "0");
- }
- /*
- * Attach a task to the given cgroup and wait for a cgroup frozen event.
- * All transient events (e.g. populated) are ignored.
- */
- static int cg_enter_and_wait_for_frozen(const char *cgroup, int pid,
- bool frozen)
- {
- int fd, ret = -1;
- int attempts;
- fd = cg_prepare_for_wait(cgroup);
- if (fd < 0)
- return fd;
- ret = cg_enter(cgroup, pid);
- if (ret)
- goto out;
- for (attempts = 0; attempts < 10; attempts++) {
- ret = cg_wait_for(fd);
- if (ret)
- break;
- ret = cg_check_frozen(cgroup, frozen);
- if (ret)
- continue;
- }
- out:
- close(fd);
- return ret;
- }
- /*
- * Freeze the given cgroup and wait for the inotify signal.
- * If there are no events in 10 seconds, treat this as an error.
- * Then check that the cgroup is in the desired state.
- */
- static int cg_freeze_wait(const char *cgroup, bool freeze)
- {
- int fd, ret = -1;
- fd = cg_prepare_for_wait(cgroup);
- if (fd < 0)
- return fd;
- ret = cg_freeze_nowait(cgroup, freeze);
- if (ret) {
- debug("Error: cg_freeze_nowait() failed\n");
- goto out;
- }
- ret = cg_wait_for(fd);
- if (ret)
- goto out;
- ret = cg_check_frozen(cgroup, freeze);
- out:
- close(fd);
- return ret;
- }
- /*
- * A simple process running in a sleep loop until being
- * re-parented.
- */
- static int child_fn(const char *cgroup, void *arg)
- {
- int ppid = getppid();
- while (getppid() == ppid)
- usleep(1000);
- return getppid() == ppid;
- }
- /*
- * A simple test for the cgroup freezer: populated the cgroup with 100
- * running processes and freeze it. Then unfreeze it. Then it kills all
- * processes and destroys the cgroup.
- */
- static int test_cgfreezer_simple(const char *root)
- {
- int ret = KSFT_FAIL;
- char *cgroup = NULL;
- int i;
- cgroup = cg_name(root, "cg_test_simple");
- if (!cgroup)
- goto cleanup;
- if (cg_create(cgroup))
- goto cleanup;
- for (i = 0; i < 100; i++)
- cg_run_nowait(cgroup, child_fn, NULL);
- if (cg_wait_for_proc_count(cgroup, 100))
- goto cleanup;
- if (cg_check_frozen(cgroup, false))
- goto cleanup;
- if (cg_freeze_wait(cgroup, true))
- goto cleanup;
- if (cg_freeze_wait(cgroup, false))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup)
- cg_destroy(cgroup);
- free(cgroup);
- return ret;
- }
- /*
- * The test creates the following hierarchy:
- * A
- * / / \ \
- * B E I K
- * /\ |
- * C D F
- * |
- * G
- * |
- * H
- *
- * with a process in C, H and 3 processes in K.
- * Then it tries to freeze and unfreeze the whole tree.
- */
- static int test_cgfreezer_tree(const char *root)
- {
- char *cgroup[10] = {0};
- int ret = KSFT_FAIL;
- int i;
- cgroup[0] = cg_name(root, "cg_test_tree_A");
- if (!cgroup[0])
- goto cleanup;
- cgroup[1] = cg_name(cgroup[0], "B");
- if (!cgroup[1])
- goto cleanup;
- cgroup[2] = cg_name(cgroup[1], "C");
- if (!cgroup[2])
- goto cleanup;
- cgroup[3] = cg_name(cgroup[1], "D");
- if (!cgroup[3])
- goto cleanup;
- cgroup[4] = cg_name(cgroup[0], "E");
- if (!cgroup[4])
- goto cleanup;
- cgroup[5] = cg_name(cgroup[4], "F");
- if (!cgroup[5])
- goto cleanup;
- cgroup[6] = cg_name(cgroup[5], "G");
- if (!cgroup[6])
- goto cleanup;
- cgroup[7] = cg_name(cgroup[6], "H");
- if (!cgroup[7])
- goto cleanup;
- cgroup[8] = cg_name(cgroup[0], "I");
- if (!cgroup[8])
- goto cleanup;
- cgroup[9] = cg_name(cgroup[0], "K");
- if (!cgroup[9])
- goto cleanup;
- for (i = 0; i < 10; i++)
- if (cg_create(cgroup[i]))
- goto cleanup;
- cg_run_nowait(cgroup[2], child_fn, NULL);
- cg_run_nowait(cgroup[7], child_fn, NULL);
- cg_run_nowait(cgroup[9], child_fn, NULL);
- cg_run_nowait(cgroup[9], child_fn, NULL);
- cg_run_nowait(cgroup[9], child_fn, NULL);
- /*
- * Wait until all child processes will enter
- * corresponding cgroups.
- */
- if (cg_wait_for_proc_count(cgroup[2], 1) ||
- cg_wait_for_proc_count(cgroup[7], 1) ||
- cg_wait_for_proc_count(cgroup[9], 3))
- goto cleanup;
- /*
- * Freeze B.
- */
- if (cg_freeze_wait(cgroup[1], true))
- goto cleanup;
- /*
- * Freeze F.
- */
- if (cg_freeze_wait(cgroup[5], true))
- goto cleanup;
- /*
- * Freeze G.
- */
- if (cg_freeze_wait(cgroup[6], true))
- goto cleanup;
- /*
- * Check that A and E are not frozen.
- */
- if (cg_check_frozen(cgroup[0], false))
- goto cleanup;
- if (cg_check_frozen(cgroup[4], false))
- goto cleanup;
- /*
- * Freeze A. Check that A, B and E are frozen.
- */
- if (cg_freeze_wait(cgroup[0], true))
- goto cleanup;
- if (cg_check_frozen(cgroup[1], true))
- goto cleanup;
- if (cg_check_frozen(cgroup[4], true))
- goto cleanup;
- /*
- * Unfreeze B, F and G
- */
- if (cg_freeze_nowait(cgroup[1], false))
- goto cleanup;
- if (cg_freeze_nowait(cgroup[5], false))
- goto cleanup;
- if (cg_freeze_nowait(cgroup[6], false))
- goto cleanup;
- /*
- * Check that C and H are still frozen.
- */
- if (cg_check_frozen(cgroup[2], true))
- goto cleanup;
- if (cg_check_frozen(cgroup[7], true))
- goto cleanup;
- /*
- * Unfreeze A. Check that A, C and K are not frozen.
- */
- if (cg_freeze_wait(cgroup[0], false))
- goto cleanup;
- if (cg_check_frozen(cgroup[2], false))
- goto cleanup;
- if (cg_check_frozen(cgroup[9], false))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- for (i = 9; i >= 0 && cgroup[i]; i--) {
- cg_destroy(cgroup[i]);
- free(cgroup[i]);
- }
- return ret;
- }
- /*
- * A fork bomb emulator.
- */
- static int forkbomb_fn(const char *cgroup, void *arg)
- {
- int ppid;
- fork();
- fork();
- ppid = getppid();
- while (getppid() == ppid)
- usleep(1000);
- return getppid() == ppid;
- }
- /*
- * The test runs a fork bomb in a cgroup and tries to freeze it.
- * Then it kills all processes and checks that cgroup isn't populated
- * anymore.
- */
- static int test_cgfreezer_forkbomb(const char *root)
- {
- int ret = KSFT_FAIL;
- char *cgroup = NULL;
- cgroup = cg_name(root, "cg_forkbomb_test");
- if (!cgroup)
- goto cleanup;
- if (cg_create(cgroup))
- goto cleanup;
- cg_run_nowait(cgroup, forkbomb_fn, NULL);
- usleep(100000);
- if (cg_freeze_wait(cgroup, true))
- goto cleanup;
- if (cg_killall(cgroup))
- goto cleanup;
- if (cg_wait_for_proc_count(cgroup, 0))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup)
- cg_destroy(cgroup);
- free(cgroup);
- return ret;
- }
- /*
- * The test creates a cgroups and freezes it. Then it creates a child cgroup
- * and populates it with a task. After that it checks that the child cgroup
- * is frozen and the parent cgroup remains frozen too.
- */
- static int test_cgfreezer_mkdir(const char *root)
- {
- int ret = KSFT_FAIL;
- char *parent, *child = NULL;
- int pid;
- parent = cg_name(root, "cg_test_mkdir_A");
- if (!parent)
- goto cleanup;
- child = cg_name(parent, "cg_test_mkdir_B");
- if (!child)
- goto cleanup;
- if (cg_create(parent))
- goto cleanup;
- if (cg_freeze_wait(parent, true))
- goto cleanup;
- if (cg_create(child))
- goto cleanup;
- pid = cg_run_nowait(child, child_fn, NULL);
- if (pid < 0)
- goto cleanup;
- if (cg_wait_for_proc_count(child, 1))
- goto cleanup;
- if (cg_check_frozen(child, true))
- goto cleanup;
- if (cg_check_frozen(parent, true))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (child)
- cg_destroy(child);
- free(child);
- if (parent)
- cg_destroy(parent);
- free(parent);
- return ret;
- }
- /*
- * The test creates two nested cgroups, freezes the parent
- * and removes the child. Then it checks that the parent cgroup
- * remains frozen and it's possible to create a new child
- * without unfreezing. The new child is frozen too.
- */
- static int test_cgfreezer_rmdir(const char *root)
- {
- int ret = KSFT_FAIL;
- char *parent, *child = NULL;
- parent = cg_name(root, "cg_test_rmdir_A");
- if (!parent)
- goto cleanup;
- child = cg_name(parent, "cg_test_rmdir_B");
- if (!child)
- goto cleanup;
- if (cg_create(parent))
- goto cleanup;
- if (cg_create(child))
- goto cleanup;
- if (cg_freeze_wait(parent, true))
- goto cleanup;
- if (cg_destroy(child))
- goto cleanup;
- if (cg_check_frozen(parent, true))
- goto cleanup;
- if (cg_create(child))
- goto cleanup;
- if (cg_check_frozen(child, true))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (child)
- cg_destroy(child);
- free(child);
- if (parent)
- cg_destroy(parent);
- free(parent);
- return ret;
- }
- /*
- * The test creates two cgroups: A and B, runs a process in A
- * and performs several migrations:
- * 1) A (running) -> B (frozen)
- * 2) B (frozen) -> A (running)
- * 3) A (frozen) -> B (frozen)
- *
- * On each step it checks the actual state of both cgroups.
- */
- static int test_cgfreezer_migrate(const char *root)
- {
- int ret = KSFT_FAIL;
- char *cgroup[2] = {0};
- int pid;
- cgroup[0] = cg_name(root, "cg_test_migrate_A");
- if (!cgroup[0])
- goto cleanup;
- cgroup[1] = cg_name(root, "cg_test_migrate_B");
- if (!cgroup[1])
- goto cleanup;
- if (cg_create(cgroup[0]))
- goto cleanup;
- if (cg_create(cgroup[1]))
- goto cleanup;
- pid = cg_run_nowait(cgroup[0], child_fn, NULL);
- if (pid < 0)
- goto cleanup;
- if (cg_wait_for_proc_count(cgroup[0], 1))
- goto cleanup;
- /*
- * Migrate from A (running) to B (frozen)
- */
- if (cg_freeze_wait(cgroup[1], true))
- goto cleanup;
- if (cg_enter_and_wait_for_frozen(cgroup[1], pid, true))
- goto cleanup;
- if (cg_check_frozen(cgroup[0], false))
- goto cleanup;
- /*
- * Migrate from B (frozen) to A (running)
- */
- if (cg_enter_and_wait_for_frozen(cgroup[0], pid, false))
- goto cleanup;
- if (cg_check_frozen(cgroup[1], true))
- goto cleanup;
- /*
- * Migrate from A (frozen) to B (frozen)
- */
- if (cg_freeze_wait(cgroup[0], true))
- goto cleanup;
- if (cg_enter_and_wait_for_frozen(cgroup[1], pid, true))
- goto cleanup;
- if (cg_check_frozen(cgroup[0], true))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup[0])
- cg_destroy(cgroup[0]);
- free(cgroup[0]);
- if (cgroup[1])
- cg_destroy(cgroup[1]);
- free(cgroup[1]);
- return ret;
- }
- /*
- * The test checks that ptrace works with a tracing process in a frozen cgroup.
- */
- static int test_cgfreezer_ptrace(const char *root)
- {
- int ret = KSFT_FAIL;
- char *cgroup = NULL;
- siginfo_t siginfo;
- int pid;
- cgroup = cg_name(root, "cg_test_ptrace");
- if (!cgroup)
- goto cleanup;
- if (cg_create(cgroup))
- goto cleanup;
- pid = cg_run_nowait(cgroup, child_fn, NULL);
- if (pid < 0)
- goto cleanup;
- if (cg_wait_for_proc_count(cgroup, 1))
- goto cleanup;
- if (cg_freeze_wait(cgroup, true))
- goto cleanup;
- if (ptrace(PTRACE_SEIZE, pid, NULL, NULL))
- goto cleanup;
- if (ptrace(PTRACE_INTERRUPT, pid, NULL, NULL))
- goto cleanup;
- waitpid(pid, NULL, 0);
- /*
- * Cgroup has to remain frozen, however the test task
- * is in traced state.
- */
- if (cg_check_frozen(cgroup, true))
- goto cleanup;
- if (ptrace(PTRACE_GETSIGINFO, pid, NULL, &siginfo))
- goto cleanup;
- if (ptrace(PTRACE_DETACH, pid, NULL, NULL))
- goto cleanup;
- if (cg_check_frozen(cgroup, true))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup)
- cg_destroy(cgroup);
- free(cgroup);
- return ret;
- }
- /*
- * Check if the process is stopped.
- */
- static int proc_check_stopped(int pid)
- {
- char buf[PAGE_SIZE];
- int len;
- len = proc_read_text(pid, 0, "stat", buf, sizeof(buf));
- if (len == -1) {
- debug("Can't get %d stat\n", pid);
- return -1;
- }
- if (strstr(buf, "(test_freezer) T ") == NULL) {
- debug("Process %d in the unexpected state: %s\n", pid, buf);
- return -1;
- }
- return 0;
- }
- /*
- * Test that it's possible to freeze a cgroup with a stopped process.
- */
- static int test_cgfreezer_stopped(const char *root)
- {
- int pid, ret = KSFT_FAIL;
- char *cgroup = NULL;
- cgroup = cg_name(root, "cg_test_stopped");
- if (!cgroup)
- goto cleanup;
- if (cg_create(cgroup))
- goto cleanup;
- pid = cg_run_nowait(cgroup, child_fn, NULL);
- if (cg_wait_for_proc_count(cgroup, 1))
- goto cleanup;
- if (kill(pid, SIGSTOP))
- goto cleanup;
- if (cg_check_frozen(cgroup, false))
- goto cleanup;
- if (cg_freeze_wait(cgroup, true))
- goto cleanup;
- if (cg_freeze_wait(cgroup, false))
- goto cleanup;
- if (proc_check_stopped(pid))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup)
- cg_destroy(cgroup);
- free(cgroup);
- return ret;
- }
- /*
- * Test that it's possible to freeze a cgroup with a ptraced process.
- */
- static int test_cgfreezer_ptraced(const char *root)
- {
- int pid, ret = KSFT_FAIL;
- char *cgroup = NULL;
- siginfo_t siginfo;
- cgroup = cg_name(root, "cg_test_ptraced");
- if (!cgroup)
- goto cleanup;
- if (cg_create(cgroup))
- goto cleanup;
- pid = cg_run_nowait(cgroup, child_fn, NULL);
- if (cg_wait_for_proc_count(cgroup, 1))
- goto cleanup;
- if (ptrace(PTRACE_SEIZE, pid, NULL, NULL))
- goto cleanup;
- if (ptrace(PTRACE_INTERRUPT, pid, NULL, NULL))
- goto cleanup;
- waitpid(pid, NULL, 0);
- if (cg_check_frozen(cgroup, false))
- goto cleanup;
- if (cg_freeze_wait(cgroup, true))
- goto cleanup;
- /*
- * cg_check_frozen(cgroup, true) will fail here,
- * because the task in in the TRACEd state.
- */
- if (cg_freeze_wait(cgroup, false))
- goto cleanup;
- if (ptrace(PTRACE_GETSIGINFO, pid, NULL, &siginfo))
- goto cleanup;
- if (ptrace(PTRACE_DETACH, pid, NULL, NULL))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup)
- cg_destroy(cgroup);
- free(cgroup);
- return ret;
- }
- static int vfork_fn(const char *cgroup, void *arg)
- {
- int pid = vfork();
- if (pid == 0)
- while (true)
- sleep(1);
- return pid;
- }
- /*
- * Test that it's possible to freeze a cgroup with a process,
- * which called vfork() and is waiting for a child.
- */
- static int test_cgfreezer_vfork(const char *root)
- {
- int ret = KSFT_FAIL;
- char *cgroup = NULL;
- cgroup = cg_name(root, "cg_test_vfork");
- if (!cgroup)
- goto cleanup;
- if (cg_create(cgroup))
- goto cleanup;
- cg_run_nowait(cgroup, vfork_fn, NULL);
- if (cg_wait_for_proc_count(cgroup, 2))
- goto cleanup;
- if (cg_freeze_wait(cgroup, true))
- goto cleanup;
- ret = KSFT_PASS;
- cleanup:
- if (cgroup)
- cg_destroy(cgroup);
- free(cgroup);
- return ret;
- }
- #define T(x) { x, #x }
- struct cgfreezer_test {
- int (*fn)(const char *root);
- const char *name;
- } tests[] = {
- T(test_cgfreezer_simple),
- T(test_cgfreezer_tree),
- T(test_cgfreezer_forkbomb),
- T(test_cgfreezer_mkdir),
- T(test_cgfreezer_rmdir),
- T(test_cgfreezer_migrate),
- T(test_cgfreezer_ptrace),
- T(test_cgfreezer_stopped),
- T(test_cgfreezer_ptraced),
- T(test_cgfreezer_vfork),
- };
- #undef T
- int main(int argc, char *argv[])
- {
- char root[PATH_MAX];
- int i, ret = EXIT_SUCCESS;
- if (cg_find_unified_root(root, sizeof(root)))
- ksft_exit_skip("cgroup v2 isn't mounted\n");
- for (i = 0; i < ARRAY_SIZE(tests); i++) {
- switch (tests[i].fn(root)) {
- case KSFT_PASS:
- ksft_test_result_pass("%s\n", tests[i].name);
- break;
- case KSFT_SKIP:
- ksft_test_result_skip("%s\n", tests[i].name);
- break;
- default:
- ret = EXIT_FAILURE;
- ksft_test_result_fail("%s\n", tests[i].name);
- break;
- }
- }
- return ret;
- }
|