2014-01-17 10:18:39 -05:00
|
|
|
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
2014-01-15 14:39:08 -05:00
|
|
|
// 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.
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
//! Code pertaining to cleanup of temporaries as well as execution of
|
|
|
|
//! drop glue. See discussion in `doc.rs` for a high-level summary.
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-11-06 00:05:53 -08:00
|
|
|
pub use self::ScopeId::*;
|
|
|
|
pub use self::CleanupScopeKind::*;
|
|
|
|
pub use self::EarlyExitLabel::*;
|
|
|
|
pub use self::Heap::*;
|
|
|
|
|
2014-07-07 17:58:01 -07:00
|
|
|
use llvm::{BasicBlockRef, ValueRef};
|
2014-11-15 20:30:33 -05:00
|
|
|
use trans::base;
|
|
|
|
use trans::build;
|
|
|
|
use trans::callee;
|
|
|
|
use trans::common;
|
|
|
|
use trans::common::{Block, FunctionContext, ExprId, NodeInfo};
|
|
|
|
use trans::debuginfo;
|
|
|
|
use trans::glue;
|
2014-11-18 14:22:59 +01:00
|
|
|
use middle::region;
|
2014-11-15 20:30:33 -05:00
|
|
|
use trans::type_::Type;
|
2014-09-13 21:09:25 +03:00
|
|
|
use middle::ty::{mod, Ty};
|
2014-10-15 02:25:34 -04:00
|
|
|
use std::fmt;
|
2014-01-15 14:39:08 -05:00
|
|
|
use syntax::ast;
|
|
|
|
use util::ppaux::Repr;
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
pub struct CleanupScope<'blk, 'tcx: 'blk> {
|
2014-01-15 14:39:08 -05:00
|
|
|
// The id of this cleanup scope. If the id is None,
|
|
|
|
// this is a *temporary scope* that is pushed during trans to
|
|
|
|
// cleanup miscellaneous garbage that trans may generate whose
|
|
|
|
// lifetime is a subset of some expression. See module doc for
|
|
|
|
// more details.
|
2014-09-06 19:13:04 +03:00
|
|
|
kind: CleanupScopeKind<'blk, 'tcx>,
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
// Cleanups to run upon scope exit.
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanups: Vec<CleanupObj<'tcx>>,
|
2014-01-15 14:39:08 -05:00
|
|
|
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
// The debug location any drop calls generated for this scope will be
|
|
|
|
// associated with.
|
|
|
|
debug_loc: Option<NodeInfo>,
|
|
|
|
|
2014-03-19 23:16:56 +11:00
|
|
|
cached_early_exits: Vec<CachedEarlyExit>,
|
2014-01-15 14:39:08 -05:00
|
|
|
cached_landing_pad: Option<BasicBlockRef>,
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy, Show)]
|
2014-01-15 14:39:08 -05:00
|
|
|
pub struct CustomScopeIndex {
|
2014-03-28 10:05:27 -07:00
|
|
|
index: uint
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-10-06 17:30:54 -07:00
|
|
|
pub const EXIT_BREAK: uint = 0;
|
|
|
|
pub const EXIT_LOOP: uint = 1;
|
|
|
|
pub const EXIT_MAX: uint = 2;
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
pub enum CleanupScopeKind<'blk, 'tcx: 'blk> {
|
2014-01-15 14:39:08 -05:00
|
|
|
CustomScopeKind,
|
|
|
|
AstScopeKind(ast::NodeId),
|
2014-09-06 19:13:04 +03:00
|
|
|
LoopScopeKind(ast::NodeId, [Block<'blk, 'tcx>, ..EXIT_MAX])
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
impl<'blk, 'tcx: 'blk> fmt::Show for CleanupScopeKind<'blk, 'tcx> {
|
2014-11-17 11:29:38 -08:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2014-10-15 02:25:34 -04:00
|
|
|
match *self {
|
|
|
|
CustomScopeKind => write!(f, "CustomScopeKind"),
|
|
|
|
AstScopeKind(nid) => write!(f, "AstScopeKind({})", nid),
|
|
|
|
LoopScopeKind(nid, ref blks) => {
|
|
|
|
try!(write!(f, "LoopScopeKind({}, [", nid));
|
|
|
|
for blk in blks.iter() {
|
|
|
|
try!(write!(f, "{:p}, ", blk));
|
|
|
|
}
|
|
|
|
write!(f, "])")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy, PartialEq, Show)]
|
2014-03-02 15:26:39 -08:00
|
|
|
pub enum EarlyExitLabel {
|
2014-01-15 14:39:08 -05:00
|
|
|
UnwindExit,
|
|
|
|
ReturnExit,
|
|
|
|
LoopExit(ast::NodeId, uint)
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy)]
|
2014-03-02 15:26:39 -08:00
|
|
|
pub struct CachedEarlyExit {
|
2014-01-15 14:39:08 -05:00
|
|
|
label: EarlyExitLabel,
|
|
|
|
cleanup_block: BasicBlockRef,
|
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
pub trait Cleanup<'tcx> {
|
2014-07-25 14:31:05 +02:00
|
|
|
fn must_unwind(&self) -> bool;
|
2014-01-15 14:39:08 -05:00
|
|
|
fn clean_on_unwind(&self) -> bool;
|
2014-10-13 16:12:38 +02:00
|
|
|
fn is_lifetime_end(&self) -> bool;
|
2014-09-29 22:11:30 +03:00
|
|
|
fn trans<'blk>(&self,
|
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
debug_loc: Option<NodeInfo>)
|
|
|
|
-> Block<'blk, 'tcx>;
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
pub type CleanupObj<'tcx> = Box<Cleanup<'tcx>+'tcx>;
|
2014-08-27 21:46:52 -04:00
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy, Show)]
|
2014-01-15 14:39:08 -05:00
|
|
|
pub enum ScopeId {
|
|
|
|
AstScope(ast::NodeId),
|
|
|
|
CustomScope(CustomScopeIndex)
|
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
impl<'blk, 'tcx> CleanupMethods<'blk, 'tcx> for FunctionContext<'blk, 'tcx> {
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Invoked when we start to trans the code contained within a new cleanup scope.
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
fn push_ast_cleanup_scope(&self, debug_loc: NodeInfo) {
|
2014-01-15 14:39:08 -05:00
|
|
|
debug!("push_ast_cleanup_scope({})",
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
self.ccx.tcx().map.node_to_string(debug_loc.id));
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
// FIXME(#2202) -- currently closure bodies have a parent
|
|
|
|
// region, which messes up the assertion below, since there
|
|
|
|
// are no cleanup scopes on the stack at the start of
|
|
|
|
// trans'ing a closure body. I think though that this should
|
|
|
|
// eventually be fixed by closure bodies not having a parent
|
|
|
|
// region, though that's a touch unclear, and it might also be
|
|
|
|
// better just to narrow this assertion more (i.e., by
|
|
|
|
// excluding id's that correspond to closure bodies only). For
|
|
|
|
// now we just say that if there is already an AST scope on the stack,
|
|
|
|
// this new AST scope had better be its immediate child.
|
|
|
|
let top_scope = self.top_ast_scope();
|
|
|
|
if top_scope.is_some() {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
assert_eq!(self.ccx
|
|
|
|
.tcx()
|
|
|
|
.region_maps
|
2014-11-18 14:22:59 +01:00
|
|
|
.opt_encl_scope(region::CodeExtent::from_node_id(debug_loc.id))
|
|
|
|
.map(|s|s.node_id()),
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
top_scope);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
self.push_scope(CleanupScope::new(AstScopeKind(debug_loc.id),
|
|
|
|
Some(debug_loc)));
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn push_loop_cleanup_scope(&self,
|
|
|
|
id: ast::NodeId,
|
2014-09-06 19:13:04 +03:00
|
|
|
exits: [Block<'blk, 'tcx>, ..EXIT_MAX]) {
|
2014-01-15 14:39:08 -05:00
|
|
|
debug!("push_loop_cleanup_scope({})",
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tcx().map.node_to_string(id));
|
2014-01-15 14:39:08 -05:00
|
|
|
assert_eq!(Some(id), self.top_ast_scope());
|
|
|
|
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
// Just copy the debuginfo source location from the enclosing scope
|
|
|
|
let debug_loc = self.scopes
|
|
|
|
.borrow()
|
|
|
|
.last()
|
|
|
|
.unwrap()
|
|
|
|
.debug_loc;
|
|
|
|
|
|
|
|
self.push_scope(CleanupScope::new(LoopScopeKind(id, exits), debug_loc));
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn push_custom_cleanup_scope(&self) -> CustomScopeIndex {
|
|
|
|
let index = self.scopes_len();
|
|
|
|
debug!("push_custom_cleanup_scope(): {}", index);
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
|
|
|
|
// Just copy the debuginfo source location from the enclosing scope
|
|
|
|
let debug_loc = self.scopes
|
|
|
|
.borrow()
|
|
|
|
.last()
|
|
|
|
.map(|opt_scope| opt_scope.debug_loc)
|
|
|
|
.unwrap_or(None);
|
|
|
|
|
|
|
|
self.push_scope(CleanupScope::new(CustomScopeKind, debug_loc));
|
|
|
|
CustomScopeIndex { index: index }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn push_custom_cleanup_scope_with_debug_loc(&self,
|
|
|
|
debug_loc: NodeInfo)
|
|
|
|
-> CustomScopeIndex {
|
|
|
|
let index = self.scopes_len();
|
|
|
|
debug!("push_custom_cleanup_scope(): {}", index);
|
|
|
|
|
|
|
|
self.push_scope(CleanupScope::new(CustomScopeKind, Some(debug_loc)));
|
2014-01-15 14:39:08 -05:00
|
|
|
CustomScopeIndex { index: index }
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Removes the cleanup scope for id `cleanup_scope`, which must be at the top of the cleanup
|
|
|
|
/// stack, and generates the code to do its cleanups for normal exit.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn pop_and_trans_ast_cleanup_scope(&self,
|
2014-09-06 19:13:04 +03:00
|
|
|
bcx: Block<'blk, 'tcx>,
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope: ast::NodeId)
|
2014-09-06 19:13:04 +03:00
|
|
|
-> Block<'blk, 'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
debug!("pop_and_trans_ast_cleanup_scope({})",
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tcx().map.node_to_string(cleanup_scope));
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
assert!(self.top_scope(|s| s.kind.is_ast_with_id(cleanup_scope)));
|
|
|
|
|
|
|
|
let scope = self.pop_scope();
|
|
|
|
self.trans_scope_cleanups(bcx, &scope)
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Removes the loop cleanup scope for id `cleanup_scope`, which must be at the top of the
|
|
|
|
/// cleanup stack. Does not generate any cleanup code, since loop scopes should exit by
|
|
|
|
/// branching to a block generated by `normal_exit_block`.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn pop_loop_cleanup_scope(&self,
|
|
|
|
cleanup_scope: ast::NodeId) {
|
|
|
|
debug!("pop_loop_cleanup_scope({})",
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tcx().map.node_to_string(cleanup_scope));
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
assert!(self.top_scope(|s| s.kind.is_loop_with_id(cleanup_scope)));
|
|
|
|
|
|
|
|
let _ = self.pop_scope();
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Removes the top cleanup scope from the stack without executing its cleanups. The top
|
|
|
|
/// cleanup scope must be the temporary scope `custom_scope`.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn pop_custom_cleanup_scope(&self,
|
|
|
|
custom_scope: CustomScopeIndex) {
|
|
|
|
debug!("pop_custom_cleanup_scope({})", custom_scope.index);
|
|
|
|
assert!(self.is_valid_to_pop_custom_scope(custom_scope));
|
|
|
|
let _ = self.pop_scope();
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Removes the top cleanup scope from the stack, which must be a temporary scope, and
|
|
|
|
/// generates the code to do its cleanups for normal exit.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn pop_and_trans_custom_cleanup_scope(&self,
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
custom_scope: CustomScopeIndex)
|
|
|
|
-> Block<'blk, 'tcx> {
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("pop_and_trans_custom_cleanup_scope({})", custom_scope);
|
2014-01-15 14:39:08 -05:00
|
|
|
assert!(self.is_valid_to_pop_custom_scope(custom_scope));
|
|
|
|
|
|
|
|
let scope = self.pop_scope();
|
|
|
|
self.trans_scope_cleanups(bcx, &scope)
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns the id of the top-most loop scope
|
2014-01-15 14:39:08 -05:00
|
|
|
fn top_loop_scope(&self) -> ast::NodeId {
|
2014-03-20 19:49:20 -07:00
|
|
|
for scope in self.scopes.borrow().iter().rev() {
|
2014-11-29 16:41:21 -05:00
|
|
|
if let LoopScopeKind(id, _) = scope.kind {
|
|
|
|
return id;
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
2014-03-15 22:29:34 +02:00
|
|
|
self.ccx.sess().bug("no loop scope found");
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns a block to branch to which will perform all pending cleanups and then
|
|
|
|
/// break/continue (depending on `exit`) out of the loop with id `cleanup_scope`
|
2014-09-06 19:13:04 +03:00
|
|
|
fn normal_exit_block(&'blk self,
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope: ast::NodeId,
|
|
|
|
exit: uint) -> BasicBlockRef {
|
|
|
|
self.trans_cleanups_to_exit_scope(LoopExit(cleanup_scope, exit))
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns a block to branch to which will perform all pending cleanups and then return from
|
|
|
|
/// this function
|
2014-09-06 19:13:04 +03:00
|
|
|
fn return_exit_block(&'blk self) -> BasicBlockRef {
|
2014-01-15 14:39:08 -05:00
|
|
|
self.trans_cleanups_to_exit_scope(ReturnExit)
|
|
|
|
}
|
|
|
|
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
fn schedule_lifetime_end(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef) {
|
|
|
|
let drop = box LifetimeEnd {
|
|
|
|
ptr: val,
|
|
|
|
};
|
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_lifetime_end({}, val={})",
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
cleanup_scope,
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tn().val_to_string(val));
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
self.schedule_clean(cleanup_scope, drop as CleanupObj);
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a (deep) drop of `val`, which is a pointer to an instance of `ty`
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_drop_mem(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>) {
|
2014-03-15 22:29:34 +02:00
|
|
|
if !ty::type_needs_drop(self.ccx.tcx(), ty) { return; }
|
2014-04-25 01:08:02 -07:00
|
|
|
let drop = box DropValue {
|
2014-01-15 14:39:08 -05:00
|
|
|
is_immediate: false,
|
2014-07-25 14:31:05 +02:00
|
|
|
must_unwind: ty::type_needs_unwind_cleanup(self.ccx.tcx(), ty),
|
2014-01-15 14:39:08 -05:00
|
|
|
val: val,
|
2014-07-04 17:55:51 -07:00
|
|
|
ty: ty,
|
|
|
|
zero: false
|
2014-01-15 14:39:08 -05:00
|
|
|
};
|
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_drop_mem({}, val={}, ty={})",
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope,
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tn().val_to_string(val),
|
2014-03-15 22:29:34 +02:00
|
|
|
ty.repr(self.ccx.tcx()));
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
self.schedule_clean(cleanup_scope, drop as CleanupObj);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a (deep) drop and zero-ing of `val`, which is a pointer to an instance of `ty`
|
2014-07-04 17:55:51 -07:00
|
|
|
fn schedule_drop_and_zero_mem(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>) {
|
2014-07-04 17:55:51 -07:00
|
|
|
if !ty::type_needs_drop(self.ccx.tcx(), ty) { return; }
|
|
|
|
let drop = box DropValue {
|
|
|
|
is_immediate: false,
|
2014-07-25 14:31:05 +02:00
|
|
|
must_unwind: ty::type_needs_unwind_cleanup(self.ccx.tcx(), ty),
|
2014-07-04 17:55:51 -07:00
|
|
|
val: val,
|
|
|
|
ty: ty,
|
|
|
|
zero: true
|
|
|
|
};
|
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_drop_and_zero_mem({}, val={}, ty={}, zero={})",
|
2014-07-04 17:55:51 -07:00
|
|
|
cleanup_scope,
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tn().val_to_string(val),
|
2014-07-04 17:55:51 -07:00
|
|
|
ty.repr(self.ccx.tcx()),
|
|
|
|
true);
|
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
self.schedule_clean(cleanup_scope, drop as CleanupObj);
|
2014-07-04 17:55:51 -07:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a (deep) drop of `val`, which is an instance of `ty`
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_drop_immediate(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>) {
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-03-15 22:29:34 +02:00
|
|
|
if !ty::type_needs_drop(self.ccx.tcx(), ty) { return; }
|
2014-04-25 01:08:02 -07:00
|
|
|
let drop = box DropValue {
|
2014-01-15 14:39:08 -05:00
|
|
|
is_immediate: true,
|
2014-07-25 14:31:05 +02:00
|
|
|
must_unwind: ty::type_needs_unwind_cleanup(self.ccx.tcx(), ty),
|
2014-01-15 14:39:08 -05:00
|
|
|
val: val,
|
2014-07-04 17:55:51 -07:00
|
|
|
ty: ty,
|
|
|
|
zero: false
|
2014-01-15 14:39:08 -05:00
|
|
|
};
|
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_drop_immediate({}, val={}, ty={})",
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope,
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tn().val_to_string(val),
|
2014-03-15 22:29:34 +02:00
|
|
|
ty.repr(self.ccx.tcx()));
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
self.schedule_clean(cleanup_scope, drop as CleanupObj);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a call to `free(val)`. Note that this is a shallow operation.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_free_value(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-05-21 00:18:10 -04:00
|
|
|
heap: Heap,
|
2014-09-29 22:11:30 +03:00
|
|
|
content_ty: Ty<'tcx>) {
|
2014-05-21 00:18:10 -04:00
|
|
|
let drop = box FreeValue { ptr: val, heap: heap, content_ty: content_ty };
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_free_value({}, val={}, heap={})",
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope,
|
2014-09-05 09:18:53 -07:00
|
|
|
self.ccx.tn().val_to_string(val),
|
2014-01-15 14:39:08 -05:00
|
|
|
heap);
|
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
self.schedule_clean(cleanup_scope, drop as CleanupObj);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a call to `free(val)`. Note that this is a shallow operation.
|
2014-09-05 03:39:15 -04:00
|
|
|
fn schedule_free_slice(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
|
|
|
size: ValueRef,
|
|
|
|
align: ValueRef,
|
|
|
|
heap: Heap) {
|
|
|
|
let drop = box FreeSlice { ptr: val, size: size, align: align, heap: heap };
|
|
|
|
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_free_slice({}, val={}, heap={})",
|
2014-09-05 03:39:15 -04:00
|
|
|
cleanup_scope,
|
|
|
|
self.ccx.tn().val_to_string(val),
|
|
|
|
heap);
|
|
|
|
|
|
|
|
self.schedule_clean(cleanup_scope, drop as CleanupObj);
|
|
|
|
}
|
|
|
|
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_clean(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanup: CleanupObj<'tcx>) {
|
2014-01-15 14:39:08 -05:00
|
|
|
match cleanup_scope {
|
|
|
|
AstScope(id) => self.schedule_clean_in_ast_scope(id, cleanup),
|
|
|
|
CustomScope(id) => self.schedule_clean_in_custom_scope(id, cleanup),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a cleanup to occur upon exit from `cleanup_scope`. If `cleanup_scope` is not
|
|
|
|
/// provided, then the cleanup is scheduled in the topmost scope, which must be a temporary
|
|
|
|
/// scope.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_clean_in_ast_scope(&self,
|
|
|
|
cleanup_scope: ast::NodeId,
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanup: CleanupObj<'tcx>) {
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("schedule_clean_in_ast_scope(cleanup_scope={})",
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope);
|
|
|
|
|
2014-09-14 20:27:36 -07:00
|
|
|
for scope in self.scopes.borrow_mut().iter_mut().rev() {
|
2014-01-15 14:39:08 -05:00
|
|
|
if scope.kind.is_ast_with_id(cleanup_scope) {
|
|
|
|
scope.cleanups.push(cleanup);
|
|
|
|
scope.clear_cached_exits();
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
// will be adding a cleanup to some enclosing scope
|
|
|
|
scope.clear_cached_exits();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-15 22:29:34 +02:00
|
|
|
self.ccx.sess().bug(
|
2014-02-06 10:38:08 +01:00
|
|
|
format!("no cleanup scope {} found",
|
2014-12-10 19:46:38 -08:00
|
|
|
self.ccx.tcx().map.node_to_string(cleanup_scope))[]);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Schedules a cleanup to occur in the top-most scope, which must be a temporary scope.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_clean_in_custom_scope(&self,
|
|
|
|
custom_scope: CustomScopeIndex,
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanup: CleanupObj<'tcx>) {
|
2014-01-15 14:39:08 -05:00
|
|
|
debug!("schedule_clean_in_custom_scope(custom_scope={})",
|
|
|
|
custom_scope.index);
|
|
|
|
|
|
|
|
assert!(self.is_valid_custom_scope(custom_scope));
|
|
|
|
|
|
|
|
let mut scopes = self.scopes.borrow_mut();
|
2014-10-23 08:42:21 -07:00
|
|
|
let scope = &mut (*scopes)[custom_scope.index];
|
2014-01-15 14:39:08 -05:00
|
|
|
scope.cleanups.push(cleanup);
|
|
|
|
scope.clear_cached_exits();
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns true if there are pending cleanups that should execute on panic.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn needs_invoke(&self) -> bool {
|
2014-03-20 19:49:20 -07:00
|
|
|
self.scopes.borrow().iter().rev().any(|s| s.needs_invoke())
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns a basic block to branch to in the event of a panic. This block will run the panic
|
|
|
|
/// cleanups and eventually invoke the LLVM `Resume` instruction.
|
2014-09-06 19:13:04 +03:00
|
|
|
fn get_landing_pad(&'blk self) -> BasicBlockRef {
|
2014-01-15 14:39:08 -05:00
|
|
|
let _icx = base::push_ctxt("get_landing_pad");
|
|
|
|
|
|
|
|
debug!("get_landing_pad");
|
|
|
|
|
|
|
|
let orig_scopes_len = self.scopes_len();
|
|
|
|
assert!(orig_scopes_len > 0);
|
|
|
|
|
2014-10-09 15:17:22 -04:00
|
|
|
// Remove any scopes that do not have cleanups on panic:
|
2014-03-19 23:16:56 +11:00
|
|
|
let mut popped_scopes = vec!();
|
2014-01-15 14:39:08 -05:00
|
|
|
while !self.top_scope(|s| s.needs_invoke()) {
|
|
|
|
debug!("top scope does not need invoke");
|
|
|
|
popped_scopes.push(self.pop_scope());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for an existing landing pad in the new topmost scope:
|
|
|
|
let llbb = self.get_or_create_landing_pad();
|
|
|
|
|
|
|
|
// Push the scopes we removed back on:
|
2013-12-23 16:20:52 +01:00
|
|
|
loop {
|
|
|
|
match popped_scopes.pop() {
|
|
|
|
Some(scope) => self.push_scope(scope),
|
|
|
|
None => break
|
|
|
|
}
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(self.scopes_len(), orig_scopes_len);
|
|
|
|
|
|
|
|
return llbb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
impl<'blk, 'tcx> CleanupHelperMethods<'blk, 'tcx> for FunctionContext<'blk, 'tcx> {
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns the id of the current top-most AST scope, if any.
|
2014-01-15 14:39:08 -05:00
|
|
|
fn top_ast_scope(&self) -> Option<ast::NodeId> {
|
2014-03-20 19:49:20 -07:00
|
|
|
for scope in self.scopes.borrow().iter().rev() {
|
2014-01-15 14:39:08 -05:00
|
|
|
match scope.kind {
|
|
|
|
CustomScopeKind | LoopScopeKind(..) => {}
|
|
|
|
AstScopeKind(i) => {
|
|
|
|
return Some(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
fn top_nonempty_cleanup_scope(&self) -> Option<uint> {
|
2014-03-20 19:49:20 -07:00
|
|
|
self.scopes.borrow().iter().rev().position(|s| !s.cleanups.is_empty())
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn is_valid_to_pop_custom_scope(&self, custom_scope: CustomScopeIndex) -> bool {
|
|
|
|
self.is_valid_custom_scope(custom_scope) &&
|
2014-03-20 19:49:20 -07:00
|
|
|
custom_scope.index == self.scopes.borrow().len() - 1
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn is_valid_custom_scope(&self, custom_scope: CustomScopeIndex) -> bool {
|
|
|
|
let scopes = self.scopes.borrow();
|
2014-03-20 19:49:20 -07:00
|
|
|
custom_scope.index < scopes.len() &&
|
2014-10-14 23:05:01 -07:00
|
|
|
(*scopes)[custom_scope.index].kind.is_temp()
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Generates the cleanups for `scope` into `bcx`
|
2014-01-15 14:39:08 -05:00
|
|
|
fn trans_scope_cleanups(&self, // cannot borrow self, will recurse
|
2014-09-06 19:13:04 +03:00
|
|
|
bcx: Block<'blk, 'tcx>,
|
2014-09-29 22:11:30 +03:00
|
|
|
scope: &CleanupScope<'blk, 'tcx>) -> Block<'blk, 'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
let mut bcx = bcx;
|
|
|
|
if !bcx.unreachable.get() {
|
2014-01-23 20:41:57 +01:00
|
|
|
for cleanup in scope.cleanups.iter().rev() {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
bcx = cleanup.trans(bcx, scope.debug_loc);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
bcx
|
|
|
|
}
|
|
|
|
|
|
|
|
fn scopes_len(&self) -> uint {
|
2014-03-20 19:49:20 -07:00
|
|
|
self.scopes.borrow().len()
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
fn push_scope(&self, scope: CleanupScope<'blk, 'tcx>) {
|
2014-03-20 19:49:20 -07:00
|
|
|
self.scopes.borrow_mut().push(scope)
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
fn pop_scope(&self) -> CleanupScope<'blk, 'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
debug!("popping cleanup scope {}, {} scopes remaining",
|
|
|
|
self.top_scope(|s| s.block_name("")),
|
|
|
|
self.scopes_len() - 1);
|
|
|
|
|
2014-03-20 19:49:20 -07:00
|
|
|
self.scopes.borrow_mut().pop().unwrap()
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-12-09 13:44:51 -05:00
|
|
|
fn top_scope<R, F>(&self, f: F) -> R where F: FnOnce(&CleanupScope<'blk, 'tcx>) -> R {
|
2014-03-20 19:49:20 -07:00
|
|
|
f(self.scopes.borrow().last().unwrap())
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Used when the caller wishes to jump to an early exit, such as a return, break, continue, or
|
|
|
|
/// unwind. This function will generate all cleanups between the top of the stack and the exit
|
|
|
|
/// `label` and return a basic block that the caller can branch to.
|
|
|
|
///
|
|
|
|
/// For example, if the current stack of cleanups were as follows:
|
|
|
|
///
|
|
|
|
/// AST 22
|
|
|
|
/// Custom 1
|
|
|
|
/// AST 23
|
|
|
|
/// Loop 23
|
|
|
|
/// Custom 2
|
|
|
|
/// AST 24
|
|
|
|
///
|
|
|
|
/// and the `label` specifies a break from `Loop 23`, then this function would generate a
|
|
|
|
/// series of basic blocks as follows:
|
|
|
|
///
|
|
|
|
/// Cleanup(AST 24) -> Cleanup(Custom 2) -> break_blk
|
|
|
|
///
|
|
|
|
/// where `break_blk` is the block specified in `Loop 23` as the target for breaks. The return
|
|
|
|
/// value would be the first basic block in that sequence (`Cleanup(AST 24)`). The caller could
|
|
|
|
/// then branch to `Cleanup(AST 24)` and it will perform all cleanups and finally branch to the
|
|
|
|
/// `break_blk`.
|
2014-09-06 19:13:04 +03:00
|
|
|
fn trans_cleanups_to_exit_scope(&'blk self,
|
2014-01-15 14:39:08 -05:00
|
|
|
label: EarlyExitLabel)
|
|
|
|
-> BasicBlockRef {
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("trans_cleanups_to_exit_scope label={} scopes={}",
|
2014-01-15 14:39:08 -05:00
|
|
|
label, self.scopes_len());
|
|
|
|
|
|
|
|
let orig_scopes_len = self.scopes_len();
|
|
|
|
let mut prev_llbb;
|
2014-03-19 23:16:56 +11:00
|
|
|
let mut popped_scopes = vec!();
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
// First we pop off all the cleanup stacks that are
|
|
|
|
// traversed until the exit is reached, pushing them
|
|
|
|
// onto the side vector `popped_scopes`. No code is
|
|
|
|
// generated at this time.
|
|
|
|
//
|
|
|
|
// So, continuing the example from above, we would wind up
|
|
|
|
// with a `popped_scopes` vector of `[AST 24, Custom 2]`.
|
|
|
|
// (Presuming that there are no cached exits)
|
|
|
|
loop {
|
|
|
|
if self.scopes_len() == 0 {
|
|
|
|
match label {
|
|
|
|
UnwindExit => {
|
|
|
|
// Generate a block that will `Resume`.
|
|
|
|
let prev_bcx = self.new_block(true, "resume", None);
|
|
|
|
let personality = self.personality.get().expect(
|
|
|
|
"create_landing_pad() should have set this");
|
|
|
|
build::Resume(prev_bcx,
|
|
|
|
build::Load(prev_bcx, personality));
|
|
|
|
prev_llbb = prev_bcx.llbb;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnExit => {
|
|
|
|
prev_llbb = self.get_llreturn();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
LoopExit(id, _) => {
|
2014-03-15 22:29:34 +02:00
|
|
|
self.ccx.sess().bug(format!(
|
2014-10-15 02:25:34 -04:00
|
|
|
"cannot exit from scope {}, \
|
2014-12-10 19:46:38 -08:00
|
|
|
not in scope", id)[]);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we have already cached the unwinding of this
|
|
|
|
// scope for this label. If so, we can stop popping scopes
|
|
|
|
// and branch to the cached label, since it contains the
|
|
|
|
// cleanups for any subsequent scopes.
|
|
|
|
match self.top_scope(|s| s.cached_early_exit(label)) {
|
|
|
|
Some(cleanup_block) => {
|
|
|
|
prev_llbb = cleanup_block;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
None => { }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pop off the scope, since we will be generating
|
|
|
|
// unwinding code for it. If we are searching for a loop exit,
|
|
|
|
// and this scope is that loop, then stop popping and set
|
|
|
|
// `prev_llbb` to the appropriate exit block from the loop.
|
|
|
|
popped_scopes.push(self.pop_scope());
|
2013-12-23 15:08:23 +01:00
|
|
|
let scope = popped_scopes.last().unwrap();
|
2014-01-15 14:39:08 -05:00
|
|
|
match label {
|
|
|
|
UnwindExit | ReturnExit => { }
|
|
|
|
LoopExit(id, exit) => {
|
|
|
|
match scope.kind.early_exit_block(id, exit) {
|
|
|
|
Some(exitllbb) => {
|
|
|
|
prev_llbb = exitllbb;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
None => { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
debug!("trans_cleanups_to_exit_scope: popped {} scopes",
|
|
|
|
popped_scopes.len());
|
|
|
|
|
|
|
|
// Now push the popped scopes back on. As we go,
|
|
|
|
// we track in `prev_llbb` the exit to which this scope
|
|
|
|
// should branch when it's done.
|
|
|
|
//
|
|
|
|
// So, continuing with our example, we will start out with
|
|
|
|
// `prev_llbb` being set to `break_blk` (or possibly a cached
|
|
|
|
// early exit). We will then pop the scopes from `popped_scopes`
|
|
|
|
// and generate a basic block for each one, prepending it in the
|
|
|
|
// series and updating `prev_llbb`. So we begin by popping `Custom 2`
|
|
|
|
// and generating `Cleanup(Custom 2)`. We make `Cleanup(Custom 2)`
|
|
|
|
// branch to `prev_llbb == break_blk`, giving us a sequence like:
|
|
|
|
//
|
|
|
|
// Cleanup(Custom 2) -> prev_llbb
|
|
|
|
//
|
|
|
|
// We then pop `AST 24` and repeat the process, giving us the sequence:
|
|
|
|
//
|
|
|
|
// Cleanup(AST 24) -> Cleanup(Custom 2) -> prev_llbb
|
|
|
|
//
|
|
|
|
// At this point, `popped_scopes` is empty, and so the final block
|
|
|
|
// that we return to the user is `Cleanup(AST 24)`.
|
|
|
|
while !popped_scopes.is_empty() {
|
2013-12-23 16:20:52 +01:00
|
|
|
let mut scope = popped_scopes.pop().unwrap();
|
2014-01-15 14:39:08 -05:00
|
|
|
|
2014-08-29 15:21:28 +12:00
|
|
|
if scope.cleanups.iter().any(|c| cleanup_is_suitable_for(&**c, label))
|
2014-01-15 14:39:08 -05:00
|
|
|
{
|
|
|
|
let name = scope.block_name("clean");
|
|
|
|
debug!("generating cleanups for {}", name);
|
2014-05-09 18:45:36 -07:00
|
|
|
let bcx_in = self.new_block(label.is_unwind(),
|
2014-12-10 19:46:38 -08:00
|
|
|
name[],
|
2014-05-09 18:45:36 -07:00
|
|
|
None);
|
2014-01-15 14:39:08 -05:00
|
|
|
let mut bcx_out = bcx_in;
|
2014-01-23 20:41:57 +01:00
|
|
|
for cleanup in scope.cleanups.iter().rev() {
|
2014-08-29 15:21:28 +12:00
|
|
|
if cleanup_is_suitable_for(&**cleanup, label) {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
bcx_out = cleanup.trans(bcx_out,
|
|
|
|
scope.debug_loc);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
build::Br(bcx_out, prev_llbb);
|
|
|
|
prev_llbb = bcx_in.llbb;
|
|
|
|
} else {
|
|
|
|
debug!("no suitable cleanups in {}",
|
|
|
|
scope.block_name("clean"));
|
|
|
|
}
|
|
|
|
|
|
|
|
scope.add_cached_early_exit(label, prev_llbb);
|
|
|
|
self.push_scope(scope);
|
|
|
|
}
|
|
|
|
|
|
|
|
debug!("trans_cleanups_to_exit_scope: prev_llbb={}", prev_llbb);
|
|
|
|
|
|
|
|
assert_eq!(self.scopes_len(), orig_scopes_len);
|
|
|
|
prev_llbb
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Creates a landing pad for the top scope, if one does not exist. The landing pad will
|
|
|
|
/// perform all cleanups necessary for an unwind and then `resume` to continue error
|
|
|
|
/// propagation:
|
|
|
|
///
|
|
|
|
/// landing_pad -> ... cleanups ... -> [resume]
|
|
|
|
///
|
|
|
|
/// (The cleanups and resume instruction are created by `trans_cleanups_to_exit_scope()`, not
|
|
|
|
/// in this function itself.)
|
2014-09-06 19:13:04 +03:00
|
|
|
fn get_or_create_landing_pad(&'blk self) -> BasicBlockRef {
|
2014-01-15 14:39:08 -05:00
|
|
|
let pad_bcx;
|
|
|
|
|
|
|
|
debug!("get_or_create_landing_pad");
|
|
|
|
|
|
|
|
// Check if a landing pad block exists; if not, create one.
|
|
|
|
{
|
|
|
|
let mut scopes = self.scopes.borrow_mut();
|
2014-09-14 20:27:36 -07:00
|
|
|
let last_scope = scopes.last_mut().unwrap();
|
2014-01-15 14:39:08 -05:00
|
|
|
match last_scope.cached_landing_pad {
|
|
|
|
Some(llbb) => { return llbb; }
|
|
|
|
None => {
|
|
|
|
let name = last_scope.block_name("unwind");
|
2014-12-10 19:46:38 -08:00
|
|
|
pad_bcx = self.new_block(true, name[], None);
|
2014-01-15 14:39:08 -05:00
|
|
|
last_scope.cached_landing_pad = Some(pad_bcx.llbb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The landing pad return type (the type being propagated). Not sure what
|
|
|
|
// this represents but it's determined by the personality function and
|
|
|
|
// this is what the EH proposal example uses.
|
2014-03-15 22:29:34 +02:00
|
|
|
let llretty = Type::struct_(self.ccx,
|
2014-11-17 21:39:01 +13:00
|
|
|
&[Type::i8p(self.ccx), Type::i32(self.ccx)],
|
2014-03-15 22:29:34 +02:00
|
|
|
false);
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
// The exception handling personality function.
|
2014-05-19 09:30:09 -07:00
|
|
|
//
|
|
|
|
// If our compilation unit has the `eh_personality` lang item somewhere
|
|
|
|
// within it, then we just need to translate that. Otherwise, we're
|
|
|
|
// building an rlib which will depend on some upstream implementation of
|
|
|
|
// this function, so we just codegen a generic reference to it. We don't
|
|
|
|
// specify any of the types for the function, we just make it a symbol
|
|
|
|
// that LLVM can later use.
|
|
|
|
let llpersonality = match pad_bcx.tcx().lang_items.eh_personality() {
|
|
|
|
Some(def_id) => callee::trans_fn_ref(pad_bcx, def_id, ExprId(0)),
|
|
|
|
None => {
|
2014-09-05 09:18:53 -07:00
|
|
|
let mut personality = self.ccx.eh_personality().borrow_mut();
|
2014-05-19 09:30:09 -07:00
|
|
|
match *personality {
|
|
|
|
Some(llpersonality) => llpersonality,
|
|
|
|
None => {
|
|
|
|
let fty = Type::variadic_func(&[], &Type::i32(self.ccx));
|
2014-05-05 10:07:49 +03:00
|
|
|
let f = base::decl_cdecl_fn(self.ccx,
|
2014-05-19 09:30:09 -07:00
|
|
|
"rust_eh_personality",
|
|
|
|
fty,
|
|
|
|
ty::mk_i32());
|
|
|
|
*personality = Some(f);
|
|
|
|
f
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2014-01-15 14:39:08 -05:00
|
|
|
|
|
|
|
// The only landing pad clause will be 'cleanup'
|
|
|
|
let llretval = build::LandingPad(pad_bcx, llretty, llpersonality, 1u);
|
|
|
|
|
|
|
|
// The landing pad block is a cleanup
|
|
|
|
build::SetCleanup(pad_bcx, llretval);
|
|
|
|
|
|
|
|
// We store the retval in a function-central alloca, so that calls to
|
|
|
|
// Resume can find it.
|
|
|
|
match self.personality.get() {
|
|
|
|
Some(addr) => {
|
|
|
|
build::Store(pad_bcx, llretval, addr);
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
let addr = base::alloca(pad_bcx, common::val_ty(llretval), "");
|
|
|
|
self.personality.set(Some(addr));
|
|
|
|
build::Store(pad_bcx, llretval, addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the cleanup block and branch to it.
|
|
|
|
let cleanup_llbb = self.trans_cleanups_to_exit_scope(UnwindExit);
|
|
|
|
build::Br(pad_bcx, cleanup_llbb);
|
|
|
|
|
|
|
|
return pad_bcx.llbb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
impl<'blk, 'tcx> CleanupScope<'blk, 'tcx> {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
fn new(kind: CleanupScopeKind<'blk, 'tcx>,
|
|
|
|
debug_loc: Option<NodeInfo>)
|
|
|
|
-> CleanupScope<'blk, 'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
CleanupScope {
|
|
|
|
kind: kind,
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
debug_loc: debug_loc,
|
2014-03-19 23:16:56 +11:00
|
|
|
cleanups: vec!(),
|
|
|
|
cached_early_exits: vec!(),
|
2014-01-15 14:39:08 -05:00
|
|
|
cached_landing_pad: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clear_cached_exits(&mut self) {
|
2014-03-19 23:16:56 +11:00
|
|
|
self.cached_early_exits = vec!();
|
2014-01-15 14:39:08 -05:00
|
|
|
self.cached_landing_pad = None;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn cached_early_exit(&self,
|
|
|
|
label: EarlyExitLabel)
|
|
|
|
-> Option<BasicBlockRef> {
|
|
|
|
self.cached_early_exits.iter().
|
|
|
|
find(|e| e.label == label).
|
|
|
|
map(|e| e.cleanup_block)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_cached_early_exit(&mut self,
|
|
|
|
label: EarlyExitLabel,
|
|
|
|
blk: BasicBlockRef) {
|
|
|
|
self.cached_early_exits.push(
|
|
|
|
CachedEarlyExit { label: label,
|
|
|
|
cleanup_block: blk });
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// True if this scope has cleanups that need unwinding
|
2014-01-15 14:39:08 -05:00
|
|
|
fn needs_invoke(&self) -> bool {
|
|
|
|
|
|
|
|
self.cached_landing_pad.is_some() ||
|
2014-07-25 14:31:05 +02:00
|
|
|
self.cleanups.iter().any(|c| c.must_unwind())
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// Returns a suitable name to use for the basic block that handles this cleanup scope
|
2014-05-22 16:57:53 -07:00
|
|
|
fn block_name(&self, prefix: &str) -> String {
|
2014-01-15 14:39:08 -05:00
|
|
|
match self.kind {
|
2014-05-27 20:44:58 -07:00
|
|
|
CustomScopeKind => format!("{}_custom_", prefix),
|
|
|
|
AstScopeKind(id) => format!("{}_ast_{}_", prefix, id),
|
|
|
|
LoopScopeKind(id, _) => format!("{}_loop_{}_", prefix, id),
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
2014-10-13 16:12:38 +02:00
|
|
|
|
|
|
|
pub fn drop_non_lifetime_clean(&mut self) {
|
|
|
|
self.cleanups.retain(|c| c.is_lifetime_end());
|
|
|
|
}
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
impl<'blk, 'tcx> CleanupScopeKind<'blk, 'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
fn is_temp(&self) -> bool {
|
|
|
|
match *self {
|
|
|
|
CustomScopeKind => true,
|
|
|
|
LoopScopeKind(..) | AstScopeKind(..) => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_ast_with_id(&self, id: ast::NodeId) -> bool {
|
|
|
|
match *self {
|
|
|
|
CustomScopeKind | LoopScopeKind(..) => false,
|
|
|
|
AstScopeKind(i) => i == id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_loop_with_id(&self, id: ast::NodeId) -> bool {
|
|
|
|
match *self {
|
|
|
|
CustomScopeKind | AstScopeKind(..) => false,
|
|
|
|
LoopScopeKind(i, _) => i == id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-25 21:17:11 -05:00
|
|
|
/// If this is a loop scope with id `id`, return the early exit block `exit`, else `None`
|
2014-01-15 14:39:08 -05:00
|
|
|
fn early_exit_block(&self,
|
|
|
|
id: ast::NodeId,
|
|
|
|
exit: uint) -> Option<BasicBlockRef> {
|
|
|
|
match *self {
|
|
|
|
LoopScopeKind(i, ref exits) if id == i => Some(exits[exit].llbb),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EarlyExitLabel {
|
|
|
|
fn is_unwind(&self) -> bool {
|
|
|
|
match *self {
|
|
|
|
UnwindExit => true,
|
|
|
|
_ => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Cleanup types
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy)]
|
2014-09-29 22:11:30 +03:00
|
|
|
pub struct DropValue<'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
is_immediate: bool,
|
2014-07-25 14:31:05 +02:00
|
|
|
must_unwind: bool,
|
2014-01-15 14:39:08 -05:00
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>,
|
2014-07-04 17:55:51 -07:00
|
|
|
zero: bool
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
impl<'tcx> Cleanup<'tcx> for DropValue<'tcx> {
|
2014-07-25 14:31:05 +02:00
|
|
|
fn must_unwind(&self) -> bool {
|
|
|
|
self.must_unwind
|
|
|
|
}
|
|
|
|
|
2014-01-15 14:39:08 -05:00
|
|
|
fn clean_on_unwind(&self) -> bool {
|
2014-07-25 14:31:05 +02:00
|
|
|
self.must_unwind
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-10-13 16:12:38 +02:00
|
|
|
fn is_lifetime_end(&self) -> bool {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
fn trans<'blk>(&self,
|
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
debug_loc: Option<NodeInfo>)
|
|
|
|
-> Block<'blk, 'tcx> {
|
2014-07-04 17:55:51 -07:00
|
|
|
let bcx = if self.is_immediate {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
glue::drop_ty_immediate(bcx, self.val, self.ty, debug_loc)
|
2014-01-15 14:39:08 -05:00
|
|
|
} else {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
glue::drop_ty(bcx, self.val, self.ty, debug_loc)
|
2014-07-04 17:55:51 -07:00
|
|
|
};
|
|
|
|
if self.zero {
|
|
|
|
base::zero_mem(bcx, self.val, self.ty);
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
2014-07-04 17:55:51 -07:00
|
|
|
bcx
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy, Show)]
|
2014-04-06 13:54:41 +03:00
|
|
|
pub enum Heap {
|
|
|
|
HeapExchange
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy)]
|
2014-09-29 22:11:30 +03:00
|
|
|
pub struct FreeValue<'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
ptr: ValueRef,
|
2014-04-06 13:54:41 +03:00
|
|
|
heap: Heap,
|
2014-09-29 22:11:30 +03:00
|
|
|
content_ty: Ty<'tcx>
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
impl<'tcx> Cleanup<'tcx> for FreeValue<'tcx> {
|
2014-07-25 14:31:05 +02:00
|
|
|
fn must_unwind(&self) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2014-01-15 14:39:08 -05:00
|
|
|
fn clean_on_unwind(&self) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2014-10-13 16:12:38 +02:00
|
|
|
fn is_lifetime_end(&self) -> bool {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
fn trans<'blk>(&self,
|
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
debug_loc: Option<NodeInfo>)
|
|
|
|
-> Block<'blk, 'tcx> {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
apply_debug_loc(bcx.fcx, debug_loc);
|
|
|
|
|
2014-01-15 14:39:08 -05:00
|
|
|
match self.heap {
|
2014-04-06 13:54:41 +03:00
|
|
|
HeapExchange => {
|
2014-05-21 00:18:10 -04:00
|
|
|
glue::trans_exchange_free_ty(bcx, self.ptr, self.content_ty)
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy)]
|
2014-09-05 03:39:15 -04:00
|
|
|
pub struct FreeSlice {
|
|
|
|
ptr: ValueRef,
|
|
|
|
size: ValueRef,
|
|
|
|
align: ValueRef,
|
|
|
|
heap: Heap,
|
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
impl<'tcx> Cleanup<'tcx> for FreeSlice {
|
2014-09-05 03:39:15 -04:00
|
|
|
fn must_unwind(&self) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clean_on_unwind(&self) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2014-10-13 16:12:38 +02:00
|
|
|
fn is_lifetime_end(&self) -> bool {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2014-12-12 11:09:32 -05:00
|
|
|
fn trans<'blk>(&self,
|
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
debug_loc: Option<NodeInfo>)
|
|
|
|
-> Block<'blk, 'tcx> {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
apply_debug_loc(bcx.fcx, debug_loc);
|
|
|
|
|
2014-09-05 03:39:15 -04:00
|
|
|
match self.heap {
|
|
|
|
HeapExchange => {
|
|
|
|
glue::trans_exchange_free_dyn(bcx, self.ptr, self.size, self.align)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-14 23:14:38 -05:00
|
|
|
#[deriving(Copy)]
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
pub struct LifetimeEnd {
|
|
|
|
ptr: ValueRef,
|
|
|
|
}
|
|
|
|
|
2014-09-29 22:11:30 +03:00
|
|
|
impl<'tcx> Cleanup<'tcx> for LifetimeEnd {
|
2014-07-25 14:31:05 +02:00
|
|
|
fn must_unwind(&self) -> bool {
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2014-07-25 14:31:05 +02:00
|
|
|
fn clean_on_unwind(&self) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2014-10-13 16:12:38 +02:00
|
|
|
fn is_lifetime_end(&self) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2014-12-12 11:09:32 -05:00
|
|
|
fn trans<'blk>(&self,
|
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
debug_loc: Option<NodeInfo>)
|
|
|
|
-> Block<'blk, 'tcx> {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
apply_debug_loc(bcx.fcx, debug_loc);
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
base::call_lifetime_end(bcx, self.ptr);
|
|
|
|
bcx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-06 05:07:47 +02:00
|
|
|
pub fn temporary_scope(tcx: &ty::ctxt,
|
2014-01-15 14:39:08 -05:00
|
|
|
id: ast::NodeId)
|
|
|
|
-> ScopeId {
|
|
|
|
match tcx.region_maps.temporary_scope(id) {
|
|
|
|
Some(scope) => {
|
2014-11-18 14:22:59 +01:00
|
|
|
let r = AstScope(scope.node_id());
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("temporary_scope({}) = {}", id, r);
|
2014-01-15 14:39:08 -05:00
|
|
|
r
|
|
|
|
}
|
|
|
|
None => {
|
2014-05-16 10:45:16 -07:00
|
|
|
tcx.sess.bug(format!("no temporary scope available for expr {}",
|
2014-12-10 19:46:38 -08:00
|
|
|
id)[])
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-06 05:07:47 +02:00
|
|
|
pub fn var_scope(tcx: &ty::ctxt,
|
2014-01-15 14:39:08 -05:00
|
|
|
id: ast::NodeId)
|
|
|
|
-> ScopeId {
|
2014-11-18 14:22:59 +01:00
|
|
|
let r = AstScope(tcx.region_maps.var_scope(id).node_id());
|
2014-10-15 02:25:34 -04:00
|
|
|
debug!("var_scope({}) = {}", id, r);
|
2014-01-15 14:39:08 -05:00
|
|
|
r
|
|
|
|
}
|
|
|
|
|
|
|
|
fn cleanup_is_suitable_for(c: &Cleanup,
|
|
|
|
label: EarlyExitLabel) -> bool {
|
|
|
|
!label.is_unwind() || c.clean_on_unwind()
|
|
|
|
}
|
|
|
|
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
fn apply_debug_loc(fcx: &FunctionContext, debug_loc: Option<NodeInfo>) {
|
|
|
|
match debug_loc {
|
|
|
|
Some(ref src_loc) => {
|
|
|
|
debuginfo::set_source_location(fcx, src_loc.id, src_loc.span);
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
debuginfo::clear_source_location(fcx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-15 14:39:08 -05:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// These traits just exist to put the methods into this file.
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
pub trait CleanupMethods<'blk, 'tcx> {
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
fn push_ast_cleanup_scope(&self, id: NodeInfo);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn push_loop_cleanup_scope(&self,
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
id: ast::NodeId,
|
|
|
|
exits: [Block<'blk, 'tcx>, ..EXIT_MAX]);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn push_custom_cleanup_scope(&self) -> CustomScopeIndex;
|
debuginfo: Make sure that all calls to drop glue are associated with debug locations.
This commit makes rustc emit debug locations for all call
and invoke statements in LLVM IR, if they are contained
within a function that debuginfo is enabled for. This is
important because LLVM does not handle the case where a
function body containing debuginfo is inlined into another
function with debuginfo, but the inlined call statement
does not have a debug location. In this case, LLVM will
not know where (in terms of source code coordinates) the
function was inlined to and we end up with some statements
still linked to the source locations in there original,
non-inlined function without any indication that they are
indeed an inline-copy. Later, when generating DWARF from
the IR, LLVM will interpret this as corrupt IR and abort.
Unfortunately, the undesirable case described above can
still occur when using LTO. If there is a crate compiled
without debuginfo calling into a crate compiled with
debuginfo, we again end up with the conditions triggering
the error. This is why some LTO tests still fail with the
dreaded assertion, if the standard library was built with
debuginfo enabled.
That is, `RUSTFLAGS_STAGE2=-g make rustc-stage2` will
succeed but `RUSTFLAGS_STAGE2=-g make check` will still
fail after this commit has been merged. This is a problem
that has to be dealt with separately.
Fixes #17201
Fixes #15816
Fixes #15156
2014-09-24 08:49:38 +02:00
|
|
|
fn push_custom_cleanup_scope_with_debug_loc(&self,
|
|
|
|
debug_loc: NodeInfo)
|
|
|
|
-> CustomScopeIndex;
|
2014-01-15 14:39:08 -05:00
|
|
|
fn pop_and_trans_ast_cleanup_scope(&self,
|
2014-09-06 19:13:04 +03:00
|
|
|
bcx: Block<'blk, 'tcx>,
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope: ast::NodeId)
|
2014-09-06 19:13:04 +03:00
|
|
|
-> Block<'blk, 'tcx>;
|
2014-01-15 14:39:08 -05:00
|
|
|
fn pop_loop_cleanup_scope(&self,
|
|
|
|
cleanup_scope: ast::NodeId);
|
|
|
|
fn pop_custom_cleanup_scope(&self,
|
|
|
|
custom_scope: CustomScopeIndex);
|
|
|
|
fn pop_and_trans_custom_cleanup_scope(&self,
|
2014-09-06 19:13:04 +03:00
|
|
|
bcx: Block<'blk, 'tcx>,
|
2014-01-15 14:39:08 -05:00
|
|
|
custom_scope: CustomScopeIndex)
|
2014-09-06 19:13:04 +03:00
|
|
|
-> Block<'blk, 'tcx>;
|
2014-01-15 14:39:08 -05:00
|
|
|
fn top_loop_scope(&self) -> ast::NodeId;
|
2014-09-06 19:13:04 +03:00
|
|
|
fn normal_exit_block(&'blk self,
|
2014-01-15 14:39:08 -05:00
|
|
|
cleanup_scope: ast::NodeId,
|
|
|
|
exit: uint) -> BasicBlockRef;
|
2014-09-06 19:13:04 +03:00
|
|
|
fn return_exit_block(&'blk self) -> BasicBlockRef;
|
Emit LLVM lifetime intrinsics to improve stack usage and codegen in general
Lifetime intrinsics help to reduce stack usage, because LLVM can apply
stack coloring to reuse the stack slots of dead allocas for new ones.
For example these functions now both use the same amount of stack, while
previous `bar()` used five times as much as `foo()`:
````rust
fn foo() {
println("{}", 5);
}
fn bar() {
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
println("{}", 5);
}
````
On top of that, LLVM can also optimize out certain operations when it
knows that memory is dead after a certain point. For example, it can
sometimes remove the zeroing used to cancel the drop glue. This is
possible when the glue drop itself was already removed because the
zeroing dominated the drop glue call. For example in:
````rust
pub fn bar(x: (Box<int>, int)) -> (Box<int>, int) {
x
}
````
With optimizations, this currently results in:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.memset.p0i8.i64(i8* %2, i8 0, i64 16, i32 8, i1 false)
ret void
}
````
But with lifetime intrinsics we get:
````llvm
define void @_ZN3bar20h330fa42547df8179niaE({ i64*, i64 }* noalias nocapture nonnull sret, { i64*, i64 }* noalias nocapture nonnull) unnamed_addr #0 {
"_ZN29_$LP$Box$LT$int$GT$$C$int$RP$39glue_drop.$x22glue_drop$x22$LP$1347$RP$17h88cf42702e5a322aE.exit":
%2 = bitcast { i64*, i64 }* %1 to i8*
%3 = bitcast { i64*, i64 }* %0 to i8*
tail call void @llvm.memcpy.p0i8.p0i8.i64(i8* %3, i8* %2, i64 16, i32 8, i1 false)
tail call void @llvm.lifetime.end(i64 16, i8* %2)
ret void
}
````
Fixes #15665
2014-05-01 19:32:07 +02:00
|
|
|
fn schedule_lifetime_end(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_drop_mem(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>);
|
2014-07-04 17:55:51 -07:00
|
|
|
fn schedule_drop_and_zero_mem(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_drop_immediate(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-09-29 22:11:30 +03:00
|
|
|
ty: Ty<'tcx>);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_free_value(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
2014-05-21 00:18:10 -04:00
|
|
|
heap: Heap,
|
2014-09-29 22:11:30 +03:00
|
|
|
content_ty: Ty<'tcx>);
|
2014-09-05 03:39:15 -04:00
|
|
|
fn schedule_free_slice(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
|
|
|
val: ValueRef,
|
|
|
|
size: ValueRef,
|
|
|
|
align: ValueRef,
|
|
|
|
heap: Heap);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_clean(&self,
|
|
|
|
cleanup_scope: ScopeId,
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanup: CleanupObj<'tcx>);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_clean_in_ast_scope(&self,
|
|
|
|
cleanup_scope: ast::NodeId,
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanup: CleanupObj<'tcx>);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn schedule_clean_in_custom_scope(&self,
|
|
|
|
custom_scope: CustomScopeIndex,
|
2014-09-29 22:11:30 +03:00
|
|
|
cleanup: CleanupObj<'tcx>);
|
2014-01-15 14:39:08 -05:00
|
|
|
fn needs_invoke(&self) -> bool;
|
2014-09-06 19:13:04 +03:00
|
|
|
fn get_landing_pad(&'blk self) -> BasicBlockRef;
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|
|
|
|
|
2014-09-06 19:13:04 +03:00
|
|
|
trait CleanupHelperMethods<'blk, 'tcx> {
|
2014-01-15 14:39:08 -05:00
|
|
|
fn top_ast_scope(&self) -> Option<ast::NodeId>;
|
|
|
|
fn top_nonempty_cleanup_scope(&self) -> Option<uint>;
|
|
|
|
fn is_valid_to_pop_custom_scope(&self, custom_scope: CustomScopeIndex) -> bool;
|
|
|
|
fn is_valid_custom_scope(&self, custom_scope: CustomScopeIndex) -> bool;
|
|
|
|
fn trans_scope_cleanups(&self,
|
2014-09-06 19:13:04 +03:00
|
|
|
bcx: Block<'blk, 'tcx>,
|
|
|
|
scope: &CleanupScope<'blk, 'tcx>) -> Block<'blk, 'tcx>;
|
|
|
|
fn trans_cleanups_to_exit_scope(&'blk self,
|
2014-01-15 14:39:08 -05:00
|
|
|
label: EarlyExitLabel)
|
|
|
|
-> BasicBlockRef;
|
2014-09-06 19:13:04 +03:00
|
|
|
fn get_or_create_landing_pad(&'blk self) -> BasicBlockRef;
|
2014-01-15 14:39:08 -05:00
|
|
|
fn scopes_len(&self) -> uint;
|
2014-09-06 19:13:04 +03:00
|
|
|
fn push_scope(&self, scope: CleanupScope<'blk, 'tcx>);
|
|
|
|
fn pop_scope(&self) -> CleanupScope<'blk, 'tcx>;
|
2014-12-09 13:44:51 -05:00
|
|
|
fn top_scope<R, F>(&self, f: F) -> R where F: FnOnce(&CleanupScope<'blk, 'tcx>) -> R;
|
2014-01-15 14:39:08 -05:00
|
|
|
}
|