1use crate::errno::Errno;
9use crate::sys::signal::Signal;
10use crate::Result;
11
12use libc::{c_int, c_ulong, c_void};
13use std::convert::TryFrom;
14use std::ffi::{CStr, CString};
15use std::num::NonZeroUsize;
16use std::ptr::NonNull;
17
18libc_enum! {
19 #[repr(i32)]
22 #[non_exhaustive]
23 #[allow(non_camel_case_types)]
24 pub enum PrctlMCEKillPolicy {
25 PR_MCE_KILL_EARLY,
27 PR_MCE_KILL_LATE,
29 PR_MCE_KILL_DEFAULT,
31 }
32 impl TryFrom<i32>
33}
34
35fn prctl_set_bool(option: c_int, status: bool) -> Result<()> {
36 let res = unsafe { libc::prctl(option, status as c_ulong, 0, 0, 0) };
37 Errno::result(res).map(drop)
38}
39
40fn prctl_get_bool(option: c_int) -> Result<bool> {
41 let res = unsafe { libc::prctl(option, 0, 0, 0, 0) };
42 Errno::result(res).map(|res| res != 0)
43}
44
45pub fn set_child_subreaper(attribute: bool) -> Result<()> {
47 prctl_set_bool(libc::PR_SET_CHILD_SUBREAPER, attribute)
48}
49
50pub fn get_child_subreaper() -> Result<bool> {
52 let mut subreaper: c_int = 0;
54
55 let res = unsafe {
56 libc::prctl(libc::PR_GET_CHILD_SUBREAPER, &mut subreaper, 0, 0, 0)
57 };
58
59 Errno::result(res).map(|_| subreaper != 0)
60}
61
62pub fn set_dumpable(attribute: bool) -> Result<()> {
64 prctl_set_bool(libc::PR_SET_DUMPABLE, attribute)
65}
66
67pub fn get_dumpable() -> Result<bool> {
69 prctl_get_bool(libc::PR_GET_DUMPABLE)
70}
71
72pub fn set_keepcaps(attribute: bool) -> Result<()> {
75 prctl_set_bool(libc::PR_SET_KEEPCAPS, attribute)
76}
77
78pub fn get_keepcaps() -> Result<bool> {
80 prctl_get_bool(libc::PR_GET_KEEPCAPS)
81}
82
83pub fn clear_mce_kill() -> Result<()> {
85 let res = unsafe {
86 libc::prctl(libc::PR_MCE_KILL, libc::PR_MCE_KILL_CLEAR, 0, 0, 0)
87 };
88
89 Errno::result(res).map(drop)
90}
91
92pub fn set_mce_kill(policy: PrctlMCEKillPolicy) -> Result<()> {
94 let res = unsafe {
95 libc::prctl(
96 libc::PR_MCE_KILL,
97 libc::PR_MCE_KILL_SET,
98 policy as c_ulong,
99 0,
100 0,
101 )
102 };
103
104 Errno::result(res).map(drop)
105}
106
107pub fn get_mce_kill() -> Result<PrctlMCEKillPolicy> {
109 let res = unsafe { libc::prctl(libc::PR_MCE_KILL_GET, 0, 0, 0, 0) };
110
111 match Errno::result(res) {
112 Ok(val) => Ok(PrctlMCEKillPolicy::try_from(val)?),
113 Err(e) => Err(e),
114 }
115}
116
117pub fn set_pdeathsig<T: Into<Option<Signal>>>(signal: T) -> Result<()> {
120 let sig = match signal.into() {
121 Some(s) => s as c_int,
122 None => 0,
123 };
124
125 let res = unsafe { libc::prctl(libc::PR_SET_PDEATHSIG, sig, 0, 0, 0) };
126
127 Errno::result(res).map(drop)
128}
129
130pub fn get_pdeathsig() -> Result<Option<Signal>> {
132 let mut sig: c_int = 0;
134
135 let res = unsafe { libc::prctl(libc::PR_GET_PDEATHSIG, &mut sig, 0, 0, 0) };
136
137 match Errno::result(res) {
138 Ok(_) => Ok(match sig {
139 0 => None,
140 _ => Some(Signal::try_from(sig)?),
141 }),
142 Err(e) => Err(e),
143 }
144}
145
146pub fn set_name(name: &CStr) -> Result<()> {
148 let res = unsafe { libc::prctl(libc::PR_SET_NAME, name.as_ptr(), 0, 0, 0) };
149
150 Errno::result(res).map(drop)
151}
152
153pub fn get_name() -> Result<CString> {
155 let buf = [0u8; 16];
157
158 let res = unsafe { libc::prctl(libc::PR_GET_NAME, &buf, 0, 0, 0) };
159
160 Errno::result(res).and_then(|_| {
161 CStr::from_bytes_until_nul(&buf)
162 .map(CStr::to_owned)
163 .map_err(|_| Errno::EINVAL)
164 })
165}
166
167pub fn set_timerslack(ns: u64) -> Result<()> {
170 let res = unsafe { libc::prctl(libc::PR_SET_TIMERSLACK, ns, 0, 0, 0) };
171
172 Errno::result(res).map(drop)
173}
174
175pub fn get_timerslack() -> Result<i32> {
177 let res = unsafe { libc::prctl(libc::PR_GET_TIMERSLACK, 0, 0, 0, 0) };
178
179 Errno::result(res)
180}
181
182pub fn task_perf_events_disable() -> Result<()> {
184 let res =
185 unsafe { libc::prctl(libc::PR_TASK_PERF_EVENTS_DISABLE, 0, 0, 0, 0) };
186
187 Errno::result(res).map(drop)
188}
189
190pub fn task_perf_events_enable() -> Result<()> {
192 let res =
193 unsafe { libc::prctl(libc::PR_TASK_PERF_EVENTS_ENABLE, 0, 0, 0, 0) };
194
195 Errno::result(res).map(drop)
196}
197
198pub fn set_no_new_privs() -> Result<()> {
200 prctl_set_bool(libc::PR_SET_NO_NEW_PRIVS, true) }
202
203pub fn get_no_new_privs() -> Result<bool> {
205 prctl_get_bool(libc::PR_GET_NO_NEW_PRIVS)
206}
207
208pub fn set_thp_disable(flag: bool) -> Result<()> {
211 prctl_set_bool(libc::PR_SET_THP_DISABLE, flag)
212}
213
214pub fn get_thp_disable() -> Result<bool> {
216 prctl_get_bool(libc::PR_GET_THP_DISABLE)
217}
218
219pub fn set_vma_anon_name(addr: NonNull<c_void>, length: NonZeroUsize, name: Option<&CStr>) -> Result<()> {
221 let nameref = match name {
222 Some(n) => n.as_ptr(),
223 _ => std::ptr::null()
224 };
225 let res = unsafe { libc::prctl(libc::PR_SET_VMA, libc::PR_SET_VMA_ANON_NAME, addr.as_ptr(), length, nameref) };
226
227 Errno::result(res).map(drop)
228}