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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
use std::fs::File;
use std::io;
use std::os::unix::io::AsRawFd;
use std::path::Path;
use bitflags::bitflags;
bitflags! {
pub struct ExtraUnshareFlags: i32 {
const FILES = libc::CLONE_FILES;
const FS = libc::CLONE_FS;
const SYSVSEM = libc::CLONE_SYSVSEM;
}
}
bitflags! {
pub struct NamespaceTypes: i32 {
const NEWCGROUP = libc::CLONE_NEWCGROUP;
const NEWIPC = libc::CLONE_NEWIPC;
const NEWNET = libc::CLONE_NEWNET;
const NEWNS = libc::CLONE_NEWNS;
const NEWPID = libc::CLONE_NEWPID;
const NEWUSER = libc::CLONE_NEWUSER;
const NEWUTS = libc::CLONE_NEWUTS;
}
}
pub fn unshare(nstypes: NamespaceTypes, extra_flags: ExtraUnshareFlags) -> io::Result<()> {
crate::error::convert_nzero_ret(unsafe { libc::unshare(nstypes.bits | extra_flags.bits) })
}
pub fn setns_raw(fd: i32, nstype: NamespaceTypes) -> io::Result<()> {
crate::error::convert_nzero_ret(unsafe { libc::setns(fd, nstype.bits) })
}
pub fn setns(f: &File, nstype: NamespaceTypes) -> io::Result<()> {
setns_raw(f.as_raw_fd(), nstype)
}
pub fn join_proc_namespaces<P: AsRef<Path>>(
proc_pid_dir: P,
mut nstypes: NamespaceTypes,
) -> io::Result<()> {
let proc_ns_dir = proc_pid_dir.as_ref().join("ns");
if nstypes.contains(NamespaceTypes::NEWUSER) {
let file = File::open(&proc_ns_dir.join("user"))?;
setns(&file, NamespaceTypes::NEWUSER)?;
nstypes.remove(NamespaceTypes::NEWUSER);
}
for entry in proc_ns_dir.read_dir()? {
let entry = entry?;
if let Some(name) = entry.file_name().to_str() {
let entry_nstype = match name {
"cgroup" => NamespaceTypes::NEWCGROUP,
"ipc" => NamespaceTypes::NEWIPC,
"net" => NamespaceTypes::NEWNET,
"mnt" => NamespaceTypes::NEWNS,
"pid" => NamespaceTypes::NEWPID,
"uts" => NamespaceTypes::NEWUTS,
_ => NamespaceTypes::empty(),
};
if !entry_nstype.is_empty() && nstypes.contains(entry_nstype) {
let file = File::open(entry.path())?;
setns(&file, entry_nstype)?;
nstypes.remove(entry_nstype);
}
}
}
if !nstypes.is_empty() {
return Err(io::Error::from_raw_os_error(libc::EINVAL));
}
Ok(())
}