Skip to content

Commit 82801b5

Browse files
committedDec 27, 2016
Auto merge of #38600 - arielb1:dead-drop, r=eddyb
clear discriminant drop flag at the bottom of a drop ladder Fixes #38437. Beta-nominating because serious I-wrong. r? @eddyb
2 parents 86896ba + 521b2ea commit 82801b5

File tree

2 files changed

+93
-38
lines changed

2 files changed

+93
-38
lines changed
 

‎src/librustc_borrowck/borrowck/mir/elaborate_drops.rs

+39-38
Original file line numberDiff line numberDiff line change
@@ -481,54 +481,55 @@ impl<'b, 'tcx> ElaborateDropsCtxt<'b, 'tcx> {
481481
is_cleanup: bool)
482482
-> Vec<BasicBlock>
483483
{
484-
let mut succ = succ;
485484
let mut unwind_succ = if is_cleanup {
486485
None
487486
} else {
488487
c.unwind
489488
};
490-
let mut update_drop_flag = true;
489+
490+
let mut succ = self.new_block(
491+
c, c.is_cleanup, TerminatorKind::Goto { target: succ }
492+
);
493+
494+
// Always clear the "master" drop flag at the bottom of the
495+
// ladder. This is needed because the "master" drop flag
496+
// protects the ADT's discriminant, which is invalidated
497+
// after the ADT is dropped.
498+
self.set_drop_flag(
499+
Location { block: succ, statement_index: 0 },
500+
c.path,
501+
DropFlagState::Absent
502+
);
491503

492504
fields.iter().rev().enumerate().map(|(i, &(ref lv, path))| {
493-
let drop_block = match path {
494-
Some(path) => {
495-
debug!("drop_ladder: for std field {} ({:?})", i, lv);
496-
497-
self.elaborated_drop_block(&DropCtxt {
498-
source_info: c.source_info,
499-
is_cleanup: is_cleanup,
500-
init_data: c.init_data,
501-
lvalue: lv,
502-
path: path,
503-
succ: succ,
504-
unwind: unwind_succ,
505-
})
506-
}
507-
None => {
508-
debug!("drop_ladder: for rest field {} ({:?})", i, lv);
509-
510-
let blk = self.complete_drop(&DropCtxt {
511-
source_info: c.source_info,
512-
is_cleanup: is_cleanup,
513-
init_data: c.init_data,
514-
lvalue: lv,
515-
path: c.path,
516-
succ: succ,
517-
unwind: unwind_succ,
518-
}, update_drop_flag);
519-
520-
// the drop flag has been updated - updating
521-
// it again would clobber it.
522-
update_drop_flag = false;
523-
524-
blk
525-
}
505+
succ = if let Some(path) = path {
506+
debug!("drop_ladder: for std field {} ({:?})", i, lv);
507+
508+
self.elaborated_drop_block(&DropCtxt {
509+
source_info: c.source_info,
510+
is_cleanup: is_cleanup,
511+
init_data: c.init_data,
512+
lvalue: lv,
513+
path: path,
514+
succ: succ,
515+
unwind: unwind_succ,
516+
})
517+
} else {
518+
debug!("drop_ladder: for rest field {} ({:?})", i, lv);
519+
520+
self.complete_drop(&DropCtxt {
521+
source_info: c.source_info,
522+
is_cleanup: is_cleanup,
523+
init_data: c.init_data,
524+
lvalue: lv,
525+
path: c.path,
526+
succ: succ,
527+
unwind: unwind_succ,
528+
}, false)
526529
};
527530

528-
succ = drop_block;
529531
unwind_succ = unwind_ladder.as_ref().map(|p| p[i]);
530-
531-
drop_block
532+
succ
532533
}).collect()
533534
}
534535

‎src/test/run-pass/issue-38437.rs

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// Check that drop elaboration clears the "master" discriminant
12+
// drop flag even if it protects no fields.
13+
14+
struct Good(usize);
15+
impl Drop for Good {
16+
#[inline(never)]
17+
fn drop(&mut self) {
18+
println!("dropping Good({})", self.0);
19+
}
20+
}
21+
22+
struct Void;
23+
impl Drop for Void {
24+
#[inline(never)]
25+
fn drop(&mut self) {
26+
panic!("Suddenly, a Void appears.");
27+
}
28+
}
29+
30+
enum E {
31+
Never(Void),
32+
Fine(Good)
33+
}
34+
35+
fn main() {
36+
let mut go = true;
37+
38+
loop {
39+
let next;
40+
match go {
41+
true => next = E::Fine(Good(123)),
42+
false => return,
43+
}
44+
45+
match next {
46+
E::Never(_) => return,
47+
E::Fine(_good) => go = false,
48+
}
49+
50+
// `next` is dropped and StorageDead'd here. We must reset the
51+
// discriminant's drop flag to avoid random variants being
52+
// dropped.
53+
}
54+
}

0 commit comments

Comments
 (0)
Please sign in to comment.