4 impl<'mir, 'tcx> EvalContextExt<'mir, 'tcx> for crate::MiriEvalContext<'mir, 'tcx> {}
5 pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriEvalContextExt<'mir, 'tcx> {
6 fn emulate_foreign_item_by_name(
9 args: &[OpTy<'tcx, Tag>],
10 dest: PlaceTy<'tcx, Tag>,
11 _ret: mir::BasicBlock,
12 ) -> InterpResult<'tcx, bool> {
13 let this = self.eval_context_mut();
16 "__errno_location" => {
17 let errno_place = this.machine.last_error.unwrap();
18 this.write_scalar(errno_place.to_ref().to_scalar()?, dest)?;
23 let result = this.close(args[0])?;
24 this.write_scalar(Scalar::from_int(result, dest.layout.size), dest)?;
29 let result = this.clock_gettime(args[0], args[1])?;
30 this.write_scalar(Scalar::from_int(result, dest.layout.size), dest)?;
34 "pthread_getattr_np" => {
35 this.write_null(dest)?;
39 let sys_getrandom = this
40 .eval_path_scalar(&["libc", "SYS_getrandom"])?
41 .expect("Failed to get libc::SYS_getrandom")
42 .to_machine_usize(this)?;
45 .eval_path_scalar(&["libc", "SYS_statx"])?
46 .expect("Failed to get libc::SYS_statx")
47 .to_machine_usize(this)?;
49 match this.read_scalar(args[0])?.to_machine_usize(this)? {
50 // `libc::syscall(NR_GETRANDOM, buf.as_mut_ptr(), buf.len(), GRND_NONBLOCK)`
51 // is called if a `HashMap` is created the regular way (e.g. HashMap<K, V>).
52 id if id == sys_getrandom => {
53 // The first argument is the syscall id,
55 getrandom(this, &args[1..], dest)?;
57 id if id == sys_statx => {
58 // The first argument is the syscall id,
60 let result = this.linux_statx(args[1], args[2], args[3], args[4], args[5])?;
61 this.write_scalar(Scalar::from_int(result, dest.layout.size), dest)?;
63 id => throw_unsup_format!("miri does not support syscall ID {}", id),
68 getrandom(this, args, dest)?;
71 "sched_getaffinity" => {
72 // Return an error; `num_cpus` then falls back to `sysconf`.
73 this.write_scalar(Scalar::from_int(-1, dest.layout.size), dest)?;
76 _ => throw_unsup_format!("can't call foreign function: {}", link_name),
83 // Shims the posix 'getrandom()' syscall.
85 this: &mut MiriEvalContext<'_, 'tcx>,
86 args: &[OpTy<'tcx, Tag>],
87 dest: PlaceTy<'tcx, Tag>,
88 ) -> InterpResult<'tcx> {
89 let ptr = this.read_scalar(args[0])?.not_undef()?;
90 let len = this.read_scalar(args[1])?.to_machine_usize(this)?;
92 // The only supported flags are GRND_RANDOM and GRND_NONBLOCK,
93 // neither of which have any effect on our current PRNG.
94 let _flags = this.read_scalar(args[2])?.to_i32()?;
96 this.gen_random(ptr, len as usize)?;
97 this.write_scalar(Scalar::from_uint(len, dest.layout.size), dest)?;