From 824328c6d5d76d3db86045da7ca606789696d4f9 Mon Sep 17 00:00:00 2001 From: Ralf Jung Date: Mon, 25 Nov 2019 22:48:31 +0100 Subject: [PATCH 1/3] adjust for goto_block refactoring --- src/machine.rs | 16 +++++++--------- src/shims/dlsym.rs | 9 +++------ src/shims/foreign_items.rs | 17 ++++++++--------- src/shims/intrinsics.rs | 27 +++++++++++++-------------- src/shims/mod.rs | 11 +++++------ src/shims/panic.rs | 5 +---- 6 files changed, 37 insertions(+), 48 deletions(-) diff --git a/src/machine.rs b/src/machine.rs index cb658170357..c4053291419 100644 --- a/src/machine.rs +++ b/src/machine.rs @@ -186,11 +186,10 @@ fn find_fn( ecx: &mut InterpCx<'mir, 'tcx, Self>, instance: ty::Instance<'tcx>, args: &[OpTy<'tcx, Tag>], - dest: Option>, - ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, unwind: Option, ) -> InterpResult<'tcx, Option<&'mir mir::Body<'tcx>>> { - ecx.find_fn(instance, args, dest, ret, unwind) + ecx.find_fn(instance, args, ret, unwind) } #[inline(always)] @@ -198,10 +197,10 @@ fn call_extra_fn( ecx: &mut InterpCx<'mir, 'tcx, Self>, fn_val: Dlsym, args: &[OpTy<'tcx, Tag>], - dest: Option>, - ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, + _unwind: Option, ) -> InterpResult<'tcx> { - ecx.call_dlsym(fn_val, args, dest, ret) + ecx.call_dlsym(fn_val, args, ret) } #[inline(always)] @@ -210,11 +209,10 @@ fn call_intrinsic( span: Span, instance: ty::Instance<'tcx>, args: &[OpTy<'tcx, Tag>], - dest: Option>, - ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, unwind: Option, ) -> InterpResult<'tcx> { - ecx.call_intrinsic(span, instance, args, dest, ret, unwind) + ecx.call_intrinsic(span, instance, args, ret, unwind) } #[inline(always)] diff --git a/src/shims/dlsym.rs b/src/shims/dlsym.rs index ca53f5d2301..dfee4066da1 100644 --- a/src/shims/dlsym.rs +++ b/src/shims/dlsym.rs @@ -27,15 +27,12 @@ fn call_dlsym( &mut self, dlsym: Dlsym, args: &[OpTy<'tcx, Tag>], - dest: Option>, - ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, ) -> InterpResult<'tcx> { use self::Dlsym::*; let this = self.eval_context_mut(); - - let dest = dest.expect("we don't support any diverging dlsym"); - let ret = ret.expect("dest is `Some` but ret is `None`"); + let (dest, ret) = ret.expect("we don't support any diverging dlsym"); match dlsym { GetEntropy => { @@ -46,8 +43,8 @@ fn call_dlsym( } } - this.goto_block(Some(ret))?; this.dump_place(*dest); + this.go_to_block(ret); Ok(()) } } diff --git a/src/shims/foreign_items.rs b/src/shims/foreign_items.rs index 5edabf21566..b136260f751 100644 --- a/src/shims/foreign_items.rs +++ b/src/shims/foreign_items.rs @@ -114,8 +114,7 @@ fn emulate_foreign_item( &mut self, def_id: DefId, args: &[OpTy<'tcx, Tag>], - dest: Option>, - ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, _unwind: Option ) -> InterpResult<'tcx, Option<&'mir mir::Body<'tcx>>> { let this = self.eval_context_mut(); @@ -129,7 +128,7 @@ fn emulate_foreign_item( let tcx = &{ this.tcx.tcx }; // First: functions that diverge. - match link_name { + let (dest, ret) = match link_name { // Note that this matches calls to the *foreign* item `__rust_start_panic* - // that is, calls to `extern "Rust" { fn __rust_start_panic(...) }`. // We forward this to the underlying *implementation* in the panic runtime crate. @@ -154,15 +153,15 @@ fn emulate_foreign_item( return Err(InterpError::Exit(code).into()); } _ => { - if dest.is_none() { + if let Some(p) = ret { + p + } else { throw_unsup_format!("can't call (diverging) foreign function: {}", link_name); } } - } + }; - // Next: functions that assume a ret and dest. - let dest = dest.expect("we already checked for a dest"); - let ret = ret.expect("dest is `Some` but ret is `None`"); + // Next: functions that return. match link_name { "malloc" => { let size = this.read_scalar(args[0])?.to_machine_usize(this)?; @@ -928,8 +927,8 @@ fn emulate_foreign_item( _ => throw_unsup_format!("can't call foreign function: {}", link_name), } - this.goto_block(Some(ret))?; this.dump_place(*dest); + this.go_to_block(ret); Ok(None) } diff --git a/src/shims/intrinsics.rs b/src/shims/intrinsics.rs index 5e1f3cff1c7..00886328030 100644 --- a/src/shims/intrinsics.rs +++ b/src/shims/intrinsics.rs @@ -16,12 +16,11 @@ fn call_intrinsic( span: Span, instance: ty::Instance<'tcx>, args: &[OpTy<'tcx, Tag>], - dest: Option>, - _ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, unwind: Option ) -> InterpResult<'tcx> { let this = self.eval_context_mut(); - if this.emulate_intrinsic(span, instance, args, dest)? { + if this.emulate_intrinsic(span, instance, args, ret)? { return Ok(()); } let tcx = &{this.tcx.tcx}; @@ -32,23 +31,21 @@ fn call_intrinsic( // that might still hang around! let intrinsic_name = &*tcx.item_name(instance.def_id()).as_str(); - // Handle diverging intrinsics - match intrinsic_name { + // Handle diverging intrinsics. + let (dest, ret) = match intrinsic_name { "abort" => { // FIXME: Add a better way of indicating 'abnormal' termination, // since this is not really an 'unsupported' behavior throw_unsup_format!("the evaluated program aborted!"); } "miri_start_panic" => return this.handle_miri_start_panic(args, unwind), - _ => {} - } - - // Handle non-diverging intrinsics - // The intrinsic itself cannot diverge (otherwise, we would have handled it above), - // so if we got here without a return place that's UB (can happen e.g., for transmute returning `!`). - let dest = match dest { - Some(dest) => dest, - None => throw_ub!(Unreachable) + _ => { + if let Some(p) = ret { + p + } else { + throw_unsup_format!("unimplemented (diverging) intrinsic: {}", intrinsic_name); + } + } }; match intrinsic_name { @@ -581,6 +578,8 @@ fn call_intrinsic( name => throw_unsup_format!("unimplemented intrinsic: {}", name), } + this.dump_place(*dest); + this.go_to_block(ret); Ok(()) } } diff --git a/src/shims/mod.rs b/src/shims/mod.rs index f554c19f11c..2e8aea3f86e 100644 --- a/src/shims/mod.rs +++ b/src/shims/mod.rs @@ -16,25 +16,24 @@ fn find_fn( &mut self, instance: ty::Instance<'tcx>, args: &[OpTy<'tcx, Tag>], - dest: Option>, - ret: Option, + ret: Option<(PlaceTy<'tcx, Tag>, mir::BasicBlock)>, unwind: Option ) -> InterpResult<'tcx, Option<&'mir mir::Body<'tcx>>> { let this = self.eval_context_mut(); trace!( "eval_fn_call: {:#?}, {:?}", instance, - dest.map(|place| *place) + ret.map(|p| *p.0) ); // There are some more lang items we want to hook that CTFE does not hook (yet). if this.tcx.lang_items().align_offset_fn() == Some(instance.def.def_id()) { - let dest = dest.unwrap(); + let (dest, ret) = ret.unwrap(); let n = this .align_offset(args[0], args[1])? .unwrap_or_else(|| this.truncate(u128::max_value(), dest.layout)); this.write_scalar(Scalar::from_uint(n, dest.layout.size), dest)?; - this.goto_block(ret)?; + this.go_to_block(ret); return Ok(None); } @@ -46,7 +45,7 @@ fn find_fn( // to run extra MIR), and Ok(Some(body)) if we found MIR to run for the // foreign function // Any needed call to `goto_block` will be performed by `emulate_foreign_item`. - return this.emulate_foreign_item(instance.def_id(), args, dest, ret, unwind); + return this.emulate_foreign_item(instance.def_id(), args, ret, unwind); } // Otherwise, load the MIR. diff --git a/src/shims/panic.rs b/src/shims/panic.rs index 59ff1a870d1..42028738822 100644 --- a/src/shims/panic.rs +++ b/src/shims/panic.rs @@ -53,10 +53,7 @@ fn handle_miri_start_panic( this.machine.panic_payload = Some(scalar); // Jump to the unwind block to begin unwinding. - // We don't use `goto_block` as that is just meant for normal returns. - let next_frame = this.frame_mut(); - next_frame.block = unwind; - next_frame.stmt = 0; + this.unwind_to_block(unwind); return Ok(()) } From fbc1fb20ac606dfcb6f91fb7823272760e583f85 Mon Sep 17 00:00:00 2001 From: Ralf Jung Date: Wed, 27 Nov 2019 08:45:24 +0100 Subject: [PATCH 2/3] bump Rust version --- rust-version | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/rust-version b/rust-version index 09cac5bb059..b5ab3053c5c 100644 --- a/rust-version +++ b/rust-version @@ -1 +1 @@ -bd816fd76f4f7a040ca7ac8ca5bc556d761f96fa +809e180a76ce97340bf4354ff357bc59e3ca40b2 From a04620eb697eb064902d5c1b5f7584fa68a1bcfd Mon Sep 17 00:00:00 2001 From: Ralf Jung Date: Wed, 27 Nov 2019 09:13:37 +0100 Subject: [PATCH 3/3] only run leak test on clean platforms --- src/eval.rs | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/src/eval.rs b/src/eval.rs index 237ef99d1fe..3524029475a 100644 --- a/src/eval.rs +++ b/src/eval.rs @@ -201,16 +201,18 @@ pub fn eval_main<'tcx>(tcx: TyCtxt<'tcx>, main_id: DefId, config: MiriConfig) -> // Process the result. match res { Ok(return_code) => { - let leaks = ecx.memory.leak_report(); // Disable the leak test on some platforms where we do not // correctly implement TLS destructors. let target_os = ecx.tcx.tcx.sess.target.target.target_os.to_lowercase(); let ignore_leaks = target_os == "windows" || target_os == "macos"; - if !ignore_leaks && leaks != 0 { - tcx.sess.err("the evaluated program leaked memory"); - // Ignore the provided return code - let the reported error - // determine the return code. - return None; + if !ignore_leaks { + let leaks = ecx.memory.leak_report(); + if leaks != 0 { + tcx.sess.err("the evaluated program leaked memory"); + // Ignore the provided return code - let the reported error + // determine the return code. + return None; + } } return Some(return_code) }