Add additional migrations to handle auto-traits and clone traits
Combine all 2229 migrations under one flag name
This commit is contained in:
parent
109248a4eb
commit
9afea614bf
27 changed files with 666 additions and 83 deletions
|
@ -343,4 +343,7 @@ language_item_table! {
|
||||||
Range, sym::Range, range_struct, Target::Struct;
|
Range, sym::Range, range_struct, Target::Struct;
|
||||||
RangeToInclusive, sym::RangeToInclusive, range_to_inclusive_struct, Target::Struct;
|
RangeToInclusive, sym::RangeToInclusive, range_to_inclusive_struct, Target::Struct;
|
||||||
RangeTo, sym::RangeTo, range_to_struct, Target::Struct;
|
RangeTo, sym::RangeTo, range_to_struct, Target::Struct;
|
||||||
|
Send, sym::send, send_trait, Target::Trait;
|
||||||
|
UnwindSafe, sym::unwind_safe, unwind_safe_trait, Target::Trait;
|
||||||
|
RefUnwindSafe, sym::ref_unwind_safe, ref_unwind_safe_trait, Target::Trait;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2995,7 +2995,7 @@ declare_lint_pass! {
|
||||||
UNSUPPORTED_NAKED_FUNCTIONS,
|
UNSUPPORTED_NAKED_FUNCTIONS,
|
||||||
MISSING_ABI,
|
MISSING_ABI,
|
||||||
SEMICOLON_IN_EXPRESSIONS_FROM_MACROS,
|
SEMICOLON_IN_EXPRESSIONS_FROM_MACROS,
|
||||||
DISJOINT_CAPTURE_DROP_REORDER,
|
DISJOINT_CAPTURE_MIGRATION,
|
||||||
LEGACY_DERIVE_HELPERS,
|
LEGACY_DERIVE_HELPERS,
|
||||||
PROC_MACRO_BACK_COMPAT,
|
PROC_MACRO_BACK_COMPAT,
|
||||||
OR_PATTERNS_BACK_COMPAT,
|
OR_PATTERNS_BACK_COMPAT,
|
||||||
|
@ -3027,14 +3027,17 @@ declare_lint! {
|
||||||
}
|
}
|
||||||
|
|
||||||
declare_lint! {
|
declare_lint! {
|
||||||
/// The `disjoint_capture_drop_reorder` lint detects variables that aren't completely
|
/// The `disjoint_capture_migration` lint detects variables that aren't completely
|
||||||
/// captured when the feature `capture_disjoint_fields` is enabled and it affects the Drop
|
/// captured when the feature `capture_disjoint_fields` is enabled and it affects the Drop
|
||||||
/// order of at least one path starting at this variable.
|
/// order of at least one path starting at this variable.
|
||||||
|
/// It can also detect when a variable implements a trait, but one of its field does not and
|
||||||
|
/// the field is captured by a closure and used with the assumption that said field implements
|
||||||
|
/// the same trait as the root variable.
|
||||||
///
|
///
|
||||||
/// ### Example
|
/// ### Example of drop reorder
|
||||||
///
|
///
|
||||||
/// ```rust,compile_fail
|
/// ```rust,compile_fail
|
||||||
/// # #![deny(disjoint_capture_drop_reorder)]
|
/// # #![deny(disjoint_capture_migration)]
|
||||||
/// # #![allow(unused)]
|
/// # #![allow(unused)]
|
||||||
/// struct FancyInteger(i32);
|
/// struct FancyInteger(i32);
|
||||||
///
|
///
|
||||||
|
@ -3065,10 +3068,35 @@ declare_lint! {
|
||||||
///
|
///
|
||||||
/// In the above example `p.y` will be dropped at the end of `f` instead of with `c` if
|
/// In the above example `p.y` will be dropped at the end of `f` instead of with `c` if
|
||||||
/// the feature `capture_disjoint_fields` is enabled.
|
/// the feature `capture_disjoint_fields` is enabled.
|
||||||
pub DISJOINT_CAPTURE_DROP_REORDER,
|
///
|
||||||
|
/// ### Example of auto-trait
|
||||||
|
///
|
||||||
|
/// ```rust,compile_fail
|
||||||
|
/// #![deny(disjoint_capture_migration)]
|
||||||
|
/// use std::thread;
|
||||||
|
///
|
||||||
|
/// struct Pointer (*mut i32);
|
||||||
|
/// unsafe impl Send for Pointer {}
|
||||||
|
///
|
||||||
|
/// fn main() {
|
||||||
|
/// let mut f = 10;
|
||||||
|
/// let fptr = Pointer(&mut f as *mut i32);
|
||||||
|
/// thread::spawn(move || unsafe {
|
||||||
|
/// *fptr.0 = 20;
|
||||||
|
/// });
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// {{produces}}
|
||||||
|
///
|
||||||
|
/// ### Explanation
|
||||||
|
///
|
||||||
|
/// In the above example `fptr.0` is captured when feature `capture_disjoint_fields` is enabled.
|
||||||
|
/// The field is of type *mut i32 which doesn't implement Send, making the code invalid as the
|
||||||
|
/// field cannot be sent between thread safely.
|
||||||
|
pub DISJOINT_CAPTURE_MIGRATION,
|
||||||
Allow,
|
Allow,
|
||||||
"Drop reorder because of `capture_disjoint_fields`"
|
"Drop reorder and auto traits error because of `capture_disjoint_fields`"
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
declare_lint_pass!(UnusedDocComment => [UNUSED_DOC_COMMENTS]);
|
declare_lint_pass!(UnusedDocComment => [UNUSED_DOC_COMMENTS]);
|
||||||
|
|
|
@ -933,6 +933,7 @@ symbols! {
|
||||||
receiver,
|
receiver,
|
||||||
recursion_limit,
|
recursion_limit,
|
||||||
reexport_test_harness_main,
|
reexport_test_harness_main,
|
||||||
|
ref_unwind_safe,
|
||||||
reference,
|
reference,
|
||||||
reflect,
|
reflect,
|
||||||
reg,
|
reg,
|
||||||
|
@ -1054,6 +1055,7 @@ symbols! {
|
||||||
self_in_typedefs,
|
self_in_typedefs,
|
||||||
self_struct_ctor,
|
self_struct_ctor,
|
||||||
semitransparent,
|
semitransparent,
|
||||||
|
send,
|
||||||
send_trait,
|
send_trait,
|
||||||
shl,
|
shl,
|
||||||
shl_assign,
|
shl_assign,
|
||||||
|
@ -1275,6 +1277,7 @@ symbols! {
|
||||||
unused_qualifications,
|
unused_qualifications,
|
||||||
unwind,
|
unwind,
|
||||||
unwind_attributes,
|
unwind_attributes,
|
||||||
|
unwind_safe,
|
||||||
unwrap,
|
unwrap,
|
||||||
unwrap_or,
|
unwrap_or,
|
||||||
use_extern_macros,
|
use_extern_macros,
|
||||||
|
|
|
@ -42,11 +42,13 @@ use rustc_hir::intravisit::{self, NestedVisitorMap, Visitor};
|
||||||
use rustc_infer::infer::UpvarRegion;
|
use rustc_infer::infer::UpvarRegion;
|
||||||
use rustc_middle::hir::place::{Place, PlaceBase, PlaceWithHirId, Projection, ProjectionKind};
|
use rustc_middle::hir::place::{Place, PlaceBase, PlaceWithHirId, Projection, ProjectionKind};
|
||||||
use rustc_middle::mir::FakeReadCause;
|
use rustc_middle::mir::FakeReadCause;
|
||||||
use rustc_middle::ty::{self, Ty, TyCtxt, TypeckResults, UpvarSubsts};
|
use rustc_middle::ty::{self, TraitRef, Ty, TyCtxt, TypeckResults, UpvarSubsts};
|
||||||
use rustc_session::lint;
|
use rustc_session::lint;
|
||||||
use rustc_span::sym;
|
use rustc_span::sym;
|
||||||
use rustc_span::{MultiSpan, Span, Symbol};
|
use rustc_span::{MultiSpan, Span, Symbol};
|
||||||
|
use rustc_trait_selection::traits::{Obligation, ObligationCause};
|
||||||
|
|
||||||
|
use rustc_data_structures::stable_set::FxHashSet;
|
||||||
use rustc_index::vec::Idx;
|
use rustc_index::vec::Idx;
|
||||||
use rustc_target::abi::VariantIdx;
|
use rustc_target::abi::VariantIdx;
|
||||||
|
|
||||||
|
@ -168,7 +170,8 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
self.compute_min_captures(closure_def_id, delegate.capture_information);
|
self.compute_min_captures(closure_def_id, delegate.capture_information);
|
||||||
|
|
||||||
let closure_hir_id = self.tcx.hir().local_def_id_to_hir_id(local_def_id);
|
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) {
|
|
||||||
|
if should_do_disjoint_capture_migration_analysis(self.tcx, closure_hir_id) {
|
||||||
self.perform_2229_migration_anaysis(closure_def_id, body_id, capture_clause, span);
|
self.perform_2229_migration_anaysis(closure_def_id, body_id, capture_clause, span);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -471,7 +474,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
capture_clause: hir::CaptureBy,
|
capture_clause: hir::CaptureBy,
|
||||||
span: Span,
|
span: Span,
|
||||||
) {
|
) {
|
||||||
let need_migrations = self.compute_2229_migrations(
|
let (need_migrations, reasons) = self.compute_2229_migrations(
|
||||||
closure_def_id,
|
closure_def_id,
|
||||||
span,
|
span,
|
||||||
capture_clause,
|
capture_clause,
|
||||||
|
@ -485,12 +488,16 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
let local_def_id = closure_def_id.expect_local();
|
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);
|
let closure_hir_id = self.tcx.hir().local_def_id_to_hir_id(local_def_id);
|
||||||
self.tcx.struct_span_lint_hir(
|
self.tcx.struct_span_lint_hir(
|
||||||
lint::builtin::DISJOINT_CAPTURE_DROP_REORDER,
|
lint::builtin::DISJOINT_CAPTURE_MIGRATION,
|
||||||
closure_hir_id,
|
closure_hir_id,
|
||||||
span,
|
span,
|
||||||
|lint| {
|
|lint| {
|
||||||
let mut diagnostics_builder = lint.build(
|
let mut diagnostics_builder = lint.build(
|
||||||
"drop order affected for closure because of `capture_disjoint_fields`",
|
format!(
|
||||||
|
"{} affected for closure because of `capture_disjoint_fields`",
|
||||||
|
reasons
|
||||||
|
)
|
||||||
|
.as_str(),
|
||||||
);
|
);
|
||||||
let closure_body_span = self.tcx.hir().span(body_id.hir_id);
|
let closure_body_span = self.tcx.hir().span(body_id.hir_id);
|
||||||
let (sugg, app) =
|
let (sugg, app) =
|
||||||
|
@ -527,6 +534,188 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Combines all the reasons for 2229 migrations
|
||||||
|
fn compute_2229_migrations_reasons(
|
||||||
|
&self,
|
||||||
|
auto_trait_reasons: FxHashSet<&str>,
|
||||||
|
drop_reason: bool,
|
||||||
|
) -> String {
|
||||||
|
let mut reasons = String::new();
|
||||||
|
|
||||||
|
if auto_trait_reasons.len() > 0 {
|
||||||
|
reasons = format!(
|
||||||
|
"{} trait implementation",
|
||||||
|
auto_trait_reasons.clone().into_iter().collect::<Vec<&str>>().join(", ")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if auto_trait_reasons.len() > 0 && drop_reason {
|
||||||
|
reasons = format!("{}, and ", reasons);
|
||||||
|
}
|
||||||
|
|
||||||
|
if drop_reason {
|
||||||
|
reasons = format!("{}drop order", reasons);
|
||||||
|
}
|
||||||
|
|
||||||
|
reasons
|
||||||
|
}
|
||||||
|
|
||||||
|
fn ty_contains_trait(
|
||||||
|
&self,
|
||||||
|
ty: Ty<'tcx>,
|
||||||
|
cause: &ObligationCause<'tcx>,
|
||||||
|
trait_def_id: DefId,
|
||||||
|
) -> bool {
|
||||||
|
use crate::rustc_middle::ty::ToPredicate;
|
||||||
|
use crate::rustc_middle::ty::WithConstness;
|
||||||
|
use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt;
|
||||||
|
let tcx = self.infcx.tcx;
|
||||||
|
|
||||||
|
let trait_ref = TraitRef { def_id: trait_def_id, substs: tcx.mk_substs_trait(ty, &[]) };
|
||||||
|
|
||||||
|
let obligation = Obligation::new(
|
||||||
|
cause.clone(),
|
||||||
|
self.param_env,
|
||||||
|
trait_ref.without_const().to_predicate(tcx),
|
||||||
|
);
|
||||||
|
|
||||||
|
self.infcx.predicate_may_hold(&obligation)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 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 auto-traits
|
||||||
|
/// differ between the root variable and the captured paths.
|
||||||
|
///
|
||||||
|
/// The output list would include a root variable if:
|
||||||
|
/// - It would have been captured into the closure when `capture_disjoint_fields` wasn't
|
||||||
|
/// enabled, **and**
|
||||||
|
/// - It wasn't completely captured by the closure, **and**
|
||||||
|
/// - One of the paths captured does not implement all the auto-traits its root variable
|
||||||
|
/// implements.
|
||||||
|
fn compute_2229_migrations_for_trait(
|
||||||
|
&self,
|
||||||
|
min_captures: Option<&ty::RootVariableMinCaptureList<'tcx>>,
|
||||||
|
_closure_clause: hir::CaptureBy,
|
||||||
|
var_hir_id: hir::HirId,
|
||||||
|
) -> Option<FxHashSet<&str>> {
|
||||||
|
let root_var_min_capture_list = if let Some(root_var_min_capture_list) =
|
||||||
|
min_captures.and_then(|m| m.get(&var_hir_id))
|
||||||
|
{
|
||||||
|
root_var_min_capture_list
|
||||||
|
} else {
|
||||||
|
return None;
|
||||||
|
};
|
||||||
|
|
||||||
|
let ty = self.infcx.resolve_vars_if_possible(self.node_ty(var_hir_id));
|
||||||
|
|
||||||
|
let tcx = self.infcx.tcx;
|
||||||
|
|
||||||
|
let cause = ObligationCause::misc(self.tcx.hir().span(var_hir_id), self.body_id);
|
||||||
|
|
||||||
|
let clone_obligation_should_hold = tcx
|
||||||
|
.lang_items()
|
||||||
|
.clone_trait()
|
||||||
|
.map(|clone_trait| self.ty_contains_trait(ty, &cause, clone_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let sync_obligation_should_hold = tcx
|
||||||
|
.lang_items()
|
||||||
|
.sync_trait()
|
||||||
|
.map(|sync_trait| self.ty_contains_trait(ty, &cause, sync_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let send_obligation_should_hold = tcx
|
||||||
|
.lang_items()
|
||||||
|
.send_trait()
|
||||||
|
.map(|send_trait| self.ty_contains_trait(ty, &cause, send_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let unpin_obligation_should_hold = tcx
|
||||||
|
.lang_items()
|
||||||
|
.unpin_trait()
|
||||||
|
.map(|unpin_trait| self.ty_contains_trait(ty, &cause, unpin_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let unwind_safe_obligation_should_hold = tcx
|
||||||
|
.lang_items()
|
||||||
|
.unwind_safe_trait()
|
||||||
|
.map(|unwind_safe_trait| self.ty_contains_trait(ty, &cause, unwind_safe_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let ref_unwind_safe_obligation_should_hold = tcx
|
||||||
|
.lang_items()
|
||||||
|
.ref_unwind_safe_trait()
|
||||||
|
.map(|ref_unwind_safe_trait| self.ty_contains_trait(ty, &cause, ref_unwind_safe_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
|
||||||
|
// Check whether catpured fields also implement the trait
|
||||||
|
let mut auto_trait_reasons = FxHashSet::default();
|
||||||
|
|
||||||
|
for capture in root_var_min_capture_list.iter() {
|
||||||
|
let ty = capture.place.ty();
|
||||||
|
|
||||||
|
let clone_obligation_holds_for_capture = tcx
|
||||||
|
.lang_items()
|
||||||
|
.clone_trait()
|
||||||
|
.map(|clone_trait| self.ty_contains_trait(ty, &cause, clone_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let sync_obligation_holds_for_capture = tcx
|
||||||
|
.lang_items()
|
||||||
|
.sync_trait()
|
||||||
|
.map(|sync_trait| self.ty_contains_trait(ty, &cause, sync_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let send_obligation_holds_for_capture = tcx
|
||||||
|
.lang_items()
|
||||||
|
.send_trait()
|
||||||
|
.map(|send_trait| self.ty_contains_trait(ty, &cause, send_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let unpin_obligation_holds_for_capture = tcx
|
||||||
|
.lang_items()
|
||||||
|
.unpin_trait()
|
||||||
|
.map(|unpin_trait| self.ty_contains_trait(ty, &cause, unpin_trait))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let unwind_safe_obligation_holds_for_capture = tcx
|
||||||
|
.lang_items()
|
||||||
|
.unwind_safe_trait()
|
||||||
|
.map(|unwind_safe| self.ty_contains_trait(ty, &cause, unwind_safe))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let ref_unwind_safe_obligation_holds_for_capture = tcx
|
||||||
|
.lang_items()
|
||||||
|
.ref_unwind_safe_trait()
|
||||||
|
.map(|ref_unwind_safe_trait| {
|
||||||
|
self.ty_contains_trait(ty, &cause, ref_unwind_safe_trait)
|
||||||
|
})
|
||||||
|
.unwrap_or(false);
|
||||||
|
|
||||||
|
if !clone_obligation_holds_for_capture && clone_obligation_should_hold {
|
||||||
|
auto_trait_reasons.insert("`Clone`");
|
||||||
|
}
|
||||||
|
|
||||||
|
if !sync_obligation_holds_for_capture && sync_obligation_should_hold {
|
||||||
|
auto_trait_reasons.insert("`Sync`");
|
||||||
|
}
|
||||||
|
|
||||||
|
if !send_obligation_holds_for_capture && send_obligation_should_hold {
|
||||||
|
auto_trait_reasons.insert("`Send`");
|
||||||
|
}
|
||||||
|
|
||||||
|
if !unpin_obligation_holds_for_capture && unpin_obligation_should_hold {
|
||||||
|
auto_trait_reasons.insert("`Unpin`");
|
||||||
|
}
|
||||||
|
|
||||||
|
if !unwind_safe_obligation_holds_for_capture && unwind_safe_obligation_should_hold {
|
||||||
|
auto_trait_reasons.insert("`UnwindSafe`");
|
||||||
|
}
|
||||||
|
|
||||||
|
if !ref_unwind_safe_obligation_holds_for_capture
|
||||||
|
&& ref_unwind_safe_obligation_should_hold
|
||||||
|
{
|
||||||
|
auto_trait_reasons.insert("`RefUnwindSafe`");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if auto_trait_reasons.len() > 0 {
|
||||||
|
return Some(auto_trait_reasons);
|
||||||
|
}
|
||||||
|
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
/// Figures out the list of root variables (and their types) that aren't completely
|
/// 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
|
/// captured by the closure when `capture_disjoint_fields` is enabled and drop order of
|
||||||
/// some path starting at that root variable **might** be affected.
|
/// some path starting at that root variable **might** be affected.
|
||||||
|
@ -536,26 +725,18 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
/// enabled, **and**
|
/// enabled, **and**
|
||||||
/// - It wasn't completely captured by the closure, **and**
|
/// - It wasn't completely captured by the closure, **and**
|
||||||
/// - One of the paths starting at this root variable, that is not captured needs Drop.
|
/// - One of the paths starting at this root variable, that is not captured needs Drop.
|
||||||
fn compute_2229_migrations(
|
fn compute_2229_migrations_for_drop(
|
||||||
&self,
|
&self,
|
||||||
closure_def_id: DefId,
|
closure_def_id: DefId,
|
||||||
closure_span: Span,
|
closure_span: Span,
|
||||||
closure_clause: hir::CaptureBy,
|
|
||||||
min_captures: Option<&ty::RootVariableMinCaptureList<'tcx>>,
|
min_captures: Option<&ty::RootVariableMinCaptureList<'tcx>>,
|
||||||
) -> Vec<hir::HirId> {
|
closure_clause: hir::CaptureBy,
|
||||||
let upvars = if let Some(upvars) = self.tcx.upvars_mentioned(closure_def_id) {
|
var_hir_id: hir::HirId,
|
||||||
upvars
|
) -> Option<()> {
|
||||||
} else {
|
|
||||||
return vec![];
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut need_migrations = Vec::new();
|
|
||||||
|
|
||||||
for (&var_hir_id, _) in upvars.iter() {
|
|
||||||
let ty = self.infcx.resolve_vars_if_possible(self.node_ty(var_hir_id));
|
let ty = self.infcx.resolve_vars_if_possible(self.node_ty(var_hir_id));
|
||||||
|
|
||||||
if !ty.needs_drop(self.tcx, self.tcx.param_env(closure_def_id.expect_local())) {
|
if !ty.needs_drop(self.tcx, self.tcx.param_env(closure_def_id.expect_local())) {
|
||||||
continue;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
let root_var_min_capture_list = if let Some(root_var_min_capture_list) =
|
let root_var_min_capture_list = if let Some(root_var_min_capture_list) =
|
||||||
|
@ -568,21 +749,18 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
|
|
||||||
match closure_clause {
|
match closure_clause {
|
||||||
// Only migrate if closure is a move closure
|
// Only migrate if closure is a move closure
|
||||||
hir::CaptureBy::Value => need_migrations.push(var_hir_id),
|
hir::CaptureBy::Value => return Some(()),
|
||||||
|
|
||||||
hir::CaptureBy::Ref => {}
|
hir::CaptureBy::Ref => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
continue;
|
return None;
|
||||||
};
|
};
|
||||||
|
|
||||||
let projections_list = root_var_min_capture_list
|
let projections_list = root_var_min_capture_list
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|captured_place| match captured_place.info.capture_kind {
|
.filter_map(|captured_place| match captured_place.info.capture_kind {
|
||||||
// Only care about captures that are moved into the closure
|
// Only care about captures that are moved into the closure
|
||||||
ty::UpvarCapture::ByValue(..) => {
|
ty::UpvarCapture::ByValue(..) => Some(captured_place.place.projections.as_slice()),
|
||||||
Some(captured_place.place.projections.as_slice())
|
|
||||||
}
|
|
||||||
ty::UpvarCapture::ByRef(..) => None,
|
ty::UpvarCapture::ByRef(..) => None,
|
||||||
})
|
})
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
@ -601,11 +779,72 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
projections_list,
|
projections_list,
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
return Some(());
|
||||||
|
}
|
||||||
|
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 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 either drop
|
||||||
|
/// order of some path starting at that root variable **might** be affected or auto-traits
|
||||||
|
/// differ between the root variable and the captured paths.
|
||||||
|
///
|
||||||
|
/// The output list would include a root variable if:
|
||||||
|
/// - It would have been moved into the closure when `capture_disjoint_fields` wasn't
|
||||||
|
/// enabled, **and**
|
||||||
|
/// - It wasn't completely captured by the closure, **and**
|
||||||
|
/// - One of the paths starting at this root variable, that is not captured needs Drop **or**
|
||||||
|
/// - One of the paths captured does not implement all the auto-traits its root variable
|
||||||
|
/// implements.
|
||||||
|
fn compute_2229_migrations(
|
||||||
|
&self,
|
||||||
|
closure_def_id: DefId,
|
||||||
|
closure_span: Span,
|
||||||
|
closure_clause: hir::CaptureBy,
|
||||||
|
min_captures: Option<&ty::RootVariableMinCaptureList<'tcx>>,
|
||||||
|
) -> (Vec<hir::HirId>, String) {
|
||||||
|
let upvars = if let Some(upvars) = self.tcx.upvars_mentioned(closure_def_id) {
|
||||||
|
upvars
|
||||||
|
} else {
|
||||||
|
return (Vec::new(), format!(""));
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut need_migrations = Vec::new();
|
||||||
|
let mut auto_trait_reasons = FxHashSet::default();
|
||||||
|
let mut drop_reorder_reason = false;
|
||||||
|
|
||||||
|
// Perform auto-trait analysis
|
||||||
|
for (&var_hir_id, _) in upvars.iter() {
|
||||||
|
// println!("CHeck auto traits");
|
||||||
|
let mut need_some_migrations = false;
|
||||||
|
if let Some(trait_migration_cause) =
|
||||||
|
self.compute_2229_migrations_for_trait(min_captures, closure_clause, var_hir_id)
|
||||||
|
{
|
||||||
|
need_some_migrations = true;
|
||||||
|
auto_trait_reasons.extend(trait_migration_cause);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(_) = self.compute_2229_migrations_for_drop(
|
||||||
|
closure_def_id,
|
||||||
|
closure_span,
|
||||||
|
min_captures,
|
||||||
|
closure_clause,
|
||||||
|
var_hir_id,
|
||||||
|
) {
|
||||||
|
need_some_migrations = true;
|
||||||
|
drop_reorder_reason = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if need_some_migrations {
|
||||||
need_migrations.push(var_hir_id);
|
need_migrations.push(var_hir_id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
need_migrations
|
(
|
||||||
|
need_migrations,
|
||||||
|
self.compute_2229_migrations_reasons(auto_trait_reasons, drop_reorder_reason),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This is a helper function to `compute_2229_migrations_precise_pass`. Provided the type
|
/// This is a helper function to `compute_2229_migrations_precise_pass`. Provided the type
|
||||||
|
@ -1544,9 +1783,8 @@ fn var_name(tcx: TyCtxt<'_>, var_hir_id: hir::HirId) -> Symbol {
|
||||||
tcx.hir().name(var_hir_id)
|
tcx.hir().name(var_hir_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn should_do_migration_analysis(tcx: TyCtxt<'_>, closure_id: hir::HirId) -> bool {
|
fn should_do_disjoint_capture_migration_analysis(tcx: TyCtxt<'_>, closure_id: hir::HirId) -> bool {
|
||||||
let (level, _) =
|
let (level, _) = tcx.lint_level_at_node(lint::builtin::DISJOINT_CAPTURE_MIGRATION, closure_id);
|
||||||
tcx.lint_level_at_node(lint::builtin::DISJOINT_CAPTURE_DROP_REORDER, closure_id);
|
|
||||||
|
|
||||||
!matches!(level, lint::Level::Allow)
|
!matches!(level, lint::Level::Allow)
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,6 +31,7 @@ use crate::hash::Hasher;
|
||||||
/// [ub]: ../../reference/behavior-considered-undefined.html
|
/// [ub]: ../../reference/behavior-considered-undefined.html
|
||||||
#[stable(feature = "rust1", since = "1.0.0")]
|
#[stable(feature = "rust1", since = "1.0.0")]
|
||||||
#[cfg_attr(not(test), rustc_diagnostic_item = "send_trait")]
|
#[cfg_attr(not(test), rustc_diagnostic_item = "send_trait")]
|
||||||
|
#[cfg_attr(not(bootstrap), lang = "send")]
|
||||||
#[rustc_on_unimplemented(
|
#[rustc_on_unimplemented(
|
||||||
message = "`{Self}` cannot be sent between threads safely",
|
message = "`{Self}` cannot be sent between threads safely",
|
||||||
label = "`{Self}` cannot be sent between threads safely"
|
label = "`{Self}` cannot be sent between threads safely"
|
||||||
|
|
|
@ -132,6 +132,7 @@ pub fn panic_any<M: 'static + Any + Send>(msg: M) -> ! {
|
||||||
/// [`AssertUnwindSafe`] wrapper struct can be used to force this trait to be
|
/// [`AssertUnwindSafe`] wrapper struct can be used to force this trait to be
|
||||||
/// implemented for any closed over variables passed to `catch_unwind`.
|
/// implemented for any closed over variables passed to `catch_unwind`.
|
||||||
#[stable(feature = "catch_unwind", since = "1.9.0")]
|
#[stable(feature = "catch_unwind", since = "1.9.0")]
|
||||||
|
#[cfg_attr(not(bootstrap), lang = "unwind_safe")]
|
||||||
#[rustc_on_unimplemented(
|
#[rustc_on_unimplemented(
|
||||||
message = "the type `{Self}` may not be safely transferred across an unwind boundary",
|
message = "the type `{Self}` may not be safely transferred across an unwind boundary",
|
||||||
label = "`{Self}` may not be safely transferred across an unwind boundary"
|
label = "`{Self}` may not be safely transferred across an unwind boundary"
|
||||||
|
@ -147,6 +148,7 @@ pub auto trait UnwindSafe {}
|
||||||
/// This is a "helper marker trait" used to provide impl blocks for the
|
/// This is a "helper marker trait" used to provide impl blocks for the
|
||||||
/// [`UnwindSafe`] trait, for more information see that documentation.
|
/// [`UnwindSafe`] trait, for more information see that documentation.
|
||||||
#[stable(feature = "catch_unwind", since = "1.9.0")]
|
#[stable(feature = "catch_unwind", since = "1.9.0")]
|
||||||
|
#[cfg_attr(not(bootstrap), lang = "ref_unwind_safe")]
|
||||||
#[rustc_on_unimplemented(
|
#[rustc_on_unimplemented(
|
||||||
message = "the type `{Self}` may contain interior mutability and a reference may not be safely \
|
message = "the type `{Self}` may contain interior mutability and a reference may not be safely \
|
||||||
transferrable across a catch_unwind boundary",
|
transferrable across a catch_unwind boundary",
|
||||||
|
|
|
@ -0,0 +1,67 @@
|
||||||
|
// run-rustfix
|
||||||
|
#![deny(disjoint_capture_migration)]
|
||||||
|
|
||||||
|
use std::thread;
|
||||||
|
|
||||||
|
/* Test Send Trait */
|
||||||
|
struct SendPointer (*mut i32);
|
||||||
|
unsafe impl Send for SendPointer {}
|
||||||
|
|
||||||
|
fn test_send_trait() {
|
||||||
|
let mut f = 10;
|
||||||
|
let fptr = SendPointer(&mut f as *mut i32);
|
||||||
|
thread::spawn(move || { let _ = &fptr; unsafe {
|
||||||
|
//~^ ERROR: `Send` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `fptr` to be fully captured
|
||||||
|
*fptr.0 = 20;
|
||||||
|
} });
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Test Sync Trait */
|
||||||
|
struct CustomInt (*mut i32);
|
||||||
|
struct SyncPointer (CustomInt);
|
||||||
|
unsafe impl Sync for SyncPointer {}
|
||||||
|
unsafe impl Send for CustomInt {}
|
||||||
|
|
||||||
|
fn test_sync_trait() {
|
||||||
|
let mut f = 10;
|
||||||
|
let f = CustomInt(&mut f as *mut i32);
|
||||||
|
let fptr = SyncPointer(f);
|
||||||
|
thread::spawn(move || { let _ = &fptr; unsafe {
|
||||||
|
//~^ ERROR: `Sync`, `Send` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `fptr` to be fully captured
|
||||||
|
*fptr.0.0 = 20;
|
||||||
|
} });
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Test Clone Trait */
|
||||||
|
struct S(String);
|
||||||
|
struct T(i32);
|
||||||
|
|
||||||
|
struct U(S,T);
|
||||||
|
|
||||||
|
impl Clone for U {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
U(S(String::from("Hello World")), T(0))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn test_clone_trait() {
|
||||||
|
let f = U(S(String::from("Hello World")), T(0));
|
||||||
|
let c = || { let _ = &f;
|
||||||
|
//~^ ERROR: `Clone` trait implementation, and drop order affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `f` to be fully captured
|
||||||
|
let f_1 = f.1;
|
||||||
|
println!("{:?}", f_1.0);
|
||||||
|
};
|
||||||
|
|
||||||
|
let c_clone = c.clone();
|
||||||
|
|
||||||
|
c_clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
test_send_trait();
|
||||||
|
test_sync_trait();
|
||||||
|
test_clone_trait();
|
||||||
|
}
|
|
@ -0,0 +1,67 @@
|
||||||
|
// run-rustfix
|
||||||
|
#![deny(disjoint_capture_migration)]
|
||||||
|
|
||||||
|
use std::thread;
|
||||||
|
|
||||||
|
/* Test Send Trait */
|
||||||
|
struct SendPointer (*mut i32);
|
||||||
|
unsafe impl Send for SendPointer {}
|
||||||
|
|
||||||
|
fn test_send_trait() {
|
||||||
|
let mut f = 10;
|
||||||
|
let fptr = SendPointer(&mut f as *mut i32);
|
||||||
|
thread::spawn(move || unsafe {
|
||||||
|
//~^ ERROR: `Send` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `fptr` to be fully captured
|
||||||
|
*fptr.0 = 20;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Test Sync Trait */
|
||||||
|
struct CustomInt (*mut i32);
|
||||||
|
struct SyncPointer (CustomInt);
|
||||||
|
unsafe impl Sync for SyncPointer {}
|
||||||
|
unsafe impl Send for CustomInt {}
|
||||||
|
|
||||||
|
fn test_sync_trait() {
|
||||||
|
let mut f = 10;
|
||||||
|
let f = CustomInt(&mut f as *mut i32);
|
||||||
|
let fptr = SyncPointer(f);
|
||||||
|
thread::spawn(move || unsafe {
|
||||||
|
//~^ ERROR: `Sync`, `Send` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `fptr` to be fully captured
|
||||||
|
*fptr.0.0 = 20;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Test Clone Trait */
|
||||||
|
struct S(String);
|
||||||
|
struct T(i32);
|
||||||
|
|
||||||
|
struct U(S,T);
|
||||||
|
|
||||||
|
impl Clone for U {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
U(S(String::from("Hello World")), T(0))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn test_clone_trait() {
|
||||||
|
let f = U(S(String::from("Hello World")), T(0));
|
||||||
|
let c = || {
|
||||||
|
//~^ ERROR: `Clone` trait implementation, and drop order affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `f` to be fully captured
|
||||||
|
let f_1 = f.1;
|
||||||
|
println!("{:?}", f_1.0);
|
||||||
|
};
|
||||||
|
|
||||||
|
let c_clone = c.clone();
|
||||||
|
|
||||||
|
c_clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
test_send_trait();
|
||||||
|
test_sync_trait();
|
||||||
|
test_clone_trait();
|
||||||
|
}
|
|
@ -0,0 +1,69 @@
|
||||||
|
error: `Send` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
--> $DIR/auto_traits.rs:13:19
|
||||||
|
|
|
||||||
|
LL | thread::spawn(move || unsafe {
|
||||||
|
| ___________________^
|
||||||
|
LL | |
|
||||||
|
LL | |
|
||||||
|
LL | | *fptr.0 = 20;
|
||||||
|
LL | | });
|
||||||
|
| |_____^
|
||||||
|
|
|
||||||
|
note: the lint level is defined here
|
||||||
|
--> $DIR/auto_traits.rs:2:9
|
||||||
|
|
|
||||||
|
LL | #![deny(disjoint_capture_migration)]
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
help: add a dummy let to cause `fptr` to be fully captured
|
||||||
|
|
|
||||||
|
LL | thread::spawn(move || { let _ = &fptr; unsafe {
|
||||||
|
LL |
|
||||||
|
LL |
|
||||||
|
LL | *fptr.0 = 20;
|
||||||
|
LL | } });
|
||||||
|
|
|
||||||
|
|
||||||
|
error: `Sync`, `Send` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
--> $DIR/auto_traits.rs:30:19
|
||||||
|
|
|
||||||
|
LL | thread::spawn(move || unsafe {
|
||||||
|
| ___________________^
|
||||||
|
LL | |
|
||||||
|
LL | |
|
||||||
|
LL | | *fptr.0.0 = 20;
|
||||||
|
LL | | });
|
||||||
|
| |_____^
|
||||||
|
|
|
||||||
|
help: add a dummy let to cause `fptr` to be fully captured
|
||||||
|
|
|
||||||
|
LL | thread::spawn(move || { let _ = &fptr; unsafe {
|
||||||
|
LL |
|
||||||
|
LL |
|
||||||
|
LL | *fptr.0.0 = 20;
|
||||||
|
LL | } });
|
||||||
|
|
|
||||||
|
|
||||||
|
error: `Clone` trait implementation, and drop order affected for closure because of `capture_disjoint_fields`
|
||||||
|
--> $DIR/auto_traits.rs:51:13
|
||||||
|
|
|
||||||
|
LL | let c = || {
|
||||||
|
| _____________^
|
||||||
|
LL | |
|
||||||
|
LL | |
|
||||||
|
LL | | let f_1 = f.1;
|
||||||
|
LL | | println!("{:?}", f_1.0);
|
||||||
|
LL | | };
|
||||||
|
| |_____^
|
||||||
|
|
|
||||||
|
help: add a dummy let to cause `f` to be fully captured
|
||||||
|
|
|
||||||
|
LL | let c = || { let _ = &f;
|
||||||
|
LL |
|
||||||
|
LL |
|
||||||
|
LL | let f_1 = f.1;
|
||||||
|
LL | println!("{:?}", f_1.0);
|
||||||
|
LL | };
|
||||||
|
|
|
||||||
|
|
||||||
|
error: aborting due to 3 previous errors
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
|
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
//~^ NOTE: the lint level is defined here
|
//~^ NOTE: the lint level is defined here
|
||||||
|
|
||||||
// Test cases for types that implement a insignificant drop (stlib defined)
|
// Test cases for types that implement a insignificant drop (stlib defined)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
|
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
//~^ NOTE: the lint level is defined here
|
//~^ NOTE: the lint level is defined here
|
||||||
|
|
||||||
// Test cases for types that implement a insignificant drop (stlib defined)
|
// Test cases for types that implement a insignificant drop (stlib defined)
|
||||||
|
|
|
@ -14,8 +14,8 @@ LL | | };
|
||||||
note: the lint level is defined here
|
note: the lint level is defined here
|
||||||
--> $DIR/insignificant_drop.rs:3:9
|
--> $DIR/insignificant_drop.rs:3:9
|
||||||
|
|
|
|
||||||
LL | #![deny(disjoint_capture_drop_reorder)]
|
LL | #![deny(disjoint_capture_migration)]
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
help: add a dummy let to cause `t`, `t1`, `t2` to be fully captured
|
help: add a dummy let to cause `t`, `t1`, `t2` to be fully captured
|
||||||
|
|
|
|
||||||
LL | let c = || { let _ = (&t, &t1, &t2);
|
LL | let c = || { let _ = (&t, &t1, &t2);
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// run-pass
|
// run-pass
|
||||||
|
|
||||||
#![warn(disjoint_capture_drop_reorder)]
|
#![warn(disjoint_capture_migration)]
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
if let a = "" {
|
if let a = "" {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
//~^ NOTE: the lint level is defined here
|
//~^ NOTE: the lint level is defined here
|
||||||
|
|
||||||
// Test the two possible cases for automated migartion using rustfix
|
// Test the two possible cases for automated migartion using rustfix
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
//~^ NOTE: the lint level is defined here
|
//~^ NOTE: the lint level is defined here
|
||||||
|
|
||||||
// Test the two possible cases for automated migartion using rustfix
|
// Test the two possible cases for automated migartion using rustfix
|
||||||
|
|
|
@ -12,8 +12,8 @@ LL | | };
|
||||||
note: the lint level is defined here
|
note: the lint level is defined here
|
||||||
--> $DIR/migrations_rustfix.rs:2:9
|
--> $DIR/migrations_rustfix.rs:2:9
|
||||||
|
|
|
|
||||||
LL | #![deny(disjoint_capture_drop_reorder)]
|
LL | #![deny(disjoint_capture_migration)]
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
help: add a dummy let to cause `t` to be fully captured
|
help: add a dummy let to cause `t` to be fully captured
|
||||||
|
|
|
|
||||||
LL | let c = || { let _ = &t;
|
LL | let c = || { let _ = &t;
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
// run-rustfix
|
||||||
|
|
||||||
|
#![deny(disjoint_capture_migration)]
|
||||||
|
// ignore-wasm32-bare compiled with panic=abort by default
|
||||||
|
|
||||||
|
#![feature(fn_traits)]
|
||||||
|
#![feature(never_type)]
|
||||||
|
|
||||||
|
use std::panic;
|
||||||
|
|
||||||
|
fn foo_diverges() -> ! { panic!() }
|
||||||
|
|
||||||
|
fn assert_panics<F>(f: F) where F: FnOnce() {
|
||||||
|
let f = panic::AssertUnwindSafe(f);
|
||||||
|
let result = panic::catch_unwind(move || { let _ = &f;
|
||||||
|
//~^ ERROR: `UnwindSafe`, `RefUnwindSafe` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `f` to be fully captured
|
||||||
|
f.0()
|
||||||
|
});
|
||||||
|
if let Ok(..) = result {
|
||||||
|
panic!("diverging function returned");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn test_fn_ptr_panic<T>(mut t: T)
|
||||||
|
where T: Fn() -> !
|
||||||
|
{
|
||||||
|
let as_fn = <T as Fn<()>>::call;
|
||||||
|
assert_panics(|| as_fn(&t, ()));
|
||||||
|
let as_fn_mut = <T as FnMut<()>>::call_mut;
|
||||||
|
assert_panics(|| as_fn_mut(&mut t, ()));
|
||||||
|
let as_fn_once = <T as FnOnce<()>>::call_once;
|
||||||
|
assert_panics(|| as_fn_once(t, ()));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
test_fn_ptr_panic(foo_diverges);
|
||||||
|
test_fn_ptr_panic(foo_diverges as fn() -> !);
|
||||||
|
}
|
|
@ -0,0 +1,39 @@
|
||||||
|
// run-rustfix
|
||||||
|
|
||||||
|
#![deny(disjoint_capture_migration)]
|
||||||
|
// ignore-wasm32-bare compiled with panic=abort by default
|
||||||
|
|
||||||
|
#![feature(fn_traits)]
|
||||||
|
#![feature(never_type)]
|
||||||
|
|
||||||
|
use std::panic;
|
||||||
|
|
||||||
|
fn foo_diverges() -> ! { panic!() }
|
||||||
|
|
||||||
|
fn assert_panics<F>(f: F) where F: FnOnce() {
|
||||||
|
let f = panic::AssertUnwindSafe(f);
|
||||||
|
let result = panic::catch_unwind(move || {
|
||||||
|
//~^ ERROR: `UnwindSafe`, `RefUnwindSafe` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
//~| HELP: add a dummy let to cause `f` to be fully captured
|
||||||
|
f.0()
|
||||||
|
});
|
||||||
|
if let Ok(..) = result {
|
||||||
|
panic!("diverging function returned");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn test_fn_ptr_panic<T>(mut t: T)
|
||||||
|
where T: Fn() -> !
|
||||||
|
{
|
||||||
|
let as_fn = <T as Fn<()>>::call;
|
||||||
|
assert_panics(|| as_fn(&t, ()));
|
||||||
|
let as_fn_mut = <T as FnMut<()>>::call_mut;
|
||||||
|
assert_panics(|| as_fn_mut(&mut t, ()));
|
||||||
|
let as_fn_once = <T as FnOnce<()>>::call_once;
|
||||||
|
assert_panics(|| as_fn_once(t, ()));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
test_fn_ptr_panic(foo_diverges);
|
||||||
|
test_fn_ptr_panic(foo_diverges as fn() -> !);
|
||||||
|
}
|
|
@ -0,0 +1,27 @@
|
||||||
|
error: `UnwindSafe`, `RefUnwindSafe` trait implementation affected for closure because of `capture_disjoint_fields`
|
||||||
|
--> $DIR/mir_calls_to_shims.rs:15:38
|
||||||
|
|
|
||||||
|
LL | let result = panic::catch_unwind(move || {
|
||||||
|
| ______________________________________^
|
||||||
|
LL | |
|
||||||
|
LL | |
|
||||||
|
LL | | f.0()
|
||||||
|
LL | | });
|
||||||
|
| |_____^
|
||||||
|
|
|
||||||
|
note: the lint level is defined here
|
||||||
|
--> $DIR/mir_calls_to_shims.rs:3:9
|
||||||
|
|
|
||||||
|
LL | #![deny(disjoint_capture_migration)]
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
help: add a dummy let to cause `f` to be fully captured
|
||||||
|
|
|
||||||
|
LL | let result = panic::catch_unwind(move || { let _ = &f;
|
||||||
|
LL |
|
||||||
|
LL |
|
||||||
|
LL | f.0()
|
||||||
|
LL | });
|
||||||
|
|
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
// Set of test cases that don't need migrations
|
// Set of test cases that don't need migrations
|
||||||
|
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
|
|
||||||
|
|
||||||
// Copy types as copied by the closure instead of being moved into the closure
|
// Copy types as copied by the closure instead of being moved into the closure
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
|
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct Foo(i32);
|
struct Foo(i32);
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
|
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct Foo(i32);
|
struct Foo(i32);
|
||||||
|
|
|
@ -13,8 +13,8 @@ LL | | };
|
||||||
note: the lint level is defined here
|
note: the lint level is defined here
|
||||||
--> $DIR/precise.rs:3:9
|
--> $DIR/precise.rs:3:9
|
||||||
|
|
|
|
||||||
LL | #![deny(disjoint_capture_drop_reorder)]
|
LL | #![deny(disjoint_capture_migration)]
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
help: add a dummy let to cause `t` to be fully captured
|
help: add a dummy let to cause `t` to be fully captured
|
||||||
|
|
|
|
||||||
LL | let c = || { let _ = &t;
|
LL | let c = || { let _ = &t;
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// run-pass
|
// run-pass
|
||||||
|
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct Foo(i32);
|
struct Foo(i32);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
//~^ NOTE: the lint level is defined here
|
//~^ NOTE: the lint level is defined here
|
||||||
|
|
||||||
// Test cases for types that implement a significant drop (user defined)
|
// Test cases for types that implement a significant drop (user defined)
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// run-rustfix
|
// run-rustfix
|
||||||
#![deny(disjoint_capture_drop_reorder)]
|
#![deny(disjoint_capture_migration)]
|
||||||
//~^ NOTE: the lint level is defined here
|
//~^ NOTE: the lint level is defined here
|
||||||
|
|
||||||
// Test cases for types that implement a significant drop (user defined)
|
// Test cases for types that implement a significant drop (user defined)
|
||||||
|
|
|
@ -14,8 +14,8 @@ LL | | };
|
||||||
note: the lint level is defined here
|
note: the lint level is defined here
|
||||||
--> $DIR/significant_drop.rs:2:9
|
--> $DIR/significant_drop.rs:2:9
|
||||||
|
|
|
|
||||||
LL | #![deny(disjoint_capture_drop_reorder)]
|
LL | #![deny(disjoint_capture_migration)]
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
| ^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
help: add a dummy let to cause `t`, `t1`, `t2` to be fully captured
|
help: add a dummy let to cause `t`, `t1`, `t2` to be fully captured
|
||||||
|
|
|
|
||||||
LL | let c = || { let _ = (&t, &t1, &t2);
|
LL | let c = || { let _ = (&t, &t1, &t2);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue