// 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. /*! Runtime support for message passing with protocol enforcement. Pipes consist of two endpoints. One endpoint can send messages and the other can receive messages. The set of legal messages and which directions they can flow at any given point are determined by a protocol. Below is an example protocol. ~~~ proto! pingpong ( ping: send { ping -> pong } pong: recv { pong -> ping } ) ~~~ The `proto!` syntax extension will convert this into a module called `pingpong`, which includes a set of types and functions that can be used to write programs that follow the pingpong protocol. */ /* IMPLEMENTATION NOTES The initial design for this feature is available at: https://github.com/eholk/rust/wiki/Proposal-for-channel-contracts Much of the design in that document is still accurate. There are several components for the pipe implementation. First of all is the syntax extension. To see how that works, it is best see comments in libsyntax/ext/pipes.rs. This module includes two related pieces of the runtime implementation: support for unbounded and bounded protocols. The main difference between the two is the type of the buffer that is carried along in the endpoint data structures. The heart of the implementation is the packet type. It contains a header and a payload field. Much of the code in this module deals with the header field. This is where the synchronization information is stored. In the case of a bounded protocol, the header also includes a pointer to the buffer the packet is contained in. Packets represent a single message in a protocol. The payload field gets instatiated at the type of the message, which is usually an enum generated by the pipe compiler. Packets are conceptually single use, although in bounded protocols they are reused each time around the loop. Packets are usually handled through a send_packet_buffered or recv_packet_buffered object. Each packet is referenced by one send_packet and one recv_packet, and these wrappers enforce that only one end can send and only one end can receive. The structs also include a destructor that marks packets are terminated if the sender or receiver destroys the object before sending or receiving a value. The *_packet_buffered structs take two type parameters. The first is the message type for the current packet (or state). The second represents the type of the whole buffer. For bounded protocols, the protocol compiler generates a struct with a field for each protocol state. This generated struct is used as the buffer type parameter. For unbounded protocols, the buffer is simply one packet, so there is a shorthand struct called send_packet and recv_packet, where the buffer type is just `packet`. Using the same underlying structure for both bounded and unbounded protocols allows for less code duplication. */ // NB: transitionary, de-mode-ing. // tjc: allowing deprecated modes due to function issue, // re-forbid after snapshot #[forbid(deprecated_pattern)]; // Transitional -- needs snapshot #[allow(structural_records)]; use cmp::Eq; use cast::{forget, reinterpret_cast, transmute}; use either::{Either, Left, Right}; use kinds::Owned; use libc; use option; use option::{None, Option, Some, unwrap}; use pipes; use ptr; use private; use task; use vec; #[doc(hidden)] const SPIN_COUNT: uint = 0; macro_rules! move_it ( { $x:expr } => ( unsafe { let y = move *ptr::addr_of(&($x)); move y } ) ) #[doc(hidden)] enum State { Empty, Full, Blocked, Terminated } impl State : Eq { pure fn eq(&self, other: &State) -> bool { ((*self) as uint) == ((*other) as uint) } pure fn ne(&self, other: &State) -> bool { !(*self).eq(other) } } pub struct BufferHeader { // Tracks whether this buffer needs to be freed. We can probably // get away with restricting it to 0 or 1, if we're careful. mut ref_count: int, // We may want a drop, and to be careful about stringing this // thing along. } pub fn BufferHeader() -> BufferHeader { BufferHeader { ref_count: 0 } } // This is for protocols to associate extra data to thread around. #[doc(hidden)] #[cfg(stage0)] type Buffer = { header: BufferHeader, data: T, }; #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] #[cfg(stage3)] pub struct Buffer { header: BufferHeader, data: T, } struct PacketHeader { mut state: State, mut blocked_task: *rust_task, // This is a reinterpret_cast of a ~buffer, that can also be cast // to a buffer_header if need be. mut buffer: *libc::c_void, } fn PacketHeader() -> PacketHeader { PacketHeader { state: Empty, blocked_task: ptr::null(), buffer: ptr::null() } } impl PacketHeader { // Returns the old state. unsafe fn mark_blocked(this: *rust_task) -> State { rustrt::rust_task_ref(this); let old_task = swap_task(&mut self.blocked_task, this); assert old_task.is_null(); swap_state_acq(&mut self.state, Blocked) } unsafe fn unblock() { let old_task = swap_task(&mut self.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::rust_task_deref(old_task) } } match swap_state_acq(&mut self.state, Empty) { Empty | Blocked => (), Terminated => self.state = Terminated, Full => self.state = Full } } // unsafe because this can do weird things to the space/time // continuum. It ends making multiple unique pointers to the same // thing. You'll proobably want to forget them when you're done. unsafe fn buf_header() -> ~BufferHeader { assert self.buffer.is_not_null(); reinterpret_cast(&self.buffer) } fn set_buffer(b: ~Buffer) { unsafe { self.buffer = reinterpret_cast(&b); } } } #[doc(hidden)] #[cfg(stage0)] pub struct Packet { header: PacketHeader, mut payload: Option, } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] pub struct Packet { header: PacketHeader, mut payload: Option, } #[doc(hidden)] pub trait HasBuffer { fn set_buffer(b: *libc::c_void); // FIXME #4421 remove after snapshot fn set_buffer_(b: *libc::c_void); } impl Packet: HasBuffer { fn set_buffer(b: *libc::c_void) { self.header.buffer = b; } // FIXME #4421 remove after snapshot fn set_buffer_(b: *libc::c_void) { self.set_buffer(b) } } #[doc(hidden)] pub fn mk_packet() -> Packet { Packet { header: PacketHeader(), payload: None, } } #[doc(hidden)] #[cfg(stage0)] fn unibuffer() -> ~Buffer> { let b = ~{ header: BufferHeader(), data: Packet { header: PacketHeader(), payload: None, } }; unsafe { b.data.header.buffer = reinterpret_cast(&b); } move b } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] #[cfg(stage3)] fn unibuffer() -> ~Buffer> { let b = ~Buffer { header: BufferHeader(), data: Packet { header: PacketHeader(), payload: None, } }; unsafe { b.data.header.buffer = reinterpret_cast(&b); } move b } #[doc(hidden)] #[cfg(stage0)] pub fn packet() -> *Packet { let b = unibuffer(); let p = ptr::addr_of(&(b.data)); // We'll take over memory management from here. unsafe { forget(move b) } p } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] pub fn packet() -> *Packet { let b = unibuffer(); let p = ptr::addr_of(&(b.data)); // We'll take over memory management from here. unsafe { forget(move b) } p } #[doc(hidden)] pub fn entangle_buffer( buffer: ~Buffer, init: fn(*libc::c_void, x: &T) -> *Packet) -> (SendPacketBuffered, RecvPacketBuffered) { let p = init(unsafe { reinterpret_cast(&buffer) }, &buffer.data); unsafe { forget(move buffer) } (SendPacketBuffered(p), RecvPacketBuffered(p)) } #[abi = "rust-intrinsic"] #[doc(hidden)] extern mod rusti { fn atomic_xchg(dst: &mut int, src: int) -> int; fn atomic_xchg_acq(dst: &mut int, src: int) -> int; fn atomic_xchg_rel(dst: &mut int, src: int) -> int; fn atomic_xadd_acq(dst: &mut int, src: int) -> int; fn atomic_xsub_rel(dst: &mut int, src: int) -> int; } // If I call the rusti versions directly from a polymorphic function, // I get link errors. This is a bug that needs investigated more. #[doc(hidden)] pub fn atomic_xchng_rel(dst: &mut int, src: int) -> int { unsafe { rusti::atomic_xchg_rel(dst, src) } } #[doc(hidden)] pub fn atomic_add_acq(dst: &mut int, src: int) -> int { unsafe { rusti::atomic_xadd_acq(dst, src) } } #[doc(hidden)] pub fn atomic_sub_rel(dst: &mut int, src: int) -> int { unsafe { rusti::atomic_xsub_rel(dst, src) } } #[doc(hidden)] pub fn swap_task(dst: &mut *rust_task, src: *rust_task) -> *rust_task { // It might be worth making both acquire and release versions of // this. unsafe { transmute(rusti::atomic_xchg(transmute(move dst), src as int)) } } #[doc(hidden)] #[allow(non_camel_case_types)] type rust_task = libc::c_void; #[doc(hidden)] extern mod rustrt { #[rust_stack] unsafe fn rust_get_task() -> *rust_task; #[rust_stack] unsafe fn rust_task_ref(task: *rust_task); unsafe fn rust_task_deref(task: *rust_task); #[rust_stack] unsafe fn task_clear_event_reject(task: *rust_task); unsafe fn task_wait_event(this: *rust_task, killed: &mut *libc::c_void) -> bool; unsafe fn task_signal_event(target: *rust_task, event: *libc::c_void); } #[doc(hidden)] fn wait_event(this: *rust_task) -> *libc::c_void { unsafe { let mut event = ptr::null(); let killed = rustrt::task_wait_event(this, &mut event); if killed && !task::failing() { die!(~"killed") } event } } #[doc(hidden)] fn swap_state_acq(dst: &mut State, src: State) -> State { unsafe { transmute(rusti::atomic_xchg_acq(transmute(move dst), src as int)) } } #[doc(hidden)] fn swap_state_rel(dst: &mut State, src: State) -> State { unsafe { transmute(rusti::atomic_xchg_rel(transmute(move dst), src as int)) } } #[doc(hidden)] #[cfg(stage0)] pub unsafe fn get_buffer(p: *PacketHeader) -> ~Buffer { transmute((*p).buf_header()) } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] pub unsafe fn get_buffer(p: *PacketHeader) -> ~Buffer { transmute((*p).buf_header()) } // This could probably be done with SharedMutableState to avoid move_it!(). #[cfg(stage0)] struct BufferResource { buffer: ~Buffer, drop { unsafe { let b = move_it!(self.buffer); //let p = ptr::addr_of(*b); //error!("drop %?", p); let old_count = atomic_sub_rel(&mut b.header.ref_count, 1); //let old_count = atomic_xchng_rel(b.header.ref_count, 0); if old_count == 1 { // The new count is 0. // go go gadget drop glue } else { forget(move b) } } } } #[cfg(stage1)] #[cfg(stage2)] struct BufferResource { buffer: ~Buffer, drop { unsafe { let b = move_it!(self.buffer); //let p = ptr::addr_of(*b); //error!("drop %?", p); let old_count = atomic_sub_rel(&mut b.header.ref_count, 1); //let old_count = atomic_xchng_rel(b.header.ref_count, 0); if old_count == 1 { // The new count is 0. // go go gadget drop glue } else { forget(move b) } } } } #[cfg(stage0)] fn BufferResource(b: ~Buffer) -> BufferResource { //let p = ptr::addr_of(*b); //error!("take %?", p); atomic_add_acq(&mut b.header.ref_count, 1); BufferResource { // tjc: ???? buffer: move b } } #[cfg(stage1)] #[cfg(stage2)] fn BufferResource(b: ~Buffer) -> BufferResource { //let p = ptr::addr_of(*b); //error!("take %?", p); atomic_add_acq(&mut b.header.ref_count, 1); BufferResource { // tjc: ???? buffer: move b } } #[doc(hidden)] #[cfg(stage0)] pub fn send(p: SendPacketBuffered, payload: T) -> bool { let header = p.header(); let p_ = p.unwrap(); let p = unsafe { &*p_ }; assert ptr::addr_of(&(p.header)) == header; assert p.payload.is_none(); p.payload = move Some(move payload); let old_state = swap_state_rel(&mut p.header.state, Full); match old_state { Empty => { // Yay, fastpath. // The receiver will eventually clean this up. //unsafe { forget(p); } return true; } Full => die!(~"duplicate send"), Blocked => { debug!("waking up task for %?", p_); let old_task = swap_task(&mut p.header.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::task_signal_event( old_task, ptr::addr_of(&(p.header)) as *libc::c_void); rustrt::rust_task_deref(old_task); } } // The receiver will eventually clean this up. //unsafe { forget(p); } return true; } Terminated => { // The receiver will never receive this. Rely on drop_glue // to clean everything up. return false; } } } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] pub fn send(p: SendPacketBuffered, payload: T) -> bool { let header = p.header(); let p_ = p.unwrap(); let p = unsafe { &*p_ }; assert ptr::addr_of(&(p.header)) == header; assert p.payload.is_none(); p.payload = move Some(move payload); let old_state = swap_state_rel(&mut p.header.state, Full); match old_state { Empty => { // Yay, fastpath. // The receiver will eventually clean this up. //unsafe { forget(p); } return true; } Full => fail ~"duplicate send", Blocked => { debug!("waking up task for %?", p_); let old_task = swap_task(&mut p.header.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::task_signal_event( old_task, ptr::addr_of(&(p.header)) as *libc::c_void); rustrt::rust_task_deref(old_task); } } // The receiver will eventually clean this up. //unsafe { forget(p); } return true; } Terminated => { // The receiver will never receive this. Rely on drop_glue // to clean everything up. return false; } } } /** Receives a message from a pipe. Fails if the sender closes the connection. */ pub fn recv( p: RecvPacketBuffered) -> T { try_recv(move p).expect("connection closed") } /** Attempts to receive a message from a pipe. Returns `None` if the sender has closed the connection without sending a message, or `Some(T)` if a message was received. */ pub fn try_recv(p: RecvPacketBuffered) -> Option { let p_ = p.unwrap(); let p = unsafe { &*p_ }; struct DropState { p: &PacketHeader, drop { if task::failing() { self.p.state = Terminated; let old_task = swap_task(&mut self.p.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::rust_task_deref(old_task); } } } } }; let _drop_state = DropState { p: &p.header }; // optimistic path match p.header.state { Full => { let mut payload = None; payload <-> p.payload; p.header.state = Empty; return Some(option::unwrap(move payload)) }, Terminated => return None, _ => {} } // regular path let this = unsafe { rustrt::rust_get_task() }; unsafe { rustrt::task_clear_event_reject(this); rustrt::rust_task_ref(this); }; debug!("blocked = %x this = %x", p.header.blocked_task as uint, this as uint); let old_task = swap_task(&mut p.header.blocked_task, this); debug!("blocked = %x this = %x old_task = %x", p.header.blocked_task as uint, this as uint, old_task as uint); assert old_task.is_null(); let mut first = true; let mut count = SPIN_COUNT; loop { unsafe { rustrt::task_clear_event_reject(this); } let old_state = swap_state_acq(&mut p.header.state, Blocked); match old_state { Empty => { debug!("no data available on %?, going to sleep.", p_); if count == 0 { wait_event(this); } else { count -= 1; // FIXME (#524): Putting the yield here destroys a lot // of the benefit of spinning, since we still go into // the scheduler at every iteration. However, without // this everything spins too much because we end up // sometimes blocking the thing we are waiting on. task::yield(); } debug!("woke up, p.state = %?", copy p.header.state); } Blocked => if first { die!(~"blocking on already blocked packet") }, Full => { let mut payload = None; payload <-> p.payload; let old_task = swap_task(&mut p.header.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::rust_task_deref(old_task); } } p.header.state = Empty; return Some(option::unwrap(move payload)) } Terminated => { // This assert detects when we've accidentally unsafely // casted too big of a number to a state. assert old_state == Terminated; let old_task = swap_task(&mut p.header.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::rust_task_deref(old_task); } } return None; } } first = false; } } /// Returns true if messages are available. pub pure fn peek(p: &RecvPacketBuffered) -> bool { match unsafe {(*p.header()).state} { Empty | Terminated => false, Blocked => die!(~"peeking on blocked packet"), Full => true } } impl RecvPacketBuffered: Peekable { pure fn peek() -> bool { peek(&self) } } #[doc(hidden)] fn sender_terminate(p: *Packet) { let p = unsafe { &*p }; match swap_state_rel(&mut p.header.state, Terminated) { Empty => { // The receiver will eventually clean up. } Blocked => { // wake up the target let old_task = swap_task(&mut p.header.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::task_signal_event( old_task, ptr::addr_of(&(p.header)) as *libc::c_void); rustrt::rust_task_deref(old_task); } } // The receiver will eventually clean up. } Full => { // This is impossible die!(~"you dun goofed") } Terminated => { assert p.header.blocked_task.is_null(); // I have to clean up, use drop_glue } } } #[doc(hidden)] fn receiver_terminate(p: *Packet) { let p = unsafe { &*p }; match swap_state_rel(&mut p.header.state, Terminated) { Empty => { assert p.header.blocked_task.is_null(); // the sender will clean up } Blocked => { let old_task = swap_task(&mut p.header.blocked_task, ptr::null()); if !old_task.is_null() { unsafe { rustrt::rust_task_deref(old_task); assert old_task == rustrt::rust_get_task(); } } } Terminated | Full => { assert p.header.blocked_task.is_null(); // I have to clean up, use drop_glue } } } /** Returns when one of the packet headers reports data is available. This function is primarily intended for building higher level waiting functions, such as `select`, `select2`, etc. It takes a vector slice of packet_headers and returns an index into that vector. The index points to an endpoint that has either been closed by the sender or has a message waiting to be received. */ fn wait_many(pkts: &[T]) -> uint { let this = unsafe { rustrt::rust_get_task() }; unsafe { rustrt::task_clear_event_reject(this); } let mut data_avail = false; let mut ready_packet = pkts.len(); for pkts.eachi |i, p| { unsafe { let p = &*p.header(); let old = p.mark_blocked(this); match old { Full | Terminated => { data_avail = true; ready_packet = i; (*p).state = old; break; } Blocked => die!(~"blocking on blocked packet"), Empty => () } } } while !data_avail { debug!("sleeping on %? packets", pkts.len()); let event = wait_event(this) as *PacketHeader; let pos = vec::position(pkts, |p| p.header() == event); match pos { Some(i) => { ready_packet = i; data_avail = true; } None => debug!("ignoring spurious event, %?", event) } } debug!("%?", pkts[ready_packet]); for pkts.each |p| { unsafe{ (*p.header()).unblock()} } debug!("%?, %?", ready_packet, pkts[ready_packet]); unsafe { assert (*pkts[ready_packet].header()).state == Full || (*pkts[ready_packet].header()).state == Terminated; } ready_packet } /** Receives a message from one of two endpoints. The return value is `left` if the first endpoint received something, or `right` if the second endpoint receives something. In each case, the result includes the other endpoint as well so it can be used again. Below is an example of using `select2`. ~~~ match select2(a, b) { left((none, b)) { // endpoint a was closed. } right((a, none)) { // endpoint b was closed. } left((Some(_), b)) { // endpoint a received a message } right(a, Some(_)) { // endpoint b received a message. } } ~~~ Sometimes messages will be available on both endpoints at once. In this case, `select2` may return either `left` or `right`. */ pub fn select2( a: RecvPacketBuffered, b: RecvPacketBuffered) -> Either<(Option, RecvPacketBuffered), (RecvPacketBuffered, Option)> { let i = wait_many([a.header(), b.header()]); match i { 0 => Left((try_recv(move a), move b)), 1 => Right((move a, try_recv(move b))), _ => die!(~"select2 return an invalid packet") } } #[doc(hidden)] trait Selectable { pure fn header() -> *PacketHeader; } impl *PacketHeader: Selectable { pure fn header() -> *PacketHeader { self } } /// Returns the index of an endpoint that is ready to receive. pub fn selecti(endpoints: &[T]) -> uint { wait_many(endpoints) } /// Returns 0 or 1 depending on which endpoint is ready to receive pub fn select2i(a: &A, b: &B) -> Either<(), ()> { match wait_many([a.header(), b.header()]) { 0 => Left(()), 1 => Right(()), _ => die!(~"wait returned unexpected index") } } /** Waits on a set of endpoints. Returns a message, its index, and a list of the remaining endpoints. */ pub fn select(endpoints: ~[RecvPacketBuffered]) -> (uint, Option, ~[RecvPacketBuffered]) { let ready = wait_many(endpoints.map(|p| p.header())); let mut remaining = move endpoints; let port = remaining.swap_remove(ready); let result = try_recv(move port); (ready, move result, move remaining) } /** The sending end of a pipe. It can be used to send exactly one message. */ #[cfg(stage0)] pub type SendPacket = SendPacketBuffered>; #[cfg(stage1)] #[cfg(stage2)] pub type SendPacket = SendPacketBuffered>; #[doc(hidden)] #[cfg(stage0)] pub fn SendPacket(p: *Packet) -> SendPacket { SendPacketBuffered(p) } #[cfg(stage1)] #[cfg(stage2)] pub fn SendPacket(p: *Packet) -> SendPacket { SendPacketBuffered(p) } #[cfg(stage0)] pub struct SendPacketBuffered { mut p: Option<*Packet>, mut buffer: Option>, drop { //if self.p != none { // debug!("drop send %?", option::get(self.p)); //} if self.p != None { let mut p = None; p <-> self.p; sender_terminate(option::unwrap(move p)) } //unsafe { error!("send_drop: %?", // if self.buffer == none { // "none" // } else { "some" }); } } } #[cfg(stage1)] #[cfg(stage2)] pub struct SendPacketBuffered { mut p: Option<*Packet>, mut buffer: Option>, } impl SendPacketBuffered : ::ops::Drop { fn finalize(&self) { //if self.p != none { // debug!("drop send %?", option::get(self.p)); //} if self.p != None { let mut p = None; p <-> self.p; sender_terminate(option::unwrap(move p)) } //unsafe { error!("send_drop: %?", // if self.buffer == none { // "none" // } else { "some" }); } } } #[cfg(stage0)] pub fn SendPacketBuffered(p: *Packet) -> SendPacketBuffered { //debug!("take send %?", p); SendPacketBuffered { p: Some(p), buffer: unsafe { Some(BufferResource( get_buffer(ptr::addr_of(&((*p).header))))) } } } #[cfg(stage1)] #[cfg(stage2)] pub fn SendPacketBuffered(p: *Packet) -> SendPacketBuffered { //debug!("take send %?", p); SendPacketBuffered { p: Some(p), buffer: unsafe { Some(BufferResource( get_buffer(ptr::addr_of(&((*p).header))))) } } } #[cfg(stage0)] impl SendPacketBuffered { fn unwrap() -> *Packet { let mut p = None; p <-> self.p; option::unwrap(move p) } pure fn header() -> *PacketHeader { match self.p { Some(packet) => unsafe { let packet = &*packet; let header = ptr::addr_of(&(packet.header)); //forget(packet); header }, None => die!(~"packet already consumed") } } fn reuse_buffer() -> BufferResource { //error!("send reuse_buffer"); let mut tmp = None; tmp <-> self.buffer; option::unwrap(move tmp) } } #[cfg(stage1)] #[cfg(stage2)] impl SendPacketBuffered { fn unwrap() -> *Packet { let mut p = None; p <-> self.p; option::unwrap(move p) } pure fn header() -> *PacketHeader { match self.p { Some(packet) => unsafe { let packet = &*packet; let header = ptr::addr_of(&(packet.header)); //forget(packet); header }, None => die!(~"packet already consumed") } } fn reuse_buffer() -> BufferResource { //error!("send reuse_buffer"); let mut tmp = None; tmp <-> self.buffer; option::unwrap(move tmp) } } /// Represents the receive end of a pipe. It can receive exactly one /// message. #[cfg(stage0)] pub type RecvPacket = RecvPacketBuffered>; #[cfg(stage1)] #[cfg(stage2)] pub type RecvPacket = RecvPacketBuffered>; #[doc(hidden)] #[cfg(stage0)] pub fn RecvPacket(p: *Packet) -> RecvPacket { RecvPacketBuffered(p) } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] pub fn RecvPacket(p: *Packet) -> RecvPacket { RecvPacketBuffered(p) } #[cfg(stage0)] pub struct RecvPacketBuffered { mut p: Option<*Packet>, mut buffer: Option>, drop { //if self.p != none { // debug!("drop recv %?", option::get(self.p)); //} if self.p != None { let mut p = None; p <-> self.p; receiver_terminate(option::unwrap(move p)) } //unsafe { error!("recv_drop: %?", // if self.buffer == none { // "none" // } else { "some" }); } } } #[cfg(stage1)] #[cfg(stage2)] pub struct RecvPacketBuffered { mut p: Option<*Packet>, mut buffer: Option>, } impl RecvPacketBuffered : ::ops::Drop { fn finalize(&self) { //if self.p != none { // debug!("drop recv %?", option::get(self.p)); //} if self.p != None { let mut p = None; p <-> self.p; receiver_terminate(option::unwrap(move p)) } //unsafe { error!("recv_drop: %?", // if self.buffer == none { // "none" // } else { "some" }); } } } impl RecvPacketBuffered { fn unwrap() -> *Packet { let mut p = None; p <-> self.p; option::unwrap(move p) } fn reuse_buffer() -> BufferResource { //error!("recv reuse_buffer"); let mut tmp = None; tmp <-> self.buffer; option::unwrap(move tmp) } } impl RecvPacketBuffered : Selectable { pure fn header() -> *PacketHeader { match self.p { Some(packet) => unsafe { let packet = &*packet; let header = ptr::addr_of(&(packet.header)); //forget(packet); header }, None => die!(~"packet already consumed") } } } #[cfg(stage0)] pub fn RecvPacketBuffered(p: *Packet) -> RecvPacketBuffered { //debug!("take recv %?", p); RecvPacketBuffered { p: Some(p), buffer: unsafe { Some(BufferResource( get_buffer(ptr::addr_of(&((*p).header))))) } } } #[cfg(stage1)] #[cfg(stage2)] pub fn RecvPacketBuffered(p: *Packet) -> RecvPacketBuffered { //debug!("take recv %?", p); RecvPacketBuffered { p: Some(p), buffer: unsafe { Some(BufferResource( get_buffer(ptr::addr_of(&((*p).header))))) } } } #[doc(hidden)] #[cfg(stage0)] pub fn entangle() -> (SendPacket, RecvPacket) { let p = packet(); (SendPacket(p), RecvPacket(p)) } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] pub fn entangle() -> (SendPacket, RecvPacket) { let p = packet(); (SendPacket(p), RecvPacket(p)) } /** Spawn a task to provide a service. It takes an initialization function that produces a send and receive endpoint. The send endpoint is returned to the caller and the receive endpoint is passed to the new task. */ pub fn spawn_service( init: extern fn() -> (SendPacketBuffered, RecvPacketBuffered), service: fn~(v: RecvPacketBuffered)) -> SendPacketBuffered { let (client, server) = init(); // This is some nasty gymnastics required to safely move the pipe // into a new task. let server = ~mut Some(move server); do task::spawn |move service, move server| { let mut server_ = None; server_ <-> *server; service(option::unwrap(move server_)) } move client } /** Like `spawn_service_recv`, but for protocols that start in the receive state. */ pub fn spawn_service_recv( init: extern fn() -> (RecvPacketBuffered, SendPacketBuffered), service: fn~(v: SendPacketBuffered)) -> RecvPacketBuffered { let (client, server) = init(); // This is some nasty gymnastics required to safely move the pipe // into a new task. let server = ~mut Some(move server); do task::spawn |move service, move server| { let mut server_ = None; server_ <-> *server; service(option::unwrap(move server_)) } move client } // Streams - Make pipes a little easier in general. proto! streamp ( Open:send { data(T) -> Open } ) /// A trait for things that can send multiple messages. pub trait GenericChan { /// Sends a message. fn send(x: T); } /// Things that can send multiple messages and can detect when the receiver /// is closed pub trait GenericSmartChan { /// Sends a message, or report if the receiver has closed the connection. fn try_send(x: T) -> bool; } /// A trait for things that can receive multiple messages. pub trait GenericPort { /// Receives a message, or fails if the connection closes. fn recv() -> T; /** Receives a message, or returns `none` if the connection is closed or closes. */ fn try_recv() -> Option; } /// Ports that can `peek` pub trait Peekable { /// Returns true if a message is available pure fn peek() -> bool; } #[doc(hidden)] #[cfg(stage0)] struct Chan_ { mut endp: Option> } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] struct Chan_ { mut endp: Option> } /// An endpoint that can send many messages. #[cfg(stage0)] pub enum Chan { Chan_(Chan_) } #[cfg(stage1)] #[cfg(stage2)] pub enum Chan { Chan_(Chan_) } #[doc(hidden)] #[cfg(stage0)] struct Port_ { mut endp: Option>, } #[doc(hidden)] #[cfg(stage1)] #[cfg(stage2)] struct Port_ { mut endp: Option>, } /// An endpoint that can receive many messages. #[cfg(stage0)] pub enum Port { Port_(Port_) } #[cfg(stage1)] #[cfg(stage2)] pub enum Port { Port_(Port_) } /** Creates a `(chan, port)` pair. These allow sending or receiving an unlimited number of messages. */ pub fn stream() -> (Port, Chan) { let (c, s) = streamp::init(); (Port_(Port_ { endp: Some(s) }), Chan_(Chan_{ endp: Some(c) })) } impl Chan: GenericChan { fn send(x: T) { let mut endp = None; endp <-> self.endp; self.endp = Some( streamp::client::data(unwrap(move endp), move x)) } } impl Chan: GenericSmartChan { fn try_send(x: T) -> bool { let mut endp = None; endp <-> self.endp; match move streamp::client::try_data(unwrap(move endp), move x) { Some(move next) => { self.endp = Some(move next); true } None => false } } } impl Port: GenericPort { fn recv() -> T { let mut endp = None; endp <-> self.endp; let streamp::data(x, endp) = pipes::recv(unwrap(move endp)); self.endp = Some(move endp); move x } fn try_recv() -> Option { let mut endp = None; endp <-> self.endp; match move pipes::try_recv(unwrap(move endp)) { Some(streamp::data(move x, move endp)) => { self.endp = Some(move endp); Some(move x) } None => None } } } impl Port: Peekable { pure fn peek() -> bool { unsafe { let mut endp = None; endp <-> self.endp; let peek = match &endp { &Some(ref endp) => pipes::peek(endp), &None => die!(~"peeking empty stream") }; self.endp <-> endp; peek } } } impl Port: Selectable { pure fn header() -> *PacketHeader { unsafe { match self.endp { Some(ref endp) => endp.header(), None => die!(~"peeking empty stream") } } } } /// Treat many ports as one. #[cfg(stage0)] pub struct PortSet { mut ports: ~[pipes::Port], } #[cfg(stage1)] #[cfg(stage2)] pub struct PortSet { mut ports: ~[pipes::Port], } pub fn PortSet() -> PortSet{ PortSet { ports: ~[] } } impl PortSet { fn add(port: pipes::Port) { self.ports.push(move port) } fn chan() -> Chan { let (po, ch) = stream(); self.add(move po); move ch } } impl PortSet : GenericPort { fn try_recv() -> Option { let mut result = None; // we have to swap the ports array so we aren't borrowing // aliasable mutable memory. let mut ports = ~[]; ports <-> self.ports; while result.is_none() && ports.len() > 0 { let i = wait_many(ports); match move ports[i].try_recv() { Some(move m) => { result = Some(move m); } None => { // Remove this port. let _ = ports.swap_remove(i); } } } ports <-> self.ports; move result } fn recv() -> T { self.try_recv().expect("port_set: endpoints closed") } } impl PortSet : Peekable { pure fn peek() -> bool { // It'd be nice to use self.port.each, but that version isn't // pure. for vec::each(self.ports) |p| { if p.peek() { return true } } false } } /// A channel that can be shared between many senders. #[cfg(stage0)] pub type SharedChan = private::Exclusive>; #[cfg(stage1)] #[cfg(stage2)] pub type SharedChan = private::Exclusive>; impl SharedChan: GenericChan { fn send(x: T) { let mut xx = Some(move x); do self.with_imm |chan| { let mut x = None; x <-> xx; chan.send(option::unwrap(move x)) } } } impl SharedChan: GenericSmartChan { fn try_send(x: T) -> bool { let mut xx = Some(move x); do self.with_imm |chan| { let mut x = None; x <-> xx; chan.try_send(option::unwrap(move x)) } } } /// Converts a `chan` into a `shared_chan`. pub fn SharedChan(c: Chan) -> SharedChan { private::exclusive(move c) } /// Receive a message from one of two endpoints. pub trait Select2 { /// Receive a message or return `None` if a connection closes. fn try_select() -> Either, Option>; /// Receive a message or fail if a connection closes. fn select() -> Either; } impl, Right: Selectable GenericPort> (Left, Right): Select2 { fn select() -> Either { match self { (ref lp, ref rp) => match select2i(lp, rp) { Left(()) => Left (lp.recv()), Right(()) => Right(rp.recv()) } } } fn try_select() -> Either, Option> { match self { (ref lp, ref rp) => match select2i(lp, rp) { Left(()) => Left (lp.try_recv()), Right(()) => Right(rp.try_recv()) } } } } proto! oneshot ( Oneshot:send { send(T) -> ! } ) /// The send end of a oneshot pipe. #[cfg(stage0)] pub type ChanOne = oneshot::client::Oneshot; #[cfg(stage1)] #[cfg(stage2)] pub type ChanOne = oneshot::client::Oneshot; /// The receive end of a oneshot pipe. #[cfg(stage0)] pub type PortOne = oneshot::server::Oneshot; #[cfg(stage1)] #[cfg(stage2)] pub type PortOne = oneshot::server::Oneshot; /// Initialiase a (send-endpoint, recv-endpoint) oneshot pipe pair. pub fn oneshot() -> (PortOne, ChanOne) { let (chan, port) = oneshot::init(); (port, chan) } impl PortOne { fn recv(self) -> T { recv_one(self) } fn try_recv(self) -> Option { try_recv_one(self) } } impl ChanOne { fn send(self, data: T) { send_one(self, data) } fn try_send(self, data: T) -> bool { try_send_one(self, data) } } /** * Receive a message from a oneshot pipe, failing if the connection was * closed. */ pub fn recv_one(port: PortOne) -> T { let oneshot::send(message) = recv(move port); move message } /// Receive a message from a oneshot pipe unless the connection was closed. pub fn try_recv_one (port: PortOne) -> Option { let message = try_recv(move port); if message.is_none() { None } else { let oneshot::send(message) = option::unwrap(move message); Some(move message) } } /// Send a message on a oneshot pipe, failing if the connection was closed. pub fn send_one(chan: ChanOne, data: T) { oneshot::client::send(move chan, move data); } /** * Send a message on a oneshot pipe, or return false if the connection was * closed. */ pub fn try_send_one(chan: ChanOne, data: T) -> bool { oneshot::client::try_send(move chan, move data).is_some() } pub mod rt { use option::{None, Option, Some}; // These are used to hide the option constructors from the // compiler because their names are changing pub fn make_some(val: T) -> Option { Some(move val) } pub fn make_none() -> Option { None } } #[cfg(test)] pub mod test { use either::{Either, Left, Right}; use pipes::{Chan, Port, oneshot, recv_one, stream}; use pipes; #[test] pub fn test_select2() { let (p1, c1) = pipes::stream(); let (p2, c2) = pipes::stream(); c1.send(~"abc"); match (move p1, move p2).select() { Right(_) => die!(), _ => () } c2.send(123); } #[test] pub fn test_oneshot() { let (c, p) = oneshot::init(); oneshot::client::send(move c, ()); recv_one(move p) } #[test] fn test_peek_terminated() { let (port, chan): (Port, Chan) = stream(); { // Destroy the channel let _chan = move chan; } assert !port.peek(); } }