[CRIU] [PATCH v3 52/55] restorer: Set NStids in all pid_ns for thread before we create it.
Kirill Tkhai
ktkhai at virtuozzo.com
Mon Apr 10 01:23:42 PDT 2017
If there is multi-level pid, ask helpers to populate /proc last pids
in their active pid namespaces. So, thread will be created with right
NStids.
Signed-off-by: Kirill Tkhai <ktkhai at virtuozzo.com>
---
criu/cr-restore.c | 24 ++++++++++++++++--------
criu/include/restorer.h | 4 +++-
criu/pie/restorer.c | 44 ++++++++++++++++++++++++++++++--------------
3 files changed, 49 insertions(+), 23 deletions(-)
diff --git a/criu/cr-restore.c b/criu/cr-restore.c
index 2402f6402..47bfba312 100644
--- a/criu/cr-restore.c
+++ b/criu/cr-restore.c
@@ -3198,12 +3198,13 @@ static int sigreturn_restore(pid_t pid, struct task_restore_args *task_args, uns
struct vm_area_list self_vmas;
struct vm_area_list *vmas = &rsti(current)->vmas;
- int i, siginfo_n;
+ int i, k, siginfo_n;
unsigned long creds_pos = 0;
unsigned long creds_pos_next;
sigset_t blockmask;
+ struct ns_id *pid_ns;
pr_info("Restore via sigreturn\n");
@@ -3393,6 +3394,12 @@ static int sigreturn_restore(pid_t pid, struct task_restore_args *task_args, uns
task_args->sigchld_act = sigchld_act;
strncpy(task_args->comm, core->tc->comm, sizeof(task_args->comm));
+ pid_ns = lookup_ns_by_id(current->ids->pid_ns_id, &pid_ns_desc);
+ BUG_ON(!pid_ns);
+ for (i = current->pid->level - 1; i >= 0; i--, pid_ns = pid_ns->parent)
+ task_args->pid_ns_id[i] = pid_ns->id;
+ if (current->pid->level != MAX_NS_NESTING)
+ task_args->pid_ns_id[current->pid->level] = 0; /* Delimiter */
/*
* Fill up per-thread data.
@@ -3404,14 +3411,15 @@ static int sigreturn_restore(pid_t pid, struct task_restore_args *task_args, uns
struct rt_sigframe *sigframe;
k_rtsigset_t *blkset = NULL;
- thread_args[i].pid = current->threads[i]->ns[0].virt;
+ for (k = 0; k < current->pid->level; k++)
+ thread_args[i].pid[k] = current->threads[i]->ns[k].virt;
thread_args[i].siginfo_n = siginfo_priv_nr[i];
thread_args[i].siginfo = task_args->siginfo;
thread_args[i].siginfo += siginfo_n;
siginfo_n += thread_args[i].siginfo_n;
/* skip self */
- if (thread_args[i].pid == pid) {
+ if (thread_args[i].pid[0] == pid) {
task_args->t = thread_args + i;
tcore = core;
blkset = (void *)&tcore->tc->blk_sigset;
@@ -3421,15 +3429,15 @@ static int sigreturn_restore(pid_t pid, struct task_restore_args *task_args, uns
blkset = (void *)&tcore->thread_core->blk_sigset;
}
- if ((tcore->tc || tcore->ids) && thread_args[i].pid != pid) {
+ if ((tcore->tc || tcore->ids) && thread_args[i].pid[0] != pid) {
pr_err("Thread has optional fields present %d\n",
- thread_args[i].pid);
+ thread_args[i].pid[0]);
ret = -1;
}
if (ret < 0) {
pr_err("Can't read core data for thread %d\n",
- thread_args[i].pid);
+ thread_args[i].pid[0]);
goto err;
}
@@ -3459,7 +3467,7 @@ static int sigreturn_restore(pid_t pid, struct task_restore_args *task_args, uns
if (construct_sigframe(sigframe, sigframe, blkset, tcore))
goto err;
- if (thread_args[i].pid != pid)
+ if (thread_args[i].pid[0] != pid)
core_entry__free_unpacked(tcore, NULL);
pr_info("Thread %4d stack %8p rt_sigframe %8p\n",
@@ -3516,7 +3524,7 @@ static int sigreturn_restore(pid_t pid, struct task_restore_args *task_args, uns
"task_args->nr_threads: %d\n"
"task_args->clone_restore_fn: %p\n"
"task_args->thread_args: %p\n",
- task_args, task_args->t->pid,
+ task_args, task_args->t->pid[0],
task_args->nr_threads,
task_args->clone_restore_fn,
task_args->thread_args);
diff --git a/criu/include/restorer.h b/criu/include/restorer.h
index 002912334..701c72ffb 100644
--- a/criu/include/restorer.h
+++ b/criu/include/restorer.h
@@ -17,6 +17,7 @@
#include "posix-timer.h"
#include "timerfd.h"
#include "shmem.h"
+#include "namespaces.h"
#include "parasite-vdso.h"
#include "fault-injection.h"
@@ -78,7 +79,7 @@ struct thread_creds_args {
struct thread_restore_args {
struct restore_mem_zone *mz;
- int pid;
+ pid_t pid[MAX_NS_NESTING];
UserRegsEntry gpregs;
u64 clear_tid_addr;
@@ -167,6 +168,7 @@ struct task_restore_args {
auxv_t mm_saved_auxv[AT_VECTOR_SIZE];
u32 mm_saved_auxv_size;
char comm[TASK_COMM_LEN];
+ int pid_ns_id[MAX_NS_NESTING];
/*
* proc_fd is a handle to /proc that the restorer blob can use to open
diff --git a/criu/pie/restorer.c b/criu/pie/restorer.c
index 67d5a0132..13e8192ca 100644
--- a/criu/pie/restorer.c
+++ b/criu/pie/restorer.c
@@ -480,10 +480,14 @@ long __export_restore_thread(struct thread_restore_args *args)
k_rtsigset_t to_block;
unsigned long new_sp;
int my_pid = sys_gettid();
- int ret;
+ int i, ret;
+
+ for (i = 0; i < MAX_NS_NESTING; i++)
+ if (args->pid[i] == 0)
+ break;
- if (my_pid != args->pid) {
- pr_err("Thread pid mismatch %d/%d\n", my_pid, args->pid);
+ if (my_pid != args->pid[i-1]) {
+ pr_err("Thread pid mismatch %d/%d\n", my_pid, args->pid[i-1]);
goto core_restore_end;
}
@@ -1140,7 +1144,7 @@ static int wait_zombies(struct task_restore_args *task_args)
long __export_restore_task(struct task_restore_args *args)
{
long ret = -1;
- int i;
+ int i, k;
VmaEntry *vma_entry;
unsigned long va;
@@ -1444,20 +1448,32 @@ long __export_restore_task(struct task_restore_args *args)
for (i = 0; i < args->nr_threads; i++) {
char last_pid_buf[16], *s;
-
/* skip self */
- if (thread_args[i].pid == args->t->pid)
+ if (thread_args[i].pid[0] == args->t->pid[0])
continue;
- new_sp = restorer_stack(thread_args[i].mz);
- last_pid_len = std_vprint_num(last_pid_buf, sizeof(last_pid_buf), thread_args[i].pid - 1, &s);
- sys_lseek(fd, 0, SEEK_SET);
- ret = sys_write(fd, s, last_pid_len);
- if (ret < 0) {
- pr_err("Can't set last_pid %ld/%s\n", ret, last_pid_buf);
- sys_close(fd);
- goto core_restore_end;
+ if (thread_args[i].pid[1] == 0) {
+ /* One level pid ns hierarhy */
+ last_pid_len = std_vprint_num(last_pid_buf, sizeof(last_pid_buf), thread_args[i].pid[0] - 1, &s);
+ sys_lseek(fd, 0, SEEK_SET);
+ ret = sys_write(fd, s, last_pid_len);
+ if (ret < 0) {
+ pr_err("Can't set last_pid %ld/%s\n", ret, last_pid_buf);
+ sys_close(fd);
+ goto core_restore_end;
+ }
+ } else {
+ for (k = 0; k < MAX_NS_NESTING; k++) {
+ if (thread_args[i].pid[k] == 0)
+ break;
+ if (request_set_next_pid(args->pid_ns_id[k], thread_args[i].pid[k], args->transport_fd) < 0) {
+ pr_err("Can't request to set pid\n");
+ sys_close(fd);
+ goto core_restore_end;
+ }
+ }
}
+ new_sp = restorer_stack(thread_args[i].mz);
/*
* To achieve functionality like libc's clone()
More information about the CRIU
mailing list