Skip to content

Commit

Permalink
PR fixup
Browse files Browse the repository at this point in the history
  • Loading branch information
arora-aman committed Jan 19, 2021
1 parent 936b70e commit b75dd0d
Show file tree
Hide file tree
Showing 3 changed files with 48 additions and 41 deletions.
85 changes: 46 additions & 39 deletions compiler/rustc_typeck/src/check/upvar.rs
Original file line number Diff line number Diff line change
Expand Up @@ -162,34 +162,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {

let closure_hir_id = self.tcx.hir().local_def_id_to_hir_id(local_def_id);
if should_do_migration_analysis(self.tcx, closure_hir_id) {
let need_migrations = self.compute_2229_migrations_first_pass(
closure_def_id,
span,
capture_clause,
body,
self.typeck_results.borrow().closure_min_captures.get(&closure_def_id),
);

if !need_migrations.is_empty() {
let need_migrations_hir_id =
need_migrations.iter().map(|m| m.0).collect::<Vec<_>>();

let migrations_text =
migration_suggestion_for_2229(self.tcx, &need_migrations_hir_id);

self.tcx.struct_span_lint_hir(
lint::builtin::DISJOINT_CAPTURE_DROP_REORDER,
closure_hir_id,
span,
|lint| {
let mut diagnostics_builder = lint.build(
"Drop order affected for closure because of `capture_disjoint_fields`",
);
diagnostics_builder.note(&migrations_text);
diagnostics_builder.emit();
},
);
}
self.perform_2229_migration_anaysis(closure_def_id, capture_clause, span, body);
}

// We now fake capture information for all variables that are mentioned within the closure
Expand Down Expand Up @@ -510,6 +483,45 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
debug!("For closure={:?}, min_captures={:#?}", closure_def_id, root_var_min_capture_list);
}

/// Perform the migration analysis for RFC 2229, and emit lint
/// `disjoint_capture_drop_reorder` if needed.
fn perform_2229_migration_anaysis(
&self,
closure_def_id: DefId,
capture_clause: hir::CaptureBy,
span: Span,
body: &'tcx hir::Body<'tcx>,
) {
let need_migrations = self.compute_2229_migrations_first_pass(
closure_def_id,
span,
capture_clause,
body,
self.typeck_results.borrow().closure_min_captures.get(&closure_def_id),
);

if !need_migrations.is_empty() {
let need_migrations_hir_id = need_migrations.iter().map(|m| m.0).collect::<Vec<_>>();

let migrations_text = migration_suggestion_for_2229(self.tcx, &need_migrations_hir_id);

let local_def_id = closure_def_id.expect_local();
let closure_hir_id = self.tcx.hir().local_def_id_to_hir_id(local_def_id);
self.tcx.struct_span_lint_hir(
lint::builtin::DISJOINT_CAPTURE_DROP_REORDER,
closure_hir_id,
span,
|lint| {
let mut diagnostics_builder = lint.build(
"Drop order affected for closure because of `capture_disjoint_fields`",
);
diagnostics_builder.note(&migrations_text);
diagnostics_builder.emit();
},
);
}
}

/// Figures out the list of root variables (and their types) that aren't completely
/// captured by the closure when `capture_disjoint_fields` is enabled and drop order of
/// some path starting at that root variable **might** be affected.
Expand Down Expand Up @@ -572,17 +584,12 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {

let is_moved = root_var_min_capture_list
.iter()
.find(|capture| matches!(capture.info.capture_kind, ty::UpvarCapture::ByValue(_)))
.is_some();

// 1. If we capture more than one path starting at the root variabe then the root variable
// isn't being captured in its entirety
// 2. If we only capture one path starting at the root variable, it's still possible
// that it isn't the root variable completely.
if is_moved
&& ((root_var_min_capture_list.len() > 1)
|| (root_var_min_capture_list[0].place.projections.len() > 0))
{
.any(|capture| matches!(capture.info.capture_kind, ty::UpvarCapture::ByValue(_)));

let is_not_completely_captured =
root_var_min_capture_list.iter().any(|capture| capture.place.projections.len() > 0);

if is_moved && is_not_completely_captured {
need_migrations.push((var_hir_id, ty));
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,7 @@ fn test6_move_closures_non_copy_types_might_need_migration() {
// Note we need migration here only because the non-copy (because Drop type) is captured,
// otherwise we won't need to, since we can get away with just by ref capture in that case.
fn test7_drop_non_drop_aggregate_need_migration() {
let t = (String::new(), 0i32);
let t = (String::new(), String::new(), 0i32);

let c = || {
//~^ERROR: Drop order affected for closure because of `capture_disjoint_fields`
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -85,7 +85,7 @@ fn test4_type_contains_drop_need_migration() {
// Note we need migration here only because the non-copy (because Drop type) is captured,
// otherwise we won't need to, since we can get away with just by ref capture in that case.
fn test5_drop_non_drop_aggregate_need_migration() {
let t = (Foo(0), 0i32);
let t = (Foo(0), Foo(0), 0i32);

let c = || {
//~^ERROR: Drop order affected for closure because of `capture_disjoint_fields`
Expand Down

0 comments on commit b75dd0d

Please sign in to comment.