From 36ccda92a611a9a6c30793098c3a1580cbcfc2cd Mon Sep 17 00:00:00 2001 From: Oliver Schneider Date: Sun, 22 Jul 2018 22:09:05 +0200 Subject: [PATCH] Abort instead of UB if promotion fails --- src/librustc_codegen_llvm/mir/operand.rs | 11 ++++-- src/test/run-pass/invalid_const_promotion.rs | 38 ++++++++++++++++++++ 2 files changed, 46 insertions(+), 3 deletions(-) create mode 100644 src/test/run-pass/invalid_const_promotion.rs diff --git a/src/librustc_codegen_llvm/mir/operand.rs b/src/librustc_codegen_llvm/mir/operand.rs index 9f32b41cb13..0c5e2b5d88e 100644 --- a/src/librustc_codegen_llvm/mir/operand.rs +++ b/src/librustc_codegen_llvm/mir/operand.rs @@ -17,7 +17,7 @@ use rustc::ty::layout::{self, Align, LayoutOf, TyLayout}; use rustc_data_structures::indexed_vec::Idx; use base; -use common::{self, CodegenCx, C_null, C_undef, C_usize}; +use common::{self, CodegenCx, C_undef, C_usize}; use builder::{Builder, MemFlags}; use value::Value; use type_of::LayoutLlvmExt; @@ -413,7 +413,10 @@ impl<'a, 'tcx> FunctionCx<'a, 'tcx> { .unwrap_or_else(|err| { match constant.literal { mir::Literal::Promoted { .. } => { - // FIXME: generate a panic here + // this is unreachable as long as runtime + // and compile-time agree on values + // With floats that won't always be true + // so we generate an abort below }, mir::Literal::Value { .. } => { err.report_as_error( @@ -422,10 +425,12 @@ impl<'a, 'tcx> FunctionCx<'a, 'tcx> { ); }, } + let fnname = bx.cx.get_intrinsic(&("llvm.trap")); + bx.call(fnname, &[], None); // We've errored, so we don't have to produce working code. let layout = bx.cx.layout_of(ty); PlaceRef::new_sized( - C_null(layout.llvm_type(bx.cx).ptr_to()), + C_undef(layout.llvm_type(bx.cx).ptr_to()), layout, layout.align, ).load(bx) diff --git a/src/test/run-pass/invalid_const_promotion.rs b/src/test/run-pass/invalid_const_promotion.rs new file mode 100644 index 00000000000..29a4b921992 --- /dev/null +++ b/src/test/run-pass/invalid_const_promotion.rs @@ -0,0 +1,38 @@ +// Copyright 2018 The Rust Project Developers. See the COPYRIGHT +// file at the top-level directory of this distribution and at +// http://rust-lang.org/COPYRIGHT. +// +// Licensed under the Apache License, Version 2.0 or the MIT license +// , at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +// ignore-wasm32 +// ignore-emscripten + +#![feature(const_fn)] +#![allow(const_err)] + +use std::env; +use std::process::{Command, Stdio}; + +const fn bar() -> usize { 0 - 1 } + +fn foo() { + let _: &'static _ = &bar(); +} + +fn main() { + let args: Vec = env::args().collect(); + if args.len() > 1 && args[1] == "test" { + foo(); + return; + } + + let mut p = Command::new(&args[0]) + .stdout(Stdio::piped()) + .stdin(Stdio::piped()) + .arg("test").output().unwrap(); + assert!(!p.status.success()); +}