forked from rust-lang/rust
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fix Subst construction: use subst from adt_def rather than Drop impl'…
…s subst. This addresses issue pointed out by niko that prior code would break if the declaration order for generics does not match how they are fed into the instantiation of the type itself. (Added some tests exercising this scenario.)
- Loading branch information
Showing
4 changed files
with
247 additions
and
13 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
// 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. | ||
|
||
#![feature(generic_param_attrs)] | ||
#![feature(dropck_eyepatch)] | ||
|
||
// The point of this test is to test uses of `#[may_dangle]` attribute | ||
// where the formal declaration order (in the impl generics) does not | ||
// match the actual usage order (in the type instantiation). | ||
// | ||
// See also dropck-eyepatch.rs for more information about the general | ||
// structure of the test. | ||
|
||
use std::fmt; | ||
|
||
struct Dt<A: fmt::Debug>(&'static str, A); | ||
struct Dr<'a, B:'a+fmt::Debug>(&'static str, &'a B); | ||
struct Pt<A: fmt::Debug, B: fmt::Debug>(&'static str, A, B); | ||
struct Pr<'a, 'b, B:'a+'b+fmt::Debug>(&'static str, &'a B, &'b B); | ||
struct St<A: fmt::Debug>(&'static str, A); | ||
struct Sr<'a, B:'a+fmt::Debug>(&'static str, &'a B); | ||
|
||
impl<A: fmt::Debug> Drop for Dt<A> { | ||
fn drop(&mut self) { println!("drop {} {:?}", self.0, self.1); } | ||
} | ||
impl<'a, B: fmt::Debug> Drop for Dr<'a, B> { | ||
fn drop(&mut self) { println!("drop {} {:?}", self.0, self.1); } | ||
} | ||
unsafe impl<B: fmt::Debug, #[may_dangle] A: fmt::Debug> Drop for Pt<A, B> { | ||
// (unsafe to access self.1 due to #[may_dangle] on A) | ||
fn drop(&mut self) { println!("drop {} {:?}", self.0, self.2); } | ||
} | ||
unsafe impl<'b, #[may_dangle] 'a, B: fmt::Debug> Drop for Pr<'a, 'b, B> { | ||
// (unsafe to access self.1 due to #[may_dangle] on 'a) | ||
fn drop(&mut self) { println!("drop {} {:?}", self.0, self.2); } | ||
} | ||
|
||
fn main() { | ||
use std::cell::Cell; | ||
let c_long; | ||
let (c, mut dt, mut dr, mut pt, mut pr, st, sr) | ||
: (Cell<_>, Dt<_>, Dr<_>, Pt<_, _>, Pr<_>, St<_>, Sr<_>); | ||
c_long = Cell::new(1); | ||
c = Cell::new(1); | ||
|
||
// No error: sufficiently long-lived state can be referenced in dtors | ||
dt = Dt("dt", &c_long); | ||
dr = Dr("dr", &c_long); | ||
// Error: destructor order imprecisely modelled | ||
dt = Dt("dt", &c); //~ ERROR `c` does not live long enough | ||
dr = Dr("dr", &c); //~ ERROR `c` does not live long enough | ||
|
||
// No error: Drop impl asserts .1 (A and &'a _) are not accessed | ||
pt = Pt("pt", &c, &c_long); | ||
pr = Pr("pr", &c, &c_long); | ||
|
||
// Error: Drop impl's assertion does not apply to `B` nor `&'b _` | ||
pt = Pt("pt", &c_long, &c); //~ ERROR `c` does not live long enough | ||
pr = Pr("pr", &c_long, &c); //~ ERROR `c` does not live long enough | ||
|
||
// No error: St and Sr have no destructor. | ||
st = St("st", &c); | ||
sr = Sr("sr", &c); | ||
|
||
println!("{:?}", (dt.0, dr.0, pt.0, pr.0, st.0, sr.0)); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
// 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. | ||
|
||
#![feature(generic_param_attrs)] | ||
#![feature(dropck_eyepatch)] | ||
|
||
// The point of this test is to test uses of `#[may_dangle]` attribute | ||
// where the formal declaration order (in the impl generics) does not | ||
// match the actual usage order (in the type instantiation). | ||
// | ||
// See also dropck-eyepatch.rs for more information about the general | ||
// structure of the test. | ||
|
||
trait Foo { fn foo(&self, _: &str); } | ||
|
||
struct Dt<A: Foo>(&'static str, A); | ||
struct Dr<'a, B:'a+Foo>(&'static str, &'a B); | ||
struct Pt<A: Foo, B: Foo>(&'static str, A, B); | ||
struct Pr<'a, 'b, B:'a+'b+Foo>(&'static str, &'a B, &'b B); | ||
struct St<A: Foo>(&'static str, A); | ||
struct Sr<'a, B:'a+Foo>(&'static str, &'a B); | ||
|
||
impl<A: Foo> Drop for Dt<A> { | ||
fn drop(&mut self) { println!("drop {}", self.0); self.1.foo(self.0); } | ||
} | ||
impl<'a, B: Foo> Drop for Dr<'a, B> { | ||
fn drop(&mut self) { println!("drop {}", self.0); self.1.foo(self.0); } | ||
} | ||
unsafe impl<B: Foo, #[may_dangle] A: Foo> Drop for Pt<A, B> { | ||
// (unsafe to access self.1 due to #[may_dangle] on A) | ||
fn drop(&mut self) { println!("drop {}", self.0); self.2.foo(self.0); } | ||
} | ||
unsafe impl<'b, #[may_dangle] 'a, B: Foo> Drop for Pr<'a, 'b, B> { | ||
// (unsafe to access self.1 due to #[may_dangle] on 'a) | ||
fn drop(&mut self) { println!("drop {}", self.0); self.2.foo(self.0); } | ||
} | ||
|
||
fn main() { | ||
use std::cell::RefCell; | ||
|
||
impl Foo for RefCell<String> { | ||
fn foo(&self, s: &str) { | ||
let s2 = format!("{}|{}", *self.borrow(), s); | ||
*self.borrow_mut() = s2; | ||
} | ||
} | ||
|
||
impl<'a, T:Foo> Foo for &'a T { | ||
fn foo(&self, s: &str) { | ||
(*self).foo(s); | ||
} | ||
} | ||
|
||
struct CheckOnDrop(RefCell<String>, &'static str); | ||
impl Drop for CheckOnDrop { | ||
fn drop(&mut self) { assert_eq!(*self.0.borrow(), self.1); } | ||
} | ||
|
||
let c_long; | ||
let (c, dt, dr, pt, pr, st, sr) | ||
: (CheckOnDrop, Dt<_>, Dr<_>, Pt<_, _>, Pr<_>, St<_>, Sr<_>); | ||
c_long = CheckOnDrop(RefCell::new("c_long".to_string()), | ||
"c_long|pr|pt|dr|dt"); | ||
c = CheckOnDrop(RefCell::new("c".to_string()), | ||
"c"); | ||
|
||
// No error: sufficiently long-lived state can be referenced in dtors | ||
dt = Dt("dt", &c_long.0); | ||
dr = Dr("dr", &c_long.0); | ||
|
||
// No error: Drop impl asserts .1 (A and &'a _) are not accessed | ||
pt = Pt("pt", &c.0, &c_long.0); | ||
pr = Pr("pr", &c.0, &c_long.0); | ||
|
||
// No error: St and Sr have no destructor. | ||
st = St("st", &c.0); | ||
sr = Sr("sr", &c.0); | ||
|
||
println!("{:?}", (dt.0, dr.0, pt.0, pr.0, st.0, sr.0)); | ||
assert_eq!(*c_long.0.borrow(), "c_long"); | ||
assert_eq!(*c.0.borrow(), "c"); | ||
} |