Rollup merge of #23074 - michaelwoerister:constants-debug-locs, r=alexcrichton
With this PR in-place constants are handled correctly with respect to debug location assignment. The PR also adds an (unrelated) test case for debug locations in `extern \"C\"` functions. Fixes #22432
This commit is contained in:
commit
d77fc9fefc
5 changed files with 181 additions and 3 deletions
|
@ -173,7 +173,7 @@ pub fn get_const_expr<'a, 'tcx>(ccx: &CrateContext<'a, 'tcx>,
|
||||||
&**expr
|
&**expr
|
||||||
} else {
|
} else {
|
||||||
ccx.sess().span_bug(ref_expr.span,
|
ccx.sess().span_bug(ref_expr.span,
|
||||||
&format!("get_const_val given non-constant item {}",
|
&format!("get_const_expr given non-constant item {}",
|
||||||
item.repr(ccx.tcx())));
|
item.repr(ccx.tcx())));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -697,6 +697,7 @@ struct FunctionDebugContextData {
|
||||||
fn_metadata: DISubprogram,
|
fn_metadata: DISubprogram,
|
||||||
argument_counter: Cell<uint>,
|
argument_counter: Cell<uint>,
|
||||||
source_locations_enabled: Cell<bool>,
|
source_locations_enabled: Cell<bool>,
|
||||||
|
source_location_override: Cell<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
enum VariableAccess<'a> {
|
enum VariableAccess<'a> {
|
||||||
|
@ -1176,6 +1177,12 @@ pub fn set_source_location(fcx: &FunctionContext,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
FunctionDebugContext::RegularContext(box ref function_debug_context) => {
|
FunctionDebugContext::RegularContext(box ref function_debug_context) => {
|
||||||
|
if function_debug_context.source_location_override.get() {
|
||||||
|
// Just ignore any attempts to set a new debug location while
|
||||||
|
// the override is active.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let cx = fcx.ccx;
|
let cx = fcx.ccx;
|
||||||
|
|
||||||
debug!("set_source_location: {}", cx.sess().codemap().span_to_string(span));
|
debug!("set_source_location: {}", cx.sess().codemap().span_to_string(span));
|
||||||
|
@ -1194,6 +1201,35 @@ pub fn set_source_location(fcx: &FunctionContext,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// This function makes sure that all debug locations emitted while executing
|
||||||
|
/// `wrapped_function` are set to the given `debug_loc`.
|
||||||
|
pub fn with_source_location_override<F, R>(fcx: &FunctionContext,
|
||||||
|
debug_loc: DebugLoc,
|
||||||
|
wrapped_function: F) -> R
|
||||||
|
where F: FnOnce() -> R
|
||||||
|
{
|
||||||
|
match fcx.debug_context {
|
||||||
|
FunctionDebugContext::DebugInfoDisabled => {
|
||||||
|
wrapped_function()
|
||||||
|
}
|
||||||
|
FunctionDebugContext::FunctionWithoutDebugInfo => {
|
||||||
|
set_debug_location(fcx.ccx, UnknownLocation);
|
||||||
|
wrapped_function()
|
||||||
|
}
|
||||||
|
FunctionDebugContext::RegularContext(box ref function_debug_context) => {
|
||||||
|
if function_debug_context.source_location_override.get() {
|
||||||
|
wrapped_function()
|
||||||
|
} else {
|
||||||
|
debug_loc.apply(fcx);
|
||||||
|
function_debug_context.source_location_override.set(true);
|
||||||
|
let result = wrapped_function();
|
||||||
|
function_debug_context.source_location_override.set(false);
|
||||||
|
result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Clears the current debug location.
|
/// Clears the current debug location.
|
||||||
///
|
///
|
||||||
/// Instructions generated hereafter won't be assigned a source location.
|
/// Instructions generated hereafter won't be assigned a source location.
|
||||||
|
@ -1414,6 +1450,7 @@ pub fn create_function_debug_context<'a, 'tcx>(cx: &CrateContext<'a, 'tcx>,
|
||||||
fn_metadata: fn_metadata,
|
fn_metadata: fn_metadata,
|
||||||
argument_counter: Cell::new(1),
|
argument_counter: Cell::new(1),
|
||||||
source_locations_enabled: Cell::new(false),
|
source_locations_enabled: Cell::new(false),
|
||||||
|
source_location_override: Cell::new(false),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -147,7 +147,7 @@ pub fn trans_into<'blk, 'tcx>(bcx: Block<'blk, 'tcx>,
|
||||||
ast::ExprPath(..) => {
|
ast::ExprPath(..) => {
|
||||||
match bcx.def(expr.id) {
|
match bcx.def(expr.id) {
|
||||||
def::DefConst(did) => {
|
def::DefConst(did) => {
|
||||||
let expr = consts::get_const_expr(bcx.ccx(), did, expr);
|
let const_expr = consts::get_const_expr(bcx.ccx(), did, expr);
|
||||||
// Temporarily get cleanup scopes out of the way,
|
// Temporarily get cleanup scopes out of the way,
|
||||||
// as they require sub-expressions to be contained
|
// as they require sub-expressions to be contained
|
||||||
// inside the current AST scope.
|
// inside the current AST scope.
|
||||||
|
@ -155,7 +155,13 @@ pub fn trans_into<'blk, 'tcx>(bcx: Block<'blk, 'tcx>,
|
||||||
// can't have destructors.
|
// can't have destructors.
|
||||||
let scopes = mem::replace(&mut *bcx.fcx.scopes.borrow_mut(),
|
let scopes = mem::replace(&mut *bcx.fcx.scopes.borrow_mut(),
|
||||||
vec![]);
|
vec![]);
|
||||||
bcx = trans_into(bcx, expr, dest);
|
// Lock emitted debug locations to the location of
|
||||||
|
// the constant reference expression.
|
||||||
|
debuginfo::with_source_location_override(bcx.fcx,
|
||||||
|
expr.debug_loc(),
|
||||||
|
|| {
|
||||||
|
bcx = trans_into(bcx, const_expr, dest)
|
||||||
|
});
|
||||||
let scopes = mem::replace(&mut *bcx.fcx.scopes.borrow_mut(),
|
let scopes = mem::replace(&mut *bcx.fcx.scopes.borrow_mut(),
|
||||||
scopes);
|
scopes);
|
||||||
assert!(scopes.is_empty());
|
assert!(scopes.is_empty());
|
||||||
|
|
67
src/test/debuginfo/constant-debug-locs.rs
Normal file
67
src/test/debuginfo/constant-debug-locs.rs
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
// Copyright 2013-2015 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 <LICENSE-APACHE or
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||||
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||||
|
// option. This file may not be copied, modified, or distributed
|
||||||
|
// except according to those terms.
|
||||||
|
|
||||||
|
// ignore-android: FIXME(#10381)
|
||||||
|
// min-lldb-version: 310
|
||||||
|
|
||||||
|
// compile-flags:-g
|
||||||
|
|
||||||
|
#![allow(unused_variables)]
|
||||||
|
#![allow(dead_code)]
|
||||||
|
#![omit_gdb_pretty_printer_section]
|
||||||
|
|
||||||
|
// This test makes sure that the compiler doesn't crash when trying to assign
|
||||||
|
// debug locations to const-expressions.
|
||||||
|
|
||||||
|
use std::sync::MUTEX_INIT;
|
||||||
|
use std::cell::UnsafeCell;
|
||||||
|
|
||||||
|
const CONSTANT: u64 = 3 + 4;
|
||||||
|
|
||||||
|
struct Struct {
|
||||||
|
a: isize,
|
||||||
|
b: usize,
|
||||||
|
}
|
||||||
|
const STRUCT: Struct = Struct { a: 1, b: 2 };
|
||||||
|
|
||||||
|
struct TupleStruct(u32);
|
||||||
|
const TUPLE_STRUCT: TupleStruct = TupleStruct(4);
|
||||||
|
|
||||||
|
enum Enum {
|
||||||
|
Variant1(char),
|
||||||
|
Variant2 { a: u8 },
|
||||||
|
Variant3
|
||||||
|
}
|
||||||
|
|
||||||
|
const VARIANT1: Enum = Enum::Variant1('v');
|
||||||
|
const VARIANT2: Enum = Enum::Variant2 { a: 2 };
|
||||||
|
const VARIANT3: Enum = Enum::Variant3;
|
||||||
|
|
||||||
|
const STRING: &'static str = "String";
|
||||||
|
|
||||||
|
const VEC: [u32; 8] = [0; 8];
|
||||||
|
|
||||||
|
const NESTED: (Struct, TupleStruct) = (STRUCT, TUPLE_STRUCT);
|
||||||
|
|
||||||
|
const UNSAFE_CELL: UnsafeCell<bool> = UnsafeCell { value: false };
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let mut _constant = CONSTANT;
|
||||||
|
let mut _struct = STRUCT;
|
||||||
|
let mut _tuple_struct = TUPLE_STRUCT;
|
||||||
|
let mut _variant1 = VARIANT1;
|
||||||
|
let mut _variant2 = VARIANT2;
|
||||||
|
let mut _variant3 = VARIANT3;
|
||||||
|
let mut _string = STRING;
|
||||||
|
let mut _vec = VEC;
|
||||||
|
let mut _nested = NESTED;
|
||||||
|
let mut _extern = MUTEX_INIT;
|
||||||
|
let mut _unsafe_cell = UNSAFE_CELL;
|
||||||
|
}
|
68
src/test/debuginfo/extern-c-fn.rs
Normal file
68
src/test/debuginfo/extern-c-fn.rs
Normal file
|
@ -0,0 +1,68 @@
|
||||||
|
// Copyright 2013-2014 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 <LICENSE-APACHE or
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||||
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||||
|
// option. This file may not be copied, modified, or distributed
|
||||||
|
// except according to those terms.
|
||||||
|
|
||||||
|
// min-lldb-version: 310
|
||||||
|
|
||||||
|
// compile-flags:-g
|
||||||
|
|
||||||
|
// === GDB TESTS ===================================================================================
|
||||||
|
// gdb-command:run
|
||||||
|
|
||||||
|
// gdb-command:print s
|
||||||
|
// gdb-check:$1 = [...]"abcd"
|
||||||
|
// gdb-command:print len
|
||||||
|
// gdb-check:$2 = 20
|
||||||
|
// gdb-command:print local0
|
||||||
|
// gdb-check:$3 = 19
|
||||||
|
// gdb-command:print local1
|
||||||
|
// gdb-check:$4 = true
|
||||||
|
// gdb-command:print local2
|
||||||
|
// gdb-check:$5 = 20.5
|
||||||
|
|
||||||
|
// gdb-command:continue
|
||||||
|
|
||||||
|
// === LLDB TESTS ==================================================================================
|
||||||
|
// lldb-command:run
|
||||||
|
|
||||||
|
// lldb-command:print len
|
||||||
|
// lldb-check:[...]$0 = 20
|
||||||
|
// lldb-command:print local0
|
||||||
|
// lldb-check:[...]$1 = 19
|
||||||
|
// lldb-command:print local1
|
||||||
|
// lldb-check:[...]$2 = true
|
||||||
|
// lldb-command:print local2
|
||||||
|
// lldb-check:[...]$3 = 20.5
|
||||||
|
|
||||||
|
// lldb-command:continue
|
||||||
|
|
||||||
|
#![allow(unused_variables)]
|
||||||
|
#![allow(dead_code)]
|
||||||
|
#![omit_gdb_pretty_printer_section]
|
||||||
|
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern "C" fn fn_with_c_abi(s: *const u8, len: i32) -> i32 {
|
||||||
|
let local0 = len - 1;
|
||||||
|
let local1 = len > 2;
|
||||||
|
let local2 = (len as f64) + 0.5;
|
||||||
|
|
||||||
|
zzz(); // #break
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
unsafe {
|
||||||
|
fn_with_c_abi(b"abcd\0".as_ptr(), 20);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline(never)]
|
||||||
|
fn zzz() {()}
|
Loading…
Add table
Add a link
Reference in a new issue