Skip to content

Commit d9ea132

Browse files
committedApr 22, 2019
Explain error when yielding a reference to a local variable
1 parent c21fbfe commit d9ea132

File tree

6 files changed

+107
-26
lines changed

6 files changed

+107
-26
lines changed
 

‎src/librustc_mir/borrow_check/error_reporting.rs

+24-10
Original file line numberDiff line numberDiff line change
@@ -826,18 +826,21 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
826826

827827
let borrow_span = borrow_spans.var_or_use();
828828
if let BorrowExplanation::MustBeValidFor {
829-
category: ConstraintCategory::Return,
829+
category,
830830
span,
831831
ref opt_place_desc,
832832
from_closure: false,
833833
..
834834
} = explanation {
835-
return self.report_cannot_return_reference_to_local(
835+
if let Some(diag) = self.try_report_cannot_return_reference_to_local(
836836
borrow,
837837
borrow_span,
838838
span,
839+
category,
839840
opt_place_desc.as_ref(),
840-
);
841+
) {
842+
return diag;
843+
}
841844
}
842845

843846
let mut err = self.infcx.tcx.path_does_not_live_long_enough(
@@ -1015,17 +1018,20 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
10151018
);
10161019

10171020
if let BorrowExplanation::MustBeValidFor {
1018-
category: ConstraintCategory::Return,
1021+
category,
10191022
span,
10201023
from_closure: false,
10211024
..
10221025
} = explanation {
1023-
return self.report_cannot_return_reference_to_local(
1026+
if let Some(diag) = self.try_report_cannot_return_reference_to_local(
10241027
borrow,
10251028
proper_span,
10261029
span,
1030+
category,
10271031
None,
1028-
);
1032+
) {
1033+
return diag;
1034+
}
10291035
}
10301036

10311037
let tcx = self.infcx.tcx;
@@ -1064,15 +1070,22 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
10641070
err
10651071
}
10661072

1067-
fn report_cannot_return_reference_to_local(
1073+
fn try_report_cannot_return_reference_to_local(
10681074
&self,
10691075
borrow: &BorrowData<'tcx>,
10701076
borrow_span: Span,
10711077
return_span: Span,
1078+
category: ConstraintCategory,
10721079
opt_place_desc: Option<&String>,
1073-
) -> DiagnosticBuilder<'cx> {
1080+
) -> Option<DiagnosticBuilder<'cx>> {
10741081
let tcx = self.infcx.tcx;
10751082

