Merge remote-tracking branch 'origin/master' into gen
This commit is contained in:
commit
06ce77cb39
@ -2025,5 +2025,6 @@ register_diagnostics! {
|
|||||||
E0490, // a value of type `..` is borrowed for too long
|
E0490, // a value of type `..` is borrowed for too long
|
||||||
E0495, // cannot infer an appropriate lifetime due to conflicting requirements
|
E0495, // cannot infer an appropriate lifetime due to conflicting requirements
|
||||||
E0566, // conflicting representation hints
|
E0566, // conflicting representation hints
|
||||||
|
E0623, // lifetime mismatch where both parameters are anonymous regions
|
||||||
E0625, // generators cannot have explicit arguments
|
E0625, // generators cannot have explicit arguments
|
||||||
}
|
}
|
||||||
|
200
src/librustc/infer/error_reporting/anon_anon_conflict.rs
Normal file
200
src/librustc/infer/error_reporting/anon_anon_conflict.rs
Normal file
@ -0,0 +1,200 @@
|
|||||||
|
// Copyright 2012-2013 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.
|
||||||
|
|
||||||
|
//! Error Reporting for Anonymous Region Lifetime Errors
|
||||||
|
//! where both the regions are anonymous.
|
||||||
|
use hir;
|
||||||
|
use infer::InferCtxt;
|
||||||
|
use ty::{self, Region};
|
||||||
|
use infer::region_inference::RegionResolutionError::*;
|
||||||
|
use infer::region_inference::RegionResolutionError;
|
||||||
|
use hir::map as hir_map;
|
||||||
|
use middle::resolve_lifetime as rl;
|
||||||
|
use hir::intravisit::{self, Visitor, NestedVisitorMap};
|
||||||
|
|
||||||
|
impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
||||||
|
// This method prints the error message for lifetime errors when both the concerned regions
|
||||||
|
// are anonymous.
|
||||||
|
// Consider a case where we have
|
||||||
|
// fn foo(x: &mut Vec<&u8>, y: &u8)
|
||||||
|
// { x.push(y); }.
|
||||||
|
// The example gives
|
||||||
|
// fn foo(x: &mut Vec<&u8>, y: &u8) {
|
||||||
|
// --- --- these references must have the same lifetime
|
||||||
|
// x.push(y);
|
||||||
|
// ^ data from `y` flows into `x` here
|
||||||
|
// It will later be extended to trait objects and structs.
|
||||||
|
pub fn try_report_anon_anon_conflict(&self, error: &RegionResolutionError<'tcx>) -> bool {
|
||||||
|
|
||||||
|
let (span, sub, sup) = match *error {
|
||||||
|
ConcreteFailure(ref origin, sub, sup) => (origin.span(), sub, sup),
|
||||||
|
_ => return false, // inapplicable
|
||||||
|
};
|
||||||
|
|
||||||
|
// Determine whether the sub and sup consist of both anonymous (elided) regions.
|
||||||
|
let (ty1, ty2) = if self.is_suitable_anonymous_region(sup).is_some() &&
|
||||||
|
self.is_suitable_anonymous_region(sub).is_some() {
|
||||||
|
if let (Some(anon_reg1), Some(anon_reg2)) =
|
||||||
|
(self.is_suitable_anonymous_region(sup), self.is_suitable_anonymous_region(sub)) {
|
||||||
|
let ((_, br1), (_, br2)) = (anon_reg1, anon_reg2);
|
||||||
|
if self.find_anon_type(sup, &br1).is_some() &&
|
||||||
|
self.find_anon_type(sub, &br2).is_some() {
|
||||||
|
(self.find_anon_type(sup, &br1).unwrap(),
|
||||||
|
self.find_anon_type(sub, &br2).unwrap())
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return false; // inapplicable
|
||||||
|
};
|
||||||
|
|
||||||
|
if let (Some(sup_arg), Some(sub_arg)) =
|
||||||
|
(self.find_arg_with_anonymous_region(sup, sup),
|
||||||
|
self.find_arg_with_anonymous_region(sub, sub)) {
|
||||||
|
let ((anon_arg1, _, _, _), (anon_arg2, _, _, _)) = (sup_arg, sub_arg);
|
||||||
|
|
||||||
|
let span_label_var1 = if let Some(simple_name) = anon_arg1.pat.simple_name() {
|
||||||
|
format!(" from `{}` ", simple_name)
|
||||||
|
} else {
|
||||||
|
format!(" ")
|
||||||
|
};
|
||||||
|
|
||||||
|
let span_label_var2 = if let Some(simple_name) = anon_arg2.pat.simple_name() {
|
||||||
|
format!(" into `{}` ", simple_name)
|
||||||
|
} else {
|
||||||
|
format!(" ")
|
||||||
|
};
|
||||||
|
|
||||||
|
struct_span_err!(self.tcx.sess, span, E0623, "lifetime mismatch")
|
||||||
|
.span_label(ty1.span,
|
||||||
|
format!("these references must have the same lifetime"))
|
||||||
|
.span_label(ty2.span, format!(""))
|
||||||
|
.span_label(span,
|
||||||
|
format!("data{}flows{}here", span_label_var1, span_label_var2))
|
||||||
|
.emit();
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This function calls the `visit_ty` method for the parameters
|
||||||
|
/// corresponding to the anonymous regions. The `nested_visitor.found_type`
|
||||||
|
/// contains the anonymous type.
|
||||||
|
///
|
||||||
|
/// # Arguments
|
||||||
|
///
|
||||||
|
/// region - the anonymous region corresponding to the anon_anon conflict
|
||||||
|
/// br - the bound region corresponding to the above region which is of type `BrAnon(_)`
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
/// ```
|
||||||
|
/// fn foo(x: &mut Vec<&u8>, y: &u8)
|
||||||
|
/// { x.push(y); }
|
||||||
|
/// ```
|
||||||
|
/// The function returns the nested type corresponding to the anonymous region
|
||||||
|
/// for e.g. `&u8` and Vec<`&u8`.
|
||||||
|
fn find_anon_type(&self, region: Region<'tcx>, br: &ty::BoundRegion) -> Option<&hir::Ty> {
|
||||||
|
if let Some(anon_reg) = self.is_suitable_anonymous_region(region) {
|
||||||
|
let (def_id, _) = anon_reg;
|
||||||
|
if let Some(node_id) = self.tcx.hir.as_local_node_id(def_id) {
|
||||||
|
let ret_ty = self.tcx.type_of(def_id);
|
||||||
|
if let ty::TyFnDef(_, _) = ret_ty.sty {
|
||||||
|
if let hir_map::NodeItem(it) = self.tcx.hir.get(node_id) {
|
||||||
|
if let hir::ItemFn(ref fndecl, _, _, _, _, _) = it.node {
|
||||||
|
return fndecl
|
||||||
|
.inputs
|
||||||
|
.iter()
|
||||||
|
.filter_map(|arg| {
|
||||||
|
let mut nested_visitor = FindNestedTypeVisitor {
|
||||||
|
infcx: &self,
|
||||||
|
hir_map: &self.tcx.hir,
|
||||||
|
bound_region: *br,
|
||||||
|
found_type: None,
|
||||||
|
};
|
||||||
|
nested_visitor.visit_ty(&**arg);
|
||||||
|
if nested_visitor.found_type.is_some() {
|
||||||
|
nested_visitor.found_type
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The FindNestedTypeVisitor captures the corresponding `hir::Ty` of the
|
||||||
|
// anonymous region. The example above would lead to a conflict between
|
||||||
|
// the two anonymous lifetimes for &u8 in x and y respectively. This visitor
|
||||||
|
// would be invoked twice, once for each lifetime, and would
|
||||||
|
// walk the types like &mut Vec<&u8> and &u8 looking for the HIR
|
||||||
|
// where that lifetime appears. This allows us to highlight the
|
||||||
|
// specific part of the type in the error message.
|
||||||
|
struct FindNestedTypeVisitor<'a, 'gcx: 'a + 'tcx, 'tcx: 'a> {
|
||||||
|
infcx: &'a InferCtxt<'a, 'gcx, 'tcx>,
|
||||||
|
hir_map: &'a hir::map::Map<'gcx>,
|
||||||
|
// The bound_region corresponding to the Refree(freeregion)
|
||||||
|
// associated with the anonymous region we are looking for.
|
||||||
|
bound_region: ty::BoundRegion,
|
||||||
|
// The type where the anonymous lifetime appears
|
||||||
|
// for e.g. Vec<`&u8`> and <`&u8`>
|
||||||
|
found_type: Option<&'gcx hir::Ty>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, 'gcx, 'tcx> Visitor<'gcx> for FindNestedTypeVisitor<'a, 'gcx, 'tcx> {
|
||||||
|
fn nested_visit_map<'this>(&'this mut self) -> NestedVisitorMap<'this, 'gcx> {
|
||||||
|
NestedVisitorMap::OnlyBodies(&self.hir_map)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_ty(&mut self, arg: &'gcx hir::Ty) {
|
||||||
|
// Find the index of the anonymous region that was part of the
|
||||||
|
// error. We will then search the function parameters for a bound
|
||||||
|
// region at the right depth with the same index.
|
||||||
|
let br_index = match self.bound_region {
|
||||||
|
ty::BrAnon(index) => index,
|
||||||
|
_ => return,
|
||||||
|
};
|
||||||
|
|
||||||
|
match arg.node {
|
||||||
|
hir::TyRptr(ref lifetime, _) => {
|
||||||
|
match self.infcx.tcx.named_region_map.defs.get(&lifetime.id) {
|
||||||
|
// the lifetime of the TyRptr
|
||||||
|
Some(&rl::Region::LateBoundAnon(debuijn_index, anon_index)) => {
|
||||||
|
if debuijn_index.depth == 1 && anon_index == br_index {
|
||||||
|
self.found_type = Some(arg);
|
||||||
|
return; // we can stop visiting now
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(&rl::Region::Static) |
|
||||||
|
Some(&rl::Region::EarlyBound(_, _)) |
|
||||||
|
Some(&rl::Region::LateBound(_, _)) |
|
||||||
|
Some(&rl::Region::Free(_, _)) |
|
||||||
|
None => {
|
||||||
|
debug!("no arg found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
// walk the embedded contents: e.g., if we are visiting `Vec<&Foo>`,
|
||||||
|
// go on to visit `&Foo`
|
||||||
|
intravisit::walk_ty(self, arg);
|
||||||
|
}
|
||||||
|
}
|
@ -76,7 +76,9 @@ use errors::{DiagnosticBuilder, DiagnosticStyledString};
|
|||||||
mod note;
|
mod note;
|
||||||
|
|
||||||
mod need_type_info;
|
mod need_type_info;
|
||||||
|
mod util;
|
||||||
mod named_anon_conflict;
|
mod named_anon_conflict;
|
||||||
|
mod anon_anon_conflict;
|
||||||
|
|
||||||
impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
|
impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
|
||||||
pub fn note_and_explain_region(self,
|
pub fn note_and_explain_region(self,
|
||||||
@ -270,29 +272,34 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
|||||||
for error in errors {
|
for error in errors {
|
||||||
debug!("report_region_errors: error = {:?}", error);
|
debug!("report_region_errors: error = {:?}", error);
|
||||||
|
|
||||||
if !self.try_report_named_anon_conflict(&error) {
|
if !self.try_report_named_anon_conflict(&error) &&
|
||||||
match error.clone() {
|
!self.try_report_anon_anon_conflict(&error) {
|
||||||
// These errors could indicate all manner of different
|
|
||||||
// problems with many different solutions. Rather
|
match error.clone() {
|
||||||
// than generate a "one size fits all" error, what we
|
// These errors could indicate all manner of different
|
||||||
// attempt to do is go through a number of specific
|
// problems with many different solutions. Rather
|
||||||
// scenarios and try to find the best way to present
|
// than generate a "one size fits all" error, what we
|
||||||
// the error. If all of these fails, we fall back to a rather
|
// attempt to do is go through a number of specific
|
||||||
// general bit of code that displays the error information
|
// scenarios and try to find the best way to present
|
||||||
ConcreteFailure(origin, sub, sup) => {
|
// the error. If all of these fails, we fall back to a rather
|
||||||
self.report_concrete_failure(origin, sub, sup).emit();
|
// general bit of code that displays the error information
|
||||||
}
|
ConcreteFailure(origin, sub, sup) => {
|
||||||
GenericBoundFailure(kind, param_ty, sub) => {
|
|
||||||
self.report_generic_bound_failure(kind, param_ty, sub);
|
self.report_concrete_failure(origin, sub, sup).emit();
|
||||||
}
|
}
|
||||||
SubSupConflict(var_origin,
|
|
||||||
sub_origin, sub_r,
|
GenericBoundFailure(kind, param_ty, sub) => {
|
||||||
sup_origin, sup_r) => {
|
self.report_generic_bound_failure(kind, param_ty, sub);
|
||||||
|
}
|
||||||
|
|
||||||
|
SubSupConflict(var_origin, sub_origin, sub_r, sup_origin, sup_r) => {
|
||||||
self.report_sub_sup_conflict(var_origin,
|
self.report_sub_sup_conflict(var_origin,
|
||||||
sub_origin, sub_r,
|
sub_origin,
|
||||||
sup_origin, sup_r);
|
sub_r,
|
||||||
}
|
sup_origin,
|
||||||
}
|
sup_r);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,69 +8,14 @@
|
|||||||
// 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.
|
||||||
|
|
||||||
//! Error Reporting for Anonymous Region Lifetime Errors.
|
//! Error Reporting for Anonymous Region Lifetime Errors
|
||||||
use hir;
|
//! where one region is named and the other is anonymous.
|
||||||
use infer::InferCtxt;
|
use infer::InferCtxt;
|
||||||
use ty::{self, Region};
|
use ty;
|
||||||
use infer::region_inference::RegionResolutionError::*;
|
use infer::region_inference::RegionResolutionError::*;
|
||||||
use infer::region_inference::RegionResolutionError;
|
use infer::region_inference::RegionResolutionError;
|
||||||
use hir::map as hir_map;
|
|
||||||
use hir::def_id::DefId;
|
|
||||||
|
|
||||||
impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
||||||
// This method walks the Type of the function body arguments using
|
|
||||||
// `fold_regions()` function and returns the
|
|
||||||
// &hir::Arg of the function argument corresponding to the anonymous
|
|
||||||
// region and the Ty corresponding to the named region.
|
|
||||||
// Currently only the case where the function declaration consists of
|
|
||||||
// one named region and one anonymous region is handled.
|
|
||||||
// Consider the example `fn foo<'a>(x: &'a i32, y: &i32) -> &'a i32`
|
|
||||||
// Here, we would return the hir::Arg for y, we return the type &'a
|
|
||||||
// i32, which is the type of y but with the anonymous region replaced
|
|
||||||
// with 'a, the corresponding bound region and is_first which is true if
|
|
||||||
// the hir::Arg is the first argument in the function declaration.
|
|
||||||
fn find_arg_with_anonymous_region
|
|
||||||
(&self,
|
|
||||||
anon_region: Region<'tcx>,
|
|
||||||
named_region: Region<'tcx>)
|
|
||||||
-> Option<(&hir::Arg, ty::Ty<'tcx>, ty::BoundRegion, bool)> {
|
|
||||||
|
|
||||||
match *anon_region {
|
|
||||||
ty::ReFree(ref free_region) => {
|
|
||||||
let id = free_region.scope;
|
|
||||||
let node_id = self.tcx.hir.as_local_node_id(id).unwrap();
|
|
||||||
let body_id = self.tcx.hir.maybe_body_owned_by(node_id).unwrap();
|
|
||||||
let body = self.tcx.hir.body(body_id);
|
|
||||||
if let Some(tables) = self.in_progress_tables {
|
|
||||||
body.arguments
|
|
||||||
.iter()
|
|
||||||
.enumerate()
|
|
||||||
.filter_map(|(index, arg)| {
|
|
||||||
let ty = tables.borrow().node_id_to_type(arg.id);
|
|
||||||
let mut found_anon_region = false;
|
|
||||||
let new_arg_ty = self.tcx
|
|
||||||
.fold_regions(&ty, &mut false, |r, _| if *r == *anon_region {
|
|
||||||
found_anon_region = true;
|
|
||||||
named_region
|
|
||||||
} else {
|
|
||||||
r
|
|
||||||
});
|
|
||||||
if found_anon_region {
|
|
||||||
let is_first = index == 0;
|
|
||||||
Some((arg, new_arg_ty, free_region.bound_region, is_first))
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.next()
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// This method generates the error message for the case when
|
// This method generates the error message for the case when
|
||||||
// the function arguments consist of a named region and an anonymous
|
// the function arguments consist of a named region and an anonymous
|
||||||
// region and corresponds to `ConcreteFailure(..)`
|
// region and corresponds to `ConcreteFailure(..)`
|
||||||
@ -86,7 +31,7 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
|||||||
// only introduced anonymous regions in parameters) as well as a
|
// only introduced anonymous regions in parameters) as well as a
|
||||||
// version new_ty of its type where the anonymous region is replaced
|
// version new_ty of its type where the anonymous region is replaced
|
||||||
// with the named one.
|
// with the named one.
|
||||||
let (named, (arg, new_ty, br, is_first), scope_def_id) =
|
let (named, (arg, new_ty, br, is_first), (scope_def_id, _)) =
|
||||||
if sub.is_named_region() && self.is_suitable_anonymous_region(sup).is_some() {
|
if sub.is_named_region() && self.is_suitable_anonymous_region(sup).is_some() {
|
||||||
(sub,
|
(sub,
|
||||||
self.find_arg_with_anonymous_region(sup, sub).unwrap(),
|
self.find_arg_with_anonymous_region(sup, sub).unwrap(),
|
||||||
@ -145,48 +90,4 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
|||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// This method returns whether the given Region is Anonymous
|
|
||||||
// and returns the DefId corresponding to the region.
|
|
||||||
pub fn is_suitable_anonymous_region(&self, region: Region<'tcx>) -> Option<DefId> {
|
|
||||||
match *region {
|
|
||||||
ty::ReFree(ref free_region) => {
|
|
||||||
match free_region.bound_region {
|
|
||||||
ty::BrAnon(..) => {
|
|
||||||
let anonymous_region_binding_scope = free_region.scope;
|
|
||||||
let node_id = self.tcx
|
|
||||||
.hir
|
|
||||||
.as_local_node_id(anonymous_region_binding_scope)
|
|
||||||
.unwrap();
|
|
||||||
match self.tcx.hir.find(node_id) {
|
|
||||||
Some(hir_map::NodeItem(..)) |
|
|
||||||
Some(hir_map::NodeTraitItem(..)) => {
|
|
||||||
// proceed ahead //
|
|
||||||
}
|
|
||||||
Some(hir_map::NodeImplItem(..)) => {
|
|
||||||
let container_id = self.tcx
|
|
||||||
.associated_item(anonymous_region_binding_scope)
|
|
||||||
.container
|
|
||||||
.id();
|
|
||||||
if self.tcx.impl_trait_ref(container_id).is_some() {
|
|
||||||
// For now, we do not try to target impls of traits. This is
|
|
||||||
// because this message is going to suggest that the user
|
|
||||||
// change the fn signature, but they may not be free to do so,
|
|
||||||
// since the signature must match the trait.
|
|
||||||
//
|
|
||||||
// FIXME(#42706) -- in some cases, we could do better here.
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => return None, // inapplicable
|
|
||||||
// we target only top-level functions
|
|
||||||
}
|
|
||||||
return Some(anonymous_region_binding_scope);
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
120
src/librustc/infer/error_reporting/util.rs
Normal file
120
src/librustc/infer/error_reporting/util.rs
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
// Copyright 2012-2013 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.
|
||||||
|
|
||||||
|
//! Helper functions corresponding to lifetime errors due to
|
||||||
|
//! anonymous regions.
|
||||||
|
use hir;
|
||||||
|
use infer::InferCtxt;
|
||||||
|
use ty::{self, Region};
|
||||||
|
use hir::def_id::DefId;
|
||||||
|
use hir::map as hir_map;
|
||||||
|
|
||||||
|
impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
|
||||||
|
// This method walks the Type of the function body arguments using
|
||||||
|
// `fold_regions()` function and returns the
|
||||||
|
// &hir::Arg of the function argument corresponding to the anonymous
|
||||||
|
// region and the Ty corresponding to the named region.
|
||||||
|
// Currently only the case where the function declaration consists of
|
||||||
|
// one named region and one anonymous region is handled.
|
||||||
|
// Consider the example `fn foo<'a>(x: &'a i32, y: &i32) -> &'a i32`
|
||||||
|
// Here, we would return the hir::Arg for y, we return the type &'a
|
||||||
|
// i32, which is the type of y but with the anonymous region replaced
|
||||||
|
// with 'a, the corresponding bound region and is_first which is true if
|
||||||
|
// the hir::Arg is the first argument in the function declaration.
|
||||||
|
pub fn find_arg_with_anonymous_region
|
||||||
|
(&self,
|
||||||
|
anon_region: Region<'tcx>,
|
||||||
|
replace_region: Region<'tcx>)
|
||||||
|
-> Option<(&hir::Arg, ty::Ty<'tcx>, ty::BoundRegion, bool)> {
|
||||||
|
|
||||||
|
if let ty::ReFree(ref free_region) = *anon_region {
|
||||||
|
|
||||||
|
let id = free_region.scope;
|
||||||
|
let hir = &self.tcx.hir;
|
||||||
|
if let Some(node_id) = hir.as_local_node_id(id) {
|
||||||
|
if let Some(body_id) = hir.maybe_body_owned_by(node_id) {
|
||||||
|
let body = hir.body(body_id);
|
||||||
|
if let Some(tables) = self.in_progress_tables {
|
||||||
|
body.arguments
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.filter_map(|(index, arg)| {
|
||||||
|
let ty = tables.borrow().node_id_to_type(arg.id);
|
||||||
|
let mut found_anon_region = false;
|
||||||
|
let new_arg_ty = self.tcx
|
||||||
|
.fold_regions(&ty, &mut false, |r, _| if *r == *anon_region {
|
||||||
|
found_anon_region = true;
|
||||||
|
replace_region
|
||||||
|
} else {
|
||||||
|
r
|
||||||
|
});
|
||||||
|
if found_anon_region {
|
||||||
|
let is_first = index == 0;
|
||||||
|
Some((arg, new_arg_ty, free_region.bound_region, is_first))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.next()
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This method returns whether the given Region is Anonymous
|
||||||
|
// and returns the DefId and the BoundRegion corresponding to the given region.
|
||||||
|
pub fn is_suitable_anonymous_region(&self,
|
||||||
|
region: Region<'tcx>)
|
||||||
|
-> Option<(DefId, ty::BoundRegion)> {
|
||||||
|
if let ty::ReFree(ref free_region) = *region {
|
||||||
|
if let ty::BrAnon(..) = free_region.bound_region{
|
||||||
|
let anonymous_region_binding_scope = free_region.scope;
|
||||||
|
let node_id = self.tcx
|
||||||
|
.hir
|
||||||
|
.as_local_node_id(anonymous_region_binding_scope)
|
||||||
|
.unwrap();
|
||||||
|
match self.tcx.hir.find(node_id) {
|
||||||
|
Some(hir_map::NodeItem(..)) |
|
||||||
|
Some(hir_map::NodeTraitItem(..)) => {
|
||||||
|
// Success -- proceed to return Some below
|
||||||
|
}
|
||||||
|
Some(hir_map::NodeImplItem(..)) => {
|
||||||
|
let container_id = self.tcx
|
||||||
|
.associated_item(anonymous_region_binding_scope)
|
||||||
|
.container
|
||||||
|
.id();
|
||||||
|
if self.tcx.impl_trait_ref(container_id).is_some() {
|
||||||
|
// For now, we do not try to target impls of traits. This is
|
||||||
|
// because this message is going to suggest that the user
|
||||||
|
// change the fn signature, but they may not be free to do so,
|
||||||
|
// since the signature must match the trait.
|
||||||
|
//
|
||||||
|
// FIXME(#42706) -- in some cases, we could do better here.
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => return None, // inapplicable
|
||||||
|
// we target only top-level functions
|
||||||
|
}
|
||||||
|
return Some((anonymous_region_binding_scope, free_region.bound_region));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
@ -210,17 +210,20 @@ macro_rules! make_mir_visitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn visit_ty(&mut self,
|
fn visit_ty(&mut self,
|
||||||
ty: & $($mutability)* Ty<'tcx>) {
|
ty: & $($mutability)* Ty<'tcx>,
|
||||||
|
_: Lookup) {
|
||||||
self.super_ty(ty);
|
self.super_ty(ty);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_substs(&mut self,
|
fn visit_substs(&mut self,
|
||||||
substs: & $($mutability)* &'tcx Substs<'tcx>) {
|
substs: & $($mutability)* &'tcx Substs<'tcx>,
|
||||||
|
_: Location) {
|
||||||
self.super_substs(substs);
|
self.super_substs(substs);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_closure_substs(&mut self,
|
fn visit_closure_substs(&mut self,
|
||||||
substs: & $($mutability)* ClosureSubsts<'tcx>) {
|
substs: & $($mutability)* ClosureSubsts<'tcx>,
|
||||||
|
_: Location) {
|
||||||
self.super_closure_substs(substs);
|
self.super_closure_substs(substs);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -270,7 +273,11 @@ macro_rules! make_mir_visitor {
|
|||||||
self.visit_visibility_scope_data(scope);
|
self.visit_visibility_scope_data(scope);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.visit_ty(&$($mutability)* mir.return_ty);
|
let lookup = Lookup::Src(SourceInfo {
|
||||||
|
span: mir.span,
|
||||||
|
scope: ARGUMENT_VISIBILITY_SCOPE,
|
||||||
|
});
|
||||||
|
self.visit_ty(&$($mutability)* mir.return_ty, lookup);
|
||||||
|
|
||||||
for local_decl in &$($mutability)* mir.local_decls {
|
for local_decl in &$($mutability)* mir.local_decls {
|
||||||
self.visit_local_decl(local_decl);
|
self.visit_local_decl(local_decl);
|
||||||
@ -389,7 +396,7 @@ macro_rules! make_mir_visitor {
|
|||||||
ref values,
|
ref values,
|
||||||
ref targets } => {
|
ref targets } => {
|
||||||
self.visit_operand(discr, source_location);
|
self.visit_operand(discr, source_location);
|
||||||
self.visit_ty(switch_ty);
|
self.visit_ty(switch_ty, Lookup::Loc(source_location));
|
||||||
for value in &values[..] {
|
for value in &values[..] {
|
||||||
self.visit_const_int(value, source_location);
|
self.visit_const_int(value, source_location);
|
||||||
}
|
}
|
||||||
@ -505,7 +512,7 @@ macro_rules! make_mir_visitor {
|
|||||||
ref $($mutability)* operand,
|
ref $($mutability)* operand,
|
||||||
ref $($mutability)* ty) => {
|
ref $($mutability)* ty) => {
|
||||||
self.visit_operand(operand, location);
|
self.visit_operand(operand, location);
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Loc(location));
|
||||||
}
|
}
|
||||||
|
|
||||||
Rvalue::BinaryOp(_bin_op,
|
Rvalue::BinaryOp(_bin_op,
|
||||||
@ -527,7 +534,7 @@ macro_rules! make_mir_visitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Rvalue::NullaryOp(_op, ref $($mutability)* ty) => {
|
Rvalue::NullaryOp(_op, ref $($mutability)* ty) => {
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Loc(location));
|
||||||
}
|
}
|
||||||
|
|
||||||
Rvalue::Aggregate(ref $($mutability)* kind,
|
Rvalue::Aggregate(ref $($mutability)* kind,
|
||||||
@ -535,7 +542,7 @@ macro_rules! make_mir_visitor {
|
|||||||
let kind = &$($mutability)* **kind;
|
let kind = &$($mutability)* **kind;
|
||||||
match *kind {
|
match *kind {
|
||||||
AggregateKind::Array(ref $($mutability)* ty) => {
|
AggregateKind::Array(ref $($mutability)* ty) => {
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Loc(location));
|
||||||
}
|
}
|
||||||
AggregateKind::Tuple => {
|
AggregateKind::Tuple => {
|
||||||
}
|
}
|
||||||
@ -543,12 +550,12 @@ macro_rules! make_mir_visitor {
|
|||||||
_variant_index,
|
_variant_index,
|
||||||
ref $($mutability)* substs,
|
ref $($mutability)* substs,
|
||||||
_active_field_index) => {
|
_active_field_index) => {
|
||||||
self.visit_substs(substs);
|
self.visit_substs(substs, location);
|
||||||
}
|
}
|
||||||
AggregateKind::Closure(ref $($mutability)* def_id,
|
AggregateKind::Closure(ref $($mutability)* def_id,
|
||||||
ref $($mutability)* closure_substs) => {
|
ref $($mutability)* closure_substs) => {
|
||||||
self.visit_def_id(def_id, location);
|
self.visit_def_id(def_id, location);
|
||||||
self.visit_closure_substs(closure_substs);
|
self.visit_closure_substs(closure_substs, location);
|
||||||
}
|
}
|
||||||
AggregateKind::Generator(ref $($mutability)* def_id,
|
AggregateKind::Generator(ref $($mutability)* def_id,
|
||||||
ref $($mutability)* closure_substs) => {
|
ref $($mutability)* closure_substs) => {
|
||||||
@ -603,7 +610,7 @@ macro_rules! make_mir_visitor {
|
|||||||
ref $($mutability)* ty,
|
ref $($mutability)* ty,
|
||||||
} = *static_;
|
} = *static_;
|
||||||
self.visit_def_id(def_id, location);
|
self.visit_def_id(def_id, location);
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Loc(location));
|
||||||
}
|
}
|
||||||
|
|
||||||
fn super_projection(&mut self,
|
fn super_projection(&mut self,
|
||||||
@ -633,7 +640,7 @@ macro_rules! make_mir_visitor {
|
|||||||
ProjectionElem::Subslice { from: _, to: _ } => {
|
ProjectionElem::Subslice { from: _, to: _ } => {
|
||||||
}
|
}
|
||||||
ProjectionElem::Field(_field, ref $($mutability)* ty) => {
|
ProjectionElem::Field(_field, ref $($mutability)* ty) => {
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Loc(location));
|
||||||
}
|
}
|
||||||
ProjectionElem::Index(ref $($mutability)* operand) => {
|
ProjectionElem::Index(ref $($mutability)* operand) => {
|
||||||
self.visit_operand(operand, location);
|
self.visit_operand(operand, location);
|
||||||
@ -658,7 +665,7 @@ macro_rules! make_mir_visitor {
|
|||||||
is_user_variable: _,
|
is_user_variable: _,
|
||||||
} = *local_decl;
|
} = *local_decl;
|
||||||
|
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Src(*source_info));
|
||||||
self.visit_source_info(source_info);
|
self.visit_source_info(source_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -681,7 +688,7 @@ macro_rules! make_mir_visitor {
|
|||||||
} = *constant;
|
} = *constant;
|
||||||
|
|
||||||
self.visit_span(span);
|
self.visit_span(span);
|
||||||
self.visit_ty(ty);
|
self.visit_ty(ty, Lookup::Loc(location));
|
||||||
self.visit_literal(literal, location);
|
self.visit_literal(literal, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -692,7 +699,7 @@ macro_rules! make_mir_visitor {
|
|||||||
Literal::Item { ref $($mutability)* def_id,
|
Literal::Item { ref $($mutability)* def_id,
|
||||||
ref $($mutability)* substs } => {
|
ref $($mutability)* substs } => {
|
||||||
self.visit_def_id(def_id, location);
|
self.visit_def_id(def_id, location);
|
||||||
self.visit_substs(substs);
|
self.visit_substs(substs, location);
|
||||||
}
|
}
|
||||||
Literal::Value { ref $($mutability)* value } => {
|
Literal::Value { ref $($mutability)* value } => {
|
||||||
self.visit_const_val(value, location);
|
self.visit_const_val(value, location);
|
||||||
@ -757,6 +764,11 @@ macro_rules! make_mir_visitor {
|
|||||||
make_mir_visitor!(Visitor,);
|
make_mir_visitor!(Visitor,);
|
||||||
make_mir_visitor!(MutVisitor,mut);
|
make_mir_visitor!(MutVisitor,mut);
|
||||||
|
|
||||||
|
pub enum Lookup {
|
||||||
|
Loc(Location),
|
||||||
|
Src(SourceInfo),
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||||
pub enum LvalueContext<'tcx> {
|
pub enum LvalueContext<'tcx> {
|
||||||
// Appears as LHS of an assignment
|
// Appears as LHS of an assignment
|
||||||
|
@ -17,7 +17,7 @@ use rustc::hir::def_id::DefId;
|
|||||||
use rustc::middle::region::CodeExtent;
|
use rustc::middle::region::CodeExtent;
|
||||||
use rustc::mir::*;
|
use rustc::mir::*;
|
||||||
use rustc::mir::transform::MirSource;
|
use rustc::mir::transform::MirSource;
|
||||||
use rustc::mir::visit::MutVisitor;
|
use rustc::mir::visit::{MutVisitor, Lookup};
|
||||||
use rustc::ty::{self, Ty, TyCtxt};
|
use rustc::ty::{self, Ty, TyCtxt};
|
||||||
use rustc::ty::subst::Substs;
|
use rustc::ty::subst::Substs;
|
||||||
use rustc::util::nodemap::NodeMap;
|
use rustc::util::nodemap::NodeMap;
|
||||||
@ -156,7 +156,7 @@ struct GlobalizeMir<'a, 'gcx: 'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, 'gcx: 'tcx, 'tcx> MutVisitor<'tcx> for GlobalizeMir<'a, 'gcx> {
|
impl<'a, 'gcx: 'tcx, 'tcx> MutVisitor<'tcx> for GlobalizeMir<'a, 'gcx> {
|
||||||
fn visit_ty(&mut self, ty: &mut Ty<'tcx>) {
|
fn visit_ty(&mut self, ty: &mut Ty<'tcx>, _: Lookup) {
|
||||||
if let Some(lifted) = self.tcx.lift(ty) {
|
if let Some(lifted) = self.tcx.lift(ty) {
|
||||||
*ty = lifted;
|
*ty = lifted;
|
||||||
} else {
|
} else {
|
||||||
@ -166,7 +166,7 @@ impl<'a, 'gcx: 'tcx, 'tcx> MutVisitor<'tcx> for GlobalizeMir<'a, 'gcx> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_substs(&mut self, substs: &mut &'tcx Substs<'tcx>) {
|
fn visit_substs(&mut self, substs: &mut &'tcx Substs<'tcx>, _: Location) {
|
||||||
if let Some(lifted) = self.tcx.lift(substs) {
|
if let Some(lifted) = self.tcx.lift(substs) {
|
||||||
*substs = lifted;
|
*substs = lifted;
|
||||||
} else {
|
} else {
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
use rustc::ty::subst::Substs;
|
use rustc::ty::subst::Substs;
|
||||||
use rustc::ty::{Ty, TyCtxt, ClosureSubsts};
|
use rustc::ty::{Ty, TyCtxt, ClosureSubsts};
|
||||||
use rustc::mir::*;
|
use rustc::mir::*;
|
||||||
use rustc::mir::visit::MutVisitor;
|
use rustc::mir::visit::{MutVisitor, Lookup};
|
||||||
use rustc::mir::transform::{MirPass, MirSource};
|
use rustc::mir::transform::{MirPass, MirSource};
|
||||||
|
|
||||||
struct EraseRegionsVisitor<'a, 'tcx: 'a> {
|
struct EraseRegionsVisitor<'a, 'tcx: 'a> {
|
||||||
@ -31,12 +31,12 @@ impl<'a, 'tcx> EraseRegionsVisitor<'a, 'tcx> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, 'tcx> MutVisitor<'tcx> for EraseRegionsVisitor<'a, 'tcx> {
|
impl<'a, 'tcx> MutVisitor<'tcx> for EraseRegionsVisitor<'a, 'tcx> {
|
||||||
fn visit_ty(&mut self, ty: &mut Ty<'tcx>) {
|
fn visit_ty(&mut self, ty: &mut Ty<'tcx>, _: Lookup) {
|
||||||
let old_ty = *ty;
|
let old_ty = *ty;
|
||||||
*ty = self.tcx.erase_regions(&old_ty);
|
*ty = self.tcx.erase_regions(&old_ty);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_substs(&mut self, substs: &mut &'tcx Substs<'tcx>) {
|
fn visit_substs(&mut self, substs: &mut &'tcx Substs<'tcx>, _: Location) {
|
||||||
*substs = self.tcx.erase_regions(&{*substs});
|
*substs = self.tcx.erase_regions(&{*substs});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -62,7 +62,8 @@ impl<'a, 'tcx> MutVisitor<'tcx> for EraseRegionsVisitor<'a, 'tcx> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn visit_closure_substs(&mut self,
|
fn visit_closure_substs(&mut self,
|
||||||
substs: &mut ClosureSubsts<'tcx>) {
|
substs: &mut ClosureSubsts<'tcx>,
|
||||||
|
_: Location) {
|
||||||
*substs = self.tcx.erase_regions(substs);
|
*substs = self.tcx.erase_regions(substs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -288,7 +288,8 @@ impl<'a, 'tcx> mir_visit::Visitor<'tcx> for StatCollector<'a, 'tcx> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn visit_closure_substs(&mut self,
|
fn visit_closure_substs(&mut self,
|
||||||
substs: &ClosureSubsts<'tcx>) {
|
substs: &ClosureSubsts<'tcx>,
|
||||||
|
_: Location) {
|
||||||
self.record("ClosureSubsts", substs);
|
self.record("ClosureSubsts", substs);
|
||||||
self.super_closure_substs(substs);
|
self.super_closure_substs(substs);
|
||||||
}
|
}
|
||||||
|
15
src/test/ui/lifetime-errors/ex3-both-anon-regions-2.rs
Normal file
15
src/test/ui/lifetime-errors/ex3-both-anon-regions-2.rs
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright 2016 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.
|
||||||
|
|
||||||
|
fn foo((v, w): (&u8, &u8), x: &u8) {
|
||||||
|
v = x;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() { }
|
10
src/test/ui/lifetime-errors/ex3-both-anon-regions-2.stderr
Normal file
10
src/test/ui/lifetime-errors/ex3-both-anon-regions-2.stderr
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
error[E0623]: lifetime mismatch
|
||||||
|
--> $DIR/ex3-both-anon-regions-2.rs:12:9
|
||||||
|
|
|
||||||
|
11 | fn foo((v, w): (&u8, &u8), x: &u8) {
|
||||||
|
| --- --- these references must have the same lifetime
|
||||||
|
12 | v = x;
|
||||||
|
| ^ data from `x` flows here
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
15
src/test/ui/lifetime-errors/ex3-both-anon-regions-3.rs
Normal file
15
src/test/ui/lifetime-errors/ex3-both-anon-regions-3.rs
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright 2016 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.
|
||||||
|
|
||||||
|
fn foo((v, w): (&u8, &u8), (x, y): (&u8, &u8)) {
|
||||||
|
v = x;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() { }
|
10
src/test/ui/lifetime-errors/ex3-both-anon-regions-3.stderr
Normal file
10
src/test/ui/lifetime-errors/ex3-both-anon-regions-3.stderr
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
error[E0623]: lifetime mismatch
|
||||||
|
--> $DIR/ex3-both-anon-regions-3.rs:12:9
|
||||||
|
|
|
||||||
|
11 | fn foo((v, w): (&u8, &u8), (x, y): (&u8, &u8)) {
|
||||||
|
| --- --- these references must have the same lifetime
|
||||||
|
12 | v = x;
|
||||||
|
| ^ data flows here
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
13
src/test/ui/lifetime-errors/ex3-both-anon-regions-4.rs
Normal file
13
src/test/ui/lifetime-errors/ex3-both-anon-regions-4.rs
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
// Copyright 2016 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.
|
||||||
|
|
||||||
|
fn foo(z: &mut Vec<(&u8,&u8)>, (x, y): (&u8, &u8)) {
|
||||||
|
z.push((x,y));
|
||||||
|
}
|
20
src/test/ui/lifetime-errors/ex3-both-anon-regions-4.stderr
Normal file
20
src/test/ui/lifetime-errors/ex3-both-anon-regions-4.stderr
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
error[E0601]: main function not found
|
||||||
|
|
||||||
|
error[E0623]: lifetime mismatch
|
||||||
|
--> $DIR/ex3-both-anon-regions-4.rs:12:13
|
||||||
|
|
|
||||||
|
11 | fn foo(z: &mut Vec<(&u8,&u8)>, (x, y): (&u8, &u8)) {
|
||||||
|
| --- --- these references must have the same lifetime
|
||||||
|
12 | z.push((x,y));
|
||||||
|
| ^ data flows into `z` here
|
||||||
|
|
||||||
|
error[E0623]: lifetime mismatch
|
||||||
|
--> $DIR/ex3-both-anon-regions-4.rs:12:15
|
||||||
|
|
|
||||||
|
11 | fn foo(z: &mut Vec<(&u8,&u8)>, (x, y): (&u8, &u8)) {
|
||||||
|
| --- --- these references must have the same lifetime
|
||||||
|
12 | z.push((x,y));
|
||||||
|
| ^ data flows into `z` here
|
||||||
|
|
||||||
|
error: aborting due to 3 previous errors
|
||||||
|
|
15
src/test/ui/lifetime-errors/ex3-both-anon-regions.rs
Normal file
15
src/test/ui/lifetime-errors/ex3-both-anon-regions.rs
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright 2016 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.
|
||||||
|
|
||||||
|
fn foo(x: &mut Vec<&u8>, y: &u8) {
|
||||||
|
x.push(y);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() { }
|
10
src/test/ui/lifetime-errors/ex3-both-anon-regions.stderr
Normal file
10
src/test/ui/lifetime-errors/ex3-both-anon-regions.stderr
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
error[E0623]: lifetime mismatch
|
||||||
|
--> $DIR/ex3-both-anon-regions.rs:12:12
|
||||||
|
|
|
||||||
|
11 | fn foo(x: &mut Vec<&u8>, y: &u8) {
|
||||||
|
| --- --- these references must have the same lifetime
|
||||||
|
12 | x.push(y);
|
||||||
|
| ^ data from `y` flows into `x` here
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
Loading…
x
Reference in New Issue
Block a user