2
0
mirror of https://github.com/checkpoint-restore/criu synced 2025-08-31 14:25:49 +00:00
Files
criu/test/zdtm/static/cgroup04.c
Michał Mirosław 131e464a05 zdtm: cgroup04: Improve error messages.
Make the errno values reported by cgroup04 always correct and showing
relevant parameters.
Constify constant strings, while at it.

Signed-off-by: Michał Mirosław <emmir@google.com>
2023-10-22 13:29:25 -07:00

172 lines
3.7 KiB
C

#include <unistd.h>
#include <stdbool.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/mount.h>
#include <limits.h>
#include "zdtmtst.h"
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
const char *test_doc = "Check that some cgroups properties in kernel controllers are preserved";
const char *test_author = "Tycho Andersen <tycho.andersen@canonical.com>";
char *dirname;
TEST_OPTION(dirname, string, "cgroup directory name", 1);
static const char *const cgname = "zdtmtst";
int mount_and_add(const char *controller, const char *path, const char *prop, const char *value)
{
char aux[1024], paux[1024], subdir[1024];
if (mkdir(dirname, 0700) < 0 && errno != EEXIST) {
pr_perror("Can't make dir %s", dirname);
return -1;
}
sprintf(subdir, "%s/%s", dirname, controller);
if (mkdir(subdir, 0700) < 0) {
pr_perror("Can't make dir %s", subdir);
return -1;
}
if (mount("none", subdir, "cgroup", 0, controller)) {
pr_perror("Can't mount cgroup controller %s at %s", controller, subdir);
goto err_rd;
}
ssprintf(paux, "%s/%s", subdir, path);
mkdir(paux, 0600);
ssprintf(paux, "%s/%s/%s", subdir, path, prop);
if (write_value(paux, value) < 0)
goto err_rs;
sprintf(aux, "%d", getpid());
ssprintf(paux, "%s/%s/tasks", subdir, path);
if (write_value(paux, aux) < 0)
goto err_rs;
ssprintf(paux, "%s/%s/special_prop_check", subdir, path);
if (mkdir(paux, 0600) < 0)
pr_perror("Can't make dir %s", paux);
return 0;
err_rs:
umount(dirname);
err_rd:
rmdir(dirname);
return -1;
}
bool checkval(char *path, char *val)
{
char buf[1024];
int fd, n;
fd = open(path, O_RDONLY);
if (fd < 0) {
pr_perror("open %s", path);
return false;
}
n = read(fd, buf, sizeof(buf) - 1);
if (n < 0)
pr_perror("read %s", path);
close(fd);
if (n < 0)
return false;
buf[n] = 0;
if (strcmp(val, buf)) {
pr_err("got %s expected %s\n", buf, val);
return false;
}
return true;
}
int main(int argc, char **argv)
{
int ret = -1, i;
char buf[1024], path[PATH_MAX];
struct stat sb;
const char *const dev_allow[] = {
"c *:* m", "b *:* m", "c 1:3 rwm", "c 1:5 rwm", "c 1:7 rwm", "c 5:0 rwm",
"c 5:2 rwm", "c 1:8 rwm", "c 1:9 rwm", "c 136:* rwm", "c 10:229 rwm",
};
test_init(argc, argv);
if (mount_and_add("devices", cgname, "devices.deny", "a") < 0)
goto out;
/* need to allow /dev/null for restore */
sprintf(path, "%s/devices/%s/devices.allow", dirname, cgname);
for (i = 0; i < ARRAY_SIZE(dev_allow); i++) {
if (write_value(path, dev_allow[i]) < 0)
goto out;
}
if (mount_and_add("memory", cgname, "memory.limit_in_bytes", "268435456") < 0)
goto out;
test_daemon();
test_waitsig();
buf[0] = 0;
for (i = 0; i < ARRAY_SIZE(dev_allow); i++) {
strcat(buf, dev_allow[i]);
strcat(buf, "\n");
}
sprintf(path, "%s/devices/%s/devices.list", dirname, cgname);
if (!checkval(path, buf)) {
errno = 0;
fail();
goto out;
}
sprintf(path, "%s/memory/%s/memory.limit_in_bytes", dirname, cgname);
if (!checkval(path, "268435456\n")) {
errno = 0;
fail();
goto out;
}
sprintf(path, "%s/devices/%s/special_prop_check", dirname, cgname);
if (stat(path, &sb) < 0) {
fail("special_prop_check doesn't exist?");
goto out;
}
if (!S_ISDIR(sb.st_mode)) {
errno = 0;
fail("special_prop_check not a directory?");
goto out;
}
pass();
ret = 0;
out:
sprintf(path, "%s/devices/%s/special_prop_check", dirname, cgname);
rmdir(path);
sprintf(path, "%s/devices/%s", dirname, cgname);
rmdir(path);
sprintf(path, "%s/devices", dirname);
umount(path);
sprintf(path, "%s/memory/%s", dirname, cgname);
rmdir(path);
sprintf(path, "%s/memory", dirname);
umount(path);
return ret;
}