1083+
let return_kind = match category {
1084+
ConstraintCategory::Return => "return",
1085+
ConstraintCategory::Yield => "yield",
1086+
_ => return None,
1087+
};
1088+
10761089
// FIXME use a better heuristic than Spans
10771090
let reference_desc = if return_span == self.mir.source_info(borrow.reserve_location).span {
10781091
"reference to"
@@ -1110,7 +1123,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
11101123
let local = if let Place::Base(PlaceBase::Local(local)) = *root_place {
11111124
local
11121125
} else {
1113-
bug!("report_cannot_return_reference_to_local: not a local")
1126+
bug!("try_report_cannot_return_reference_to_local: not a local")
11141127
};
11151128
match self.mir.local_kind(local) {
11161129
LocalKind::ReturnPointer | LocalKind::Temp => {
@@ -1131,6 +1144,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
11311144

11321145
let mut err = tcx.cannot_return_reference_to_local(
11331146
return_span,
1147+
return_kind,
11341148
reference_desc,
11351149
&place_desc,
11361150
Origin::Mir,
@@ -1140,7 +1154,7 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
11401154
err.span_label(borrow_span, note);
11411155
}
11421156

1143-
err
1157+
Some(err)
11441158
}
11451159

11461160
fn report_escaping_closure_capture(

‎src/librustc_mir/borrow_check/nll/explain_borrow/mod.rs

+4
Original file line numberDiff line numberDiff line change
@@ -313,9 +313,13 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
313313
opt_place_desc,
314314
}
315315
} else {
316+
debug!("explain_why_borrow_contains_point: \
317+
Could not generate a region name");
316318
BorrowExplanation::Unexplained
317319
}
318320
} else {
321+
debug!("explain_why_borrow_contains_point: \
322+
Could not generate an error region vid");
319323
BorrowExplanation::Unexplained
320324
}
321325
}

‎src/librustc_mir/borrow_check/nll/region_infer/error_reporting/region_name.rs

+66-5
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ crate enum RegionNameSource {
3232
MatchedAdtAndSegment(Span),
3333
AnonRegionFromUpvar(Span, String),
3434
AnonRegionFromOutput(Span, String, String),
35+
AnonRegionFromYieldTy(Span, String),
3536
}
3637

3738
impl RegionName {
@@ -46,7 +47,8 @@ impl RegionName {
4647
RegionNameSource::MatchedHirTy(..) |
4748
RegionNameSource::MatchedAdtAndSegment(..) |
4849
RegionNameSource::AnonRegionFromUpvar(..) |
49-
RegionNameSource::AnonRegionFromOutput(..) => false,
50+
RegionNameSource::AnonRegionFromOutput(..) |
51+
RegionNameSource::AnonRegionFromYieldTy(..) => false,
5052
}
5153
}
5254

@@ -103,6 +105,12 @@ impl RegionName {
103105
format!("return type{} is {}", mir_description, type_name),
104106
);
105107
},
108+
RegionNameSource::AnonRegionFromYieldTy(span, type_name) => {
109+
diag.span_label(
110+
*span,
111+
format!("yield type is {}", type_name),
112+
);
113+
}
106114
RegionNameSource::Static => {},
107115
}
108116
}
@@ -167,6 +175,11 @@ impl<'tcx> RegionInferenceContext<'tcx> {
167175
self.give_name_if_anonymous_region_appears_in_output(
168176
infcx, mir, mir_def_id, fr, counter,
169177
)
178+
})
179+
.or_else(|| {
180+
self.give_name_if_anonymous_region_appears_in_yield_ty(
181+
infcx, mir, mir_def_id, fr, counter,
182+
)
170183
});
171184

172185
debug!("give_region_a_name: gave name {:?}", value);
@@ -673,10 +686,7 @@ impl<'tcx> RegionInferenceContext<'tcx> {
673686
"give_name_if_anonymous_region_appears_in_output: return_ty = {:?}",
674687
return_ty
675688
);
676-
if !infcx
677-
.tcx
678-
.any_free_region_meets(&return_ty, |r| r.to_region_vid() == fr)
679-
{
689+
if !tcx.any_free_region_meets(&return_ty, |r| r.to_region_vid() == fr) {
680690
return None;
681691
}
682692

@@ -721,6 +731,57 @@ impl<'tcx> RegionInferenceContext<'tcx> {
721731
})
722732
}
723733

