mirror of
https://github.com/rust-lang/rust.git
synced 2025-02-21 19:33:16 +00:00
two-phase borrows: support multiple activations in one statement
The need for this has arisen since the introduction of two-phase borrows on method autorefs. Fixes 49635 Fixes 49662
This commit is contained in:
parent
17fea66ba4
commit
0801bd1d0e
@ -64,10 +64,7 @@ pub struct Borrows<'a, 'gcx: 'tcx, 'tcx: 'a> {
|
||||
assigned_map: FxHashMap<Place<'tcx>, FxHashSet<BorrowIndex>>,
|
||||
|
||||
/// Locations which activate borrows.
|
||||
/// NOTE: A given location may activate more than one borrow in the future
|
||||
/// when more general two-phase borrow support is introduced, but for now we
|
||||
/// only need to store one borrow index
|
||||
activation_map: FxHashMap<Location, BorrowIndex>,
|
||||
activation_map: FxHashMap<Location, FxHashSet<BorrowIndex>>,
|
||||
|
||||
/// Every borrow has a region; this maps each such regions back to
|
||||
/// its borrow-indexes.
|
||||
@ -174,7 +171,7 @@ impl<'a, 'gcx, 'tcx> Borrows<'a, 'gcx, 'tcx> {
|
||||
idx_vec: IndexVec<BorrowIndex, BorrowData<'tcx>>,
|
||||
location_map: FxHashMap<Location, BorrowIndex>,
|
||||
assigned_map: FxHashMap<Place<'tcx>, FxHashSet<BorrowIndex>>,
|
||||
activation_map: FxHashMap<Location, BorrowIndex>,
|
||||
activation_map: FxHashMap<Location, FxHashSet<BorrowIndex>>,
|
||||
region_map: FxHashMap<Region<'tcx>, FxHashSet<BorrowIndex>>,
|
||||
local_map: FxHashMap<mir::Local, FxHashSet<BorrowIndex>>,
|
||||
region_span_map: FxHashMap<RegionKind, Span>,
|
||||
@ -211,12 +208,7 @@ impl<'a, 'gcx, 'tcx> Borrows<'a, 'gcx, 'tcx> {
|
||||
let idx = self.idx_vec.push(borrow);
|
||||
self.location_map.insert(location, idx);
|
||||
|
||||
// This assert is a good sanity check until more general 2-phase borrow
|
||||
// support is introduced. See NOTE on the activation_map field for more
|
||||
assert!(!self.activation_map.contains_key(&activate_location),
|
||||
"More than one activation introduced at the same location.");
|
||||
self.activation_map.insert(activate_location, idx);
|
||||
|
||||
insert(&mut self.activation_map, &activate_location, idx);
|
||||
insert(&mut self.assigned_map, assigned_place, idx);
|
||||
insert(&mut self.region_map, ®ion, idx);
|
||||
if let Some(local) = root_local(borrowed_place) {
|
||||
@ -552,9 +544,11 @@ impl<'a, 'gcx, 'tcx> Borrows<'a, 'gcx, 'tcx> {
|
||||
location: Location) {
|
||||
// Handle activations
|
||||
match self.activation_map.get(&location) {
|
||||
Some(&activated) => {
|
||||
debug!("activating borrow {:?}", activated);
|
||||
sets.gen(&ReserveOrActivateIndex::active(activated))
|
||||
Some(activations) => {
|
||||
for activated in activations {
|
||||
debug!("activating borrow {:?}", activated);
|
||||
sets.gen(&ReserveOrActivateIndex::active(*activated))
|
||||
}
|
||||
}
|
||||
None => {}
|
||||
}
|
||||
|
35
src/test/ui/borrowck/two-phase-multiple-activations.rs
Normal file
35
src/test/ui/borrowck/two-phase-multiple-activations.rs
Normal file
@ -0,0 +1,35 @@
|
||||
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
|
||||
// file at the top-level directory of this distribution and at
|
||||
// http://rust-lang.org/COPYRIGHT.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 <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.
|
||||
|
||||
// revisions: lxl nll
|
||||
//[lxl]compile-flags: -Z borrowck=mir -Z two-phase-borrows
|
||||
//[nll]compile-flags: -Z borrowck=mir -Z two-phase-borrows -Z nll
|
||||
|
||||
// run-pass
|
||||
|
||||
use std::io::Result;
|
||||
|
||||
struct Foo {}
|
||||
|
||||
pub trait FakeRead {
|
||||
fn read_to_end(&mut self, buf: &mut Vec<u8>) -> Result<usize>;
|
||||
}
|
||||
|
||||
impl FakeRead for Foo {
|
||||
fn read_to_end(&mut self, buf: &mut Vec<u8>) -> Result<usize> {
|
||||
Ok(4)
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let mut a = Foo {};
|
||||
let mut v = Vec::new();
|
||||
a.read_to_end(&mut v);
|
||||
}
|
Loading…
Reference in New Issue
Block a user