| // Reject mixing cyclic structure and Drop when using TypedArena. | 
 | // | 
 | // (Compare against dropck-vec-cycle-checked.rs) | 
 | // | 
 | // (Also compare against ui-fulldeps/dropck-tarena-unsound-drop.rs, | 
 | //  which is a reduction of this code to more directly show the reason | 
 | //  for the error message we see here.) | 
 |  | 
 | #![feature(rustc_private)] | 
 |  | 
 | extern crate rustc_arena; | 
 |  | 
 | use rustc_arena::TypedArena; | 
 | use std::cell::Cell; | 
 | use id::Id; | 
 |  | 
 | mod s { | 
 |     use std::sync::atomic::{AtomicUsize, Ordering}; | 
 |  | 
 |     static S_COUNT: AtomicUsize = AtomicUsize::new(0); | 
 |  | 
 |     pub fn next_count() -> usize { | 
 |         S_COUNT.fetch_add(1, Ordering::SeqCst) + 1 | 
 |     } | 
 | } | 
 |  | 
 | mod id { | 
 |     use s; | 
 |     #[derive(Debug)] | 
 |     pub struct Id { | 
 |         orig_count: usize, | 
 |         count: usize, | 
 |     } | 
 |  | 
 |     impl Id { | 
 |         pub fn new() -> Id { | 
 |             let c = s::next_count(); | 
 |             println!("building Id {}", c); | 
 |             Id { orig_count: c, count: c } | 
 |         } | 
 |         pub fn count(&self) -> usize { | 
 |             println!("Id::count on {} returns {}", self.orig_count, self.count); | 
 |             self.count | 
 |         } | 
 |     } | 
 |  | 
 |     impl Drop for Id { | 
 |         fn drop(&mut self) { | 
 |             println!("dropping Id {}", self.count); | 
 |             self.count = 0; | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | trait HasId { | 
 |     fn count(&self) -> usize; | 
 | } | 
 |  | 
 | #[derive(Debug)] | 
 | struct CheckId<T:HasId> { | 
 |     v: T | 
 | } | 
 |  | 
 | #[allow(non_snake_case)] | 
 | fn CheckId<T:HasId>(t: T) -> CheckId<T> { CheckId{ v: t } } | 
 |  | 
 | impl<T:HasId> Drop for CheckId<T> { | 
 |     fn drop(&mut self) { | 
 |         assert!(self.v.count() > 0); | 
 |     } | 
 | } | 
 |  | 
 | #[derive(Debug)] | 
 | struct C<'a> { | 
 |     id: Id, | 
 |     v: Vec<CheckId<Cell<Option<&'a C<'a>>>>>, | 
 | } | 
 |  | 
 | impl<'a> HasId for Cell<Option<&'a C<'a>>> { | 
 |     fn count(&self) -> usize { | 
 |         match self.get() { | 
 |             None => 1, | 
 |             Some(c) => c.id.count(), | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | impl<'a> C<'a> { | 
 |     fn new() -> C<'a> { | 
 |         C { id: Id::new(), v: Vec::new() } | 
 |     } | 
 | } | 
 |  | 
 | fn f<'a>(arena: &'a TypedArena<C<'a>>) { | 
 |     let c1 = arena.alloc(C::new()); | 
 |     let c2 = arena.alloc(C::new()); | 
 |     let c3 = arena.alloc(C::new()); | 
 |  | 
 |     c1.v.push(CheckId(Cell::new(None))); | 
 |     c1.v.push(CheckId(Cell::new(None))); | 
 |     c2.v.push(CheckId(Cell::new(None))); | 
 |     c2.v.push(CheckId(Cell::new(None))); | 
 |     c3.v.push(CheckId(Cell::new(None))); | 
 |     c3.v.push(CheckId(Cell::new(None))); | 
 |  | 
 |     c1.v[0].v.set(Some(c2)); | 
 |     c1.v[1].v.set(Some(c3)); | 
 |     c2.v[0].v.set(Some(c2)); | 
 |     c2.v[1].v.set(Some(c3)); | 
 |     c3.v[0].v.set(Some(c1)); | 
 |     c3.v[1].v.set(Some(c2)); | 
 | } | 
 |  | 
 | fn main() { | 
 |     let arena = TypedArena::default(); | 
 |     f(&arena); | 
 | } //~^ ERROR `arena` does not live long enough |