Auto merge of #54782 - pnkfelix:issue-54556-semi-on-tail-diagnostic, r=nikomatsakis
NLL: temps in block tail expression diagnostic This change adds a diagnostic that explains when temporaries in a block tail expression live longer than block local variables that they borrow, and attempts to suggest turning the tail expresion into a statement (either by adding a semicolon at the end, when its result value is clearly unused, or by introducing a `let`-binding for the result value and then returning that). Fix #54556
This commit is contained in:
commit
dbecb7a644
@ -29,6 +29,7 @@ impl_stable_hash_for!(struct mir::LocalDecl<'tcx> {
|
||||
source_info,
|
||||
visibility_scope,
|
||||
internal,
|
||||
is_block_tail,
|
||||
is_user_variable
|
||||
});
|
||||
impl_stable_hash_for!(struct mir::UpvarDecl { debug_name, var_hir_id, by_ref, mutability });
|
||||
|
@ -638,6 +638,26 @@ mod binding_form_impl {
|
||||
}
|
||||
}
|
||||
|
||||
/// `BlockTailInfo` is attached to the `LocalDecl` for temporaries
|
||||
/// created during evaluation of expressions in a block tail
|
||||
/// expression; that is, a block like `{ STMT_1; STMT_2; EXPR }`.
|
||||
///
|
||||
/// It is used to improve diagnostics when such temporaries are
|
||||
/// involved in borrow_check errors, e.g. explanations of where the
|
||||
/// temporaries come from, when their destructors are run, and/or how
|
||||
/// one might revise the code to satisfy the borrow checker's rules.
|
||||
#[derive(Clone, Debug, RustcEncodable, RustcDecodable)]
|
||||
pub struct BlockTailInfo {
|
||||
/// If `true`, then the value resulting from evaluating this tail
|
||||
/// expression is ignored by the block's expression context.
|
||||
///
|
||||
/// Examples include `{ ...; tail };` and `let _ = { ...; tail };`
|
||||
/// but not e.g. `let _x = { ...; tail };`
|
||||
pub tail_result_is_ignored: bool,
|
||||
}
|
||||
|
||||
impl_stable_hash_for!(struct BlockTailInfo { tail_result_is_ignored });
|
||||
|
||||
/// A MIR local.
|
||||
///
|
||||
/// This can be a binding declared by the user, a temporary inserted by the compiler, a function
|
||||
@ -677,6 +697,12 @@ pub struct LocalDecl<'tcx> {
|
||||
/// generator.
|
||||
pub internal: bool,
|
||||
|
||||
/// If this local is a temporary and `is_block_tail` is `Some`,
|
||||
/// then it is a temporary created for evaluation of some
|
||||
/// subexpression of some block's tail expression (with no
|
||||
/// intervening statement context).
|
||||
pub is_block_tail: Option<BlockTailInfo>,
|
||||
|
||||
/// Type of this local.
|
||||
pub ty: Ty<'tcx>,
|
||||
|
||||
@ -825,10 +851,19 @@ impl<'tcx> LocalDecl<'tcx> {
|
||||
Self::new_local(ty, Mutability::Mut, false, span)
|
||||
}
|
||||
|
||||
/// Create a new immutable `LocalDecl` for a temporary.
|
||||
/// Converts `self` into same `LocalDecl` except tagged as immutable.
|
||||
#[inline]
|
||||
pub fn new_immutable_temp(ty: Ty<'tcx>, span: Span) -> Self {
|
||||
Self::new_local(ty, Mutability::Not, false, span)
|
||||
pub fn immutable(mut self) -> Self {
|
||||
self.mutability = Mutability::Not;
|
||||
self
|
||||
}
|
||||
|
||||
/// Converts `self` into same `LocalDecl` except tagged as internal temporary.
|
||||
#[inline]
|
||||
pub fn block_tail(mut self, info: BlockTailInfo) -> Self {
|
||||
assert!(self.is_block_tail.is_none());
|
||||
self.is_block_tail = Some(info);
|
||||
self
|
||||
}
|
||||
|
||||
/// Create a new `LocalDecl` for a internal temporary.
|
||||
@ -856,6 +891,7 @@ impl<'tcx> LocalDecl<'tcx> {
|
||||
visibility_scope: OUTERMOST_SOURCE_SCOPE,
|
||||
internal,
|
||||
is_user_variable: None,
|
||||
is_block_tail: None,
|
||||
}
|
||||
}
|
||||
|
||||
@ -874,6 +910,7 @@ impl<'tcx> LocalDecl<'tcx> {
|
||||
},
|
||||
visibility_scope: OUTERMOST_SOURCE_SCOPE,
|
||||
internal: false,
|
||||
is_block_tail: None,
|
||||
name: None, // FIXME maybe we do want some name here?
|
||||
is_user_variable: None,
|
||||
}
|
||||
@ -2668,6 +2705,7 @@ pub enum ClosureOutlivesSubject<'tcx> {
|
||||
*/
|
||||
|
||||
CloneTypeFoldableAndLiftImpls! {
|
||||
BlockTailInfo,
|
||||
Mutability,
|
||||
SourceInfo,
|
||||
UpvarDecl,
|
||||
@ -2711,6 +2749,7 @@ BraceStructTypeFoldableImpl! {
|
||||
user_ty,
|
||||
name,
|
||||
source_info,
|
||||
is_block_tail,
|
||||
visibility_scope,
|
||||
}
|
||||
}
|
||||
|
@ -728,6 +728,7 @@ macro_rules! make_mir_visitor {
|
||||
ref $($mutability)* visibility_scope,
|
||||
internal: _,
|
||||
is_user_variable: _,
|
||||
is_block_tail: _,
|
||||
} = *local_decl;
|
||||
|
||||
self.visit_ty(ty, TyContext::LocalDecl {
|
||||
|
@ -262,7 +262,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
move_spans.var_span_label(&mut err, "move occurs due to use in closure");
|
||||
|
||||
self.explain_why_borrow_contains_point(context, borrow, None)
|
||||
.emit(self.infcx.tcx, &mut err, String::new());
|
||||
.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
err.buffer(&mut self.errors_buffer);
|
||||
}
|
||||
|
||||
@ -299,7 +299,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
});
|
||||
|
||||
self.explain_why_borrow_contains_point(context, borrow, None)
|
||||
.emit(self.infcx.tcx, &mut err, String::new());
|
||||
.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
err.buffer(&mut self.errors_buffer);
|
||||
}
|
||||
|
||||
@ -483,7 +483,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
}
|
||||
|
||||
self.explain_why_borrow_contains_point(context, issued_borrow, None)
|
||||
.emit(self.infcx.tcx, &mut err, first_borrow_desc.to_string());
|
||||
.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, first_borrow_desc);
|
||||
|
||||
err.buffer(&mut self.errors_buffer);
|
||||
}
|
||||
@ -638,7 +638,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
|
||||
if let BorrowExplanation::MustBeValidFor(..) = explanation {
|
||||
} else {
|
||||
explanation.emit(self.infcx.tcx, &mut err, String::new());
|
||||
explanation.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
}
|
||||
} else {
|
||||
err.span_label(borrow_span, "borrowed value does not live long enough");
|
||||
@ -649,7 +649,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
|
||||
borrow_spans.args_span_label(&mut err, "value captured here");
|
||||
|
||||
explanation.emit(self.infcx.tcx, &mut err, String::new());
|
||||
explanation.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
}
|
||||
|
||||
err
|
||||
@ -709,7 +709,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
_ => {}
|
||||
}
|
||||
|
||||
explanation.emit(self.infcx.tcx, &mut err, String::new());
|
||||
explanation.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
|
||||
err.buffer(&mut self.errors_buffer);
|
||||
}
|
||||
@ -770,13 +770,13 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
match explanation {
|
||||
BorrowExplanation::UsedLater(..)
|
||||
| BorrowExplanation::UsedLaterInLoop(..)
|
||||
| BorrowExplanation::UsedLaterWhenDropped(..) => {
|
||||
| BorrowExplanation::UsedLaterWhenDropped { .. } => {
|
||||
// Only give this note and suggestion if it could be relevant.
|
||||
err.note("consider using a `let` binding to create a longer lived value");
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
explanation.emit(self.infcx.tcx, &mut err, String::new());
|
||||
explanation.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
|
||||
borrow_spans.args_span_label(&mut err, "value captured here");
|
||||
|
||||
@ -913,7 +913,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
loan_spans.var_span_label(&mut err, "borrow occurs due to use in closure");
|
||||
|
||||
self.explain_why_borrow_contains_point(context, loan, None)
|
||||
.emit(self.infcx.tcx, &mut err, String::new());
|
||||
.add_explanation_to_diagnostic(self.infcx.tcx, self.mir, &mut err, "");
|
||||
|
||||
err.buffer(&mut self.errors_buffer);
|
||||
}
|
||||
|
@ -12,18 +12,22 @@ use borrow_check::borrow_set::BorrowData;
|
||||
use borrow_check::error_reporting::UseSpans;
|
||||
use borrow_check::nll::region_infer::Cause;
|
||||
use borrow_check::{Context, MirBorrowckCtxt, WriteKind};
|
||||
use rustc::ty::{Region, TyCtxt};
|
||||
use rustc::mir::{FakeReadCause, Location, Operand, Place, StatementKind, TerminatorKind};
|
||||
use rustc::ty::{self, Region, TyCtxt};
|
||||
use rustc::mir::{FakeReadCause, Local, Location, Mir, Operand};
|
||||
use rustc::mir::{Place, StatementKind, TerminatorKind};
|
||||
use rustc_errors::DiagnosticBuilder;
|
||||
use syntax_pos::Span;
|
||||
use syntax_pos::symbol::Symbol;
|
||||
|
||||
mod find_use;
|
||||
|
||||
pub(in borrow_check) enum BorrowExplanation<'tcx> {
|
||||
UsedLater(LaterUseKind, Span),
|
||||
UsedLaterInLoop(LaterUseKind, Span),
|
||||
UsedLaterWhenDropped(Span, Symbol, bool),
|
||||
UsedLaterWhenDropped {
|
||||
drop_loc: Location,
|
||||
dropped_local: Local,
|
||||
should_note_order: bool,
|
||||
},
|
||||
MustBeValidFor(Region<'tcx>),
|
||||
Unexplained,
|
||||
}
|
||||
@ -37,24 +41,25 @@ pub(in borrow_check) enum LaterUseKind {
|
||||
}
|
||||
|
||||
impl<'tcx> BorrowExplanation<'tcx> {
|
||||
pub(in borrow_check) fn emit<'cx, 'gcx>(
|
||||
pub(in borrow_check) fn add_explanation_to_diagnostic<'cx, 'gcx>(
|
||||
&self,
|
||||
tcx: TyCtxt<'cx, 'gcx, 'tcx>,
|
||||
mir: &Mir<'tcx>,
|
||||
err: &mut DiagnosticBuilder<'_>,
|
||||
borrow_desc: String,
|
||||
borrow_desc: &str,
|
||||
) {
|
||||
match *self {
|
||||
BorrowExplanation::UsedLater(later_use_kind, var_or_use_span) => {
|
||||
let message = borrow_desc + match later_use_kind {
|
||||
let message = match later_use_kind {
|
||||
LaterUseKind::ClosureCapture => "borrow later captured here by closure",
|
||||
LaterUseKind::Call => "borrow later used by call",
|
||||
LaterUseKind::FakeLetRead => "borrow later stored here",
|
||||
LaterUseKind::Other => "borrow later used here",
|
||||
};
|
||||
err.span_label(var_or_use_span, message);
|
||||
err.span_label(var_or_use_span, format!("{}{}", borrow_desc, message));
|
||||
},
|
||||
BorrowExplanation::UsedLaterInLoop(later_use_kind, var_or_use_span) => {
|
||||
let message = borrow_desc + match later_use_kind {
|
||||
let message = match later_use_kind {
|
||||
LaterUseKind::ClosureCapture => {
|
||||
"borrow captured here by closure, in later iteration of loop"
|
||||
},
|
||||
@ -62,29 +67,82 @@ impl<'tcx> BorrowExplanation<'tcx> {
|
||||
LaterUseKind::FakeLetRead => "borrow later stored here",
|
||||
LaterUseKind::Other => "borrow used here, in later iteration of loop",
|
||||
};
|
||||
err.span_label(var_or_use_span, message);
|
||||
err.span_label(var_or_use_span, format!("{}{}", borrow_desc, message));
|
||||
},
|
||||
BorrowExplanation::UsedLaterWhenDropped(span, local_name, should_note_order) => {
|
||||
err.span_label(
|
||||
span,
|
||||
format!(
|
||||
"{}borrow later used here, when `{}` is dropped",
|
||||
borrow_desc,
|
||||
local_name,
|
||||
),
|
||||
);
|
||||
BorrowExplanation::UsedLaterWhenDropped { drop_loc, dropped_local,
|
||||
should_note_order } =>
|
||||
{
|
||||
let local_decl = &mir.local_decls[dropped_local];
|
||||
let (dtor_desc, type_desc) = match local_decl.ty.sty {
|
||||
// If type is an ADT that implements Drop, then
|
||||
// simplify output by reporting just the ADT name.
|
||||
ty::Adt(adt, _substs) if adt.has_dtor(tcx) && !adt.is_box() =>
|
||||
("`Drop` code", format!("type `{}`", tcx.item_path_str(adt.did))),
|
||||
|
||||
if should_note_order {
|
||||
err.note(
|
||||
"values in a scope are dropped \
|
||||
in the opposite order they are defined",
|
||||
);
|
||||
// Otherwise, just report the whole type (and use
|
||||
// the intentionally fuzzy phrase "destructor")
|
||||
ty::Closure(..) =>
|
||||
("destructor", format!("closure")),
|
||||
ty::Generator(..) =>
|
||||
("destructor", format!("generator")),
|
||||
|
||||
_ => ("destructor", format!("type `{}`", local_decl.ty)),
|
||||
};
|
||||
|
||||
match local_decl.name {
|
||||
Some(local_name) => {
|
||||
let message =
|
||||
format!("{B}borrow might be used here, when `{LOC}` is dropped \
|
||||
and runs the {DTOR} for {TYPE}",
|
||||
B=borrow_desc, LOC=local_name, TYPE=type_desc, DTOR=dtor_desc);
|
||||
err.span_label(mir.source_info(drop_loc).span, message);
|
||||
|
||||
if should_note_order {
|
||||
err.note(
|
||||
"values in a scope are dropped \
|
||||
in the opposite order they are defined",
|
||||
);
|
||||
}
|
||||
}
|
||||
None => {
|
||||
err.span_label(local_decl.source_info.span,
|
||||
format!("a temporary with access to the {B}borrow \
|
||||
is created here ...",
|
||||
B=borrow_desc));
|
||||
let message =
|
||||
format!("... and the {B}borrow might be used here, \
|
||||
when that temporary is dropped \
|
||||
and runs the {DTOR} for {TYPE}",
|
||||
B=borrow_desc, TYPE=type_desc, DTOR=dtor_desc);
|
||||
err.span_label(mir.source_info(drop_loc).span, message);
|
||||
|
||||
if let Some(info) = &local_decl.is_block_tail {
|
||||
// FIXME: use span_suggestion instead, highlighting the
|
||||
// whole block tail expression.
|
||||
let msg = if info.tail_result_is_ignored {
|
||||
"The temporary is part of an expression at the end of a block. \
|
||||
Consider adding semicolon after the expression so its temporaries \
|
||||
are dropped sooner, before the local variables declared by the \
|
||||
block are dropped."
|
||||
} else {
|
||||
"The temporary is part of an expression at the end of a block. \
|
||||
Consider forcing this temporary to be dropped sooner, before \
|
||||
the block's local variables are dropped. \
|
||||
For example, you could save the expression's value in a new \
|
||||
local variable `x` and then make `x` be the expression \
|
||||
at the end of the block."
|
||||
};
|
||||
|
||||
err.note(msg);
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
}
|
||||
|
||||
BorrowExplanation::MustBeValidFor(region) => {
|
||||
tcx.note_and_explain_free_region(
|
||||
err,
|
||||
&(borrow_desc + "borrowed value must be valid for "),
|
||||
&format!("{}{}", borrow_desc, "borrowed value must be valid for "),
|
||||
region,
|
||||
"...",
|
||||
);
|
||||
@ -95,7 +153,7 @@ impl<'tcx> BorrowExplanation<'tcx> {
|
||||
}
|
||||
|
||||
impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
/// Adds annotations to `err` explaining *why* the borrow contains the
|
||||
/// Returns structured explanation for *why* the borrow contains the
|
||||
/// point from `context`. This is key for the "3-point errors"
|
||||
/// [described in the NLL RFC][d].
|
||||
///
|
||||
@ -106,7 +164,6 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
/// - `kind_place`: if Some, this describes the statement that triggered the error.
|
||||
/// - first half is the kind of write, if any, being performed
|
||||
/// - second half is the place being accessed
|
||||
/// - `err`: where the error annotations are going to be added
|
||||
///
|
||||
/// [d]: https://rust-lang.github.io/rfcs/2094-nll.html#leveraging-intuition-framing-errors-in-terms-of-points
|
||||
pub(in borrow_check) fn explain_why_borrow_contains_point(
|
||||
@ -116,8 +173,8 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
kind_place: Option<(WriteKind, &Place<'tcx>)>,
|
||||
) -> BorrowExplanation<'tcx> {
|
||||
debug!(
|
||||
"find_why_borrow_contains_point(context={:?}, borrow={:?})",
|
||||
context, borrow,
|
||||
"explain_why_borrow_contains_point(context={:?}, borrow={:?}, kind_place={:?})",
|
||||
context, borrow, kind_place
|
||||
);
|
||||
|
||||
let regioncx = &self.nonlexical_regioncx;
|
||||
@ -154,32 +211,30 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
||||
}
|
||||
}
|
||||
|
||||
Some(Cause::DropVar(local, location)) => match &mir.local_decls[local].name {
|
||||
Some(local_name) => {
|
||||
let mut should_note_order = false;
|
||||
if let Some((WriteKind::StorageDeadOrDrop, place)) = kind_place {
|
||||
if let Place::Local(borrowed_local) = place {
|
||||
let dropped_local_scope = mir.local_decls[local].visibility_scope;
|
||||
let borrowed_local_scope =
|
||||
mir.local_decls[*borrowed_local].visibility_scope;
|
||||
Some(Cause::DropVar(local, location)) => {
|
||||
let mut should_note_order = false;
|
||||
if mir.local_decls[local].name.is_some() {
|
||||
if let Some((WriteKind::StorageDeadOrDrop, place)) = kind_place {
|
||||
if let Place::Local(borrowed_local) = place {
|
||||
let dropped_local_scope = mir.local_decls[local].visibility_scope;
|
||||
let borrowed_local_scope =
|
||||
mir.local_decls[*borrowed_local].visibility_scope;
|
||||
|
||||
if mir.is_sub_scope(borrowed_local_scope, dropped_local_scope)
|
||||
&& local != *borrowed_local
|
||||
{
|
||||
should_note_order = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
if mir.is_sub_scope(borrowed_local_scope, dropped_local_scope)
|
||||
&& local != *borrowed_local
|
||||
{
|
||||
should_note_order = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
BorrowExplanation::UsedLaterWhenDropped(
|
||||
mir.source_info(location).span,
|
||||
*local_name,
|
||||
should_note_order
|
||||
)
|
||||
},
|
||||
|
||||
None => BorrowExplanation::Unexplained,
|
||||
},
|
||||
BorrowExplanation::UsedLaterWhenDropped {
|
||||
drop_loc: location,
|
||||
dropped_local: local,
|
||||
should_note_order,
|
||||
}
|
||||
}
|
||||
|
||||
None => if let Some(region) = regioncx.to_error_region(region_sub) {
|
||||
BorrowExplanation::MustBeValidFor(region)
|
||||
|
@ -8,7 +8,7 @@
|
||||
// option. This file may not be copied, modified, or distributed
|
||||
// except according to those terms.
|
||||
|
||||
use build::{BlockAnd, BlockAndExtension, Builder};
|
||||
use build::{BlockAnd, BlockAndExtension, BlockFrame, Builder};
|
||||
use build::ForGuard::OutsideGuard;
|
||||
use build::matches::ArmHasGuard;
|
||||
use hair::*;
|
||||
@ -93,6 +93,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
let Stmt { kind, opt_destruction_scope } = this.hir.mirror(stmt);
|
||||
match kind {
|
||||
StmtKind::Expr { scope, expr } => {
|
||||
this.block_context.push(BlockFrame::Statement { ignores_expr_result: true });
|
||||
unpack!(block = this.in_opt_scope(
|
||||
opt_destruction_scope.map(|de|(de, source_info)), block, |this| {
|
||||
let si = (scope, source_info);
|
||||
@ -109,6 +110,13 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
initializer,
|
||||
lint_level
|
||||
} => {
|
||||
let ignores_expr_result = if let PatternKind::Wild = *pattern.kind {
|
||||
true
|
||||
} else {
|
||||
false
|
||||
};
|
||||
this.block_context.push(BlockFrame::Statement { ignores_expr_result });
|
||||
|
||||
// Enter the remainder scope, i.e. the bindings' destruction scope.
|
||||
this.push_scope((remainder_scope, source_info));
|
||||
let_scope_stack.push(remainder_scope);
|
||||
@ -155,19 +163,40 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
let popped = this.block_context.pop();
|
||||
assert!(popped.map_or(false, |bf|bf.is_statement()));
|
||||
}
|
||||
|
||||
// Then, the block may have an optional trailing expression which is a “return” value
|
||||
// of the block.
|
||||
// of the block, which is stored into `destination`.
|
||||
let tcx = this.hir.tcx();
|
||||
let destination_ty = destination.ty(&this.local_decls, tcx).to_ty(tcx);
|
||||
if let Some(expr) = expr {
|
||||
let tail_result_is_ignored = destination_ty.is_unit() ||
|
||||
match this.block_context.last() {
|
||||
// no context: conservatively assume result is read
|
||||
None => false,
|
||||
|
||||
// sub-expression: block result feeds into some computation
|
||||
Some(BlockFrame::SubExpr) => false,
|
||||
|
||||
// otherwise: use accumualated is_ignored state.
|
||||
Some(BlockFrame::TailExpr { tail_result_is_ignored: ignored }) |
|
||||
Some(BlockFrame::Statement { ignores_expr_result: ignored }) => *ignored,
|
||||
};
|
||||
this.block_context.push(BlockFrame::TailExpr { tail_result_is_ignored });
|
||||
|
||||
unpack!(block = this.into(destination, block, expr));
|
||||
let popped = this.block_context.pop();
|
||||
|
||||
assert!(popped.map_or(false, |bf|bf.is_tail_expr()));
|
||||
} else {
|
||||
// If a block has no trailing expression, then it is given an implicit return type.
|
||||
// This return type is usually `()`, unless the block is diverging, in which case the
|
||||
// return type is `!`. For the unit type, we need to actually return the unit, but in
|
||||
// the case of `!`, no return value is required, as the block will never return.
|
||||
let tcx = this.hir.tcx();
|
||||
let ty = destination.ty(&this.local_decls, tcx).to_ty(tcx);
|
||||
if ty.is_unit() {
|
||||
if destination_ty.is_unit() {
|
||||
// We only want to assign an implicit `()` as the return value of the block if the
|
||||
// block does not diverge. (Otherwise, we may try to assign a unit to a `!`-type.)
|
||||
this.cfg.push_assign_unit(block, source_info, destination);
|
||||
|
@ -10,7 +10,7 @@
|
||||
|
||||
//! See docs in build/expr/mod.rs
|
||||
|
||||
use build::{BlockAnd, BlockAndExtension, Builder};
|
||||
use build::{BlockAnd, BlockAndExtension, BlockFrame, Builder};
|
||||
use hair::*;
|
||||
use rustc::middle::region;
|
||||
use rustc::mir::*;
|
||||
@ -59,14 +59,30 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
}
|
||||
|
||||
let expr_ty = expr.ty;
|
||||
let temp = if mutability == Mutability::Not {
|
||||
this.local_decls
|
||||
.push(LocalDecl::new_immutable_temp(expr_ty, expr_span))
|
||||
} else {
|
||||
this.local_decls
|
||||
.push(LocalDecl::new_temp(expr_ty, expr_span))
|
||||
};
|
||||
let temp = {
|
||||
let mut local_decl = LocalDecl::new_temp(expr_ty, expr_span);
|
||||
if mutability == Mutability::Not {
|
||||
local_decl = local_decl.immutable();
|
||||
}
|
||||
|
||||
debug!("creating temp {:?} with block_context: {:?}", local_decl, this.block_context);
|
||||
// Find out whether this temp is being created within the
|
||||
// tail expression of a block whose result is ignored.
|
||||
for bf in this.block_context.iter().rev() {
|
||||
match bf {
|
||||
BlockFrame::SubExpr => continue,
|
||||
BlockFrame::Statement { .. } => break,
|
||||
&BlockFrame::TailExpr { tail_result_is_ignored } => {
|
||||
local_decl = local_decl.block_tail(BlockTailInfo {
|
||||
tail_result_is_ignored
|
||||
});
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
this.local_decls.push(local_decl)
|
||||
};
|
||||
if !expr_ty.is_never() {
|
||||
this.cfg.push(
|
||||
block,
|
||||
|
@ -11,7 +11,7 @@
|
||||
//! See docs in build/expr/mod.rs
|
||||
|
||||
use build::expr::category::{Category, RvalueFunc};
|
||||
use build::{BlockAnd, BlockAndExtension, Builder};
|
||||
use build::{BlockAnd, BlockAndExtension, BlockFrame, Builder};
|
||||
use hair::*;
|
||||
use rustc::mir::*;
|
||||
use rustc::ty;
|
||||
@ -39,7 +39,17 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
let expr_span = expr.span;
|
||||
let source_info = this.source_info(expr_span);
|
||||
|
||||
match expr.kind {
|
||||
let expr_is_block_or_scope = match expr.kind {
|
||||
ExprKind::Block { .. } => true,
|
||||
ExprKind::Scope { .. } => true,
|
||||
_ => false,
|
||||
};
|
||||
|
||||
if !expr_is_block_or_scope {
|
||||
this.block_context.push(BlockFrame::SubExpr);
|
||||
}
|
||||
|
||||
let block_and = match expr.kind {
|
||||
ExprKind::Scope {
|
||||
region_scope,
|
||||
lint_level,
|
||||
@ -302,6 +312,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
visibility_scope: source_info.scope,
|
||||
internal: true,
|
||||
is_user_variable: None,
|
||||
is_block_tail: None,
|
||||
});
|
||||
let ptr_temp = Place::Local(ptr_temp);
|
||||
let block = unpack!(this.into(&ptr_temp, block, ptr));
|
||||
@ -414,6 +425,13 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
.push_assign(block, source_info, destination, rvalue);
|
||||
block.unit()
|
||||
}
|
||||
};
|
||||
|
||||
if !expr_is_block_or_scope {
|
||||
let popped = this.block_context.pop();
|
||||
assert!(popped.is_some());
|
||||
}
|
||||
|
||||
block_and
|
||||
}
|
||||
}
|
||||
|
@ -9,7 +9,7 @@
|
||||
// except according to those terms.
|
||||
|
||||
use build::scope::BreakableScope;
|
||||
use build::{BlockAnd, BlockAndExtension, Builder};
|
||||
use build::{BlockAnd, BlockAndExtension, BlockFrame, Builder};
|
||||
use hair::*;
|
||||
use rustc::mir::*;
|
||||
|
||||
@ -20,6 +20,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
let source_info = this.source_info(expr.span);
|
||||
// Handle a number of expressions that don't need a destination at all. This
|
||||
// avoids needing a mountain of temporary `()` variables.
|
||||
let expr2 = expr.clone();
|
||||
match expr.kind {
|
||||
ExprKind::Scope {
|
||||
region_scope,
|
||||
@ -40,19 +41,23 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
// is better for borrowck interaction with overloaded
|
||||
// operators like x[j] = x[i].
|
||||
|
||||
debug!("stmt_expr Assign block_context.push(SubExpr) : {:?}", expr2);
|
||||
this.block_context.push(BlockFrame::SubExpr);
|
||||
|
||||
// Generate better code for things that don't need to be
|
||||
// dropped.
|
||||
if this.hir.needs_drop(lhs.ty) {
|
||||
let rhs = unpack!(block = this.as_local_operand(block, rhs));
|
||||
let lhs = unpack!(block = this.as_place(block, lhs));
|
||||
unpack!(block = this.build_drop_and_replace(block, lhs_span, lhs, rhs));
|
||||
block.unit()
|
||||
} else {
|
||||
let rhs = unpack!(block = this.as_local_rvalue(block, rhs));
|
||||
let lhs = unpack!(block = this.as_place(block, lhs));
|
||||
this.cfg.push_assign(block, source_info, &lhs, rhs);
|
||||
block.unit()
|
||||
}
|
||||
|
||||
this.block_context.pop();
|
||||
block.unit()
|
||||
}
|
||||
ExprKind::AssignOp { op, lhs, rhs } => {
|
||||
// FIXME(#28160) there is an interesting semantics
|
||||
@ -66,6 +71,9 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
let lhs = this.hir.mirror(lhs);
|
||||
let lhs_ty = lhs.ty;
|
||||
|
||||
debug!("stmt_expr AssignOp block_context.push(SubExpr) : {:?}", expr2);
|
||||
this.block_context.push(BlockFrame::SubExpr);
|
||||
|
||||
// As above, RTL.
|
||||
let rhs = unpack!(block = this.as_local_operand(block, rhs));
|
||||
let lhs = unpack!(block = this.as_place(block, lhs));
|
||||
@ -85,6 +93,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
);
|
||||
this.cfg.push_assign(block, source_info, &lhs, result);
|
||||
|
||||
this.block_context.pop();
|
||||
block.unit()
|
||||
}
|
||||
ExprKind::Continue { label } => {
|
||||
@ -114,7 +123,10 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
(break_block, region_scope, break_destination.clone())
|
||||
};
|
||||
if let Some(value) = value {
|
||||
unpack!(block = this.into(&destination, block, value))
|
||||
debug!("stmt_expr Break val block_context.push(SubExpr) : {:?}", expr2);
|
||||
this.block_context.push(BlockFrame::SubExpr);
|
||||
unpack!(block = this.into(&destination, block, value));
|
||||
this.block_context.pop();
|
||||
} else {
|
||||
this.cfg.push_assign_unit(block, source_info, &destination)
|
||||
}
|
||||
@ -123,7 +135,13 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
}
|
||||
ExprKind::Return { value } => {
|
||||
block = match value {
|
||||
Some(value) => unpack!(this.into(&Place::Local(RETURN_PLACE), block, value)),
|
||||
Some(value) => {
|
||||
debug!("stmt_expr Return val block_context.push(SubExpr) : {:?}", expr2);
|
||||
this.block_context.push(BlockFrame::SubExpr);
|
||||
let result = unpack!(this.into(&Place::Local(RETURN_PLACE), block, value));
|
||||
this.block_context.pop();
|
||||
result
|
||||
}
|
||||
None => {
|
||||
this.cfg
|
||||
.push_assign_unit(block, source_info, &Place::Local(RETURN_PLACE));
|
||||
@ -140,6 +158,8 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
outputs,
|
||||
inputs,
|
||||
} => {
|
||||
debug!("stmt_expr InlineAsm block_context.push(SubExpr) : {:?}", expr2);
|
||||
this.block_context.push(BlockFrame::SubExpr);
|
||||
let outputs = outputs
|
||||
.into_iter()
|
||||
.map(|output| unpack!(block = this.as_place(block, output)))
|
||||
@ -161,6 +181,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
},
|
||||
},
|
||||
);
|
||||
this.block_context.pop();
|
||||
block.unit()
|
||||
}
|
||||
_ => {
|
||||
|
@ -1485,6 +1485,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
source_info,
|
||||
visibility_scope,
|
||||
internal: false,
|
||||
is_block_tail: None,
|
||||
is_user_variable: Some(ClearCrossCrate::Set(BindingForm::Var(VarBindingForm {
|
||||
binding_mode,
|
||||
// hypothetically, `visit_bindings` could try to unzip
|
||||
@ -1518,6 +1519,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
visibility_scope,
|
||||
// FIXME: should these secretly injected ref_for_guard's be marked as `internal`?
|
||||
internal: false,
|
||||
is_block_tail: None,
|
||||
is_user_variable: Some(ClearCrossCrate::Set(BindingForm::RefForGuard)),
|
||||
});
|
||||
LocalsForNode::ForGuard {
|
||||
|
@ -281,6 +281,57 @@ fn liberated_closure_env_ty<'a, 'gcx, 'tcx>(tcx: TyCtxt<'a, 'gcx, 'tcx>,
|
||||
tcx.liberate_late_bound_regions(closure_def_id, &closure_env_ty)
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq, Eq)]
|
||||
pub enum BlockFrame {
|
||||
/// Evaluation is currently within a statement.
|
||||
///
|
||||
/// Examples include:
|
||||
/// 1. `EXPR;`
|
||||
/// 2. `let _ = EXPR;`
|
||||
/// 3. `let x = EXPR;`
|
||||
Statement {
|
||||
/// If true, then statement discards result from evaluating
|
||||
/// the expression (such as examples 1 and 2 above).
|
||||
ignores_expr_result: bool
|
||||
},
|
||||
|
||||
/// Evaluation is currently within the tail expression of a block.
|
||||
///
|
||||
/// Example: `{ STMT_1; STMT_2; EXPR }`
|
||||
TailExpr {
|
||||
/// If true, then the surrounding context of the block ignores
|
||||
/// the result of evaluating the block's tail expression.
|
||||
///
|
||||
/// Example: `let _ = { STMT_1; EXPR };`
|
||||
tail_result_is_ignored: bool
|
||||
},
|
||||
|
||||
/// Generic mark meaning that the block occurred as a subexpression
|
||||
/// where the result might be used.
|
||||
///
|
||||
/// Examples: `foo(EXPR)`, `match EXPR { ... }`
|
||||
SubExpr,
|
||||
}
|
||||
|
||||
impl BlockFrame {
|
||||
fn is_tail_expr(&self) -> bool {
|
||||
match *self {
|
||||
BlockFrame::TailExpr { .. } => true,
|
||||
|
||||
BlockFrame::Statement { .. } |
|
||||
BlockFrame::SubExpr => false,
|
||||
}
|
||||
}
|
||||
fn is_statement(&self) -> bool {
|
||||
match *self {
|
||||
BlockFrame::Statement { .. } => true,
|
||||
|
||||
BlockFrame::TailExpr { .. } |
|
||||
BlockFrame::SubExpr => false,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct Builder<'a, 'gcx: 'a+'tcx, 'tcx: 'a> {
|
||||
hir: Cx<'a, 'gcx, 'tcx>,
|
||||
cfg: CFG<'tcx>,
|
||||
@ -292,6 +343,20 @@ struct Builder<'a, 'gcx: 'a+'tcx, 'tcx: 'a> {
|
||||
/// see the `scope` module for more details
|
||||
scopes: Vec<scope::Scope<'tcx>>,
|
||||
|
||||
/// the block-context: each time we build the code within an hair::Block,
|
||||
/// we push a frame here tracking whether we are building a statement or
|
||||
/// if we are pushing the tail expression of the block. This is used to
|
||||
/// embed information in generated temps about whether they were created
|
||||
/// for a block tail expression or not.
|
||||
///
|
||||
/// It would be great if we could fold this into `self.scopes`
|
||||
/// somehow; but right now I think that is very tightly tied to
|
||||
/// the code generation in ways that we cannot (or should not)
|
||||
/// start just throwing new entries onto that vector in order to
|
||||
/// distinguish the context of EXPR1 from the context of EXPR2 in
|
||||
/// `{ STMTS; EXPR1 } + EXPR2`
|
||||
block_context: Vec<BlockFrame>,
|
||||
|
||||
/// The current unsafe block in scope, even if it is hidden by
|
||||
/// a PushUnsafeBlock
|
||||
unpushed_unsafe: Safety,
|
||||
@ -695,6 +760,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
fn_span: span,
|
||||
arg_count,
|
||||
scopes: vec![],
|
||||
block_context: vec![],
|
||||
source_scopes: IndexVec::new(),
|
||||
source_scope: OUTERMOST_SOURCE_SCOPE,
|
||||
source_scope_local_data: IndexVec::new(),
|
||||
@ -781,6 +847,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||
name,
|
||||
internal: false,
|
||||
is_user_variable: None,
|
||||
is_block_tail: None,
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -148,6 +148,7 @@ fn temp_decl(mutability: Mutability, ty: Ty, span: Span) -> LocalDecl {
|
||||
visibility_scope: source_info.scope,
|
||||
internal: false,
|
||||
is_user_variable: None,
|
||||
is_block_tail: None,
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -308,6 +308,7 @@ fn replace_result_variable<'tcx>(
|
||||
source_info,
|
||||
visibility_scope: source_info.scope,
|
||||
internal: false,
|
||||
is_block_tail: None,
|
||||
is_user_variable: None,
|
||||
};
|
||||
let new_ret_local = Local::new(mir.local_decls.len());
|
||||
@ -662,6 +663,7 @@ fn create_generator_drop_shim<'a, 'tcx>(
|
||||
source_info,
|
||||
visibility_scope: source_info.scope,
|
||||
internal: false,
|
||||
is_block_tail: None,
|
||||
is_user_variable: None,
|
||||
};
|
||||
|
||||
@ -679,6 +681,7 @@ fn create_generator_drop_shim<'a, 'tcx>(
|
||||
source_info,
|
||||
visibility_scope: source_info.scope,
|
||||
internal: false,
|
||||
is_block_tail: None,
|
||||
is_user_variable: None,
|
||||
};
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c_shortest` dropped here while still borrowed
|
||||
| borrow later used here, when `dt` is dropped
|
||||
| borrow might be used here, when `dt` is dropped and runs the `Drop` code for type `other::Dt`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c_shortest` dropped here while still borrowed
|
||||
| borrow later used here, when `dt` is dropped
|
||||
| borrow might be used here, when `dt` is dropped and runs the `Drop` code for type `Dt`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c_shortest` dropped here while still borrowed
|
||||
| borrow later used here, when `dt` is dropped
|
||||
| borrow might be used here, when `dt` is dropped and runs the `Drop` code for type `Dt`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `v` dropped here while still borrowed
|
||||
| borrow later used here, when `v` is dropped
|
||||
| borrow might be used here, when `v` is dropped and runs the `Drop` code for type `Wrap`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `y` dropped here while still borrowed
|
||||
| borrow later used here, when `x` is dropped
|
||||
| borrow might be used here, when `x` is dropped and runs the `Drop` code for type `Foo`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -2,10 +2,17 @@ error[E0597]: `a` does not live long enough
|
||||
--> $DIR/borrowing.rs:18:18
|
||||
|
|
||||
LL | unsafe { (|| yield &a).resume() }
|
||||
| ^^^^^^^^^^^^^ borrowed value does not live long enough
|
||||
| ^^^^^^^^^^^^^
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
LL | //~^ ERROR: `a` does not live long enough
|
||||
LL | };
|
||||
| - `a` dropped here while still borrowed
|
||||
| -- ... and the borrow might be used here, when that temporary is dropped and runs the destructor for generator
|
||||
| |
|
||||
| `a` dropped here while still borrowed
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error[E0597]: `a` does not live long enough
|
||||
--> $DIR/borrowing.rs:24:9
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `*cell` dropped here while still borrowed
|
||||
| borrow later used here, when `gen` is dropped
|
||||
| borrow might be used here, when `gen` is dropped and runs the destructor for generator
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -27,7 +27,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `ref_` dropped here while still borrowed
|
||||
| borrow later used here, when `gen` is dropped
|
||||
| borrow might be used here, when `gen` is dropped and runs the destructor for generator
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -11,7 +11,7 @@ LL | c.push(Box::new(|| y = 0));
|
||||
| second mutable borrow occurs here
|
||||
LL | //~^ ERROR cannot borrow `y` as mutable more than once at a time
|
||||
LL | }
|
||||
| - first borrow later used here, when `c` is dropped
|
||||
| - first borrow might be used here, when `c` is dropped and runs the destructor for type `std::cell::RefCell<std::vec::Vec<std::boxed::Box<dyn std::ops::FnMut()>>>`
|
||||
|
||||
error[E0499]: cannot borrow `y` as mutable more than once at a time
|
||||
--> $DIR/issue-18783.rs:26:29
|
||||
@ -26,7 +26,7 @@ LL | Push::push(&c, Box::new(|| y = 0));
|
||||
| second mutable borrow occurs here
|
||||
LL | //~^ ERROR cannot borrow `y` as mutable more than once at a time
|
||||
LL | }
|
||||
| - first borrow later used here, when `c` is dropped
|
||||
| - first borrow might be used here, when `c` is dropped and runs the destructor for type `std::cell::RefCell<std::vec::Vec<std::boxed::Box<dyn std::ops::FnMut()>>>`
|
||||
|
||||
error: aborting due to 2 previous errors
|
||||
|
||||
|
@ -3,9 +3,15 @@ error[E0502]: cannot borrow `heap` as immutable because it is also borrowed as m
|
||||
|
|
||||
LL | let borrow = heap.peek_mut();
|
||||
| ---- mutable borrow occurs here
|
||||
...
|
||||
LL |
|
||||
LL | match (borrow, ()) {
|
||||
| ------------ a temporary with access to the mutable borrow is created here ...
|
||||
LL | (Some(_), ()) => {
|
||||
LL | println!("{:?}", heap); //~ ERROR cannot borrow `heap` as immutable
|
||||
| ^^^^ immutable borrow occurs here
|
||||
...
|
||||
LL | };
|
||||
| - ... and the mutable borrow might be used here, when that temporary is dropped and runs the destructor for type `(std::option::Option<std::collections::binary_heap::PeekMut<'_, i32>>, ())`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | v[0] += 1; //~ ERROR cannot assign to `v[..]` because it is borrowe
|
||||
| ^^^^^^^^^ assignment to borrowed `v[..]` occurs here
|
||||
...
|
||||
LL | }
|
||||
| - borrow later used here, when `p` is dropped
|
||||
| - borrow might be used here, when `p` is dropped and runs the `Drop` code for type `WrapMayNotDangle`
|
||||
|
||||
error[E0506]: cannot assign to `v[..]` because it is borrowed
|
||||
--> $DIR/drop-no-may-dangle.rs:33:5
|
||||
@ -19,7 +19,7 @@ LL | let p: WrapMayNotDangle<&usize> = WrapMayNotDangle { value: &v[0] };
|
||||
LL | v[0] += 1; //~ ERROR cannot assign to `v[..]` because it is borrowed
|
||||
| ^^^^^^^^^ assignment to borrowed `v[..]` occurs here
|
||||
LL | }
|
||||
| - borrow later used here, when `p` is dropped
|
||||
| - borrow might be used here, when `p` is dropped and runs the `Drop` code for type `WrapMayNotDangle`
|
||||
|
||||
error: aborting due to 2 previous errors
|
||||
|
||||
|
20
src/test/ui/nll/issue-21114-ebfull.rs
Normal file
20
src/test/ui/nll/issue-21114-ebfull.rs
Normal file
@ -0,0 +1,20 @@
|
||||
// (this works, but only in NLL)
|
||||
// compile-pass
|
||||
#![feature(nll)]
|
||||
|
||||
use std::collections::HashMap;
|
||||
use std::sync::Mutex;
|
||||
|
||||
fn i_used_to_be_able_to(foo: &Mutex<HashMap<usize, usize>>) -> Vec<(usize, usize)> {
|
||||
let mut foo = foo.lock().unwrap();
|
||||
|
||||
foo.drain().collect()
|
||||
}
|
||||
|
||||
fn but_after_nightly_update_now_i_gotta(foo: &Mutex<HashMap<usize, usize>>) -> Vec<(usize, usize)> {
|
||||
let mut foo = foo.lock().unwrap();
|
||||
|
||||
return foo.drain().collect();
|
||||
}
|
||||
|
||||
fn main() {}
|
19
src/test/ui/nll/issue-21114-kixunil.rs
Normal file
19
src/test/ui/nll/issue-21114-kixunil.rs
Normal file
@ -0,0 +1,19 @@
|
||||
// (this works, but only in NLL)
|
||||
// compile-pass
|
||||
#![feature(nll)]
|
||||
|
||||
fn from_stdin(min: u64) -> Vec<u64> {
|
||||
use std::io::BufRead;
|
||||
|
||||
let stdin = std::io::stdin();
|
||||
let stdin = stdin.lock();
|
||||
|
||||
stdin.lines()
|
||||
.map(Result::unwrap)
|
||||
.map(|val| val.parse())
|
||||
.map(Result::unwrap)
|
||||
.filter(|val| *val >= min)
|
||||
.collect()
|
||||
}
|
||||
|
||||
fn main() {}
|
20
src/test/ui/nll/issue-54556-niconii.nll.stderr
Normal file
20
src/test/ui/nll/issue-54556-niconii.nll.stderr
Normal file
@ -0,0 +1,20 @@
|
||||
error[E0597]: `counter` does not live long enough
|
||||
--> $DIR/issue-54556-niconii.rs:22:20
|
||||
|
|
||||
LL | if let Ok(_) = counter.lock() { }
|
||||
| ^^^^^^^-------
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
...
|
||||
LL | }
|
||||
| -
|
||||
| |
|
||||
| `counter` dropped here while still borrowed
|
||||
| ... and the borrow might be used here, when that temporary is dropped and runs the destructor for type `std::result::Result<MutexGuard<'_>, ()>`
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
31
src/test/ui/nll/issue-54556-niconii.rs
Normal file
31
src/test/ui/nll/issue-54556-niconii.rs
Normal file
@ -0,0 +1,31 @@
|
||||
// This is a reduction of a concrete test illustrating a case that was
|
||||
// annoying to Rust developer niconii (see comment thread on #21114).
|
||||
//
|
||||
// With resolving issue #54556, pnkfelix hopes that the new diagnostic
|
||||
// output produced by NLL helps to *explain* the semantic significance
|
||||
// of temp drop order, and thus why inserting a semi-colon after the
|
||||
// `if let` expression in `main` works.
|
||||
|
||||
struct Mutex;
|
||||
struct MutexGuard<'a>(&'a Mutex);
|
||||
|
||||
impl Drop for Mutex { fn drop(&mut self) { println!("Mutex::drop"); } }
|
||||
impl<'a> Drop for MutexGuard<'a> { fn drop(&mut self) { println!("MutexGuard::drop"); } }
|
||||
|
||||
impl Mutex {
|
||||
fn lock(&self) -> Result<MutexGuard, ()> { Ok(MutexGuard(self)) }
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let counter = Mutex;
|
||||
|
||||
if let Ok(_) = counter.lock() { }
|
||||
|
||||
// With this code as written, the dynamic semantics here implies
|
||||
// that `Mutex::drop` for `counter` runs *before*
|
||||
// `MutexGuard::drop`, which would be unsound since `MutexGuard`
|
||||
// still has a reference to `counter`.
|
||||
//
|
||||
// The goal of #54556 is to explain that within a compiler
|
||||
// diagnostic.
|
||||
}
|
14
src/test/ui/nll/issue-54556-niconii.stderr
Normal file
14
src/test/ui/nll/issue-54556-niconii.stderr
Normal file
@ -0,0 +1,14 @@
|
||||
error[E0597]: `counter` does not live long enough
|
||||
--> $DIR/issue-54556-niconii.rs:22:20
|
||||
|
|
||||
LL | if let Ok(_) = counter.lock() { }
|
||||
| ^^^^^^^ borrowed value does not live long enough
|
||||
...
|
||||
LL | }
|
||||
| - `counter` dropped here while still borrowed
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are created
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
19
src/test/ui/nll/issue-54556-stephaneyfx.nll.stderr
Normal file
19
src/test/ui/nll/issue-54556-stephaneyfx.nll.stderr
Normal file
@ -0,0 +1,19 @@
|
||||
error[E0597]: `stmt` does not live long enough
|
||||
--> $DIR/issue-54556-stephaneyfx.rs:27:21
|
||||
|
|
||||
LL | let rows = Rows(&stmt);
|
||||
| ^^^^^ borrowed value does not live long enough
|
||||
LL | rows.map(|row| row).next()
|
||||
| ------------------- a temporary with access to the borrow is created here ...
|
||||
...
|
||||
LL | }
|
||||
| -
|
||||
| |
|
||||
| `stmt` dropped here while still borrowed
|
||||
| ... and the borrow might be used here, when that temporary is dropped and runs the destructor for type `std::iter::Map<Rows<'_>, [closure@$DIR/issue-54556-stephaneyfx.rs:28:14: 28:23]>`
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
35
src/test/ui/nll/issue-54556-stephaneyfx.rs
Normal file
35
src/test/ui/nll/issue-54556-stephaneyfx.rs
Normal file
@ -0,0 +1,35 @@
|
||||
// This is a reduction of a concrete test illustrating a case that was
|
||||
// annoying to Rust developer stephaneyfx (see issue #46413).
|
||||
//
|
||||
// With resolving issue #54556, pnkfelix hopes that the new diagnostic
|
||||
// output produced by NLL helps to *explain* the semantic significance
|
||||
// of temp drop order, and thus why storing the result in `x` and then
|
||||
// returning `x` works.
|
||||
|
||||
pub struct Statement;
|
||||
|
||||
pub struct Rows<'stmt>(&'stmt Statement);
|
||||
|
||||
impl<'stmt> Drop for Rows<'stmt> {
|
||||
fn drop(&mut self) {}
|
||||
}
|
||||
|
||||
impl<'stmt> Iterator for Rows<'stmt> {
|
||||
type Item = String;
|
||||
|
||||
fn next(&mut self) -> Option<Self::Item> {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
fn get_names() -> Option<String> {
|
||||
let stmt = Statement;
|
||||
let rows = Rows(&stmt);
|
||||
rows.map(|row| row).next()
|
||||
// let x = rows.map(|row| row).next();
|
||||
// x
|
||||
//
|
||||
// Removing the map works too as does removing the Drop impl.
|
||||
}
|
||||
|
||||
fn main() {}
|
14
src/test/ui/nll/issue-54556-stephaneyfx.stderr
Normal file
14
src/test/ui/nll/issue-54556-stephaneyfx.stderr
Normal file
@ -0,0 +1,14 @@
|
||||
error[E0597]: `stmt` does not live long enough
|
||||
--> $DIR/issue-54556-stephaneyfx.rs:27:22
|
||||
|
|
||||
LL | let rows = Rows(&stmt);
|
||||
| ^^^^ borrowed value does not live long enough
|
||||
...
|
||||
LL | }
|
||||
| - `stmt` dropped here while still borrowed
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are created
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
@ -0,0 +1,19 @@
|
||||
error[E0597]: `_thing1` does not live long enough
|
||||
--> $DIR/issue-54556-temps-in-tail-diagnostic.rs:5:11
|
||||
|
|
||||
LL | D(&_thing1).end()
|
||||
| --^^^^^^^^-
|
||||
| | |
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
LL | }
|
||||
| - `_thing1` dropped here while still borrowed
|
||||
LL |
|
||||
LL | ;
|
||||
| - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
23
src/test/ui/nll/issue-54556-temps-in-tail-diagnostic.rs
Normal file
23
src/test/ui/nll/issue-54556-temps-in-tail-diagnostic.rs
Normal file
@ -0,0 +1,23 @@
|
||||
fn main() {
|
||||
{
|
||||
let mut _thing1 = D(Box::new("thing1"));
|
||||
// D("other").next(&_thing1).end()
|
||||
D(&_thing1).end()
|
||||
}
|
||||
|
||||
;
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
struct D<T: std::fmt::Debug>(T);
|
||||
|
||||
impl<T: std::fmt::Debug> Drop for D<T> {
|
||||
fn drop(&mut self) {
|
||||
println!("dropping {:?})", self);
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: std::fmt::Debug> D<T> {
|
||||
fn next<U: std::fmt::Debug>(&self, _other: U) -> D<U> { D(_other) }
|
||||
fn end(&self) { }
|
||||
}
|
14
src/test/ui/nll/issue-54556-temps-in-tail-diagnostic.stderr
Normal file
14
src/test/ui/nll/issue-54556-temps-in-tail-diagnostic.stderr
Normal file
@ -0,0 +1,14 @@
|
||||
error[E0597]: `_thing1` does not live long enough
|
||||
--> $DIR/issue-54556-temps-in-tail-diagnostic.rs:5:12
|
||||
|
|
||||
LL | D(&_thing1).end()
|
||||
| ^^^^^^^ borrowed value does not live long enough
|
||||
LL | }
|
||||
| - `_thing1` dropped here while still borrowed
|
||||
LL |
|
||||
LL | ;
|
||||
| - borrowed value needs to live until here
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
113
src/test/ui/nll/issue-54556-used-vs-unused-tails.nll.stderr
Normal file
113
src/test/ui/nll/issue-54556-used-vs-unused-tails.nll.stderr
Normal file
@ -0,0 +1,113 @@
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:10:55
|
||||
|
|
||||
LL | { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // suggest `;`
|
||||
| --^^^^- - - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:12:55
|
||||
|
|
||||
LL | { { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } } ; // suggest `;`
|
||||
| --^^^^- - - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:14:55
|
||||
|
|
||||
LL | { { let mut _t1 = D(Box::new("t1")); D(&_t1).end() }; } // suggest `;`
|
||||
| --^^^^- -- ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:16:55
|
||||
|
|
||||
LL | let _ = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // suggest `;`
|
||||
| --^^^^- - - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:18:55
|
||||
|
|
||||
LL | let _u = { let mut _t1 = D(Box::new("t1")); D(&_t1).unit() } ; // suggest `;`
|
||||
| --^^^^- - - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:20:55
|
||||
|
|
||||
LL | let _x = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // `let x = ...; x`
|
||||
| --^^^^- - - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:24:55
|
||||
|
|
||||
LL | _y = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // `let x = ...; x`
|
||||
| --^^^^- - - ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:30:55
|
||||
|
|
||||
LL | fn f_local_ref() { let mut _t1 = D(Box::new("t1")); D(&_t1).unit() } // suggest `;`
|
||||
| --^^^^- -
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | | ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider adding semicolon after the expression so its temporaries are dropped sooner, before the local variables declared by the block are dropped.
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:32:55
|
||||
|
|
||||
LL | fn f() -> String { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } // `let x = ...; x`
|
||||
| --^^^^- -
|
||||
| | | |
|
||||
| | | `_t1` dropped here while still borrowed
|
||||
| | | ... and the borrow might be used here, when that temporary is dropped and runs the `Drop` code for type `D`
|
||||
| | borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error: aborting due to 9 previous errors
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
48
src/test/ui/nll/issue-54556-used-vs-unused-tails.rs
Normal file
48
src/test/ui/nll/issue-54556-used-vs-unused-tails.rs
Normal file
@ -0,0 +1,48 @@
|
||||
// Ths test case is exploring the space of how blocs with tail
|
||||
// expressions and statements can be composed, trying to keep each
|
||||
// case on one line so that we can compare them via a vertical scan
|
||||
// with the human eye.
|
||||
|
||||
// Each comment on the right side of the line is summarizing the
|
||||
// expected suggestion from the diagnostic for issue #54556.
|
||||
|
||||
fn main() {
|
||||
{ let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // suggest `;`
|
||||
|
||||
{ { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } } ; // suggest `;`
|
||||
|
||||
{ { let mut _t1 = D(Box::new("t1")); D(&_t1).end() }; } // suggest `;`
|
||||
|
||||
let _ = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // suggest `;`
|
||||
|
||||
let _u = { let mut _t1 = D(Box::new("t1")); D(&_t1).unit() } ; // suggest `;`
|
||||
|
||||
let _x = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // `let x = ...; x`
|
||||
let _x = { let mut _t1 = D(Box::new("t1")); let x = D(&_t1).end(); x } ; // no error
|
||||
|
||||
let mut _y;
|
||||
_y = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // `let x = ...; x`
|
||||
_y = { let mut _t1 = D(Box::new("t1")); let x = D(&_t1).end(); x } ; // no error
|
||||
}
|
||||
|
||||
fn f_param_ref(_t1: D<Box<&'static str>>) { D(&_t1).unit() } // no error
|
||||
|
||||
fn f_local_ref() { let mut _t1 = D(Box::new("t1")); D(&_t1).unit() } // suggest `;`
|
||||
|
||||
fn f() -> String { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } // `let x = ...; x`
|
||||
|
||||
|
||||
#[derive(Debug)]
|
||||
struct D<T: std::fmt::Debug>(T);
|
||||
|
||||
impl<T: std::fmt::Debug> Drop for D<T> {
|
||||
fn drop(&mut self) {
|
||||
println!("dropping {:?})", self);
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: std::fmt::Debug> D<T> {
|
||||
fn next<U: std::fmt::Debug>(&self, _other: U) -> D<U> { D(_other) }
|
||||
fn end(&self) -> String { format!("End({:?})", self.0) }
|
||||
fn unit(&self) { }
|
||||
}
|
86
src/test/ui/nll/issue-54556-used-vs-unused-tails.stderr
Normal file
86
src/test/ui/nll/issue-54556-used-vs-unused-tails.stderr
Normal file
@ -0,0 +1,86 @@
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:10:56
|
||||
|
|
||||
LL | { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // suggest `;`
|
||||
| ^^^ - - borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:12:56
|
||||
|
|
||||
LL | { { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } } ; // suggest `;`
|
||||
| ^^^ - - borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:14:56
|
||||
|
|
||||
LL | { { let mut _t1 = D(Box::new("t1")); D(&_t1).end() }; } // suggest `;`
|
||||
| ^^^ -- borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:16:56
|
||||
|
|
||||
LL | let _ = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // suggest `;`
|
||||
| ^^^ - - borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:18:56
|
||||
|
|
||||
LL | let _u = { let mut _t1 = D(Box::new("t1")); D(&_t1).unit() } ; // suggest `;`
|
||||
| ^^^ - - borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:20:56
|
||||
|
|
||||
LL | let _x = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // `let x = ...; x`
|
||||
| ^^^ - - borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:24:56
|
||||
|
|
||||
LL | _y = { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } ; // `let x = ...; x`
|
||||
| ^^^ - - borrowed value needs to live until here
|
||||
| | |
|
||||
| | `_t1` dropped here while still borrowed
|
||||
| borrowed value does not live long enough
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:30:56
|
||||
|
|
||||
LL | fn f_local_ref() { let mut _t1 = D(Box::new("t1")); D(&_t1).unit() } // suggest `;`
|
||||
| ^^^ - `_t1` dropped here while still borrowed
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are created
|
||||
|
||||
error[E0597]: `_t1` does not live long enough
|
||||
--> $DIR/issue-54556-used-vs-unused-tails.rs:32:56
|
||||
|
|
||||
LL | fn f() -> String { let mut _t1 = D(Box::new("t1")); D(&_t1).end() } // `let x = ...; x`
|
||||
| ^^^ - `_t1` dropped here while still borrowed
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are created
|
||||
|
||||
error: aborting due to 9 previous errors
|
||||
|
||||
For more information about this error, try `rustc --explain E0597`.
|
14
src/test/ui/nll/issue-54556-wrap-it-up.nll.stderr
Normal file
14
src/test/ui/nll/issue-54556-wrap-it-up.nll.stderr
Normal file
@ -0,0 +1,14 @@
|
||||
error[E0506]: cannot assign to `x` because it is borrowed
|
||||
--> $DIR/issue-54556-wrap-it-up.rs:27:5
|
||||
|
|
||||
LL | let wrap = Wrap { p: &mut x };
|
||||
| ------ borrow of `x` occurs here
|
||||
...
|
||||
LL | x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
| ^^^^^ assignment to borrowed `x` occurs here
|
||||
LL | }
|
||||
| - borrow might be used here, when `foo` is dropped and runs the destructor for type `Foo<'_>`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0506`.
|
28
src/test/ui/nll/issue-54556-wrap-it-up.rs
Normal file
28
src/test/ui/nll/issue-54556-wrap-it-up.rs
Normal file
@ -0,0 +1,28 @@
|
||||
// This is testing how the diagnostic from issue #54556 behaves when
|
||||
// the destructor code is attached to a place held in a field of the
|
||||
// temporary being dropped.
|
||||
//
|
||||
// Eventually it would be nice if the diagnostic would actually report
|
||||
// that specific place and its type that implements the `Drop` trait.
|
||||
// But for the short term, it is acceptable to just print out the
|
||||
// whole type of the temporary.
|
||||
|
||||
#![allow(warnings)]
|
||||
|
||||
struct Wrap<'p> { p: &'p mut i32 }
|
||||
|
||||
impl<'p> Drop for Wrap<'p> {
|
||||
fn drop(&mut self) {
|
||||
*self.p += 1;
|
||||
}
|
||||
}
|
||||
|
||||
struct Foo<'p> { a: String, b: Wrap<'p> }
|
||||
|
||||
fn main() {
|
||||
let mut x = 0;
|
||||
let wrap = Wrap { p: &mut x };
|
||||
let s = String::from("str");
|
||||
let foo = Foo { a: s, b: wrap };
|
||||
x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
}
|
12
src/test/ui/nll/issue-54556-wrap-it-up.stderr
Normal file
12
src/test/ui/nll/issue-54556-wrap-it-up.stderr
Normal file
@ -0,0 +1,12 @@
|
||||
error[E0506]: cannot assign to `x` because it is borrowed
|
||||
--> $DIR/issue-54556-wrap-it-up.rs:27:5
|
||||
|
|
||||
LL | let wrap = Wrap { p: &mut x };
|
||||
| - borrow of `x` occurs here
|
||||
...
|
||||
LL | x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
| ^^^^^ assignment to borrowed `x` occurs here
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0506`.
|
@ -8,7 +8,7 @@ LL | x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
| ^^^^^ assignment to borrowed `x` occurs here
|
||||
LL | // FIXME ^ Should not error in the future with implicit dtors, only manually implemented ones
|
||||
LL | }
|
||||
| - borrow later used here, when `foo` is dropped
|
||||
| - borrow might be used here, when `foo` is dropped and runs the destructor for type `Foo<'_>`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | let wrap = Wrap { p: &mut x };
|
||||
LL | x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
| ^^^^^ assignment to borrowed `x` occurs here
|
||||
LL | }
|
||||
| - borrow later used here, when `foo` is dropped
|
||||
| - borrow might be used here, when `foo` is dropped and runs the destructor for type `Foo<'_>`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
| ^^^^^ assignment to borrowed `x` occurs here
|
||||
LL | // FIXME ^ This currently errors and it should not.
|
||||
LL | }
|
||||
| - borrow later used here, when `foo` is dropped
|
||||
| - borrow might be used here, when `foo` is dropped and runs the destructor for type `Foo<'_>`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -6,7 +6,7 @@ LL | let wrap = Wrap { p: &mut x };
|
||||
LL | x = 1; //~ ERROR cannot assign to `x` because it is borrowed [E0506]
|
||||
| ^^^^^ assignment to borrowed `x` occurs here
|
||||
LL | }
|
||||
| - borrow later used here, when `wrap` is dropped
|
||||
| - borrow might be used here, when `wrap` is dropped and runs the `Drop` code for type `Wrap`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -2,9 +2,16 @@ error[E0597]: `*a` does not live long enough
|
||||
--> $DIR/destructor-restrictions.rs:18:10
|
||||
|
|
||||
LL | *a.borrow() + 1
|
||||
| ^ borrowed value does not live long enough
|
||||
| ^---------
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
LL | }; //~^ ERROR `*a` does not live long enough
|
||||
| - `*a` dropped here while still borrowed
|
||||
| -- ... and the borrow might be used here, when that temporary is dropped and runs the destructor for type `std::cell::Ref<'_, i32>`
|
||||
| |
|
||||
| `*a` dropped here while still borrowed
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `*m` dropped here while still borrowed
|
||||
| borrow later used here, when `m` is dropped
|
||||
| borrow might be used here, when `m` is dropped and runs the destructor for type `std::boxed::Box<dyn Trait<'_>>`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `b2` dropped here while still borrowed
|
||||
| borrow later used here, when `b1` is dropped
|
||||
| borrow might be used here, when `b1` is dropped and runs the destructor for type `B<'_>`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -22,7 +22,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `b3` dropped here while still borrowed
|
||||
| borrow later used here, when `b1` is dropped
|
||||
| borrow might be used here, when `b1` is dropped and runs the destructor for type `B<'_>`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -36,7 +36,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `b1` dropped here while still borrowed
|
||||
| borrow later used here, when `b1` is dropped
|
||||
| borrow might be used here, when `b1` is dropped and runs the destructor for type `B<'_>`
|
||||
|
||||
error: aborting due to 3 previous errors
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d2` dropped here while still borrowed
|
||||
| borrow later used here, when `d1` is dropped
|
||||
| borrow might be used here, when `d1` is dropped and runs the `Drop` code for type `D`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -22,7 +22,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d1` dropped here while still borrowed
|
||||
| borrow later used here, when `d1` is dropped
|
||||
| borrow might be used here, when `d1` is dropped and runs the `Drop` code for type `D`
|
||||
|
||||
error: aborting due to 2 previous errors
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `bomb` dropped here while still borrowed
|
||||
| borrow later used here, when `_w` is dropped
|
||||
| borrow might be used here, when `_w` is dropped and runs the destructor for type `Wrap<&[&str]>`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -21,7 +21,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `v` dropped here while still borrowed
|
||||
| borrow later used here, when `_w` is dropped
|
||||
| borrow might be used here, when `_w` is dropped and runs the destructor for closure
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c2` dropped here while still borrowed
|
||||
| borrow later used here, when `c1` is dropped
|
||||
| borrow might be used here, when `c1` is dropped and runs the destructor for type `C<'_>`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -22,7 +22,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c3` dropped here while still borrowed
|
||||
| borrow later used here, when `c1` is dropped
|
||||
| borrow might be used here, when `c1` is dropped and runs the destructor for type `C<'_>`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -36,7 +36,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c1` dropped here while still borrowed
|
||||
| borrow later used here, when `c1` is dropped
|
||||
| borrow might be used here, when `c1` is dropped and runs the destructor for type `C<'_>`
|
||||
|
||||
error: aborting due to 3 previous errors
|
||||
|
||||
|
@ -2,17 +2,32 @@ error[E0597]: `y` does not live long enough
|
||||
--> $DIR/issue-23338-locals-die-before-temps-of-body.rs:20:5
|
||||
|
|
||||
LL | y.borrow().clone()
|
||||
| ^ borrowed value does not live long enough
|
||||
| ^---------
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
LL | }
|
||||
| - `y` dropped here while still borrowed
|
||||
| -
|
||||
| |
|
||||
| `y` dropped here while still borrowed
|
||||
| ... and the borrow might be used here, when that temporary is dropped and runs the destructor for type `std::cell::Ref<'_, std::string::String>`
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error[E0597]: `y` does not live long enough
|
||||
--> $DIR/issue-23338-locals-die-before-temps-of-body.rs:27:9
|
||||
|
|
||||
LL | y.borrow().clone()
|
||||
| ^ borrowed value does not live long enough
|
||||
| ^---------
|
||||
| |
|
||||
| borrowed value does not live long enough
|
||||
| a temporary with access to the borrow is created here ...
|
||||
LL | };
|
||||
| - `y` dropped here while still borrowed
|
||||
| -- ... and the borrow might be used here, when that temporary is dropped and runs the destructor for type `std::cell::Ref<'_, std::string::String>`
|
||||
| |
|
||||
| `y` dropped here while still borrowed
|
||||
|
|
||||
= note: The temporary is part of an expression at the end of a block. Consider forcing this temporary to be dropped sooner, before the block's local variables are dropped. For example, you could save the expression's value in a new local variable `x` and then make `x` be the expression at the end of the block.
|
||||
|
||||
error: aborting due to 2 previous errors
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d1` dropped here while still borrowed
|
||||
| borrow later used here, when `_d` is dropped
|
||||
| borrow might be used here, when `_d` is dropped and runs the `Drop` code for type `D_Child`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d1` dropped here while still borrowed
|
||||
| borrow later used here, when `_d` is dropped
|
||||
| borrow might be used here, when `_d` is dropped and runs the `Drop` code for type `D_HasSelfMethod`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -20,7 +20,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d1` dropped here while still borrowed
|
||||
| borrow later used here, when `_d` is dropped
|
||||
| borrow might be used here, when `_d` is dropped and runs the `Drop` code for type `D_HasMethodWithSelfArg`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -33,7 +33,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d1` dropped here while still borrowed
|
||||
| borrow later used here, when `_d` is dropped
|
||||
| borrow might be used here, when `_d` is dropped and runs the `Drop` code for type `D_HasType`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `d1` dropped here while still borrowed
|
||||
| borrow later used here, when `d2` is dropped
|
||||
| borrow might be used here, when `d2` is dropped and runs the `Drop` code for type `D`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `container` dropped here while still borrowed
|
||||
| borrow later used here, when `container` is dropped
|
||||
| borrow might be used here, when `container` is dropped and runs the destructor for type `Container<'_>`
|
||||
|
||||
error: aborting due to previous error
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `ticking` dropped here while still borrowed
|
||||
| borrow later used here, when `zook` is dropped
|
||||
| borrow might be used here, when `zook` is dropped and runs the `Drop` code for type `Zook`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `x` dropped here while still borrowed
|
||||
| borrow later used here, when `y` is dropped
|
||||
| borrow might be used here, when `y` is dropped and runs the `Drop` code for type `std::sync::Arc`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -20,7 +20,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `x` dropped here while still borrowed
|
||||
| borrow later used here, when `y` is dropped
|
||||
| borrow might be used here, when `y` is dropped and runs the `Drop` code for type `std::rc::Rc`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| here, drop of `foo` needs exclusive access to `foo.data`, because the type `Foo<Concrete<'_>>` implements the `Drop` trait
|
||||
| borrow later used here, when `foo` is dropped
|
||||
| borrow might be used here, when `foo` is dropped and runs the `Drop` code for type `Foo`
|
||||
|
|
||||
= note: consider using a `let` binding to create a longer lived value
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `first_dropped` dropped here while still borrowed
|
||||
| borrow later used here, when `foo1` is dropped
|
||||
| borrow might be used here, when `foo1` is dropped and runs the `Drop` code for type `Foo`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `first_dropped` dropped here while still borrowed
|
||||
| borrow later used here, when `foo1` is dropped
|
||||
| borrow might be used here, when `foo1` is dropped and runs the `Drop` code for type `Foo`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `first_dropped` dropped here while still borrowed
|
||||
| borrow later used here, when `foo1` is dropped
|
||||
| borrow might be used here, when `foo1` is dropped and runs the `Drop` code for type `Foo`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -7,7 +7,7 @@ LL | let ss: &isize = &id(1);
|
||||
LL | }
|
||||
| - temporary value is freed at the end of this statement
|
||||
LL | }
|
||||
| - borrow later used here, when `blah` is dropped
|
||||
| - borrow might be used here, when `blah` is dropped and runs the destructor for type `std::boxed::Box<dyn Foo>`
|
||||
|
|
||||
= note: consider using a `let` binding to create a longer lived value
|
||||
|
||||
|
@ -62,7 +62,7 @@ LL | }
|
||||
| - `z` dropped here while still borrowed
|
||||
...
|
||||
LL | }
|
||||
| - borrow later used here, when `tx` is dropped
|
||||
| - borrow might be used here, when `tx` is dropped and runs the `Drop` code for type `std::sync::mpsc::Sender`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c2` dropped here while still borrowed
|
||||
| borrow later used here, when `c1` is dropped
|
||||
| borrow might be used here, when `c1` is dropped and runs the destructor for type `C<'_>`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -22,7 +22,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `c1` dropped here while still borrowed
|
||||
| borrow later used here, when `c1` is dropped
|
||||
| borrow might be used here, when `c1` is dropped and runs the destructor for type `C<'_>`
|
||||
|
||||
error: aborting due to 2 previous errors
|
||||
|
||||
|
@ -8,7 +8,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `x` dropped here while still borrowed
|
||||
| borrow later used here, when `v` is dropped
|
||||
| borrow might be used here, when `v` is dropped and runs the `Drop` code for type `Bag`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
@ -22,7 +22,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `y` dropped here while still borrowed
|
||||
| borrow later used here, when `v` is dropped
|
||||
| borrow might be used here, when `v` is dropped and runs the `Drop` code for type `Bag`
|
||||
|
|
||||
= note: values in a scope are dropped in the opposite order they are defined
|
||||
|
||||
|
@ -10,7 +10,7 @@ LL | }
|
||||
| -
|
||||
| |
|
||||
| `factorial` dropped here while still borrowed
|
||||
| borrow later used here, when `factorial` is dropped
|
||||
| borrow might be used here, when `factorial` is dropped and runs the destructor for type `std::option::Option<std::boxed::Box<dyn std::ops::Fn(u32) -> u32>>`
|
||||
|
||||
error[E0506]: cannot assign to `factorial` because it is borrowed
|
||||
--> $DIR/unboxed-closures-failed-recursive-fn-1.rs:30:5
|
||||
|
Loading…
Reference in New Issue
Block a user