mirror of
				https://github.com/rust-lang/rust.git
				synced 2025-11-03 22:49:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			118 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			118 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
// 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
 |