// Copyright 2012 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. // xfail-test linked failure // A port of task-killjoin to use a class with a dtor to manage // the join. use std::comm::*; use std::ptr; use std::task; struct notify { ch: Chan, v: @mut bool, } #[unsafe_destructor] impl Drop for notify { fn drop(&mut self) { unsafe { error!("notify: task=%? v=%x unwinding=%b b=%b", 0, ptr::to_unsafe_ptr(&(*(self.v))) as uint, task::failing(), *(self.v)); let b = *(self.v); self.ch.send(b); } } } fn notify(ch: Chan, v: @mut bool) -> notify { notify { ch: ch, v: v } } fn joinable(f: proc()) -> Port { fn wrapper(c: Chan, f: ||) { let b = @mut false; error!("wrapper: task=%? allocated v=%x", 0, ptr::to_unsafe_ptr(&(*b)) as uint); let _r = notify(c, b); f(); *b = true; } let (p, c) = stream(); do task::spawn_unlinked { let ccc = c; wrapper(ccc, f) } p } fn join(port: Port) -> bool { port.recv() } fn supervised() { // Deschedule to make sure the supervisor joins before we // fail. This is currently not needed because the supervisor // runs first, but I can imagine that changing. error!("supervised task=%?", 0); task::deschedule(); fail!(); } fn supervisor() { error!("supervisor task=%?", 0); let t = joinable(supervised); join(t); } pub fn main() { join(joinable(supervisor)); }