mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-23 15:23:46 +00:00
Auto merge of #54188 - lqd:fallout-53695, r=nikomatsakis
NLL: disallow creation of immediately unusable variables Fix #53695 Original description follows ---- This WIP PR is for discussing the impact of fixing #53695 by injecting a fake read in let patterns. (Travis will fail, at least the `mir-opt` suite is failing in its current state)
This commit is contained in:
commit
4591a245c7
@ -238,7 +238,8 @@ for mir::StatementKind<'gcx> {
|
|||||||
place.hash_stable(hcx, hasher);
|
place.hash_stable(hcx, hasher);
|
||||||
rvalue.hash_stable(hcx, hasher);
|
rvalue.hash_stable(hcx, hasher);
|
||||||
}
|
}
|
||||||
mir::StatementKind::ReadForMatch(ref place) => {
|
mir::StatementKind::FakeRead(ref cause, ref place) => {
|
||||||
|
cause.hash_stable(hcx, hasher);
|
||||||
place.hash_stable(hcx, hasher);
|
place.hash_stable(hcx, hasher);
|
||||||
}
|
}
|
||||||
mir::StatementKind::SetDiscriminant { ref place, variant_index } => {
|
mir::StatementKind::SetDiscriminant { ref place, variant_index } => {
|
||||||
@ -271,6 +272,8 @@ for mir::StatementKind<'gcx> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl_stable_hash_for!(enum mir::FakeReadCause { ForMatch, ForLet });
|
||||||
|
|
||||||
impl<'a, 'gcx, T> HashStable<StableHashingContext<'a>>
|
impl<'a, 'gcx, T> HashStable<StableHashingContext<'a>>
|
||||||
for mir::ValidationOperand<'gcx, T>
|
for mir::ValidationOperand<'gcx, T>
|
||||||
where T: HashStable<StableHashingContext<'a>>
|
where T: HashStable<StableHashingContext<'a>>
|
||||||
|
@ -1613,8 +1613,10 @@ pub enum StatementKind<'tcx> {
|
|||||||
Assign(Place<'tcx>, Rvalue<'tcx>),
|
Assign(Place<'tcx>, Rvalue<'tcx>),
|
||||||
|
|
||||||
/// This represents all the reading that a pattern match may do
|
/// This represents all the reading that a pattern match may do
|
||||||
/// (e.g. inspecting constants and discriminant values).
|
/// (e.g. inspecting constants and discriminant values), and the
|
||||||
ReadForMatch(Place<'tcx>),
|
/// kind of pattern it comes from. This is in order to adapt potential
|
||||||
|
/// error messages to these specific patterns.
|
||||||
|
FakeRead(FakeReadCause, Place<'tcx>),
|
||||||
|
|
||||||
/// Write the discriminant for a variant to the enum Place.
|
/// Write the discriminant for a variant to the enum Place.
|
||||||
SetDiscriminant {
|
SetDiscriminant {
|
||||||
@ -1662,6 +1664,31 @@ pub enum StatementKind<'tcx> {
|
|||||||
Nop,
|
Nop,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// The `FakeReadCause` describes the type of pattern why a `FakeRead` statement exists.
|
||||||
|
#[derive(Copy, Clone, RustcEncodable, RustcDecodable, Debug)]
|
||||||
|
pub enum FakeReadCause {
|
||||||
|
/// Inject a fake read of the borrowed input at the start of each arm's
|
||||||
|
/// pattern testing code.
|
||||||
|
///
|
||||||
|
/// This should ensure that you cannot change the variant for an enum
|
||||||
|
/// while you are in the midst of matching on it.
|
||||||
|
ForMatch,
|
||||||
|
|
||||||
|
/// Officially, the semantics of
|
||||||
|
///
|
||||||
|
/// `let pattern = <expr>;`
|
||||||
|
///
|
||||||
|
/// is that `<expr>` is evaluated into a temporary and then this temporary is
|
||||||
|
/// into the pattern.
|
||||||
|
///
|
||||||
|
/// However, if we see the simple pattern `let var = <expr>`, we optimize this to
|
||||||
|
/// evaluate `<expr>` directly into the variable `var`. This is mostly unobservable,
|
||||||
|
/// but in some cases it can affect the borrow checker, as in #53695.
|
||||||
|
/// Therefore, we insert a "fake read" here to ensure that we get
|
||||||
|
/// appropriate errors.
|
||||||
|
ForLet,
|
||||||
|
}
|
||||||
|
|
||||||
/// The `ValidationOp` describes what happens with each of the operands of a
|
/// The `ValidationOp` describes what happens with each of the operands of a
|
||||||
/// `Validate` statement.
|
/// `Validate` statement.
|
||||||
#[derive(Copy, Clone, RustcEncodable, RustcDecodable, PartialEq, Eq)]
|
#[derive(Copy, Clone, RustcEncodable, RustcDecodable, PartialEq, Eq)]
|
||||||
@ -1718,7 +1745,7 @@ impl<'tcx> Debug for Statement<'tcx> {
|
|||||||
use self::StatementKind::*;
|
use self::StatementKind::*;
|
||||||
match self.kind {
|
match self.kind {
|
||||||
Assign(ref place, ref rv) => write!(fmt, "{:?} = {:?}", place, rv),
|
Assign(ref place, ref rv) => write!(fmt, "{:?} = {:?}", place, rv),
|
||||||
ReadForMatch(ref place) => write!(fmt, "ReadForMatch({:?})", place),
|
FakeRead(ref cause, ref place) => write!(fmt, "FakeRead({:?}, {:?})", cause, place),
|
||||||
// (reuse lifetime rendering policy from ppaux.)
|
// (reuse lifetime rendering policy from ppaux.)
|
||||||
EndRegion(ref ce) => write!(fmt, "EndRegion({})", ty::ReScope(*ce)),
|
EndRegion(ref ce) => write!(fmt, "EndRegion({})", ty::ReScope(*ce)),
|
||||||
Validate(ref op, ref places) => write!(fmt, "Validate({:?}, {:?})", op, places),
|
Validate(ref op, ref places) => write!(fmt, "Validate({:?}, {:?})", op, places),
|
||||||
@ -2585,6 +2612,7 @@ CloneTypeFoldableAndLiftImpls! {
|
|||||||
Mutability,
|
Mutability,
|
||||||
SourceInfo,
|
SourceInfo,
|
||||||
UpvarDecl,
|
UpvarDecl,
|
||||||
|
FakeReadCause,
|
||||||
ValidationOp,
|
ValidationOp,
|
||||||
SourceScope,
|
SourceScope,
|
||||||
SourceScopeData,
|
SourceScopeData,
|
||||||
@ -2651,7 +2679,7 @@ BraceStructTypeFoldableImpl! {
|
|||||||
EnumTypeFoldableImpl! {
|
EnumTypeFoldableImpl! {
|
||||||
impl<'tcx> TypeFoldable<'tcx> for StatementKind<'tcx> {
|
impl<'tcx> TypeFoldable<'tcx> for StatementKind<'tcx> {
|
||||||
(StatementKind::Assign)(a, b),
|
(StatementKind::Assign)(a, b),
|
||||||
(StatementKind::ReadForMatch)(place),
|
(StatementKind::FakeRead)(cause, place),
|
||||||
(StatementKind::SetDiscriminant) { place, variant_index },
|
(StatementKind::SetDiscriminant) { place, variant_index },
|
||||||
(StatementKind::StorageLive)(a),
|
(StatementKind::StorageLive)(a),
|
||||||
(StatementKind::StorageDead)(a),
|
(StatementKind::StorageDead)(a),
|
||||||
|
@ -354,7 +354,7 @@ macro_rules! make_mir_visitor {
|
|||||||
ref $($mutability)* rvalue) => {
|
ref $($mutability)* rvalue) => {
|
||||||
self.visit_assign(block, place, rvalue, location);
|
self.visit_assign(block, place, rvalue, location);
|
||||||
}
|
}
|
||||||
StatementKind::ReadForMatch(ref $($mutability)* place) => {
|
StatementKind::FakeRead(_, ref $($mutability)* place) => {
|
||||||
self.visit_place(place,
|
self.visit_place(place,
|
||||||
PlaceContext::Inspect,
|
PlaceContext::Inspect,
|
||||||
location);
|
location);
|
||||||
|
@ -1330,7 +1330,7 @@ options! {DebuggingOptions, DebuggingSetter, basic_debugging_options,
|
|||||||
disable_nll_user_type_assert: bool = (false, parse_bool, [UNTRACKED],
|
disable_nll_user_type_assert: bool = (false, parse_bool, [UNTRACKED],
|
||||||
"disable user provided type assertion in NLL"),
|
"disable user provided type assertion in NLL"),
|
||||||
nll_dont_emit_read_for_match: bool = (false, parse_bool, [UNTRACKED],
|
nll_dont_emit_read_for_match: bool = (false, parse_bool, [UNTRACKED],
|
||||||
"in match codegen, do not include ReadForMatch statements (used by mir-borrowck)"),
|
"in match codegen, do not include FakeRead statements (used by mir-borrowck)"),
|
||||||
dont_buffer_diagnostics: bool = (false, parse_bool, [UNTRACKED],
|
dont_buffer_diagnostics: bool = (false, parse_bool, [UNTRACKED],
|
||||||
"emit diagnostics rather than buffering (breaks NLL error downgrading, sorting)."),
|
"emit diagnostics rather than buffering (breaks NLL error downgrading, sorting)."),
|
||||||
polonius: bool = (false, parse_bool, [UNTRACKED],
|
polonius: bool = (false, parse_bool, [UNTRACKED],
|
||||||
|
@ -1505,7 +1505,7 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
|
|||||||
self.emit_read_for_match()
|
self.emit_read_for_match()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// If true, make MIR codegen for `match` emit ReadForMatch
|
/// If true, make MIR codegen for `match` emit FakeRead
|
||||||
/// statements (which simulate the maximal effect of executing the
|
/// statements (which simulate the maximal effect of executing the
|
||||||
/// patterns in a match arm).
|
/// patterns in a match arm).
|
||||||
pub fn emit_read_for_match(&self) -> bool {
|
pub fn emit_read_for_match(&self) -> bool {
|
||||||
|
@ -89,7 +89,7 @@ impl FunctionCx<'a, 'll, 'tcx> {
|
|||||||
asm::codegen_inline_asm(&bx, asm, outputs, input_vals);
|
asm::codegen_inline_asm(&bx, asm, outputs, input_vals);
|
||||||
bx
|
bx
|
||||||
}
|
}
|
||||||
mir::StatementKind::ReadForMatch(_) |
|
mir::StatementKind::FakeRead(..) |
|
||||||
mir::StatementKind::EndRegion(_) |
|
mir::StatementKind::EndRegion(_) |
|
||||||
mir::StatementKind::Validate(..) |
|
mir::StatementKind::Validate(..) |
|
||||||
mir::StatementKind::AscribeUserType(..) |
|
mir::StatementKind::AscribeUserType(..) |
|
||||||
|
@ -12,7 +12,7 @@ use borrow_check::WriteKind;
|
|||||||
use rustc::middle::region::ScopeTree;
|
use rustc::middle::region::ScopeTree;
|
||||||
use rustc::mir::VarBindingForm;
|
use rustc::mir::VarBindingForm;
|
||||||
use rustc::mir::{BindingForm, BorrowKind, ClearCrossCrate, Field, Local};
|
use rustc::mir::{BindingForm, BorrowKind, ClearCrossCrate, Field, Local};
|
||||||
use rustc::mir::{LocalDecl, LocalKind, Location, Operand, Place};
|
use rustc::mir::{FakeReadCause, LocalDecl, LocalKind, Location, Operand, Place};
|
||||||
use rustc::mir::{ProjectionElem, Rvalue, Statement, StatementKind};
|
use rustc::mir::{ProjectionElem, Rvalue, Statement, StatementKind};
|
||||||
use rustc::ty;
|
use rustc::ty;
|
||||||
use rustc_data_structures::fx::FxHashSet;
|
use rustc_data_structures::fx::FxHashSet;
|
||||||
@ -1020,6 +1020,21 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
|||||||
false
|
false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the `FakeReadCause` at this location if it is a `FakeRead` statement.
|
||||||
|
pub(super) fn retrieve_fake_read_cause_for_location(
|
||||||
|
&self,
|
||||||
|
location: &Location,
|
||||||
|
) -> Option<FakeReadCause> {
|
||||||
|
let stmt = self.mir.basic_blocks()[location.block]
|
||||||
|
.statements
|
||||||
|
.get(location.statement_index)?;
|
||||||
|
if let StatementKind::FakeRead(cause, _) = stmt.kind {
|
||||||
|
Some(cause)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// The span(s) associated to a use of a place.
|
// The span(s) associated to a use of a place.
|
||||||
|
@ -501,9 +501,9 @@ impl<'cx, 'gcx, 'tcx> DataflowResultsConsumer<'cx, 'tcx> for MirBorrowckCtxt<'cx
|
|||||||
flow_state,
|
flow_state,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
StatementKind::ReadForMatch(ref place) => {
|
StatementKind::FakeRead(_, ref place) => {
|
||||||
self.access_place(
|
self.access_place(
|
||||||
ContextKind::ReadForMatch.new(location),
|
ContextKind::FakeRead.new(location),
|
||||||
(place, span),
|
(place, span),
|
||||||
(Deep, Read(ReadKind::Borrow(BorrowKind::Shared))),
|
(Deep, Read(ReadKind::Borrow(BorrowKind::Shared))),
|
||||||
LocalMutationIsAllowed::No,
|
LocalMutationIsAllowed::No,
|
||||||
@ -2229,7 +2229,7 @@ enum ContextKind {
|
|||||||
CallDest,
|
CallDest,
|
||||||
Assert,
|
Assert,
|
||||||
Yield,
|
Yield,
|
||||||
ReadForMatch,
|
FakeRead,
|
||||||
StorageDead,
|
StorageDead,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
use borrow_check::borrow_set::BorrowData;
|
use borrow_check::borrow_set::BorrowData;
|
||||||
use borrow_check::nll::region_infer::Cause;
|
use borrow_check::nll::region_infer::Cause;
|
||||||
use borrow_check::{Context, MirBorrowckCtxt, WriteKind};
|
use borrow_check::{Context, MirBorrowckCtxt, WriteKind};
|
||||||
use rustc::mir::{Local, Location, Place, TerminatorKind};
|
use rustc::mir::{FakeReadCause, Local, Location, Place, TerminatorKind};
|
||||||
use rustc_errors::DiagnosticBuilder;
|
use rustc_errors::DiagnosticBuilder;
|
||||||
use rustc::ty::Region;
|
use rustc::ty::Region;
|
||||||
|
|
||||||
@ -142,7 +142,13 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
|
|||||||
if spans.for_closure() {
|
if spans.for_closure() {
|
||||||
"borrow later captured here by closure"
|
"borrow later captured here by closure"
|
||||||
} else {
|
} else {
|
||||||
"borrow later used here"
|
// Check if the location represents a `FakeRead`, and adapt the error
|
||||||
|
// message to the `FakeReadCause` it is from: in particular,
|
||||||
|
// the ones inserted in optimized `let var = <expr>` patterns.
|
||||||
|
match self.retrieve_fake_read_cause_for_location(&location) {
|
||||||
|
Some(FakeReadCause::ForLet) => "borrow later stored here",
|
||||||
|
_ => "borrow later used here"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
err.span_label(spans.var_or_use(), message);
|
err.span_label(spans.var_or_use(), message);
|
||||||
|
@ -93,9 +93,9 @@ impl<'cg, 'cx, 'tcx, 'gcx> Visitor<'tcx> for InvalidationGenerator<'cg, 'cx, 'tc
|
|||||||
JustWrite
|
JustWrite
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
StatementKind::ReadForMatch(ref place) => {
|
StatementKind::FakeRead(_, ref place) => {
|
||||||
self.access_place(
|
self.access_place(
|
||||||
ContextKind::ReadForMatch.new(location),
|
ContextKind::FakeRead.new(location),
|
||||||
place,
|
place,
|
||||||
(Deep, Read(ReadKind::Borrow(BorrowKind::Shared))),
|
(Deep, Read(ReadKind::Borrow(BorrowKind::Shared))),
|
||||||
LocalMutationIsAllowed::No,
|
LocalMutationIsAllowed::No,
|
||||||
|
@ -995,7 +995,7 @@ impl<'a, 'gcx, 'tcx> TypeChecker<'a, 'gcx, 'tcx> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
StatementKind::ReadForMatch(_)
|
StatementKind::FakeRead(..)
|
||||||
| StatementKind::StorageLive(_)
|
| StatementKind::StorageLive(_)
|
||||||
| StatementKind::StorageDead(_)
|
| StatementKind::StorageDead(_)
|
||||||
| StatementKind::InlineAsm { .. }
|
| StatementKind::InlineAsm { .. }
|
||||||
|
@ -145,19 +145,16 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
|||||||
if let (true, Some(borrow_temp)) =
|
if let (true, Some(borrow_temp)) =
|
||||||
(tcx.emit_read_for_match(), borrowed_input_temp.clone())
|
(tcx.emit_read_for_match(), borrowed_input_temp.clone())
|
||||||
{
|
{
|
||||||
// inject a fake read of the borrowed input at
|
// Inject a fake read, see comments on `FakeReadCause::ForMatch`.
|
||||||
// the start of each arm's pattern testing
|
|
||||||
// code.
|
|
||||||
//
|
|
||||||
// This should ensure that you cannot change
|
|
||||||
// the variant for an enum while you are in
|
|
||||||
// the midst of matching on it.
|
|
||||||
let pattern_source_info = self.source_info(pattern.span);
|
let pattern_source_info = self.source_info(pattern.span);
|
||||||
self.cfg.push(
|
self.cfg.push(
|
||||||
*pre_binding_block,
|
*pre_binding_block,
|
||||||
Statement {
|
Statement {
|
||||||
source_info: pattern_source_info,
|
source_info: pattern_source_info,
|
||||||
kind: StatementKind::ReadForMatch(borrow_temp.clone()),
|
kind: StatementKind::FakeRead(
|
||||||
|
FakeReadCause::ForMatch,
|
||||||
|
borrow_temp.clone(),
|
||||||
|
),
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -264,6 +261,18 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
|||||||
let place =
|
let place =
|
||||||
self.storage_live_binding(block, var, irrefutable_pat.span, OutsideGuard);
|
self.storage_live_binding(block, var, irrefutable_pat.span, OutsideGuard);
|
||||||
unpack!(block = self.into(&place, block, initializer));
|
unpack!(block = self.into(&place, block, initializer));
|
||||||
|
|
||||||
|
|
||||||
|
// Inject a fake read, see comments on `FakeReadCause::ForLet`.
|
||||||
|
let source_info = self.source_info(irrefutable_pat.span);
|
||||||
|
self.cfg.push(
|
||||||
|
block,
|
||||||
|
Statement {
|
||||||
|
source_info,
|
||||||
|
kind: StatementKind::FakeRead(FakeReadCause::ForLet, place.clone()),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
self.schedule_drop_for_binding(var, irrefutable_pat.span, OutsideGuard);
|
self.schedule_drop_for_binding(var, irrefutable_pat.span, OutsideGuard);
|
||||||
block.unit()
|
block.unit()
|
||||||
}
|
}
|
||||||
@ -305,6 +314,15 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Inject a fake read, see comments on `FakeReadCause::ForLet`.
|
||||||
|
self.cfg.push(
|
||||||
|
block,
|
||||||
|
Statement {
|
||||||
|
source_info,
|
||||||
|
kind: StatementKind::FakeRead(FakeReadCause::ForLet, place.clone()),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
self.schedule_drop_for_binding(var, irrefutable_pat.span, OutsideGuard);
|
self.schedule_drop_for_binding(var, irrefutable_pat.span, OutsideGuard);
|
||||||
block.unit()
|
block.unit()
|
||||||
}
|
}
|
||||||
|
@ -336,7 +336,7 @@ impl<'a, 'gcx, 'tcx> BitDenotation for Borrows<'a, 'gcx, 'tcx> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mir::StatementKind::ReadForMatch(..) |
|
mir::StatementKind::FakeRead(..) |
|
||||||
mir::StatementKind::SetDiscriminant { .. } |
|
mir::StatementKind::SetDiscriminant { .. } |
|
||||||
mir::StatementKind::StorageLive(..) |
|
mir::StatementKind::StorageLive(..) |
|
||||||
mir::StatementKind::Validate(..) |
|
mir::StatementKind::Validate(..) |
|
||||||
|
@ -281,7 +281,7 @@ impl<'b, 'a, 'gcx, 'tcx> Gatherer<'b, 'a, 'gcx, 'tcx> {
|
|||||||
}
|
}
|
||||||
self.gather_rvalue(rval);
|
self.gather_rvalue(rval);
|
||||||
}
|
}
|
||||||
StatementKind::ReadForMatch(ref place) => {
|
StatementKind::FakeRead(_, ref place) => {
|
||||||
self.create_move_path(place);
|
self.create_move_path(place);
|
||||||
}
|
}
|
||||||
StatementKind::InlineAsm { ref outputs, ref inputs, ref asm } => {
|
StatementKind::InlineAsm { ref outputs, ref inputs, ref asm } => {
|
||||||
|
@ -150,9 +150,9 @@ impl<'a, 'mir, 'tcx, M: Machine<'mir, 'tcx>> EvalContext<'a, 'mir, 'tcx, M> {
|
|||||||
self.deallocate_local(old_val)?;
|
self.deallocate_local(old_val)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
// No dynamic semantics attached to `ReadForMatch`; MIR
|
// No dynamic semantics attached to `FakeRead`; MIR
|
||||||
// interpreter is solely intended for borrowck'ed code.
|
// interpreter is solely intended for borrowck'ed code.
|
||||||
ReadForMatch(..) => {}
|
FakeRead(..) => {}
|
||||||
|
|
||||||
// Validity checks.
|
// Validity checks.
|
||||||
Validate(op, ref places) => {
|
Validate(op, ref places) => {
|
||||||
|
@ -108,7 +108,7 @@ impl<'a, 'tcx> Visitor<'tcx> for UnsafetyChecker<'a, 'tcx> {
|
|||||||
self.source_info = statement.source_info;
|
self.source_info = statement.source_info;
|
||||||
match statement.kind {
|
match statement.kind {
|
||||||
StatementKind::Assign(..) |
|
StatementKind::Assign(..) |
|
||||||
StatementKind::ReadForMatch(..) |
|
StatementKind::FakeRead(..) |
|
||||||
StatementKind::SetDiscriminant { .. } |
|
StatementKind::SetDiscriminant { .. } |
|
||||||
StatementKind::StorageLive(..) |
|
StatementKind::StorageLive(..) |
|
||||||
StatementKind::StorageDead(..) |
|
StatementKind::StorageDead(..) |
|
||||||
|
@ -1090,7 +1090,7 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
|
|||||||
StatementKind::Assign(ref place, ref rvalue) => {
|
StatementKind::Assign(ref place, ref rvalue) => {
|
||||||
this.visit_assign(bb, place, rvalue, location);
|
this.visit_assign(bb, place, rvalue, location);
|
||||||
}
|
}
|
||||||
StatementKind::ReadForMatch(..) |
|
StatementKind::FakeRead(..) |
|
||||||
StatementKind::SetDiscriminant { .. } |
|
StatementKind::SetDiscriminant { .. } |
|
||||||
StatementKind::StorageLive(_) |
|
StatementKind::StorageLive(_) |
|
||||||
StatementKind::StorageDead(_) |
|
StatementKind::StorageDead(_) |
|
||||||
|
@ -216,7 +216,7 @@ fn check_statement(
|
|||||||
check_rvalue(tcx, mir, rval, span)
|
check_rvalue(tcx, mir, rval, span)
|
||||||
}
|
}
|
||||||
|
|
||||||
StatementKind::ReadForMatch(_) => Err((span, "match in const fn is unstable".into())),
|
StatementKind::FakeRead(..) => Err((span, "match in const fn is unstable".into())),
|
||||||
|
|
||||||
// just an assignment
|
// just an assignment
|
||||||
StatementKind::SetDiscriminant { .. } => Ok(()),
|
StatementKind::SetDiscriminant { .. } => Ok(()),
|
||||||
|
@ -49,7 +49,7 @@ impl RemoveNoopLandingPads {
|
|||||||
) -> bool {
|
) -> bool {
|
||||||
for stmt in &mir[bb].statements {
|
for stmt in &mir[bb].statements {
|
||||||
match stmt.kind {
|
match stmt.kind {
|
||||||
StatementKind::ReadForMatch(_) |
|
StatementKind::FakeRead(..) |
|
||||||
StatementKind::StorageLive(_) |
|
StatementKind::StorageLive(_) |
|
||||||
StatementKind::StorageDead(_) |
|
StatementKind::StorageDead(_) |
|
||||||
StatementKind::EndRegion(_) |
|
StatementKind::EndRegion(_) |
|
||||||
|
@ -157,7 +157,7 @@ fn each_block<'a, 'tcx, O>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
|
|||||||
mir::StatementKind::Assign(ref place, ref rvalue) => {
|
mir::StatementKind::Assign(ref place, ref rvalue) => {
|
||||||
(place, rvalue)
|
(place, rvalue)
|
||||||
}
|
}
|
||||||
mir::StatementKind::ReadForMatch(_) |
|
mir::StatementKind::FakeRead(..) |
|
||||||
mir::StatementKind::StorageLive(_) |
|
mir::StatementKind::StorageLive(_) |
|
||||||
mir::StatementKind::StorageDead(_) |
|
mir::StatementKind::StorageDead(_) |
|
||||||
mir::StatementKind::InlineAsm { .. } |
|
mir::StatementKind::InlineAsm { .. } |
|
||||||
|
@ -82,7 +82,7 @@ impl<'a, 'tcx> mir_visit::Visitor<'tcx> for StatCollector<'a, 'tcx> {
|
|||||||
self.record("Statement", statement);
|
self.record("Statement", statement);
|
||||||
self.record(match statement.kind {
|
self.record(match statement.kind {
|
||||||
StatementKind::Assign(..) => "StatementKind::Assign",
|
StatementKind::Assign(..) => "StatementKind::Assign",
|
||||||
StatementKind::ReadForMatch(..) => "StatementKind::ReadForMatch",
|
StatementKind::FakeRead(..) => "StatementKind::FakeRead",
|
||||||
StatementKind::EndRegion(..) => "StatementKind::EndRegion",
|
StatementKind::EndRegion(..) => "StatementKind::EndRegion",
|
||||||
StatementKind::Validate(..) => "StatementKind::Validate",
|
StatementKind::Validate(..) => "StatementKind::Validate",
|
||||||
StatementKind::SetDiscriminant { .. } => "StatementKind::SetDiscriminant",
|
StatementKind::SetDiscriminant { .. } => "StatementKind::SetDiscriminant",
|
||||||
|
@ -47,6 +47,7 @@ fn main() {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = const false;
|
// _1 = const false;
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// _3 = _1;
|
// _3 = _1;
|
||||||
@ -55,6 +56,7 @@ fn main() {
|
|||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// _4 = std::option::Option<std::boxed::Box<u32>>::None;
|
// _4 = std::option::Option<std::boxed::Box<u32>>::None;
|
||||||
// AscribeUserType(_4, o, Canonical { variables: [], value: std::option::Option<std::boxed::Box<u32>> });
|
// AscribeUserType(_4, o, Canonical { variables: [], value: std::option::Option<std::boxed::Box<u32>> });
|
||||||
|
// FakeRead(ForLet, _4);
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
// StorageLive(_6);
|
// StorageLive(_6);
|
||||||
// _6 = move _4;
|
// _6 = move _4;
|
||||||
|
@ -63,6 +63,7 @@ impl Drop for S {
|
|||||||
//
|
//
|
||||||
// bb4: {
|
// bb4: {
|
||||||
// StorageDead(_2);
|
// StorageDead(_2);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// _4 = move _1;
|
// _4 = move _1;
|
||||||
// _3 = const std::mem::drop(move _4) -> [return: bb5, unwind: bb7];
|
// _3 = const std::mem::drop(move _4) -> [return: bb5, unwind: bb7];
|
||||||
|
@ -29,8 +29,10 @@ fn main() {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = const 3i32;
|
// _1 = const 3i32;
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = &'10_1rs _1;
|
// _2 = &'10_1rs _1;
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// _0 = ();
|
// _0 = ();
|
||||||
// EndRegion('10_1rs);
|
// EndRegion('10_1rs);
|
||||||
// StorageDead(_2);
|
// StorageDead(_2);
|
||||||
|
@ -45,8 +45,10 @@ fn main() {
|
|||||||
// bb2: {
|
// bb2: {
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = const true;
|
// _2 = const true;
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// _3 = &'23_1rs _2;
|
// _3 = &'23_1rs _2;
|
||||||
|
// FakeRead(ForLet, _3);
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
// _5 = _2;
|
// _5 = _2;
|
||||||
// switchInt(move _5) -> [false: bb5, otherwise: bb4];
|
// switchInt(move _5) -> [false: bb5, otherwise: bb4];
|
||||||
@ -67,6 +69,7 @@ fn main() {
|
|||||||
// StorageDead(_5);
|
// StorageDead(_5);
|
||||||
// StorageLive(_7);
|
// StorageLive(_7);
|
||||||
// _7 = &'23_3rs _2;
|
// _7 = &'23_3rs _2;
|
||||||
|
// FakeRead(ForLet, _7);
|
||||||
// _1 = ();
|
// _1 = ();
|
||||||
// EndRegion('23_3rs);
|
// EndRegion('23_3rs);
|
||||||
// StorageDead(_7);
|
// StorageDead(_7);
|
||||||
|
@ -49,6 +49,7 @@ fn main() {
|
|||||||
// _1 = const true;
|
// _1 = const true;
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// _3 = &'26_1rs _1;
|
// _3 = &'26_1rs _1;
|
||||||
|
// FakeRead(ForLet, _3);
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
// _5 = _1;
|
// _5 = _1;
|
||||||
// switchInt(move _5) -> [false: bb5, otherwise: bb4];
|
// switchInt(move _5) -> [false: bb5, otherwise: bb4];
|
||||||
@ -69,6 +70,7 @@ fn main() {
|
|||||||
// StorageDead(_5);
|
// StorageDead(_5);
|
||||||
// StorageLive(_7);
|
// StorageLive(_7);
|
||||||
// _7 = &'26_3rs _1;
|
// _7 = &'26_3rs _1;
|
||||||
|
// FakeRead(ForLet, _7);
|
||||||
// _2 = ();
|
// _2 = ();
|
||||||
// EndRegion('26_3rs);
|
// EndRegion('26_3rs);
|
||||||
// StorageDead(_7);
|
// StorageDead(_7);
|
||||||
|
@ -45,10 +45,13 @@ fn foo(i: i32) {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = D::{{constructor}}(const 0i32,);
|
// _1 = D::{{constructor}}(const 0i32,);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = const 0i32;
|
// _2 = const 0i32;
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// _3 = &'26_2rs _2;
|
// _3 = &'26_2rs _2;
|
||||||
|
// FakeRead(ForLet, _3);
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
// _5 = (*_3);
|
// _5 = (*_3);
|
||||||
// _4 = const foo(move _5) -> [return: bb2, unwind: bb3];
|
// _4 = const foo(move _5) -> [return: bb2, unwind: bb3];
|
||||||
@ -60,6 +63,7 @@ fn foo(i: i32) {
|
|||||||
// StorageDead(_5);
|
// StorageDead(_5);
|
||||||
// StorageLive(_6);
|
// StorageLive(_6);
|
||||||
// _6 = &'26_4rs _2;
|
// _6 = &'26_4rs _2;
|
||||||
|
// FakeRead(ForLet, _6);
|
||||||
// _0 = ();
|
// _0 = ();
|
||||||
// EndRegion('26_4rs);
|
// EndRegion('26_4rs);
|
||||||
// StorageDead(_6);
|
// StorageDead(_6);
|
||||||
|
@ -38,6 +38,7 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = D::{{constructor}}(const 0i32,);
|
// _1 = D::{{constructor}}(const 0i32,);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// _4 = &'14s _1;
|
// _4 = &'14s _1;
|
||||||
|
@ -38,6 +38,7 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = D::{{constructor}}(const 0i32,);
|
// _1 = D::{{constructor}}(const 0i32,);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// _4 = &'19s _1;
|
// _4 = &'19s _1;
|
||||||
@ -73,6 +74,7 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = &'16_0rs (*(_1.0: &'19s D));
|
// _2 = &'16_0rs (*(_1.0: &'19s D));
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// _0 = ((*_2).0: i32);
|
// _0 = ((*_2).0: i32);
|
||||||
// EndRegion('16_0rs);
|
// EndRegion('16_0rs);
|
||||||
// StorageDead(_2);
|
// StorageDead(_2);
|
||||||
|
@ -37,6 +37,7 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = D::{{constructor}}(const 0i32,);
|
// _1 = D::{{constructor}}(const 0i32,);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// _3 = [closure@NodeId(22)] { d: move _1 };
|
// _3 = [closure@NodeId(22)] { d: move _1 };
|
||||||
// _2 = const foo(move _3) -> [return: bb2, unwind: bb4];
|
// _2 = const foo(move _3) -> [return: bb2, unwind: bb4];
|
||||||
@ -74,6 +75,7 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = &'16_0rs (_1.0: D);
|
// _2 = &'16_0rs (_1.0: D);
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// _0 = ((*_2).0: i32);
|
// _0 = ((*_2).0: i32);
|
||||||
// EndRegion('16_0rs);
|
// EndRegion('16_0rs);
|
||||||
// StorageDead(_2);
|
// StorageDead(_2);
|
||||||
|
@ -40,8 +40,10 @@ fn foo<F>(f: F) where F: FnOnce() -> i32 {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = D::{{constructor}}(const 0i32,);
|
// _1 = D::{{constructor}}(const 0i32,);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = &'21_1rs _1;
|
// _2 = &'21_1rs _1;
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// _4 = [closure@NodeId(22)] { r: _2 };
|
// _4 = [closure@NodeId(22)] { r: _2 };
|
||||||
// _3 = const foo(move _4) -> [return: bb2, unwind: bb3];
|
// _3 = const foo(move _4) -> [return: bb2, unwind: bb3];
|
||||||
|
@ -55,8 +55,10 @@ fn main() {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = const false;
|
// _1 = const false;
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// _2 = const 3i32;
|
// _2 = const 3i32;
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// falseUnwind -> [real: bb2, cleanup: bb1];
|
// falseUnwind -> [real: bb2, cleanup: bb1];
|
||||||
// }
|
// }
|
||||||
// bb1: {
|
// bb1: {
|
||||||
|
@ -83,6 +83,7 @@ fn query() -> bool { true }
|
|||||||
// StorageDead(_4);
|
// StorageDead(_4);
|
||||||
// _2 = S<'36_0rs> { r: move _3 };
|
// _2 = S<'36_0rs> { r: move _3 };
|
||||||
// StorageDead(_3);
|
// StorageDead(_3);
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// StorageLive(_6);
|
// StorageLive(_6);
|
||||||
// _6 = &'17s (_2.0: std::cell::Cell<std::option::Option<&'36_0rs S<'36_0rs>>>);
|
// _6 = &'17s (_2.0: std::cell::Cell<std::option::Option<&'36_0rs S<'36_0rs>>>);
|
||||||
// StorageLive(_7);
|
// StorageLive(_7);
|
||||||
|
@ -40,6 +40,7 @@ fn foo<T: Copy>(_t: T, q: &i32) -> i32 {
|
|||||||
// _5 = (move _6, move _7);
|
// _5 = (move _6, move _7);
|
||||||
// _8 = move (_5.0: &i32);
|
// _8 = move (_5.0: &i32);
|
||||||
// _9 = move (_5.1: &i32);
|
// _9 = move (_5.1: &i32);
|
||||||
|
// ...
|
||||||
// _0 = (*_8);
|
// _0 = (*_8);
|
||||||
// ...
|
// ...
|
||||||
// return;
|
// return;
|
||||||
|
@ -25,6 +25,7 @@ fn main() {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = const false;
|
// _1 = const false;
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// goto -> bb2;
|
// goto -> bb2;
|
||||||
// }
|
// }
|
||||||
// bb1: {
|
// bb1: {
|
||||||
|
@ -87,6 +87,7 @@ fn main() {
|
|||||||
// goto -> bb6;
|
// goto -> bb6;
|
||||||
// }
|
// }
|
||||||
// bb14: {
|
// bb14: {
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// StorageDead(_3);
|
// StorageDead(_3);
|
||||||
// StorageLive(_7);
|
// StorageLive(_7);
|
||||||
// _7 = &_2;
|
// _7 = &_2;
|
||||||
|
@ -42,6 +42,7 @@ fn main() {
|
|||||||
// bb5: { // The loop body (body_block)
|
// bb5: { // The loop body (body_block)
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
// _5 = const 1i32;
|
// _5 = const 1i32;
|
||||||
|
// FakeRead(ForLet, _5);
|
||||||
// StorageDead(_5);
|
// StorageDead(_5);
|
||||||
// goto -> bb4;
|
// goto -> bb4;
|
||||||
// }
|
// }
|
||||||
|
@ -66,15 +66,15 @@ fn main() {
|
|||||||
// goto -> bb13;
|
// goto -> bb13;
|
||||||
// }
|
// }
|
||||||
// bb3: { // binding3(empty) and arm3
|
// bb3: { // binding3(empty) and arm3
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb8, imaginary: bb4]; //pre_binding1
|
// falseEdges -> [real: bb8, imaginary: bb4]; //pre_binding1
|
||||||
// }
|
// }
|
||||||
// bb4: {
|
// bb4: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb12, imaginary: bb5]; //pre_binding2
|
// falseEdges -> [real: bb12, imaginary: bb5]; //pre_binding2
|
||||||
// }
|
// }
|
||||||
// bb5: {
|
// bb5: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb2, imaginary: bb6]; //pre_binding3
|
// falseEdges -> [real: bb2, imaginary: bb6]; //pre_binding3
|
||||||
// }
|
// }
|
||||||
// bb6: {
|
// bb6: {
|
||||||
@ -136,15 +136,15 @@ fn main() {
|
|||||||
// goto -> bb13;
|
// goto -> bb13;
|
||||||
// }
|
// }
|
||||||
// bb3: {
|
// bb3: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb8, imaginary: bb4]; //pre_binding1
|
// falseEdges -> [real: bb8, imaginary: bb4]; //pre_binding1
|
||||||
// }
|
// }
|
||||||
// bb4: {
|
// bb4: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb2, imaginary: bb5]; //pre_binding2
|
// falseEdges -> [real: bb2, imaginary: bb5]; //pre_binding2
|
||||||
// }
|
// }
|
||||||
// bb5: {
|
// bb5: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb12, imaginary: bb6]; //pre_binding3
|
// falseEdges -> [real: bb12, imaginary: bb6]; //pre_binding3
|
||||||
// }
|
// }
|
||||||
// bb6: {
|
// bb6: {
|
||||||
@ -202,19 +202,19 @@ fn main() {
|
|||||||
// resume;
|
// resume;
|
||||||
// }
|
// }
|
||||||
// bb2: {
|
// bb2: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb7, imaginary: bb3]; //pre_binding1
|
// falseEdges -> [real: bb7, imaginary: bb3]; //pre_binding1
|
||||||
// }
|
// }
|
||||||
// bb3: {
|
// bb3: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb11, imaginary: bb4]; //pre_binding2
|
// falseEdges -> [real: bb11, imaginary: bb4]; //pre_binding2
|
||||||
// }
|
// }
|
||||||
// bb4: {
|
// bb4: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb12, imaginary: bb5]; //pre_binding3
|
// falseEdges -> [real: bb12, imaginary: bb5]; //pre_binding3
|
||||||
// }
|
// }
|
||||||
// bb5: {
|
// bb5: {
|
||||||
// ReadForMatch(_4);
|
// FakeRead(ForMatch, _4);
|
||||||
// falseEdges -> [real: bb16, imaginary: bb6]; //pre_binding4
|
// falseEdges -> [real: bb16, imaginary: bb6]; //pre_binding4
|
||||||
// }
|
// }
|
||||||
// bb6: {
|
// bb6: {
|
||||||
|
@ -32,9 +32,9 @@ fn main() {
|
|||||||
|
|
||||||
// END RUST SOURCE
|
// END RUST SOURCE
|
||||||
// START rustc.main.nll.0.mir
|
// START rustc.main.nll.0.mir
|
||||||
// | '_#2r | U0 | {bb2[0..=3], bb3[0..=1]}
|
// | '_#2r | U0 | {bb2[0..=5], bb3[0..=1]}
|
||||||
// | '_#3r | U0 | {bb2[1..=3], bb3[0..=1]}
|
// | '_#3r | U0 | {bb2[1..=5], bb3[0..=1]}
|
||||||
// | '_#4r | U0 | {bb2[3], bb3[0..=1]}
|
// | '_#4r | U0 | {bb2[4..=5], bb3[0..=1]}
|
||||||
// END rustc.main.nll.0.mir
|
// END rustc.main.nll.0.mir
|
||||||
// START rustc.main.nll.0.mir
|
// START rustc.main.nll.0.mir
|
||||||
// let _6: &'_#4r usize;
|
// let _6: &'_#4r usize;
|
||||||
|
@ -23,6 +23,7 @@ fn main() {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = const 0i32;
|
// _1 = const 0i32;
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_3);
|
// StorageLive(_3);
|
||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
@ -30,11 +31,13 @@ fn main() {
|
|||||||
// _4 = std::option::Option<i32>::Some(move _5,);
|
// _4 = std::option::Option<i32>::Some(move _5,);
|
||||||
// StorageDead(_5);
|
// StorageDead(_5);
|
||||||
// _3 = &_4;
|
// _3 = &_4;
|
||||||
|
// FakeRead(ForLet, _3);
|
||||||
// _2 = ();
|
// _2 = ();
|
||||||
// StorageDead(_4);
|
// StorageDead(_4);
|
||||||
// StorageDead(_3);
|
// StorageDead(_3);
|
||||||
// StorageLive(_6);
|
// StorageLive(_6);
|
||||||
// _6 = const 1i32;
|
// _6 = const 1i32;
|
||||||
|
// FakeRead(ForLet, _6);
|
||||||
// _0 = ();
|
// _0 = ();
|
||||||
// StorageDead(_6);
|
// StorageDead(_6);
|
||||||
// StorageDead(_1);
|
// StorageDead(_1);
|
||||||
|
@ -67,6 +67,7 @@ fn main() {
|
|||||||
// Validate(Suspend(ReScope(Remainder { block: ItemLocalId(25), first_statement_index: 0 })), [(*_2): i32]);
|
// Validate(Suspend(ReScope(Remainder { block: ItemLocalId(25), first_statement_index: 0 })), [(*_2): i32]);
|
||||||
// _3 = &ReErased (*_2);
|
// _3 = &ReErased (*_2);
|
||||||
// Validate(Acquire, [(*_3): i32/ReScope(Remainder { block: ItemLocalId(25), first_statement_index: 0 }) (imm)]);
|
// Validate(Acquire, [(*_3): i32/ReScope(Remainder { block: ItemLocalId(25), first_statement_index: 0 }) (imm)]);
|
||||||
|
// FakeRead(ForLet, _3);
|
||||||
// _0 = (*_3);
|
// _0 = (*_3);
|
||||||
// EndRegion(ReScope(Remainder { block: ItemLocalId(25), first_statement_index: 0 }));
|
// EndRegion(ReScope(Remainder { block: ItemLocalId(25), first_statement_index: 0 }));
|
||||||
// StorageDead(_3);
|
// StorageDead(_3);
|
||||||
|
@ -28,6 +28,7 @@ fn main() {
|
|||||||
// Validate(Acquire, [_1: std::boxed::Box<[i32]>]);
|
// Validate(Acquire, [_1: std::boxed::Box<[i32]>]);
|
||||||
// StorageDead(_2);
|
// StorageDead(_2);
|
||||||
// StorageDead(_3);
|
// StorageDead(_3);
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// _0 = ();
|
// _0 = ();
|
||||||
// Validate(Release, [_1: std::boxed::Box<[i32]>]);
|
// Validate(Release, [_1: std::boxed::Box<[i32]>]);
|
||||||
// drop(_1) -> [return: bb2, unwind: bb3];
|
// drop(_1) -> [return: bb2, unwind: bb3];
|
||||||
|
@ -47,10 +47,12 @@ fn main() {
|
|||||||
// bb0: {
|
// bb0: {
|
||||||
// StorageLive(_1);
|
// StorageLive(_1);
|
||||||
// _1 = Test { x: const 0i32 };
|
// _1 = Test { x: const 0i32 };
|
||||||
|
// FakeRead(ForLet, _1);
|
||||||
// StorageLive(_2);
|
// StorageLive(_2);
|
||||||
// Validate(Suspend(ReScope(Remainder { block: ItemLocalId(20), first_statement_index: 3 })), [_1: Test]);
|
// Validate(Suspend(ReScope(Remainder { block: ItemLocalId(20), first_statement_index: 3 })), [_1: Test]);
|
||||||
// _2 = &ReErased _1;
|
// _2 = &ReErased _1;
|
||||||
// Validate(Acquire, [(*_2): Test/ReScope(Remainder { block: ItemLocalId(20), first_statement_index: 3 }) (imm)]);
|
// Validate(Acquire, [(*_2): Test/ReScope(Remainder { block: ItemLocalId(20), first_statement_index: 3 }) (imm)]);
|
||||||
|
// FakeRead(ForLet, _2);
|
||||||
// StorageLive(_4);
|
// StorageLive(_4);
|
||||||
// StorageLive(_5);
|
// StorageLive(_5);
|
||||||
// Validate(Suspend(ReScope(Node(ItemLocalId(18)))), [((*_2).0: i32): i32/ReScope(Remainder { block: ItemLocalId(20), first_statement_index: 3 }) (imm)]);
|
// Validate(Suspend(ReScope(Node(ItemLocalId(18)))), [((*_2).0: i32): i32/ReScope(Remainder { block: ItemLocalId(20), first_statement_index: 3 }) (imm)]);
|
||||||
|
25
src/test/ui/extern/extern-const.fixed
vendored
25
src/test/ui/extern/extern-const.fixed
vendored
@ -1,25 +0,0 @@
|
|||||||
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
|
|
||||||
// file at the top-level directory of this distribution and at
|
|
||||||
// http://rust-lang.org/COPYRIGHT.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
||||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
||||||
// option. This file may not be copied, modified, or distributed
|
|
||||||
// except according to those terms.
|
|
||||||
|
|
||||||
// run-rustfix
|
|
||||||
// compile-flags: -Z continue-parse-after-error
|
|
||||||
|
|
||||||
extern "C" {
|
|
||||||
static C: u8; //~ ERROR extern items cannot be `const`
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
// We suggest turning the (illegal) extern `const` into an extern `static`,
|
|
||||||
// but this also requires `unsafe` (a deny-by-default lint at comment time,
|
|
||||||
// future error; Issue #36247)
|
|
||||||
unsafe {
|
|
||||||
let _x = C;
|
|
||||||
}
|
|
||||||
}
|
|
2
src/test/ui/extern/extern-const.rs
vendored
2
src/test/ui/extern/extern-const.rs
vendored
@ -8,7 +8,7 @@
|
|||||||
// option. This file may not be copied, modified, or distributed
|
// option. This file may not be copied, modified, or distributed
|
||||||
// except according to those terms.
|
// except according to those terms.
|
||||||
|
|
||||||
// run-rustfix
|
// FIXME(#54388): re-enable rustfix later, when this test has consistent output across targets
|
||||||
// compile-flags: -Z continue-parse-after-error
|
// compile-flags: -Z continue-parse-after-error
|
||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
|
@ -10,6 +10,9 @@ LL | };
|
|||||||
error[E0597]: `a` does not live long enough
|
error[E0597]: `a` does not live long enough
|
||||||
--> $DIR/borrowing.rs:24:9
|
--> $DIR/borrowing.rs:24:9
|
||||||
|
|
|
|
||||||
|
LL | let _b = {
|
||||||
|
| -- borrow later stored here
|
||||||
|
LL | let a = 3;
|
||||||
LL | / || {
|
LL | / || {
|
||||||
LL | | yield &a
|
LL | | yield &a
|
||||||
LL | | //~^ ERROR: `a` does not live long enough
|
LL | | //~^ ERROR: `a` does not live long enough
|
||||||
@ -17,8 +20,6 @@ LL | | }
|
|||||||
| |_________^ borrowed value does not live long enough
|
| |_________^ borrowed value does not live long enough
|
||||||
LL | };
|
LL | };
|
||||||
| - `a` dropped here while still borrowed
|
| - `a` dropped here while still borrowed
|
||||||
LL | }
|
|
||||||
| - borrow later used here, when `_b` is dropped
|
|
||||||
|
|
||||||
error: aborting due to 2 previous errors
|
error: aborting due to 2 previous errors
|
||||||
|
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
error[E0597]: `i` does not live long enough
|
error[E0597]: `i` does not live long enough
|
||||||
--> $DIR/regions-steal-closure.rs:24:28
|
--> $DIR/regions-steal-closure.rs:24:28
|
||||||
|
|
|
|
||||||
|
LL | let mut cl_box = {
|
||||||
|
| ---------- borrow later stored here
|
||||||
|
LL | let mut i = 3;
|
||||||
LL | box_it(Box::new(|| i += 1)) //~ ERROR `i` does not live long enough
|
LL | box_it(Box::new(|| i += 1)) //~ ERROR `i` does not live long enough
|
||||||
| -- ^ borrowed value does not live long enough
|
| -- ^ borrowed value does not live long enough
|
||||||
| |
|
| |
|
||||||
| value captured here
|
| value captured here
|
||||||
LL | };
|
LL | };
|
||||||
| - `i` dropped here while still borrowed
|
| - `i` dropped here while still borrowed
|
||||||
LL | cl_box.cl.call_mut(());
|
|
||||||
| --------- borrow later used here
|
|
||||||
|
|
||||||
error: aborting due to previous error
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
@ -1,24 +1,24 @@
|
|||||||
error[E0597]: `a` does not live long enough
|
error[E0597]: `a` does not live long enough
|
||||||
--> $DIR/range-2.rs:17:9
|
--> $DIR/range-2.rs:17:9
|
||||||
|
|
|
|
||||||
|
LL | let r = {
|
||||||
|
| - borrow later stored here
|
||||||
|
...
|
||||||
LL | &a..&b
|
LL | &a..&b
|
||||||
| ^^ borrowed value does not live long enough
|
| ^^ borrowed value does not live long enough
|
||||||
LL | };
|
LL | };
|
||||||
| - `a` dropped here while still borrowed
|
| - `a` dropped here while still borrowed
|
||||||
...
|
|
||||||
LL | r.use_ref();
|
|
||||||
| - borrow later used here
|
|
||||||
|
|
||||||
error[E0597]: `b` does not live long enough
|
error[E0597]: `b` does not live long enough
|
||||||
--> $DIR/range-2.rs:17:13
|
--> $DIR/range-2.rs:17:13
|
||||||
|
|
|
|
||||||
|
LL | let r = {
|
||||||
|
| - borrow later stored here
|
||||||
|
...
|
||||||
LL | &a..&b
|
LL | &a..&b
|
||||||
| ^^ borrowed value does not live long enough
|
| ^^ borrowed value does not live long enough
|
||||||
LL | };
|
LL | };
|
||||||
| - `b` dropped here while still borrowed
|
| - `b` dropped here while still borrowed
|
||||||
...
|
|
||||||
LL | r.use_ref();
|
|
||||||
| - borrow later used here
|
|
||||||
|
|
||||||
error: aborting due to 2 previous errors
|
error: aborting due to 2 previous errors
|
||||||
|
|
||||||
|
@ -1,18 +1,21 @@
|
|||||||
error[E0597]: `x` does not live long enough
|
error[E0597]: `x` does not live long enough
|
||||||
--> $DIR/send-is-not-static-ensures-scoping.rs:26:17
|
--> $DIR/send-is-not-static-ensures-scoping.rs:26:17
|
||||||
|
|
|
|
||||||
|
LL | let bad = {
|
||||||
|
| --- borrow later stored here
|
||||||
|
LL | let x = 1;
|
||||||
LL | let y = &x;
|
LL | let y = &x;
|
||||||
| ^^ borrowed value does not live long enough
|
| ^^ borrowed value does not live long enough
|
||||||
...
|
...
|
||||||
LL | };
|
LL | };
|
||||||
| - `x` dropped here while still borrowed
|
| - `x` dropped here while still borrowed
|
||||||
LL |
|
|
||||||
LL | bad.join();
|
|
||||||
| --- borrow later used here
|
|
||||||
|
|
||||||
error[E0597]: `y` does not live long enough
|
error[E0597]: `y` does not live long enough
|
||||||
--> $DIR/send-is-not-static-ensures-scoping.rs:30:22
|
--> $DIR/send-is-not-static-ensures-scoping.rs:30:22
|
||||||
|
|
|
|
||||||
|
LL | let bad = {
|
||||||
|
| --- borrow later stored here
|
||||||
|
...
|
||||||
LL | scoped(|| {
|
LL | scoped(|| {
|
||||||
| -- value captured here
|
| -- value captured here
|
||||||
LL | let _z = y;
|
LL | let _z = y;
|
||||||
@ -20,9 +23,6 @@ LL | let _z = y;
|
|||||||
...
|
...
|
||||||
LL | };
|
LL | };
|
||||||
| - `y` dropped here while still borrowed
|
| - `y` dropped here while still borrowed
|
||||||
LL |
|
|
||||||
LL | bad.join();
|
|
||||||
| --- borrow later used here
|
|
||||||
|
|
||||||
error: aborting due to 2 previous errors
|
error: aborting due to 2 previous errors
|
||||||
|
|
||||||
|
@ -1,24 +1,24 @@
|
|||||||
error[E0597]: `x` does not live long enough
|
error[E0597]: `x` does not live long enough
|
||||||
--> $DIR/send-is-not-static-std-sync-2.rs:21:20
|
--> $DIR/send-is-not-static-std-sync-2.rs:21:20
|
||||||
|
|
|
|
||||||
|
LL | let lock = {
|
||||||
|
| ---- borrow later stored here
|
||||||
|
LL | let x = 1;
|
||||||
LL | Mutex::new(&x)
|
LL | Mutex::new(&x)
|
||||||
| ^^ borrowed value does not live long enough
|
| ^^ borrowed value does not live long enough
|
||||||
LL | };
|
LL | };
|
||||||
| - `x` dropped here while still borrowed
|
| - `x` dropped here while still borrowed
|
||||||
...
|
|
||||||
LL | let _dangling = *lock.lock().unwrap();
|
|
||||||
| ---- borrow later used here
|
|
||||||
|
|
||||||
error[E0597]: `x` does not live long enough
|
error[E0597]: `x` does not live long enough
|
||||||
--> $DIR/send-is-not-static-std-sync-2.rs:31:21
|
--> $DIR/send-is-not-static-std-sync-2.rs:31:21
|
||||||
|
|
|
|
||||||
|
LL | let lock = {
|
||||||
|
| ---- borrow later stored here
|
||||||
|
LL | let x = 1;
|
||||||
LL | RwLock::new(&x)
|
LL | RwLock::new(&x)
|
||||||
| ^^ borrowed value does not live long enough
|
| ^^ borrowed value does not live long enough
|
||||||
LL | };
|
LL | };
|
||||||
| - `x` dropped here while still borrowed
|
| - `x` dropped here while still borrowed
|
||||||
LL | //~^^ ERROR `x` does not live long enough
|
|
||||||
LL | let _dangling = *lock.read().unwrap();
|
|
||||||
| ---- borrow later used here
|
|
||||||
|
|
||||||
error[E0597]: `x` does not live long enough
|
error[E0597]: `x` does not live long enough
|
||||||
--> $DIR/send-is-not-static-std-sync-2.rs:41:25
|
--> $DIR/send-is-not-static-std-sync-2.rs:41:25
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
error[E0597]: `pointer` does not live long enough
|
error[E0597]: `pointer` does not live long enough
|
||||||
--> $DIR/wf-method-late-bound-regions.rs:30:18
|
--> $DIR/wf-method-late-bound-regions.rs:30:18
|
||||||
|
|
|
|
||||||
|
LL | let dangling = {
|
||||||
|
| -------- borrow later stored here
|
||||||
|
LL | let pointer = Box::new(42);
|
||||||
LL | f2.xmute(&pointer)
|
LL | f2.xmute(&pointer)
|
||||||
| ^^^^^^^^ borrowed value does not live long enough
|
| ^^^^^^^^ borrowed value does not live long enough
|
||||||
LL | };
|
LL | };
|
||||||
| - `pointer` dropped here while still borrowed
|
| - `pointer` dropped here while still borrowed
|
||||||
LL | //~^^ ERROR `pointer` does not live long enough
|
|
||||||
LL | println!("{}", dangling);
|
|
||||||
| -------- borrow later used here
|
|
||||||
|
|
||||||
error: aborting due to previous error
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
@ -1,14 +1,15 @@
|
|||||||
error[E0597]: `x` does not live long enough
|
error[E0597]: `x` does not live long enough
|
||||||
--> $DIR/unboxed-closure-region.rs:18:12
|
--> $DIR/unboxed-closure-region.rs:18:12
|
||||||
|
|
|
|
||||||
|
LL | let _f = {
|
||||||
|
| -- borrow later stored here
|
||||||
|
LL | let x = 0;
|
||||||
LL | || x //~ ERROR `x` does not live long enough
|
LL | || x //~ ERROR `x` does not live long enough
|
||||||
| -- ^ borrowed value does not live long enough
|
| -- ^ borrowed value does not live long enough
|
||||||
| |
|
| |
|
||||||
| value captured here
|
| value captured here
|
||||||
LL | };
|
LL | };
|
||||||
| - `x` dropped here while still borrowed
|
| - `x` dropped here while still borrowed
|
||||||
LL | _f;
|
|
||||||
| -- borrow later used here
|
|
||||||
|
|
||||||
error: aborting due to previous error
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user