2017-05-22 06:06:51 -05:00
|
|
|
// Copyright 2017 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.
|
|
|
|
|
|
|
|
//! This module provides one pass, `CleanEndRegions`, that reduces the
|
|
|
|
//! set of `EndRegion` statements in the MIR.
|
|
|
|
//!
|
|
|
|
//! The "pass" is actually implemented as two traversals (aka visits)
|
|
|
|
//! of the input MIR. The first traversal, `GatherBorrowedRegions`,
|
|
|
|
//! finds all of the regions in the MIR that are involved in a borrow.
|
|
|
|
//!
|
|
|
|
//! The second traversal, `DeleteTrivialEndRegions`, walks over the
|
|
|
|
//! MIR and removes any `EndRegion` that is applied to a region that
|
|
|
|
//! was not seen in the previous pass.
|
|
|
|
|
|
|
|
use rustc_data_structures::fx::FxHashSet;
|
|
|
|
|
|
|
|
use rustc::middle::region::CodeExtent;
|
|
|
|
use rustc::mir::transform::{MirPass, MirSource};
|
|
|
|
use rustc::mir::{BasicBlock, Location, Mir, Rvalue, Statement, StatementKind};
|
2017-07-11 16:30:30 -05:00
|
|
|
use rustc::mir::visit::{MutVisitor, Visitor, Lookup};
|
|
|
|
use rustc::ty::{Ty, RegionKind, TyCtxt};
|
2017-05-22 06:06:51 -05:00
|
|
|
|
|
|
|
pub struct CleanEndRegions;
|
|
|
|
|
|
|
|
struct GatherBorrowedRegions {
|
|
|
|
seen_regions: FxHashSet<CodeExtent>,
|
2017-07-11 16:30:30 -05:00
|
|
|
in_validation_statement: bool,
|
2017-05-22 06:06:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
struct DeleteTrivialEndRegions<'a> {
|
|
|
|
seen_regions: &'a FxHashSet<CodeExtent>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MirPass for CleanEndRegions {
|
|
|
|
fn run_pass<'a, 'tcx>(&self,
|
|
|
|
_tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
|
|
|
_source: MirSource,
|
|
|
|
mir: &mut Mir<'tcx>) {
|
2017-07-11 16:30:30 -05:00
|
|
|
let mut gather = GatherBorrowedRegions { seen_regions: FxHashSet(), in_validation_statement: false };
|
2017-05-22 06:06:51 -05:00
|
|
|
gather.visit_mir(mir);
|
|
|
|
|
|
|
|
let mut delete = DeleteTrivialEndRegions { seen_regions: &mut gather.seen_regions };
|
|
|
|
delete.visit_mir(mir);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'tcx> Visitor<'tcx> for GatherBorrowedRegions {
|
|
|
|
fn visit_rvalue(&mut self,
|
|
|
|
rvalue: &Rvalue<'tcx>,
|
|
|
|
location: Location) {
|
2017-07-11 16:30:30 -05:00
|
|
|
// Gather regions that are used for borrows
|
2017-05-22 06:06:51 -05:00
|
|
|
if let Rvalue::Ref(r, _, _) = *rvalue {
|
|
|
|
if let RegionKind::ReScope(ce) = *r {
|
|
|
|
self.seen_regions.insert(ce);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.super_rvalue(rvalue, location);
|
|
|
|
}
|
2017-07-11 16:30:30 -05:00
|
|
|
|
|
|
|
fn visit_statement(&mut self,
|
|
|
|
block: BasicBlock,
|
|
|
|
statement: &Statement<'tcx>,
|
|
|
|
location: Location) {
|
|
|
|
self.in_validation_statement = match statement.kind {
|
|
|
|
StatementKind::Validate(..) => true,
|
|
|
|
_ => false,
|
|
|
|
};
|
|
|
|
self.super_statement(block, statement, location);
|
|
|
|
self.in_validation_statement = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_ty(&mut self, ty: &Ty<'tcx>, _: Lookup) {
|
|
|
|
// Gather regions that occur in types inside AcquireValid/ReleaseValid statements
|
|
|
|
if self.in_validation_statement {
|
|
|
|
for re in ty.walk().flat_map(|t| t.regions()) {
|
|
|
|
match *re {
|
|
|
|
RegionKind::ReScope(ce) => { self.seen_regions.insert(ce); }
|
|
|
|
_ => {},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.super_ty(ty);
|
|
|
|
}
|
2017-05-22 06:06:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, 'tcx> MutVisitor<'tcx> for DeleteTrivialEndRegions<'a> {
|
|
|
|
fn visit_statement(&mut self,
|
|
|
|
block: BasicBlock,
|
|
|
|
statement: &mut Statement<'tcx>,
|
|
|
|
location: Location) {
|
|
|
|
let mut delete_it = false;
|
|
|
|
|
|
|
|
if let StatementKind::EndRegion(ref extent) = statement.kind {
|
|
|
|
if !self.seen_regions.contains(extent) {
|
|
|
|
delete_it = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if delete_it {
|
|
|
|
statement.kind = StatementKind::Nop;
|
|
|
|
}
|
|
|
|
self.super_statement(block, statement, location);
|
|
|
|
}
|
|
|
|
}
|