[CRIU] [PATCH 21/21] zdtm: Add pidns03 test
Kirill Tkhai
ktkhai at virtuozzo.com
Tue May 23 05:41:33 PDT 2017
Create a child in new pid_ns; then the child creats thread and zombie.
Zombie is in the second created new pid_ns. Then the great parent
setns() to its active pid_ns. So, lets draw the table:
pid_ns vs pid_for_children_ns
great parent: equal
child: not equal
child thread: equal
grand child zombie: zombies don't have pid_for_children_ns
After signal chech that everything remains the same.
Signed-off-by: Kirill Tkhai <ktkhai at virtuozzo.com>
---
test/zdtm/static/Makefile | 4 +
test/zdtm/static/pidns03.c | 200 +++++++++++++++++++++++++++++++++++++++++
test/zdtm/static/pidns03.desc | 1
3 files changed, 205 insertions(+)
create mode 100644 test/zdtm/static/pidns03.c
create mode 100644 test/zdtm/static/pidns03.desc
diff --git a/test/zdtm/static/Makefile b/test/zdtm/static/Makefile
index 178e564d7..24e1420bf 100644
--- a/test/zdtm/static/Makefile
+++ b/test/zdtm/static/Makefile
@@ -179,6 +179,7 @@ TST_NOFILE := \
pidns00 \
pidns01 \
pidns02 \
+ pidns03 \
# jobctl00 \
ifneq ($(SRCARCH),arm)
@@ -464,6 +465,9 @@ pty-console: CFLAGS += -D ZDTM_DEV_CONSOLE
userns02: CFLAGS += -D USERNS02
proc-self01: override CFLAGS += -D PROC_SELF01
+pidns03: CFLAGS += -pthread
+pidns03: LDFLAGS += -pthread
+
$(LIB): force
$(Q) $(MAKE) -C $(LIBDIR)
diff --git a/test/zdtm/static/pidns03.c b/test/zdtm/static/pidns03.c
new file mode 100644
index 000000000..1cca6e72a
--- /dev/null
+++ b/test/zdtm/static/pidns03.c
@@ -0,0 +1,200 @@
+#include <stdbool.h>
+#include <string.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <signal.h>
+#include <stdio.h>
+#include <sys/mman.h>
+#include <sys/types.h>
+#include <pthread.h>
+#include <sched.h>
+#include <sys/wait.h>
+#include <stdlib.h>
+#include <limits.h>
+#include <dirent.h>
+#include <sys/stat.h>
+
+#include "zdtmtst.h"
+#include "lock.h"
+
+const char *test_doc = "Check pid_for_children of tasks and threads restore right";
+const char *test_author = "Kirill Tkhai <ktkhai at virtuozzo.com>";
+/*
+ * Create a child in new pid_ns; then the child creats thread and zombie.
+ * Zombie is in the second created new pid_ns. Then the great parent
+ * setns() to its active pid_ns. So, lets draw the table:
+ *
+ * pid_ns vs pid_for_children_ns
+ * great parent: equal
+ * child: not equal
+ * child thread: equal
+ * grand child zombie: zombies don't have pid_for_children_ns
+ */
+
+enum {
+ FUTEX_INITIALIZED = 0,
+ CHILD_PREPARED,
+ POST_RESTORE_CHECK,
+ EMERGENCY_ABORT,
+};
+
+futex_t *futex;
+
+enum {
+ EQUAL = 0,
+ NOTEQUAL,
+ ERROR,
+};
+
+char *ret_names[] = {"equal", "not equal", "error"};
+
+static int read_ns_link(const char *file_name, ino_t *id)
+{
+ char ns_path[128];
+ struct stat st;
+
+ sprintf(ns_path, "/proc/thread-self/ns/%s", file_name);
+
+ if (stat(ns_path, &st)) {
+ pr_perror("Unable to stat %s", ns_path);
+ return -1;
+ }
+ *id = st.st_ino;
+ return 0;
+}
+
+static int compare_pid_and_pfc(void)
+{
+ ino_t pid, pid_for_children;
+
+ if (read_ns_link("pid", &pid) || read_ns_link("pid_for_children", &pid_for_children)) {
+ pr_err("Can't read link\n");
+ return ERROR;
+ }
+
+ if (pid != pid_for_children)
+ return NOTEQUAL;
+ return EQUAL;
+}
+
+static void *thread_fn(void *unused)
+{
+ int ret;
+ futex_wait_while_lt(futex, POST_RESTORE_CHECK);
+ ret = compare_pid_and_pfc();
+ if (ret != EQUAL) {
+ pr_err("thread: %s\n", ret_names[ret]);
+ return (void *)(long)-1;
+ }
+ return (void *)(long)0;
+}
+
+static int child_fn(void)
+{
+ long thread_retval;
+ pthread_t thread;
+ pid_t pid;
+ int ret;
+
+ ret = pthread_create(&thread, NULL, thread_fn, NULL);
+ if (ret) {
+ pr_perror("Can't creat thread");
+ goto err;
+ }
+
+ if (unshare(CLONE_NEWPID) < 0) {
+ pr_perror("Can't unshare");
+ goto err;
+ }
+
+ pid = fork();
+ if (pid < 0) {
+ pr_perror("Can't fork");
+ goto err;
+ } else if (!pid)
+ exit(0);
+ futex_set_and_wake(futex, CHILD_PREPARED);
+ futex_wait_while_lt(futex, POST_RESTORE_CHECK);
+
+ ret = pthread_join(thread, (void **)&thread_retval);
+ if (ret) {
+ pr_perror("Can't join thread");
+ goto err;
+ }
+
+ if (thread_retval == -1) {
+ pr_err("Thread finished with error\n");
+ goto err;
+ }
+
+ ret = compare_pid_and_pfc();
+ if (ret != NOTEQUAL) {
+ pr_err("child: %s\n", ret_names[ret]);
+ goto err;
+ }
+
+ return 0;
+err:
+ futex_set_and_wake(futex, EMERGENCY_ABORT);
+ return -1;
+}
+
+int main(int argc, char **argv)
+{
+ int status, fd, ret;
+ pid_t pid;
+
+ test_init(argc, argv);
+ futex = mmap(NULL, sizeof(*futex), PROT_WRITE | PROT_READ, MAP_SHARED | MAP_ANONYMOUS, -1, 0);
+ if (futex == MAP_FAILED) {
+ fail("mmap futex\n");
+ return 1;
+ }
+ futex_init(futex);
+
+ if (unshare(CLONE_NEWPID) < 0) {
+ fail("Can't unshare");
+ return 1;
+ }
+
+ pid = fork();
+ if (pid < 0) {
+ fail("Can't fork");
+ return 1;
+ } else if (pid == 0)
+ exit(child_fn());
+
+ fd = open("/proc/self/ns/pid", O_RDONLY);
+ if (fd < 0) {
+ fail("Can't open");
+ return 1;
+ }
+
+ if (setns(fd, CLONE_NEWPID) < 0) {
+ fail("Can't setns");
+ return 1;
+ }
+
+ futex_wait_while_lt(futex, CHILD_PREPARED);
+
+ test_daemon();
+ test_waitsig();
+
+ ret = compare_pid_and_pfc();
+ if (ret != EQUAL) {
+ fail("parent: %s\n", ret_names[ret]);
+ kill(pid, SIGKILL);
+ return 1;
+ }
+
+ futex_set_and_wake(futex, POST_RESTORE_CHECK);
+
+ errno = 0;
+ if (waitpid(pid, &status, 0) != pid || !WIFEXITED(status) || WEXITSTATUS(status)) {
+ fail("Can't wait or bad status: errno=%d, status=%d", errno, status);
+ return 1;
+ }
+
+ pass();
+ return 0;
+}
diff --git a/test/zdtm/static/pidns03.desc b/test/zdtm/static/pidns03.desc
new file mode 100644
index 000000000..8b53d6205
--- /dev/null
+++ b/test/zdtm/static/pidns03.desc
@@ -0,0 +1 @@
+{'flavor': 'uns', 'flags': 'suid', 'feature': 'ns_pid ns_get_userns ns_get_parent pid_for_children_ns'}
More information about the CRIU
mailing list