2015-11-03 06:35:09 -05:00
|
|
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
//! An analysis to determine which locals require allocas and
|
2015-11-03 06:35:09 -05:00
|
|
|
//! which do not.
|
|
|
|
|
2016-01-21 18:57:43 +02:00
|
|
|
use rustc_data_structures::bitvec::BitVector;
|
2016-06-07 17:28:36 +03:00
|
|
|
use rustc_data_structures::indexed_vec::{Idx, IndexVec};
|
2016-09-19 23:50:00 +03:00
|
|
|
use rustc::mir::{self, Location, TerminatorKind};
|
2015-11-24 14:35:34 +01:00
|
|
|
use rustc::mir::visit::{Visitor, LvalueContext};
|
2016-06-05 19:38:22 +03:00
|
|
|
use rustc::mir::traversal;
|
2016-12-11 22:19:39 -07:00
|
|
|
use common::{self, BlockAndBuilder};
|
2015-11-03 06:35:09 -05:00
|
|
|
use super::rvalue;
|
|
|
|
|
2016-12-17 19:54:32 -07:00
|
|
|
pub fn lvalue_locals<'a, 'tcx>(bcx: &BlockAndBuilder<'a, 'tcx>, mir: &mir::Mir<'tcx>) -> BitVector {
|
2016-06-20 23:55:14 +03:00
|
|
|
let mut analyzer = LocalAnalyzer::new(mir, &bcx);
|
2015-11-03 06:35:09 -05:00
|
|
|
|
|
|
|
analyzer.visit_mir(mir);
|
|
|
|
|
2016-09-25 01:38:27 +02:00
|
|
|
for (index, ty) in mir.local_decls.iter().map(|l| l.ty).enumerate() {
|
2016-12-11 22:19:39 -07:00
|
|
|
let ty = bcx.fcx().monomorphize(&ty);
|
2016-06-20 23:55:14 +03:00
|
|
|
debug!("local {} has type {:?}", index, ty);
|
2015-11-10 22:05:11 +02:00
|
|
|
if ty.is_scalar() ||
|
|
|
|
ty.is_unique() ||
|
|
|
|
ty.is_region_ptr() ||
|
Various improvements to MIR and LLVM IR Construction
Primarily affects the MIR construction, which indirectly improves LLVM
IR generation, but some LLVM IR changes have been made too.
* Handle "statement expressions" more intelligently. These are
expressions that always evaluate to `()`. Previously a temporary would
be generated as a destination to translate into, which is unnecessary.
This affects assignment, augmented assignment, `return`, `break` and
`continue`.
* Avoid inserting drops for non-drop types in more places. Scheduled
drops were already skipped for types that we knew wouldn't need
dropping at construction time. However manually-inserted drops like
those for `x` in `x = y;` were still generated. `build_drop` now takes
a type parameter like its `schedule_drop` counterpart and checks to
see if the type needs dropping.
* Avoid generating an extra temporary for an assignment where the types
involved don't need dropping. Previously an expression like
`a = b + 1;` would result in a temporary for `b + 1`. This is so the
RHS can be evaluated, then the LHS evaluated and dropped and have
everything work correctly. However, this isn't necessary if the `LHS`
doesn't need a drop, as we can just overwrite the existing value.
* Improves lvalue analysis to allow treating an `Rvalue::Use` as an
operand in certain conditions. The reason for it never being an
operand is so it can be zeroed/drop-filled, but this is only true for
types that need dropping.
The first two changes result in significantly fewer MIR blocks being
generated, as previously almost every statement would end up generating
a new block due to the drop of the `()` temporary being generated.
2016-04-15 12:36:16 +12:00
|
|
|
ty.is_simd() ||
|
|
|
|
common::type_is_zero_size(bcx.ccx(), ty)
|
2015-11-03 06:35:09 -05:00
|
|
|
{
|
|
|
|
// These sorts of types are immediates that we can store
|
|
|
|
// in an ValueRef without an alloca.
|
2015-11-10 22:05:11 +02:00
|
|
|
assert!(common::type_is_immediate(bcx.ccx(), ty) ||
|
2016-12-18 11:50:07 -07:00
|
|
|
common::type_is_fat_ptr(bcx.ccx(), ty));
|
2016-05-25 11:58:08 +03:00
|
|
|
} else if common::type_is_imm_pair(bcx.ccx(), ty) {
|
|
|
|
// We allow pairs and uses of any of their 2 fields.
|
2015-11-03 06:35:09 -05:00
|
|
|
} else {
|
|
|
|
// These sorts of types require an alloca. Note that
|
|
|
|
// type_is_immediate() may *still* be true, particularly
|
|
|
|
// for newtypes, but we currently force some types
|
|
|
|
// (e.g. structs) into an alloca unconditionally, just so
|
|
|
|
// that we don't have to deal with having two pathways
|
2015-11-03 15:50:04 -05:00
|
|
|
// (gep vs extractvalue etc).
|
2016-06-20 23:55:14 +03:00
|
|
|
analyzer.mark_as_lvalue(mir::Local::new(index));
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
analyzer.lvalue_locals
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
|
2016-12-17 19:54:32 -07:00
|
|
|
struct LocalAnalyzer<'mir, 'a: 'mir, 'tcx: 'a> {
|
Various improvements to MIR and LLVM IR Construction
Primarily affects the MIR construction, which indirectly improves LLVM
IR generation, but some LLVM IR changes have been made too.
* Handle "statement expressions" more intelligently. These are
expressions that always evaluate to `()`. Previously a temporary would
be generated as a destination to translate into, which is unnecessary.
This affects assignment, augmented assignment, `return`, `break` and
`continue`.
* Avoid inserting drops for non-drop types in more places. Scheduled
drops were already skipped for types that we knew wouldn't need
dropping at construction time. However manually-inserted drops like
those for `x` in `x = y;` were still generated. `build_drop` now takes
a type parameter like its `schedule_drop` counterpart and checks to
see if the type needs dropping.
* Avoid generating an extra temporary for an assignment where the types
involved don't need dropping. Previously an expression like
`a = b + 1;` would result in a temporary for `b + 1`. This is so the
RHS can be evaluated, then the LHS evaluated and dropped and have
everything work correctly. However, this isn't necessary if the `LHS`
doesn't need a drop, as we can just overwrite the existing value.
* Improves lvalue analysis to allow treating an `Rvalue::Use` as an
operand in certain conditions. The reason for it never being an
operand is so it can be zeroed/drop-filled, but this is only true for
types that need dropping.
The first two changes result in significantly fewer MIR blocks being
generated, as previously almost every statement would end up generating
a new block due to the drop of the `()` temporary being generated.
2016-04-15 12:36:16 +12:00
|
|
|
mir: &'mir mir::Mir<'tcx>,
|
2016-12-17 19:54:32 -07:00
|
|
|
bcx: &'mir BlockAndBuilder<'a, 'tcx>,
|
2016-06-20 23:55:14 +03:00
|
|
|
lvalue_locals: BitVector,
|
2016-01-21 18:57:43 +02:00
|
|
|
seen_assigned: BitVector
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
|
2016-12-17 19:54:32 -07:00
|
|
|
impl<'mir, 'a, 'tcx> LocalAnalyzer<'mir, 'a, 'tcx> {
|
|
|
|
fn new(mir: &'mir mir::Mir<'tcx>, bcx: &'mir BlockAndBuilder<'a, 'tcx>)
|
|
|
|
-> LocalAnalyzer<'mir, 'a, 'tcx> {
|
2016-06-20 23:55:14 +03:00
|
|
|
LocalAnalyzer {
|
Various improvements to MIR and LLVM IR Construction
Primarily affects the MIR construction, which indirectly improves LLVM
IR generation, but some LLVM IR changes have been made too.
* Handle "statement expressions" more intelligently. These are
expressions that always evaluate to `()`. Previously a temporary would
be generated as a destination to translate into, which is unnecessary.
This affects assignment, augmented assignment, `return`, `break` and
`continue`.
* Avoid inserting drops for non-drop types in more places. Scheduled
drops were already skipped for types that we knew wouldn't need
dropping at construction time. However manually-inserted drops like
those for `x` in `x = y;` were still generated. `build_drop` now takes
a type parameter like its `schedule_drop` counterpart and checks to
see if the type needs dropping.
* Avoid generating an extra temporary for an assignment where the types
involved don't need dropping. Previously an expression like
`a = b + 1;` would result in a temporary for `b + 1`. This is so the
RHS can be evaluated, then the LHS evaluated and dropped and have
everything work correctly. However, this isn't necessary if the `LHS`
doesn't need a drop, as we can just overwrite the existing value.
* Improves lvalue analysis to allow treating an `Rvalue::Use` as an
operand in certain conditions. The reason for it never being an
operand is so it can be zeroed/drop-filled, but this is only true for
types that need dropping.
The first two changes result in significantly fewer MIR blocks being
generated, as previously almost every statement would end up generating
a new block due to the drop of the `()` temporary being generated.
2016-04-15 12:36:16 +12:00
|
|
|
mir: mir,
|
|
|
|
bcx: bcx,
|
2016-09-25 01:38:27 +02:00
|
|
|
lvalue_locals: BitVector::new(mir.local_decls.len()),
|
|
|
|
seen_assigned: BitVector::new(mir.local_decls.len())
|
2016-01-21 18:57:43 +02:00
|
|
|
}
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
fn mark_as_lvalue(&mut self, local: mir::Local) {
|
|
|
|
debug!("marking {:?} as lvalue", local);
|
|
|
|
self.lvalue_locals.insert(local.index());
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
2016-01-21 18:57:43 +02:00
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
fn mark_assigned(&mut self, local: mir::Local) {
|
|
|
|
if !self.seen_assigned.insert(local.index()) {
|
|
|
|
self.mark_as_lvalue(local);
|
2016-01-21 18:57:43 +02:00
|
|
|
}
|
|
|
|
}
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
|
2016-12-17 19:54:32 -07:00
|
|
|
impl<'mir, 'a, 'tcx> Visitor<'tcx> for LocalAnalyzer<'mir, 'a, 'tcx> {
|
2015-11-03 06:35:09 -05:00
|
|
|
fn visit_assign(&mut self,
|
|
|
|
block: mir::BasicBlock,
|
|
|
|
lvalue: &mir::Lvalue<'tcx>,
|
2016-08-08 18:46:06 -07:00
|
|
|
rvalue: &mir::Rvalue<'tcx>,
|
|
|
|
location: Location) {
|
2015-11-03 06:35:09 -05:00
|
|
|
debug!("visit_assign(block={:?}, lvalue={:?}, rvalue={:?})", block, lvalue, rvalue);
|
|
|
|
|
2016-09-25 01:38:27 +02:00
|
|
|
if let mir::Lvalue::Local(index) = *lvalue {
|
2016-06-20 23:55:14 +03:00
|
|
|
self.mark_assigned(index);
|
|
|
|
if !rvalue::rvalue_creates_operand(self.mir, self.bcx, rvalue) {
|
|
|
|
self.mark_as_lvalue(index);
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
2016-06-20 23:55:14 +03:00
|
|
|
} else {
|
2016-08-08 18:46:06 -07:00
|
|
|
self.visit_lvalue(lvalue, LvalueContext::Store, location);
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
|
2016-08-08 18:46:06 -07:00
|
|
|
self.visit_rvalue(rvalue, location);
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
fn visit_terminator_kind(&mut self,
|
|
|
|
block: mir::BasicBlock,
|
2016-08-08 18:46:06 -07:00
|
|
|
kind: &mir::TerminatorKind<'tcx>,
|
|
|
|
location: Location) {
|
2016-06-20 23:55:14 +03:00
|
|
|
match *kind {
|
|
|
|
mir::TerminatorKind::Call {
|
|
|
|
func: mir::Operand::Constant(mir::Constant {
|
|
|
|
literal: mir::Literal::Item { def_id, .. }, ..
|
|
|
|
}),
|
|
|
|
ref args, ..
|
|
|
|
} if Some(def_id) == self.bcx.tcx().lang_items.box_free_fn() => {
|
|
|
|
// box_free(x) shares with `drop x` the property that it
|
|
|
|
// is not guaranteed to be statically dominated by the
|
|
|
|
// definition of x, so x must always be in an alloca.
|
|
|
|
if let mir::Operand::Consume(ref lvalue) = args[0] {
|
2016-08-08 18:46:06 -07:00
|
|
|
self.visit_lvalue(lvalue, LvalueContext::Drop, location);
|
2016-06-20 23:55:14 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
|
2016-08-08 18:46:06 -07:00
|
|
|
self.super_terminator_kind(block, kind, location);
|
2016-06-20 23:55:14 +03:00
|
|
|
}
|
|
|
|
|
2015-11-03 06:35:09 -05:00
|
|
|
fn visit_lvalue(&mut self,
|
|
|
|
lvalue: &mir::Lvalue<'tcx>,
|
2016-09-15 18:18:40 -07:00
|
|
|
context: LvalueContext<'tcx>,
|
2016-08-08 18:46:06 -07:00
|
|
|
location: Location) {
|
2015-11-03 06:35:09 -05:00
|
|
|
debug!("visit_lvalue(lvalue={:?}, context={:?})", lvalue, context);
|
|
|
|
|
2016-05-25 11:58:08 +03:00
|
|
|
// Allow uses of projections of immediate pair fields.
|
|
|
|
if let mir::Lvalue::Projection(ref proj) = *lvalue {
|
2016-09-25 01:38:27 +02:00
|
|
|
if let mir::Lvalue::Local(_) = proj.base {
|
2016-08-05 15:59:51 -07:00
|
|
|
let ty = proj.base.ty(self.mir, self.bcx.tcx());
|
|
|
|
|
2016-12-11 22:19:39 -07:00
|
|
|
let ty = self.bcx.fcx().monomorphize(&ty.to_ty(self.bcx.tcx()));
|
2016-05-25 11:58:08 +03:00
|
|
|
if common::type_is_imm_pair(self.bcx.ccx(), ty) {
|
|
|
|
if let mir::ProjectionElem::Field(..) = proj.elem {
|
|
|
|
if let LvalueContext::Consume = context {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-25 01:38:27 +02:00
|
|
|
if let mir::Lvalue::Local(index) = *lvalue {
|
2016-06-20 23:55:14 +03:00
|
|
|
match context {
|
|
|
|
LvalueContext::Call => {
|
|
|
|
self.mark_assigned(index);
|
|
|
|
}
|
2016-08-14 06:34:14 +03:00
|
|
|
|
|
|
|
LvalueContext::StorageLive |
|
|
|
|
LvalueContext::StorageDead |
|
|
|
|
LvalueContext::Consume => {}
|
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
LvalueContext::Store |
|
|
|
|
LvalueContext::Inspect |
|
|
|
|
LvalueContext::Borrow { .. } |
|
2016-09-15 18:18:40 -07:00
|
|
|
LvalueContext::Projection(..) => {
|
2016-06-20 23:55:14 +03:00
|
|
|
self.mark_as_lvalue(index);
|
|
|
|
}
|
2016-08-14 06:34:14 +03:00
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
LvalueContext::Drop => {
|
2016-08-05 15:59:51 -07:00
|
|
|
let ty = lvalue.ty(self.mir, self.bcx.tcx());
|
2016-12-11 22:19:39 -07:00
|
|
|
let ty = self.bcx.fcx().monomorphize(&ty.to_ty(self.bcx.tcx()));
|
2016-06-09 18:15:15 +03:00
|
|
|
|
2016-06-20 23:55:14 +03:00
|
|
|
// Only need the lvalue if we're actually dropping it.
|
2016-12-18 11:50:07 -07:00
|
|
|
if self.bcx.ccx().shared().type_needs_drop(ty) {
|
2016-06-20 23:55:14 +03:00
|
|
|
self.mark_as_lvalue(index);
|
2016-06-09 18:15:15 +03:00
|
|
|
}
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-09 18:14:47 +03:00
|
|
|
// A deref projection only reads the pointer, never needs the lvalue.
|
|
|
|
if let mir::Lvalue::Projection(ref proj) = *lvalue {
|
|
|
|
if let mir::ProjectionElem::Deref = proj.elem {
|
2016-08-08 18:46:06 -07:00
|
|
|
return self.visit_lvalue(&proj.base, LvalueContext::Consume, location);
|
2016-06-09 18:14:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-08 18:46:06 -07:00
|
|
|
self.super_lvalue(lvalue, context, location);
|
2015-11-03 06:35:09 -05:00
|
|
|
}
|
|
|
|
}
|
2016-05-29 22:01:06 +03:00
|
|
|
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
|
|
|
pub enum CleanupKind {
|
|
|
|
NotCleanup,
|
|
|
|
Funclet,
|
|
|
|
Internal { funclet: mir::BasicBlock }
|
|
|
|
}
|
|
|
|
|
2016-12-17 19:54:32 -07:00
|
|
|
pub fn cleanup_kinds<'a, 'tcx>(mir: &mir::Mir<'tcx>) -> IndexVec<mir::BasicBlock, CleanupKind> {
|
2016-06-07 17:28:36 +03:00
|
|
|
fn discover_masters<'tcx>(result: &mut IndexVec<mir::BasicBlock, CleanupKind>,
|
|
|
|
mir: &mir::Mir<'tcx>) {
|
2016-06-07 21:20:50 +03:00
|
|
|
for (bb, data) in mir.basic_blocks().iter_enumerated() {
|
2016-05-29 22:01:06 +03:00
|
|
|
match data.terminator().kind {
|
|
|
|
TerminatorKind::Goto { .. } |
|
|
|
|
TerminatorKind::Resume |
|
|
|
|
TerminatorKind::Return |
|
2016-06-08 19:26:19 +03:00
|
|
|
TerminatorKind::Unreachable |
|
2016-05-29 22:01:06 +03:00
|
|
|
TerminatorKind::If { .. } |
|
|
|
|
TerminatorKind::Switch { .. } |
|
|
|
|
TerminatorKind::SwitchInt { .. } => {
|
|
|
|
/* nothing to do */
|
|
|
|
}
|
|
|
|
TerminatorKind::Call { cleanup: unwind, .. } |
|
2016-05-25 08:39:32 +03:00
|
|
|
TerminatorKind::Assert { cleanup: unwind, .. } |
|
2016-05-29 22:01:06 +03:00
|
|
|
TerminatorKind::DropAndReplace { unwind, .. } |
|
|
|
|
TerminatorKind::Drop { unwind, .. } => {
|
|
|
|
if let Some(unwind) = unwind {
|
|
|
|
debug!("cleanup_kinds: {:?}/{:?} registering {:?} as funclet",
|
|
|
|
bb, data, unwind);
|
2016-06-07 17:28:36 +03:00
|
|
|
result[unwind] = CleanupKind::Funclet;
|
2016-05-29 22:01:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-07 17:28:36 +03:00
|
|
|
fn propagate<'tcx>(result: &mut IndexVec<mir::BasicBlock, CleanupKind>,
|
|
|
|
mir: &mir::Mir<'tcx>) {
|
2016-06-07 21:20:50 +03:00
|
|
|
let mut funclet_succs = IndexVec::from_elem(None, mir.basic_blocks());
|
2016-05-29 22:01:06 +03:00
|
|
|
|
|
|
|
let mut set_successor = |funclet: mir::BasicBlock, succ| {
|
2016-06-07 17:28:36 +03:00
|
|
|
match funclet_succs[funclet] {
|
2016-05-29 22:01:06 +03:00
|
|
|
ref mut s @ None => {
|
|
|
|
debug!("set_successor: updating successor of {:?} to {:?}",
|
|
|
|
funclet, succ);
|
|
|
|
*s = Some(succ);
|
|
|
|
},
|
|
|
|
Some(s) => if s != succ {
|
|
|
|
span_bug!(mir.span, "funclet {:?} has 2 parents - {:?} and {:?}",
|
|
|
|
funclet, s, succ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
for (bb, data) in traversal::reverse_postorder(mir) {
|
2016-06-07 17:28:36 +03:00
|
|
|
let funclet = match result[bb] {
|
2016-05-29 22:01:06 +03:00
|
|
|
CleanupKind::NotCleanup => continue,
|
|
|
|
CleanupKind::Funclet => bb,
|
|
|
|
CleanupKind::Internal { funclet } => funclet,
|
|
|
|
};
|
|
|
|
|
|
|
|
debug!("cleanup_kinds: {:?}/{:?}/{:?} propagating funclet {:?}",
|
2016-06-07 17:28:36 +03:00
|
|
|
bb, data, result[bb], funclet);
|
2016-05-29 22:01:06 +03:00
|
|
|
|
|
|
|
for &succ in data.terminator().successors().iter() {
|
2016-06-07 17:28:36 +03:00
|
|
|
let kind = result[succ];
|
2016-05-29 22:01:06 +03:00
|
|
|
debug!("cleanup_kinds: propagating {:?} to {:?}/{:?}",
|
|
|
|
funclet, succ, kind);
|
|
|
|
match kind {
|
|
|
|
CleanupKind::NotCleanup => {
|
2016-06-07 17:28:36 +03:00
|
|
|
result[succ] = CleanupKind::Internal { funclet: funclet };
|
2016-05-29 22:01:06 +03:00
|
|
|
}
|
|
|
|
CleanupKind::Funclet => {
|
|
|
|
set_successor(funclet, succ);
|
|
|
|
}
|
|
|
|
CleanupKind::Internal { funclet: succ_funclet } => {
|
|
|
|
if funclet != succ_funclet {
|
|
|
|
// `succ` has 2 different funclet going into it, so it must
|
|
|
|
// be a funclet by itself.
|
|
|
|
|
|
|
|
debug!("promoting {:?} to a funclet and updating {:?}", succ,
|
|
|
|
succ_funclet);
|
2016-06-07 17:28:36 +03:00
|
|
|
result[succ] = CleanupKind::Funclet;
|
2016-05-29 22:01:06 +03:00
|
|
|
set_successor(succ_funclet, succ);
|
|
|
|
set_successor(funclet, succ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-07 21:20:50 +03:00
|
|
|
let mut result = IndexVec::from_elem(CleanupKind::NotCleanup, mir.basic_blocks());
|
2016-05-29 22:01:06 +03:00
|
|
|
|
|
|
|
discover_masters(&mut result, mir);
|
|
|
|
propagate(&mut result, mir);
|
|
|
|
debug!("cleanup_kinds: result={:?}", result);
|
|
|
|
result
|
|
|
|
}
|