uts_namespace内核实现
本次调试基于linux v5.10-rc7,具体参加下方链接:
断点
1 2 (gdb) save breakpoints ust_namespace.breakpoint Saved to file 'ust_namespace.breakpoint'.
1 2 3 4 5 6 7 8 9 10 break fs/exec.c:do_execveat_common break kernel/fork.c:2573 break kernel/fork.c:kernel_clone break kernel/fork.c:2456 break kernel/fork.c:copy_process break kernel/fork.c:1929 break kernel/fork.c:2098 break kernel/nsproxy.c:copy_namespaces break kernel/nsproxy.c:162 break include/linux/nsproxy.h:109
1 2 source ust_namespace.breakpoint i b
uts_namespace_demo
通过qemu进入系统后:
1 2 cd build ./uts_namespace_demo
命名空间复制
1 2 14 breakpoint keep y 0xffffffff8105ea0d in copy_process at kernel/fork.c:2098 retval = copy_namespaces(clone_flags, p);
1 2 b kernel/nsproxy.c:copy_namespaces Breakpoint 15 at 0xffffffff81084c50: file kernel/nsproxy.c, line 153.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 int copy_namespaces (unsigned long flags, struct task_struct *tsk) { struct nsproxy *old_ns = tsk->nsproxy; struct user_namespace *user_ns = task_cred_xxx(tsk, user_ns); struct nsproxy *new_ns ; int ret; if (likely(!(flags & (CLONE_NEWNS | CLONE_NEWUTS | CLONE_NEWIPC | CLONE_NEWPID | CLONE_NEWNET | CLONE_NEWCGROUP | CLONE_NEWTIME)))) { if (likely(old_ns->time_ns_for_children == old_ns->time_ns)) { get_nsproxy(old_ns); return 0 ; } } else if (!ns_capable(user_ns, CAP_SYS_ADMIN)) return -EPERM; ... }
1 2 (gdb) b kernel/nsproxy.c:162 Breakpoint 16 at 0xffffffff81084c90: file kernel/nsproxy.c, line 163.
1 2 (gdb) b include/linux/nsproxy.h:109 Breakpoint 17 at 0xffffffff81229df2: file ./include/linux/nsproxy.h, line 109.
nsproxy数据结构
1 2 3 4 5 6 7 8 9 10 11 12 struct nsproxy { atomic_t count; struct uts_namespace *uts_ns ; struct ipc_namespace *ipc_ns ; struct mnt_namespace *mnt_ns ; struct pid_namespace *pid_ns_for_children ; struct net *net_ns ; struct time_namespace *time_ns ; struct time_namespace *time_ns_for_children ; struct cgroup_namespace *cgroup_ns ; };extern struct nsproxy init_nsproxy ;
unshare系统调用加入新的namespace
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 int unshare_nsproxy_namespaces (unsigned long unshare_flags, struct nsproxy **new_nsp, struct cred *new_cred, struct fs_struct *new_fs) { struct user_namespace *user_ns ; int err = 0 ; if (!(unshare_flags & (CLONE_NEWNS | CLONE_NEWUTS | CLONE_NEWIPC | CLONE_NEWNET | CLONE_NEWPID | CLONE_NEWCGROUP | CLONE_NEWTIME))) return 0 ; user_ns = new_cred ? new_cred->user_ns : current_user_ns(); if (!ns_capable(user_ns, CAP_SYS_ADMIN)) return -EPERM; *new_nsp = create_new_namespaces(unshare_flags, current, user_ns, new_fs ? new_fs : current->fs); if (IS_ERR(*new_nsp)) { err = PTR_ERR(*new_nsp); goto out; } out: return err; }