Auto merge of #31442 - pnkfelix:issue-30438-sidestep-dummy-node-during-expand-givens-dfs, r=nikomatsakis

Split dummy-idx node to fix expand_givens DFS

(Much more detail in commit comments.)

Fix #30438.
This commit is contained in:
bors 2016-02-08 14:56:24 +00:00
commit 04f12ef4f7
4 changed files with 107 additions and 3 deletions

View file

@ -1105,7 +1105,14 @@ impl<'a, 'tcx> RegionVarBindings<'a, 'tcx> {
for _ in 0..num_vars {
graph.add_node(());
}
let dummy_idx = graph.add_node(());
// Issue #30438: two distinct dummy nodes, one for incoming
// edges (dummy_source) and another for outgoing edges
// (dummy_sink). In `dummy -> a -> b -> dummy`, using one
// dummy node leads one to think (erroneously) there exists a
// path from `b` to `a`. Two dummy nodes sidesteps the issue.
let dummy_source = graph.add_node(());
let dummy_sink = graph.add_node(());
for (constraint, _) in constraints.iter() {
match *constraint {
@ -1115,10 +1122,10 @@ impl<'a, 'tcx> RegionVarBindings<'a, 'tcx> {
*constraint);
}
ConstrainRegSubVar(_, b_id) => {
graph.add_edge(dummy_idx, NodeIndex(b_id.index as usize), *constraint);
graph.add_edge(dummy_source, NodeIndex(b_id.index as usize), *constraint);
}
ConstrainVarSubReg(a_id, _) => {
graph.add_edge(NodeIndex(a_id.index as usize), dummy_idx, *constraint);
graph.add_edge(NodeIndex(a_id.index as usize), dummy_sink, *constraint);
}
}
}

View file

@ -0,0 +1,33 @@
// 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.
// Original regression test for Issue #30438.
use std::ops::Index;
struct Test<'a> {
s: &'a String
}
impl <'a> Index<usize> for Test<'a> {
type Output = Test<'a>;
fn index(&self, _: usize) -> &Self::Output {
return &Test { s: &self.s};
//~^ ERROR: borrowed value does not live long enough
}
}
fn main() {
let s = "Hello World".to_string();
let test = Test{s: &s};
let r = &test[0];
println!("{}", test.s); // OK since test is valid
println!("{}", r.s); // Segfault since value pointed by r has already been dropped
}

View file

@ -0,0 +1,34 @@
// 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.
// Modified regression test for Issue #30438 that exposed an
// independent issue (see discussion on ticket).
use std::ops::Index;
struct Test<'a> {
s: &'a String
}
impl <'a> Index<usize> for Test<'a> {
type Output = Test<'a>;
fn index(&self, _: usize) -> &Self::Output {
&Test { s: &self.s}
//~^ ERROR: borrowed value does not live long enough
}
}
fn main() {
let s = "Hello World".to_string();
let test = Test{s: &s};
let r = &test[0];
println!("{}", test.s); // OK since test is valid
println!("{}", r.s); // Segfault since value pointed by r has already been dropped
}

View file

@ -0,0 +1,30 @@
// 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.
// Simplfied regression test for #30438, inspired by arielb1.
trait Trait { type Out; }
struct Test<'a> { s: &'a str }
fn silly<'y, 'z>(_s: &'y Test<'z>) -> &'y <Test<'z> as Trait>::Out where 'z: 'static {
let x = Test { s: "this cannot last" };
&x
//~^ ERROR: `x` does not live long enough
}
impl<'b> Trait for Test<'b> { type Out = Test<'b>; }
fn main() {
let orig = Test { s: "Hello World" };
let r = silly(&orig);
println!("{}", orig.s); // OK since `orig` is valid
println!("{}", r.s); // Segfault (method does not return a sane value)
}