Skip to content

Commit 05748c6

Browse files
authored
Rollup merge of #107271 - Zeegomo:drop-rmw, r=oli-obk
Treat Drop as a rmw operation Previously, a Drop terminator was considered a move in MIR. This commit changes the behavior to only treat Drop as a mutable access to the dropped place. In order for this change to be correct, we need to guarantee that 1. A dropped value won't be used again 2. Places that appear in a drop won't be used again before a subsequent initialization. We can ensure this to be correct at MIR construction because Drop will only be emitted when a variable goes out of scope, thus having: * (1) as there is no way of reaching the old value. drop-elaboration will also remove any uninitialized drop. * (2) as the place can't be named following the end of the scope. However, the initialization status, previously tracked by moves, should also be tied to the execution of a Drop, hence the additional logic in the dataflow analyses. From discussion in [this thread](https://rust-lang.zulipchat.com/#narrow/stream/233931-t-compiler.2Fmajor-changes/topic/.60DROP.60.20to.20.60DROP_IF.60.20compiler-team.23558), originating from rust-lang/compiler-team#558. See also #104488 (comment)
2 parents 562581c + 68c1e2f commit 05748c6

File tree

4 files changed

+46
-9
lines changed

4 files changed

+46
-9
lines changed

compiler/rustc_mir_dataflow/src/drop_flag_effects.rs

+12-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use crate::elaborate_drops::DropFlagState;
2-
use rustc_middle::mir::{self, Body, Location};
2+
use rustc_middle::mir::{self, Body, Location, Terminator, TerminatorKind};
33
use rustc_middle::ty::{self, TyCtxt};
44
use rustc_target::abi::VariantIdx;
55

@@ -194,6 +194,17 @@ pub fn drop_flag_effects_for_location<'tcx, F>(
194194
on_all_children_bits(tcx, body, move_data, path, |mpi| callback(mpi, DropFlagState::Absent))
195195
}
196196

197+
// Drop does not count as a move but we should still consider the variable uninitialized.
198+
if let Some(Terminator { kind: TerminatorKind::Drop { place, .. }, .. }) =
199+
body.stmt_at(loc).right()
200+
{
201+
if let LookupResult::Exact(mpi) = move_data.rev_lookup.find(place.as_ref()) {
202+
on_all_children_bits(tcx, body, move_data, mpi, |mpi| {
203+
callback(mpi, DropFlagState::Absent)
204+
})
205+
}
206+
}
207+
197208
debug!("drop_flag_effects: assignment for location({:?})", loc);
198209

199210
for_location_inits(tcx, body, move_data, loc, |mpi| callback(mpi, DropFlagState::Present));

compiler/rustc_mir_dataflow/src/move_paths/builder.rs

+2-5
Original file line numberDiff line numberDiff line change
@@ -376,7 +376,8 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
376376
| TerminatorKind::Resume
377377
| TerminatorKind::Abort
378378
| TerminatorKind::GeneratorDrop
379-
| TerminatorKind::Unreachable => {}
379+
| TerminatorKind::Unreachable
380+
| TerminatorKind::Drop { .. } => {}
380381

381382
TerminatorKind::Assert { ref cond, .. } => {
382383
self.gather_operand(cond);
@@ -391,10 +392,6 @@ impl<'b, 'a, 'tcx> Gatherer<'b, 'a, 'tcx> {
391392
self.create_move_path(place);
392393
self.gather_init(place.as_ref(), InitKind::Deep);
393394
}
394-
395-
TerminatorKind::Drop { place, target: _, unwind: _ } => {
396-
self.gather_move(place);
397-
}
398395
TerminatorKind::DropAndReplace { place, ref value, .. } => {
399396
self.create_move_path(place);
400397
self.gather_operand(value);

compiler/rustc_mir_dataflow/src/value_analysis.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -223,13 +223,13 @@ pub trait ValueAnalysis<'tcx> {
223223
self.super_terminator(terminator, state)
224224
}
225225

226-
fn super_terminator(&self, terminator: &Terminator<'tcx>, _state: &mut State<Self::Value>) {
226+
fn super_terminator(&self, terminator: &Terminator<'tcx>, state: &mut State<Self::Value>) {
227227
match &terminator.kind {
228228
TerminatorKind::Call { .. } | TerminatorKind::InlineAsm { .. } => {
229229
// Effect is applied by `handle_call_return`.
230230
}
231-
TerminatorKind::Drop { .. } => {
232-
// We don't track dropped places.
231+
TerminatorKind::Drop { place, .. } => {
232+
state.flood_with(place.as_ref(), self.map(), Self::Value::bottom());
233233
}
234234
TerminatorKind::DropAndReplace { .. } | TerminatorKind::Yield { .. } => {
235235
// They would have an effect, but are not allowed in this phase.

compiler/rustc_mir_transform/src/elaborate_drops.rs

+29
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,35 @@ use rustc_span::Span;
1818
use rustc_target::abi::VariantIdx;
1919
use std::fmt;
2020

21+
/// During MIR building, Drop and DropAndReplace terminators are inserted in every place where a drop may occur.
22+
/// However, in this phase, the presence of these terminators does not guarantee that a destructor will run,
23+
/// as the target of the drop may be uninitialized.
24+
/// In general, the compiler cannot determine at compile time whether a destructor will run or not.
25+
///
26+
/// At a high level, this pass refines Drop and DropAndReplace to only run the destructor if the
27+
/// target is initialized. The way this is achievied is by inserting drop flags for every variable
28+
/// that may be dropped, and then using those flags to determine whether a destructor should run.
29+
/// This pass also removes DropAndReplace, replacing it with a Drop paired with an assign statement.
30+
/// Once this is complete, Drop terminators in the MIR correspond to a call to the "drop glue" or
31+
/// "drop shim" for the type of the dropped place.
32+
///
33+
/// This pass relies on dropped places having an associated move path, which is then used to determine
34+
/// the initialization status of the place and its descendants.
35+
/// It's worth noting that a MIR containing a Drop without an associated move path is probably ill formed,
36+
/// as it would allow running a destructor on a place behind a reference:
37+
///
38+
/// ```text
39+
// fn drop_term<T>(t: &mut T) {
40+
// mir!(
41+
// {
42+
// Drop(*t, exit)
43+
// }
44+
// exit = {
45+
// Return()
46+
// }
47+
// )
48+
// }
49+
/// ```
2150
pub struct ElaborateDrops;
2251

2352
impl<'tcx> MirPass<'tcx> for ElaborateDrops {

0 commit comments

Comments
 (0)