1use rustc_abi::CanonAbi;
2use rustc_middle::ty::Ty;
3use rustc_span::Symbol;
4use rustc_target::callconv::FnAbi;
5
6use self::shims::unix::linux::mem::EvalContextExt as _;
7use self::shims::unix::linux_like::epoll::EvalContextExt as _;
8use self::shims::unix::linux_like::eventfd::EvalContextExt as _;
9use self::shims::unix::linux_like::syscall::syscall;
10use crate::machine::{SIGRTMAX, SIGRTMIN};
11use crate::shims::unix::foreign_items::EvalContextExt as _;
12use crate::shims::unix::*;
13use crate::*;
14
15const TASK_COMM_LEN: u64 = 16;
19
20pub fn is_dyn_sym(name: &str) -> bool {
21 matches!(name, "gettid" | "statx")
22}
23
24impl<'tcx> EvalContextExt<'tcx> for crate::MiriInterpCx<'tcx> {}
25pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> {
26 fn emulate_foreign_item_inner(
27 &mut self,
28 link_name: Symbol,
29 abi: &FnAbi<'tcx, Ty<'tcx>>,
30 args: &[OpTy<'tcx>],
31 dest: &MPlaceTy<'tcx>,
32 ) -> InterpResult<'tcx, EmulateItemResult> {
33 let this = self.eval_context_mut();
34
35 match link_name.as_str() {
38 "open64" => {
40 let ([path_raw, flag], varargs) =
43 this.check_shim_sig_variadic_lenient(abi, CanonAbi::C, link_name, args)?;
44 let result = this.open(path_raw, flag, varargs)?;
45 this.write_scalar(result, dest)?;
46 }
47 "pread64" => {
48 let [fd, buf, count, offset] = this.check_shim_sig(
50 shim_sig!(extern "C" fn(i32, *mut _, usize, libc::off64_t) -> isize),
51 link_name,
52 abi,
53 args,
54 )?;
55 let fd = this.read_scalar(fd)?.to_i32()?;
56 let buf = this.read_pointer(buf)?;
57 let count = this.read_target_usize(count)?;
58 let offset = this.read_scalar(offset)?.to_int(offset.layout.size)?;
59 this.read(fd, buf, count, Some(offset), dest)?;
60 }
61 "pwrite64" => {
62 let [fd, buf, n, offset] = this.check_shim_sig(
64 shim_sig!(extern "C" fn(i32, *const _, usize, libc::off64_t) -> isize),
65 link_name,
66 abi,
67 args,
68 )?;
69 let fd = this.read_scalar(fd)?.to_i32()?;
70 let buf = this.read_pointer(buf)?;
71 let count = this.read_target_usize(n)?;
72 let offset = this.read_scalar(offset)?.to_int(offset.layout.size)?;
73 trace!("Called pwrite64({:?}, {:?}, {:?}, {:?})", fd, buf, count, offset);
74 this.write(fd, buf, count, Some(offset), dest)?;
75 }
76 "lseek64" => {
77 let [fd, offset, whence] = this.check_shim_sig(
79 shim_sig!(extern "C" fn(i32, libc::off64_t, i32) -> libc::off64_t),
80 link_name,
81 abi,
82 args,
83 )?;
84 let fd = this.read_scalar(fd)?.to_i32()?;
85 let offset = this.read_scalar(offset)?.to_int(offset.layout.size)?;
86 let whence = this.read_scalar(whence)?.to_i32()?;
87 this.lseek(fd, offset, whence, dest)?;
88 }
89 "ftruncate64" => {
90 let [fd, length] = this.check_shim_sig(
91 shim_sig!(extern "C" fn(i32, libc::off64_t) -> i32),
92 link_name,
93 abi,
94 args,
95 )?;
96 let fd = this.read_scalar(fd)?.to_i32()?;
97 let length = this.read_scalar(length)?.to_int(length.layout.size)?;
98 let result = this.ftruncate64(fd, length)?;
99 this.write_scalar(result, dest)?;
100 }
101 "posix_fallocate64" => {
102 let [fd, offset, len] = this.check_shim_sig(
103 shim_sig!(extern "C" fn(i32, libc::off64_t, libc::off64_t) -> i32),
104 link_name,
105 abi,
106 args,
107 )?;
108
109 let fd = this.read_scalar(fd)?.to_i32()?;
110 let offset = this.read_scalar(offset)?.to_i64()?;
111 let len = this.read_scalar(len)?.to_i64()?;
112
113 let result = this.posix_fallocate(fd, offset, len)?;
114 this.write_scalar(result, dest)?;
115 }
116 "readdir64" => {
117 let [dirp] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
118 this.readdir(dirp, dest)?;
119 }
120 "sync_file_range" => {
121 let [fd, offset, nbytes, flags] =
122 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
123 let result = this.sync_file_range(fd, offset, nbytes, flags)?;
124 this.write_scalar(result, dest)?;
125 }
126 "statx" => {
127 let [dirfd, pathname, flags, mask, statxbuf] =
129 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
130 let result = this.linux_statx(dirfd, pathname, flags, mask, statxbuf)?;
131 this.write_scalar(result, dest)?;
132 }
133 "epoll_create1" => {
135 let [flag] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
136 let result = this.epoll_create1(flag)?;
137 this.write_scalar(result, dest)?;
138 }
139 "epoll_ctl" => {
140 let [epfd, op, fd, event] =
141 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
142 let result = this.epoll_ctl(epfd, op, fd, event)?;
143 this.write_scalar(result, dest)?;
144 }
145 "epoll_wait" => {
146 let [epfd, events, maxevents, timeout] =
147 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
148 this.epoll_wait(epfd, events, maxevents, timeout, dest)?;
149 }
150 "eventfd" => {
151 let [val, flag] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
152 let result = this.eventfd(val, flag)?;
153 this.write_scalar(result, dest)?;
154 }
155
156 "pthread_setname_np" => {
158 let [thread, name] =
159 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
160 let res = match this.pthread_setname_np(
161 this.read_scalar(thread)?,
162 this.read_scalar(name)?,
163 TASK_COMM_LEN,
164 false,
165 )? {
166 ThreadNameResult::Ok => Scalar::from_u32(0),
167 ThreadNameResult::NameTooLong => this.eval_libc("ERANGE"),
168 ThreadNameResult::ThreadNotFound => this.eval_libc("ENOENT"),
170 };
171 this.write_scalar(res, dest)?;
172 }
173 "pthread_getname_np" => {
174 let [thread, name, len] =
175 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
176 let len = this.read_scalar(len)?;
180 let res = if len.to_target_usize(this)? >= TASK_COMM_LEN {
181 match this.pthread_getname_np(
182 this.read_scalar(thread)?,
183 this.read_scalar(name)?,
184 len,
185 false,
186 )? {
187 ThreadNameResult::Ok => Scalar::from_u32(0),
188 ThreadNameResult::NameTooLong => unreachable!(),
189 ThreadNameResult::ThreadNotFound => this.eval_libc("ENOENT"),
191 }
192 } else {
193 this.eval_libc("ERANGE")
194 };
195 this.write_scalar(res, dest)?;
196 }
197 "gettid" => {
198 let [] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
199 let result = this.unix_gettid(link_name.as_str())?;
200 this.write_scalar(result, dest)?;
201 }
202
203 "syscall" => {
205 syscall(this, link_name, abi, args, dest)?;
206 }
207
208 "mmap64" => {
210 let [addr, length, prot, flags, fd, offset] =
211 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
212 let offset = this.read_scalar(offset)?.to_i64()?;
213 let ptr = this.mmap(addr, length, prot, flags, fd, offset.into())?;
214 this.write_scalar(ptr, dest)?;
215 }
216 "mremap" => {
217 let ([old_address, old_size, new_size, flags], _) =
218 this.check_shim_sig_variadic_lenient(abi, CanonAbi::C, link_name, args)?;
219 let ptr = this.mremap(old_address, old_size, new_size, flags)?;
220 this.write_scalar(ptr, dest)?;
221 }
222 "__xpg_strerror_r" => {
223 let [errnum, buf, buflen] =
224 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
225 let result = this.strerror_r(errnum, buf, buflen)?;
226 this.write_scalar(result, dest)?;
227 }
228 "__errno_location" => {
229 let [] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
230 let errno_place = this.last_error_place()?;
231 this.write_scalar(errno_place.to_ref(this).to_scalar(), dest)?;
232 }
233 "__libc_current_sigrtmin" => {
234 let [] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
235
236 this.write_int(SIGRTMIN, dest)?;
237 }
238 "__libc_current_sigrtmax" => {
239 let [] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
240
241 this.write_int(SIGRTMAX, dest)?;
242 }
243
244 "pthread_getattr_np" if this.frame_in_std() => {
247 let [_thread, _attr] =
248 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
249 this.write_null(dest)?;
250 }
251
252 _ => return interp_ok(EmulateItemResult::NotSupported),
253 };
254
255 interp_ok(EmulateItemResult::NeedsReturn)
256 }
257}