Skip to content

Commit 77a033a

Browse files
authored
Rollup merge of rust-lang#64005 - ecstatic-morse:is-indirect, r=oli-obk
Add a `Place::is_indirect` method to determine whether a `Place` contains a `Deref` projection Working on rust-lang#63860 requires tracking some property about each local. This requires differentiating `Place`s like `x` and `x.field[index]` from ones like `*x` and `*x.field`, since the first two will always access the same region of memory as `x` while the latter two may access any region of memory. This functionality is duplicated in various places across the compiler. This PR adds a helper method to `Place` which determines whether that `Place` has a `Deref` projection at any point and changes some existing code to use the new method. I've not converted `qualify_consts.rs` to use the new method, since it's not a trivial conversion and it will get replaced anyway by rust-lang#63860. There may be other potential uses besides the two I change in this PR. r? @oli-obk
2 parents a721221 + 96ac02b commit 77a033a

File tree

3 files changed

+36
-29
lines changed

3 files changed

+36
-29
lines changed

src/librustc/mir/mod.rs

+25
Original file line numberDiff line numberDiff line change
@@ -1808,6 +1808,23 @@ pub enum ProjectionElem<V, T> {
18081808
Downcast(Option<Symbol>, VariantIdx),
18091809
}
18101810

1811+
impl<V, T> ProjectionElem<V, T> {
1812+
/// Returns `true` if the target of this projection may refer to a different region of memory
1813+
/// than the base.
1814+
fn is_indirect(&self) -> bool {
1815+
match self {
1816+
Self::Deref => true,
1817+
1818+
| Self::Field(_, _)
1819+
| Self::Index(_)
1820+
| Self::ConstantIndex { .. }
1821+
| Self::Subslice { .. }
1822+
| Self::Downcast(_, _)
1823+
=> false
1824+
}
1825+
}
1826+
}
1827+
18111828
/// Alias for projections as they appear in places, where the base is a place
18121829
/// and the index is a local.
18131830
pub type PlaceElem<'tcx> = ProjectionElem<Local, Ty<'tcx>>;
@@ -1869,6 +1886,14 @@ impl<'tcx> Place<'tcx> {
18691886
}
18701887
}
18711888

1889+
/// Returns `true` if this `Place` contains a `Deref` projection.
1890+
///
1891+
/// If `Place::is_indirect` returns false, the caller knows that the `Place` refers to the
1892+
/// same region of memory as its base.
1893+
pub fn is_indirect(&self) -> bool {
1894+
self.iterate(|_, mut projections| projections.any(|proj| proj.elem.is_indirect()))
1895+
}
1896+
18721897
/// Finds the innermost `Local` from this `Place`, *if* it is either a local itself or
18731898
/// a single deref of a local.
18741899
//

src/librustc_mir/borrow_check/path_utils.rs

+7-16
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ use crate::borrow_check::places_conflict;
33
use crate::borrow_check::AccessDepth;
44
use crate::dataflow::indexes::BorrowIndex;
55
use rustc::mir::{BasicBlock, Location, Body, Place, PlaceBase};
6-
use rustc::mir::{ProjectionElem, BorrowKind};
6+
use rustc::mir::BorrowKind;
77
use rustc::ty::{self, TyCtxt};
88
use rustc_data_structures::graph::dominators::Dominators;
99

@@ -133,20 +133,11 @@ pub(super) fn is_active<'tcx>(
133133
/// Determines if a given borrow is borrowing local data
134134
/// This is called for all Yield statements on movable generators
135135
pub(super) fn borrow_of_local_data(place: &Place<'_>) -> bool {
136-
place.iterate(|place_base, place_projection| {
137-
match place_base {
138-
PlaceBase::Static(..) => return false,
139-
PlaceBase::Local(..) => {},
140-
}
141-
142-
for proj in place_projection {
143-
// Reborrow of already borrowed data is ignored
144-
// Any errors will be caught on the initial borrow
145-
if proj.elem == ProjectionElem::Deref {
146-
return false;
147-
}
148-
}
136+
match place.base {
137+
PlaceBase::Static(_) => false,
149138

150-
true
151-
})
139+
// Reborrow of already borrowed data is ignored
140+
// Any errors will be caught on the initial borrow
141+
PlaceBase::Local(_) => !place.is_indirect(),
142+
}
152143
}

src/librustc_mir/dataflow/impls/borrowed_locals.rs

+4-13
Original file line numberDiff line numberDiff line change
@@ -93,19 +93,10 @@ struct BorrowedLocalsVisitor<'gk> {
9393
}
9494

9595
fn find_local(place: &Place<'_>) -> Option<Local> {
96-
place.iterate(|place_base, place_projection| {
97-
for proj in place_projection {
98-
if proj.elem == ProjectionElem::Deref {
99-
return None;
100-
}
101-
}
102-
103-
if let PlaceBase::Local(local) = place_base {
104-
Some(*local)
105-
} else {
106-
None
107-
}
108-
})
96+
match place.base {
97+
PlaceBase::Local(local) if !place.is_indirect() => Some(local),
98+
_ => None,
99+
}
109100
}
110101

111102
impl<'tcx> Visitor<'tcx> for BorrowedLocalsVisitor<'_> {

0 commit comments

Comments
 (0)