1use std::collections::hash_map::Entry;
2use std::io::Write;
3use std::path::Path;
4
5use rustc_abi::{Align, CanonAbi, ExternAbi, Size};
6use rustc_ast::expand::allocator::NO_ALLOC_SHIM_IS_UNSTABLE;
7use rustc_data_structures::either::Either;
8use rustc_hir::attrs::Linkage;
9use rustc_hir::def::DefKind;
10use rustc_hir::def_id::CrateNum;
11use rustc_middle::middle::codegen_fn_attrs::CodegenFnAttrFlags;
12use rustc_middle::mir::interpret::AllocInit;
13use rustc_middle::ty::{Instance, Ty};
14use rustc_middle::{mir, ty};
15use rustc_span::Symbol;
16use rustc_target::callconv::FnAbi;
17use rustc_target::spec::{Arch, Os};
18
19use super::alloc::EvalContextExt as _;
20use super::backtrace::EvalContextExt as _;
21use crate::concurrency::GenmcEvalContextExt as _;
22use crate::helpers::EvalContextExt as _;
23use crate::*;
24
25#[derive(Debug, Copy, Clone)]
27pub struct DynSym(Symbol);
28
29#[expect(clippy::should_implement_trait)]
30impl DynSym {
31 pub fn from_str(name: &str) -> Self {
32 DynSym(Symbol::intern(name))
33 }
34}
35
36impl<'tcx> EvalContextExt<'tcx> for crate::MiriInterpCx<'tcx> {}
37pub trait EvalContextExt<'tcx>: crate::MiriInterpCxExt<'tcx> {
38 fn emulate_foreign_item(
45 &mut self,
46 link_name: Symbol,
47 abi: &FnAbi<'tcx, Ty<'tcx>>,
48 args: &[OpTy<'tcx>],
49 dest: &PlaceTy<'tcx>,
50 ret: Option<mir::BasicBlock>,
51 unwind: mir::UnwindAction,
52 ) -> InterpResult<'tcx, Option<(&'tcx mir::Body<'tcx>, ty::Instance<'tcx>)>> {
53 let this = self.eval_context_mut();
54
55 if let Some(shim) = this.machine.allocator_shim_symbols.get(&link_name) {
57 match *shim {
58 Either::Left(other_fn) => {
59 let handler = this
60 .lookup_exported_symbol(other_fn)?
61 .expect("missing alloc error handler symbol");
62 return interp_ok(Some(handler));
63 }
64 Either::Right(special) => {
65 this.rust_special_allocator_method(special, link_name, abi, args, dest)?;
66 this.return_to_block(ret)?;
67 return interp_ok(None);
68 }
69 }
70 }
71
72 let dest = this.force_allocation(dest)?;
74
75 match this.emulate_foreign_item_inner(link_name, abi, args, &dest)? {
77 EmulateItemResult::NeedsReturn => {
78 trace!("{:?}", this.dump_place(&dest.clone().into()));
79 this.return_to_block(ret)?;
80 }
81 EmulateItemResult::NeedsUnwind => {
82 this.unwind_to_block(unwind)?;
84 }
85 EmulateItemResult::AlreadyJumped => (),
86 EmulateItemResult::NotSupported => {
87 if let Some(body) = this.lookup_exported_symbol(link_name)? {
88 return interp_ok(Some(body));
89 }
90
91 throw_machine_stop!(TerminationInfo::UnsupportedForeignItem(format!(
92 "can't call foreign function `{link_name}` on OS `{os}`",
93 os = this.tcx.sess.target.os,
94 )));
95 }
96 }
97
98 interp_ok(None)
99 }
100
101 fn is_dyn_sym(&self, name: &str) -> bool {
102 let this = self.eval_context_ref();
103 match &this.tcx.sess.target.os {
104 os if this.target_os_is_unix() => shims::unix::foreign_items::is_dyn_sym(name, os),
105 Os::Windows => shims::windows::foreign_items::is_dyn_sym(name),
106 _ => false,
107 }
108 }
109
110 fn emulate_dyn_sym(
112 &mut self,
113 sym: DynSym,
114 abi: &FnAbi<'tcx, Ty<'tcx>>,
115 args: &[OpTy<'tcx>],
116 dest: &PlaceTy<'tcx>,
117 ret: Option<mir::BasicBlock>,
118 unwind: mir::UnwindAction,
119 ) -> InterpResult<'tcx> {
120 let res = self.emulate_foreign_item(sym.0, abi, args, dest, ret, unwind)?;
121 assert!(res.is_none(), "DynSyms that delegate are not supported");
122 interp_ok(())
123 }
124
125 fn lookup_exported_symbol(
127 &mut self,
128 link_name: Symbol,
129 ) -> InterpResult<'tcx, Option<(&'tcx mir::Body<'tcx>, ty::Instance<'tcx>)>> {
130 let this = self.eval_context_mut();
131 let tcx = this.tcx.tcx;
132
133 let entry = this.machine.exported_symbols_cache.entry(link_name);
136 let instance = *match entry {
137 Entry::Occupied(e) => e.into_mut(),
138 Entry::Vacant(e) => {
139 struct SymbolTarget<'tcx> {
142 instance: ty::Instance<'tcx>,
143 cnum: CrateNum,
144 is_weak: bool,
145 }
146 let mut symbol_target: Option<SymbolTarget<'tcx>> = None;
147 helpers::iter_exported_symbols(tcx, |cnum, def_id| {
148 let attrs = tcx.codegen_fn_attrs(def_id);
149 if tcx.is_foreign_item(def_id) {
151 return interp_ok(());
152 }
153 if !(attrs.symbol_name.is_some()
155 || attrs.flags.contains(CodegenFnAttrFlags::NO_MANGLE)
156 || attrs.flags.contains(CodegenFnAttrFlags::RUSTC_STD_INTERNAL_SYMBOL))
157 {
158 return interp_ok(());
159 }
160
161 let instance = Instance::mono(tcx, def_id);
162 let symbol_name = tcx.symbol_name(instance).name;
163 let is_weak = attrs.linkage == Some(Linkage::WeakAny);
164 if symbol_name == link_name.as_str() {
165 if let Some(original) = &symbol_target {
166 match (is_weak, original.is_weak) {
169 (false, true) => {
170 symbol_target = Some(SymbolTarget {
173 instance: ty::Instance::mono(tcx, def_id),
174 cnum,
175 is_weak,
176 });
177 }
178 (true, false) => {
179 }
181 (true, true) | (false, false) => {
182 let original_span =
189 tcx.def_span(original.instance.def_id()).data();
190 let span = tcx.def_span(def_id).data();
191 if original_span < span {
192 throw_machine_stop!(
193 TerminationInfo::MultipleSymbolDefinitions {
194 link_name,
195 first: original_span,
196 first_crate: tcx.crate_name(original.cnum),
197 second: span,
198 second_crate: tcx.crate_name(cnum),
199 }
200 );
201 } else {
202 throw_machine_stop!(
203 TerminationInfo::MultipleSymbolDefinitions {
204 link_name,
205 first: span,
206 first_crate: tcx.crate_name(cnum),
207 second: original_span,
208 second_crate: tcx.crate_name(original.cnum),
209 }
210 );
211 }
212 }
213 }
214 } else {
215 symbol_target = Some(SymbolTarget {
216 instance: ty::Instance::mono(tcx, def_id),
217 cnum,
218 is_weak,
219 });
220 }
221 }
222 interp_ok(())
223 })?;
224
225 if let Some(SymbolTarget { instance, .. }) = symbol_target {
229 if !matches!(tcx.def_kind(instance.def_id()), DefKind::Fn | DefKind::AssocFn) {
230 throw_ub_format!(
231 "attempt to call an exported symbol that is not defined as a function"
232 );
233 }
234 }
235
236 e.insert(symbol_target.map(|SymbolTarget { instance, .. }| instance))
237 }
238 };
239 match instance {
240 None => interp_ok(None), Some(instance) => interp_ok(Some((this.load_mir(instance.def, None)?, instance))),
242 }
243 }
244}
245
246impl<'tcx> EvalContextExtPriv<'tcx> for crate::MiriInterpCx<'tcx> {}
247trait EvalContextExtPriv<'tcx>: crate::MiriInterpCxExt<'tcx> {
248 fn emulate_foreign_item_inner(
249 &mut self,
250 link_name: Symbol,
251 abi: &FnAbi<'tcx, Ty<'tcx>>,
252 args: &[OpTy<'tcx>],
253 dest: &MPlaceTy<'tcx>,
254 ) -> InterpResult<'tcx, EmulateItemResult> {
255 let this = self.eval_context_mut();
256
257 #[cfg(all(feature = "native-lib", unix))]
259 if !this.machine.native_lib.is_empty() {
260 use crate::shims::native_lib::EvalContextExt as _;
261 if this.call_native_fn(link_name, dest, args)? {
265 return interp_ok(EmulateItemResult::NeedsReturn);
266 }
267 }
268 match link_name.as_str() {
307 name if name == this.mangle_internal_symbol(NO_ALLOC_SHIM_IS_UNSTABLE) => {
309 let [] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
312 }
313
314 "miri_alloc" => {
316 let [size, align] =
317 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
318 let size = this.read_target_usize(size)?;
319 let align = this.read_target_usize(align)?;
320
321 this.check_rust_alloc_request(size, align)?;
322
323 let ptr = this.allocate_ptr(
324 Size::from_bytes(size),
325 Align::from_bytes(align).unwrap(),
326 MiriMemoryKind::Miri.into(),
327 AllocInit::Uninit,
328 )?;
329
330 this.write_pointer(ptr, dest)?;
331 }
332 "miri_dealloc" => {
333 let [ptr, old_size, align] =
334 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
335 let ptr = this.read_pointer(ptr)?;
336 let old_size = this.read_target_usize(old_size)?;
337 let align = this.read_target_usize(align)?;
338
339 this.deallocate_ptr(
341 ptr,
342 Some((Size::from_bytes(old_size), Align::from_bytes(align).unwrap())),
343 MiriMemoryKind::Miri.into(),
344 )?;
345 }
346 "miri_track_alloc" => {
347 let [ptr] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
348 let ptr = this.read_pointer(ptr)?;
349 let (alloc_id, _, _) = this.ptr_get_alloc_id(ptr, 0).map_err_kind(|_e| {
350 err_machine_stop!(TerminationInfo::Abort(format!(
351 "pointer passed to `miri_get_alloc_id` must not be dangling, got {ptr:?}"
352 )))
353 })?;
354 if this.machine.tracked_alloc_ids.insert(alloc_id) {
355 let info = this.get_alloc_info(alloc_id);
356 this.emit_diagnostic(NonHaltingDiagnostic::TrackingAlloc(
357 alloc_id, info.size, info.align,
358 ));
359 }
360 }
361 "miri_start_unwind" => {
362 let [payload] =
363 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
364 this.handle_miri_start_unwind(payload)?;
365 return interp_ok(EmulateItemResult::NeedsUnwind);
366 }
367 "miri_run_provenance_gc" => {
368 let [] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
369 this.run_provenance_gc();
370 }
371 "miri_get_alloc_id" => {
372 let [ptr] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
373 let ptr = this.read_pointer(ptr)?;
374 let (alloc_id, _, _) = this.ptr_get_alloc_id(ptr, 0).map_err_kind(|_e| {
375 err_machine_stop!(TerminationInfo::Abort(format!(
376 "pointer passed to `miri_get_alloc_id` must not be dangling, got {ptr:?}"
377 )))
378 })?;
379 this.write_scalar(Scalar::from_u64(alloc_id.0.get()), dest)?;
380 }
381 "miri_print_borrow_state" => {
382 let [id, show_unnamed] =
383 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
384 let id = this.read_scalar(id)?.to_u64()?;
385 let show_unnamed = this.read_scalar(show_unnamed)?.to_bool()?;
386 if let Some(id) = std::num::NonZero::new(id).map(AllocId)
387 && this.get_alloc_info(id).kind == AllocKind::LiveData
388 {
389 this.print_borrow_state(id, show_unnamed)?;
390 } else {
391 eprintln!("{id} is not the ID of a live data allocation");
392 }
393 }
394 "miri_pointer_name" => {
395 let [ptr, nth_parent, name] =
398 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
399 let ptr = this.read_pointer(ptr)?;
400 let nth_parent = this.read_scalar(nth_parent)?.to_u8()?;
401 let name = this.read_immediate(name)?;
402
403 let name = this.read_byte_slice(&name)?;
404 let name = String::from_utf8_lossy(name).into_owned();
408 this.give_pointer_debug_name(ptr, nth_parent, &name)?;
409 }
410 "miri_static_root" => {
411 let [ptr] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
412 let ptr = this.read_pointer(ptr)?;
413 let (alloc_id, offset, _) = this.ptr_get_alloc_id(ptr, 0)?;
414 if offset != Size::ZERO {
415 throw_unsup_format!(
416 "pointer passed to `miri_static_root` must point to beginning of an allocated block"
417 );
418 }
419 this.machine.static_roots.push(alloc_id);
420 }
421 "miri_host_to_target_path" => {
422 let [ptr, out, out_size] =
423 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
424 let ptr = this.read_pointer(ptr)?;
425 let out = this.read_pointer(out)?;
426 let out_size = this.read_scalar(out_size)?.to_target_usize(this)?;
427
428 this.check_no_isolation("`miri_host_to_target_path`")?;
430
431 let path = this.read_os_str_from_c_str(ptr)?.to_owned();
433 let (success, needed_size) =
434 this.write_path_to_c_str(Path::new(&path), out, out_size)?;
435 this.write_int(if success { 0 } else { needed_size }, dest)?;
437 }
438 "miri_thread_spawn" => {
439 let [start_routine, func_arg] =
441 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
442 let start_routine = this.read_pointer(start_routine)?;
443 let func_arg = this.read_immediate(func_arg)?;
444
445 this.start_regular_thread(
446 Some(dest.clone()),
447 start_routine,
448 ExternAbi::Rust,
449 func_arg,
450 this.machine.layouts.unit,
451 )?;
452 }
453 "miri_thread_join" => {
454 let [thread_id] = this.check_shim_sig(
455 shim_sig!(extern "Rust" fn(usize) -> bool),
456 link_name,
457 abi,
458 args,
459 )?;
460
461 let thread = this.read_target_usize(thread_id)?;
462 use crate::concurrency::thread::ThreadLookupError;
464 let thread = match this.thread_id_try_from(thread) {
465 Ok(id) | Err(ThreadLookupError::Terminated(id)) => Some(id),
466 Err(ThreadLookupError::InvalidId) => None,
467 };
468 if let Some(thread) = thread {
469 this.join_thread_exclusive(
470 thread,
471 Scalar::from_bool(true),
472 dest,
473 )?;
474 } else {
475 this.write_scalar(Scalar::from_bool(false), dest)?;
476 }
477 }
478 "miri_spin_loop" => {
480 let [] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
481
482 this.yield_active_thread();
484 }
485 "miri_backtrace_size" => {
487 this.handle_miri_backtrace_size(abi, link_name, args, dest)?;
488 }
489 "miri_get_backtrace" => {
491 this.handle_miri_get_backtrace(abi, link_name, args)?;
493 }
494 "miri_resolve_frame" => {
496 this.handle_miri_resolve_frame(abi, link_name, args, dest)?;
498 }
499 "miri_resolve_frame_names" => {
501 this.handle_miri_resolve_frame_names(abi, link_name, args)?;
502 }
503 "miri_write_to_stdout" | "miri_write_to_stderr" => {
506 let [msg] = this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
507 let msg = this.read_immediate(msg)?;
508 let msg = this.read_byte_slice(&msg)?;
509 let _ignore = match link_name.as_str() {
511 "miri_write_to_stdout" => std::io::stdout().write_all(msg),
512 "miri_write_to_stderr" => std::io::stderr().write_all(msg),
513 _ => unreachable!(),
514 };
515 }
516 "miri_promise_symbolic_alignment" => {
518 use rustc_abi::AlignFromBytesError;
519
520 let [ptr, align] =
521 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
522 let ptr = this.read_pointer(ptr)?;
523 let align = this.read_target_usize(align)?;
524 if !align.is_power_of_two() {
525 throw_unsup_format!(
526 "`miri_promise_symbolic_alignment`: alignment must be a power of 2, got {align}"
527 );
528 }
529 let align = Align::from_bytes(align).unwrap_or_else(|err| {
530 match err {
531 AlignFromBytesError::NotPowerOfTwo(_) => unreachable!(),
532 AlignFromBytesError::TooLarge(_) => Align::MAX,
534 }
535 });
536 let addr = ptr.addr();
537 if addr.bytes().strict_rem(align.bytes()) != 0 {
539 throw_unsup_format!(
540 "`miri_promise_symbolic_alignment`: pointer is not actually aligned"
541 );
542 }
543 if let Ok((alloc_id, offset, ..)) = this.ptr_try_get_alloc_id(ptr, 0) {
544 let alloc_align = this.get_alloc_info(alloc_id).align;
545 if align > alloc_align
548 && this
549 .machine
550 .symbolic_alignment
551 .get_mut()
552 .get(&alloc_id)
553 .is_none_or(|&(_, old_align)| align > old_align)
554 {
555 this.machine.symbolic_alignment.get_mut().insert(alloc_id, (offset, align));
556 }
557 }
558 }
559 "miri_genmc_assume" => {
561 let [condition] =
562 this.check_shim_sig_lenient(abi, CanonAbi::Rust, link_name, args)?;
563 if this.machine.data_race.as_genmc_ref().is_some() {
564 this.handle_genmc_verifier_assume(condition)?;
565 } else {
566 throw_unsup_format!("miri_genmc_assume is only supported in GenMC mode")
567 }
568 }
569
570 "exit" => {
572 let [code] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
573 let code = this.read_scalar(code)?.to_i32()?;
574 if let Some(genmc_ctx) = this.machine.data_race.as_genmc_ref() {
575 genmc_ctx.handle_exit(
577 this.machine.threads.active_thread(),
578 code,
579 crate::concurrency::ExitType::ExitCalled,
580 )?;
581 return interp_ok(EmulateItemResult::AlreadyJumped);
582 }
583 throw_machine_stop!(TerminationInfo::Exit { code, leak_check: false });
584 }
585 "abort" => {
586 let [] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
587 throw_machine_stop!(TerminationInfo::Abort(
588 "the program aborted execution".to_owned()
589 ));
590 }
591
592 "malloc" => {
594 let [size] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
595 let size = this.read_target_usize(size)?;
596 if size <= this.max_size_of_val().bytes() {
597 let res = this.malloc(size, AllocInit::Uninit)?;
598 this.write_pointer(res, dest)?;
599 } else {
600 if this.target_os_is_unix() {
602 this.set_last_error(LibcError("ENOMEM"))?;
603 }
604 this.write_null(dest)?;
605 }
606 }
607 "calloc" => {
608 let [items, elem_size] =
609 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
610 let items = this.read_target_usize(items)?;
611 let elem_size = this.read_target_usize(elem_size)?;
612 if let Some(size) = this.compute_size_in_bytes(Size::from_bytes(elem_size), items) {
613 let res = this.malloc(size.bytes(), AllocInit::Zero)?;
614 this.write_pointer(res, dest)?;
615 } else {
616 if this.target_os_is_unix() {
618 this.set_last_error(LibcError("ENOMEM"))?;
619 }
620 this.write_null(dest)?;
621 }
622 }
623 "free" => {
624 let [ptr] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
625 let ptr = this.read_pointer(ptr)?;
626 this.free(ptr)?;
627 }
628 "realloc" => {
629 let [old_ptr, new_size] =
630 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
631 let old_ptr = this.read_pointer(old_ptr)?;
632 let new_size = this.read_target_usize(new_size)?;
633 if new_size <= this.max_size_of_val().bytes() {
634 let res = this.realloc(old_ptr, new_size)?;
635 this.write_pointer(res, dest)?;
636 } else {
637 if this.target_os_is_unix() {
639 this.set_last_error(LibcError("ENOMEM"))?;
640 }
641 this.write_null(dest)?;
642 }
643 }
644
645 "memcmp" => {
647 let [left, right, n] =
648 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
649 let left = this.read_pointer(left)?;
650 let right = this.read_pointer(right)?;
651 let n = Size::from_bytes(this.read_target_usize(n)?);
652
653 this.ptr_get_alloc_id(left, 0)?;
655 this.ptr_get_alloc_id(right, 0)?;
656
657 let result = {
658 let left_bytes = this.read_bytes_ptr_strip_provenance(left, n)?;
659 let right_bytes = this.read_bytes_ptr_strip_provenance(right, n)?;
660
661 use std::cmp::Ordering::*;
662 match left_bytes.cmp(right_bytes) {
663 Less => -1i32,
664 Equal => 0,
665 Greater => 1,
666 }
667 };
668
669 this.write_scalar(Scalar::from_i32(result), dest)?;
670 }
671 "memrchr" => {
672 let [ptr, val, num] =
673 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
674 let ptr = this.read_pointer(ptr)?;
675 let val = this.read_scalar(val)?.to_i32()?;
676 let num = this.read_target_usize(num)?;
677 #[expect(clippy::as_conversions)]
679 let val = val as u8;
680
681 this.ptr_get_alloc_id(ptr, 0)?;
683
684 if let Some(idx) = this
685 .read_bytes_ptr_strip_provenance(ptr, Size::from_bytes(num))?
686 .iter()
687 .rev()
688 .position(|&c| c == val)
689 {
690 let idx = u64::try_from(idx).unwrap();
691 #[expect(clippy::arithmetic_side_effects)] let new_ptr = ptr.wrapping_offset(Size::from_bytes(num - idx - 1), this);
693 this.write_pointer(new_ptr, dest)?;
694 } else {
695 this.write_null(dest)?;
696 }
697 }
698 "memchr" => {
699 let [ptr, val, num] =
700 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
701 let ptr = this.read_pointer(ptr)?;
702 let val = this.read_scalar(val)?.to_i32()?;
703 let num = this.read_target_usize(num)?;
704 #[expect(clippy::as_conversions)]
706 let val = val as u8;
707
708 this.ptr_get_alloc_id(ptr, 0)?;
710
711 let idx = this
712 .read_bytes_ptr_strip_provenance(ptr, Size::from_bytes(num))?
713 .iter()
714 .position(|&c| c == val);
715 if let Some(idx) = idx {
716 let new_ptr = ptr.wrapping_offset(Size::from_bytes(idx), this);
717 this.write_pointer(new_ptr, dest)?;
718 } else {
719 this.write_null(dest)?;
720 }
721 }
722 "strlen" => {
723 let [ptr] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
724 let ptr = this.read_pointer(ptr)?;
725 let n = this.read_c_str(ptr)?.len();
727 this.write_scalar(
728 Scalar::from_target_usize(u64::try_from(n).unwrap(), this),
729 dest,
730 )?;
731 }
732 "wcslen" => {
733 let [ptr] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
734 let ptr = this.read_pointer(ptr)?;
735 let n = this.read_wchar_t_str(ptr)?.len();
737 this.write_scalar(
738 Scalar::from_target_usize(u64::try_from(n).unwrap(), this),
739 dest,
740 )?;
741 }
742 "memcpy" => {
743 let [ptr_dest, ptr_src, n] =
744 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
745 let ptr_dest = this.read_pointer(ptr_dest)?;
746 let ptr_src = this.read_pointer(ptr_src)?;
747 let n = this.read_target_usize(n)?;
748
749 this.ptr_get_alloc_id(ptr_dest, 0)?;
752 this.ptr_get_alloc_id(ptr_src, 0)?;
753
754 this.mem_copy(ptr_src, ptr_dest, Size::from_bytes(n), true)?;
755 this.write_pointer(ptr_dest, dest)?;
756 }
757 "strcpy" => {
758 let [ptr_dest, ptr_src] =
759 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
760 let ptr_dest = this.read_pointer(ptr_dest)?;
761 let ptr_src = this.read_pointer(ptr_src)?;
762
763 let n = this.read_c_str(ptr_src)?.len().strict_add(1);
770 this.mem_copy(ptr_src, ptr_dest, Size::from_bytes(n), true)?;
771 this.write_pointer(ptr_dest, dest)?;
772 }
773 "memset" => {
774 let [ptr_dest, val, n] =
775 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
776 let ptr_dest = this.read_pointer(ptr_dest)?;
777 let val = this.read_scalar(val)?.to_i32()?;
778 let n = this.read_target_usize(n)?;
779 #[expect(clippy::as_conversions)]
781 let val = val as u8;
782
783 this.ptr_get_alloc_id(ptr_dest, 0)?;
785
786 let bytes = std::iter::repeat_n(val, n.try_into().unwrap());
787 this.write_bytes_ptr(ptr_dest, bytes)?;
788 this.write_pointer(ptr_dest, dest)?;
789 }
790
791 "llvm.prefetch" => {
793 let [p, rw, loc, ty] =
794 this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
795
796 let _ = this.read_pointer(p)?;
797 let rw = this.read_scalar(rw)?.to_i32()?;
798 let loc = this.read_scalar(loc)?.to_i32()?;
799 let ty = this.read_scalar(ty)?.to_i32()?;
800
801 if ty == 1 {
802 if !matches!(rw, 0 | 1) {
806 throw_unsup_format!("invalid `rw` value passed to `llvm.prefetch`: {}", rw);
807 }
808 if !matches!(loc, 0..=3) {
809 throw_unsup_format!(
810 "invalid `loc` value passed to `llvm.prefetch`: {}",
811 loc
812 );
813 }
814 } else {
815 throw_unsup_format!("unsupported `llvm.prefetch` type argument: {}", ty);
816 }
817 }
818 name if name.starts_with("llvm.ctpop.v") => {
821 let [op] = this.check_shim_sig_lenient(abi, CanonAbi::C, link_name, args)?;
822
823 let (op, op_len) = this.project_to_simd(op)?;
824 let (dest, dest_len) = this.project_to_simd(dest)?;
825
826 assert_eq!(dest_len, op_len);
827
828 for i in 0..dest_len {
829 let op = this.read_immediate(&this.project_index(&op, i)?)?;
830 let res = op.to_scalar().to_uint(op.layout.size)?.count_ones();
833
834 this.write_scalar(
835 Scalar::from_uint(res, op.layout.size),
836 &this.project_index(&dest, i)?,
837 )?;
838 }
839 }
840
841 name if name.starts_with("llvm.x86.")
843 && matches!(this.tcx.sess.target.arch, Arch::X86 | Arch::X86_64) =>
844 {
845 return shims::x86::EvalContextExt::emulate_x86_intrinsic(
846 this, link_name, abi, args, dest,
847 );
848 }
849 name if name.starts_with("llvm.aarch64.")
850 && this.tcx.sess.target.arch == Arch::AArch64 =>
851 {
852 return shims::aarch64::EvalContextExt::emulate_aarch64_intrinsic(
853 this, link_name, abi, args, dest,
854 );
855 }
856
857 _ => {
859 if let res = shims::math::EvalContextExt::emulate_foreign_item_inner(
861 this, link_name, abi, args, dest,
862 )? && !matches!(res, EmulateItemResult::NotSupported)
863 {
864 return interp_ok(res);
865 }
866
867 return match &this.tcx.sess.target.os {
869 _ if this.target_os_is_unix() =>
870 shims::unix::foreign_items::EvalContextExt::emulate_foreign_item_inner(
871 this, link_name, abi, args, dest,
872 ),
873 Os::Windows =>
874 shims::windows::foreign_items::EvalContextExt::emulate_foreign_item_inner(
875 this, link_name, abi, args, dest,
876 ),
877 _ => interp_ok(EmulateItemResult::NotSupported),
878 };
879 }
880 };
881 interp_ok(EmulateItemResult::NeedsReturn)
884 }
885}