734+
fn give_name_if_anonymous_region_appears_in_yield_ty(
735+
&self,
736+
infcx: &InferCtxt<'_, '_, 'tcx>,
737+
mir: &Mir<'tcx>,
738+
mir_def_id: DefId,
739+
fr: RegionVid,
740+
counter: &mut usize,
741+
) -> Option<RegionName> {
742+
// Note: generators from `async fn` yield `()`, so we don't have to
743+
// worry about them here.
744+
let yield_ty = self.universal_regions.yield_ty?;
745+
debug!(
746+
"give_name_if_anonymous_region_appears_in_yield_ty: yield_ty = {:?}",
747+
yield_ty,
748+
);
749+
750+
let tcx = infcx.tcx;
751+
752+
if !tcx.any_free_region_meets(&yield_ty, |r| r.to_region_vid() == fr) {
753+
return None;
754+
}
755+
756+
let mut highlight = RegionHighlightMode::default();
757+
highlight.highlighting_region_vid(fr, *counter);
758+
let type_name = infcx.extract_type_name(&yield_ty, Some(highlight));
759+
760+
let mir_node_id = tcx.hir().as_local_node_id(mir_def_id).expect("non-local mir");
761+
762+
let yield_span = match tcx.hir().get(mir_node_id) {
763+
hir::Node::Expr(hir::Expr {
764+
node: hir::ExprKind::Closure(_, _, _, span, _),
765+
..
766+
}) => (
767+
tcx.sess.source_map().end_point(*span)
768+
),
769+
_ => mir.span,
770+
};
771+
772+
debug!(
773+
"give_name_if_anonymous_region_appears_in_yield_ty: \
774+
type_name = {:?}, yield_span = {:?}",
775+
yield_span,
776+
type_name,
777+
);
778+
779+
Some(RegionName {
780+
name: self.synthesize_region_name(counter),
781+
source: RegionNameSource::AnonRegionFromYieldTy(yield_span, type_name),
782+
})
783+
}
784+
724785
/// Creates a synthetic region named `'1`, incrementing the
725786
/// counter.
726787
fn synthesize_region_name(&self, counter: &mut usize) -> InternedString {

‎src/librustc_mir/util/borrowck_errors.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -650,6 +650,7 @@ pub trait BorrowckErrors<'cx>: Sized + Copy {
650650
fn cannot_return_reference_to_local(
651651
self,
652652
span: Span,
653+
return_kind: &str,
653654
reference_desc: &str,
654655
path_desc: &str,
655656
o: Origin,
@@ -658,15 +659,16 @@ pub trait BorrowckErrors<'cx>: Sized + Copy {
658659
self,
659660
span,
660661
E0515,
661-
"cannot return {REFERENCE} {LOCAL}{OGN}",
662+
"cannot {RETURN} {REFERENCE} {LOCAL}{OGN}",
663+
RETURN=return_kind,
662664
REFERENCE=reference_desc,
663665
LOCAL=path_desc,
664666
OGN = o
665667
);
666668

667669
err.span_label(
668670
span,
669-
format!("returns a {} data owned by the current function", reference_desc),
671+
format!("{}s a {} data owned by the current function", return_kind, reference_desc),
670672
);
671673

672674
self.cancel_if_wrong_origin(err, o)

‎src/test/ui/nll/issue-55850.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ where
2525
fn bug<'a>() -> impl Iterator<Item = &'a str> {
2626
GenIter(move || {
2727
let mut s = String::new();
28-
yield &s[..] //~ ERROR `s` does not live long enough [E0597]
28+
yield &s[..] //~ ERROR cannot yield value referencing local variable `s` [E0515]
2929
//~| ERROR borrow may still be in use when generator yields
3030
})
3131
}

‎src/test/ui/nll/issue-55850.stderr

+8-8
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
1-
error[E0597]: `s` does not live long enough
2-
--> $DIR/issue-55850.rs:28:16
1+
error[E0515]: cannot yield value referencing local variable `s`
2+
--> $DIR/issue-55850.rs:28:9
33
|
44
LL | yield &s[..]
5-
| ^ borrowed value does not live long enough
6-
LL |
7-
LL | })
8-
| - `s` dropped here while still borrowed
5+
| ^^^^^^^-^^^^
6+
| | |
7+
| | `s` is borrowed here
8+
| yields a value referencing data owned by the current function
99

1010
error[E0626]: borrow may still be in use when generator yields
1111
--> $DIR/issue-55850.rs:28:16
@@ -15,5 +15,5 @@ LL | yield &s[..]
1515

1616
error: aborting due to 2 previous errors
1717

18-
Some errors have detailed explanations: E0597, E0626.
19-
For more information about an error, try `rustc --explain E0597`.
18+
Some errors have detailed explanations: E0515, E0626.
19+
For more information about an error, try `rustc --explain E0515`.

0 commit comments

Comments
 (0)
Please sign in to comment.