05e3248a79
This can break code that looked like: impl Foo for Box<Any> { fn f(&self) { ... } } let x: Box<Any + Send> = ...; x.f(); Change such code to: impl Foo for Box<Any> { fn f(&self) { ... } } let x: Box<Any> = ...; x.f(); That is, upcast before calling methods. This is a conservative solution to #5781. A more proper treatment (see the xfail'd `trait-contravariant-self.rs`) would take variance into account. This change fixes the soundness hole. Some library changes had to be made to make this work. In particular, `Box<Any>` is no longer showable, and only `Box<Any+Send>` is showable. Eventually, this restriction can be lifted; for now, it does not prove too onerous, because `Any` is only used for propagating the result of task failure. This patch also adds a test for the variance inference work in #12828, which accidentally landed as part of DST. Closes #5781. [breaking-change]
198 lines
5.2 KiB
Rust
198 lines
5.2 KiB
Rust
// Copyright 2013 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.
|
|
|
|
use clone::Clone;
|
|
use cmp;
|
|
use collections::Collection;
|
|
use comm::{Sender, Receiver};
|
|
use io;
|
|
use option::{None, Option, Some};
|
|
use result::{Ok, Err};
|
|
use slice::{bytes, MutableVector, ImmutableVector};
|
|
use str::StrSlice;
|
|
use super::{Reader, Writer, IoResult};
|
|
use vec::Vec;
|
|
|
|
/// Allows reading from a rx.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```
|
|
/// use std::io::ChanReader;
|
|
///
|
|
/// let (tx, rx) = channel();
|
|
/// # drop(tx);
|
|
/// let mut reader = ChanReader::new(rx);
|
|
///
|
|
/// let mut buf = [0u8, ..100];
|
|
/// match reader.read(buf) {
|
|
/// Ok(nread) => println!("Read {} bytes", nread),
|
|
/// Err(e) => println!("read error: {}", e),
|
|
/// }
|
|
/// ```
|
|
pub struct ChanReader {
|
|
buf: Option<Vec<u8>>, // A buffer of bytes received but not consumed.
|
|
pos: uint, // How many of the buffered bytes have already be consumed.
|
|
rx: Receiver<Vec<u8>>, // The Receiver to pull data from.
|
|
closed: bool, // Whether the channel this Receiver connects to has been closed.
|
|
}
|
|
|
|
impl ChanReader {
|
|
/// Wraps a `Port` in a `ChanReader` structure
|
|
pub fn new(rx: Receiver<Vec<u8>>) -> ChanReader {
|
|
ChanReader {
|
|
buf: None,
|
|
pos: 0,
|
|
rx: rx,
|
|
closed: false,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Reader for ChanReader {
|
|
fn read(&mut self, buf: &mut [u8]) -> IoResult<uint> {
|
|
let mut num_read = 0;
|
|
loop {
|
|
match self.buf {
|
|
Some(ref prev) => {
|
|
let dst = buf.mut_slice_from(num_read);
|
|
let src = prev.slice_from(self.pos);
|
|
let count = cmp::min(dst.len(), src.len());
|
|
bytes::copy_memory(dst, src.slice_to(count));
|
|
num_read += count;
|
|
self.pos += count;
|
|
},
|
|
None => (),
|
|
};
|
|
if num_read == buf.len() || self.closed {
|
|
break;
|
|
}
|
|
self.pos = 0;
|
|
self.buf = self.rx.recv_opt().ok();
|
|
self.closed = self.buf.is_none();
|
|
}
|
|
if self.closed && num_read == 0 {
|
|
Err(io::standard_error(io::EndOfFile))
|
|
} else {
|
|
Ok(num_read)
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Allows writing to a tx.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```
|
|
/// # #![allow(unused_must_use)]
|
|
/// use std::io::ChanWriter;
|
|
///
|
|
/// let (tx, rx) = channel();
|
|
/// # drop(rx);
|
|
/// let mut writer = ChanWriter::new(tx);
|
|
/// writer.write("hello, world".as_bytes());
|
|
/// ```
|
|
pub struct ChanWriter {
|
|
tx: Sender<Vec<u8>>,
|
|
}
|
|
|
|
impl ChanWriter {
|
|
/// Wraps a channel in a `ChanWriter` structure
|
|
pub fn new(tx: Sender<Vec<u8>>) -> ChanWriter {
|
|
ChanWriter { tx: tx }
|
|
}
|
|
}
|
|
|
|
impl Clone for ChanWriter {
|
|
fn clone(&self) -> ChanWriter {
|
|
ChanWriter { tx: self.tx.clone() }
|
|
}
|
|
}
|
|
|
|
impl Writer for ChanWriter {
|
|
fn write(&mut self, buf: &[u8]) -> IoResult<()> {
|
|
self.tx.send_opt(Vec::from_slice(buf)).map_err(|_| {
|
|
io::IoError {
|
|
kind: io::BrokenPipe,
|
|
desc: "Pipe closed",
|
|
detail: None
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use prelude::*;
|
|
use super::*;
|
|
use io;
|
|
use task;
|
|
|
|
#[test]
|
|
fn test_rx_reader() {
|
|
let (tx, rx) = channel();
|
|
task::spawn(proc() {
|
|
tx.send(vec![1u8, 2u8]);
|
|
tx.send(vec![]);
|
|
tx.send(vec![3u8, 4u8]);
|
|
tx.send(vec![5u8, 6u8]);
|
|
tx.send(vec![7u8, 8u8]);
|
|
});
|
|
|
|
let mut reader = ChanReader::new(rx);
|
|
let mut buf = [0u8, ..3];
|
|
|
|
|
|
assert_eq!(Ok(0), reader.read([]));
|
|
|
|
assert_eq!(Ok(3), reader.read(buf));
|
|
assert_eq!(&[1,2,3], buf.as_slice());
|
|
|
|
assert_eq!(Ok(3), reader.read(buf));
|
|
assert_eq!(&[4,5,6], buf.as_slice());
|
|
|
|
assert_eq!(Ok(2), reader.read(buf));
|
|
assert_eq!(&[7,8,6], buf.as_slice());
|
|
|
|
match reader.read(buf.as_mut_slice()) {
|
|
Ok(..) => fail!(),
|
|
Err(e) => assert_eq!(e.kind, io::EndOfFile),
|
|
}
|
|
assert_eq!(&[7,8,6], buf.as_slice());
|
|
|
|
// Ensure it continues to fail in the same way.
|
|
match reader.read(buf.as_mut_slice()) {
|
|
Ok(..) => fail!(),
|
|
Err(e) => assert_eq!(e.kind, io::EndOfFile),
|
|
}
|
|
assert_eq!(&[7,8,6], buf.as_slice());
|
|
}
|
|
|
|
#[test]
|
|
fn test_chan_writer() {
|
|
let (tx, rx) = channel();
|
|
let mut writer = ChanWriter::new(tx);
|
|
writer.write_be_u32(42).unwrap();
|
|
|
|
let wanted = vec![0u8, 0u8, 0u8, 42u8];
|
|
let got = match task::try(proc() { rx.recv() }) {
|
|
Ok(got) => got,
|
|
Err(_) => fail!(),
|
|
};
|
|
assert_eq!(wanted, got);
|
|
|
|
match writer.write_u8(1) {
|
|
Ok(..) => fail!(),
|
|
Err(e) => assert_eq!(e.kind, io::BrokenPipe),
|
|
}
|
|
}
|
|
}
|