blob: 2616ad84ff8746d3fa33df2db24559b00e1c04bf [file] [log] [blame]
//@ run-pass
#![allow(unused_must_use)]
// Tests that a heterogeneous list of existential `dyn` types can be put inside an Arc
// and shared between threads as long as all types fulfill Send.
//@ needs-threads
use std::sync::Arc;
use std::sync::mpsc::channel;
use std::thread;
trait Pet {
fn name(&self, blk: Box<dyn FnMut(&str)>);
fn num_legs(&self) -> usize;
fn of_good_pedigree(&self) -> bool;
}
struct Catte {
num_whiskers: usize,
name: String,
}
struct Dogge {
bark_decibels: usize,
tricks_known: usize,
name: String,
}
struct Goldfyshe {
swim_speed: usize,
name: String,
}
impl Pet for Catte {
fn name(&self, mut blk: Box<dyn FnMut(&str)>) { blk(&self.name) }
fn num_legs(&self) -> usize { 4 }
fn of_good_pedigree(&self) -> bool { self.num_whiskers >= 4 }
}
impl Pet for Dogge {
fn name(&self, mut blk: Box<dyn FnMut(&str)>) { blk(&self.name) }
fn num_legs(&self) -> usize { 4 }
fn of_good_pedigree(&self) -> bool {
self.bark_decibels < 70 || self.tricks_known > 20
}
}
impl Pet for Goldfyshe {
fn name(&self, mut blk: Box<dyn FnMut(&str)>) { blk(&self.name) }
fn num_legs(&self) -> usize { 0 }
fn of_good_pedigree(&self) -> bool { self.swim_speed >= 500 }
}
pub fn main() {
let catte = Catte { num_whiskers: 7, name: "alonzo_church".to_string() };
let dogge1 = Dogge {
bark_decibels: 100,
tricks_known: 42,
name: "alan_turing".to_string(),
};
let dogge2 = Dogge {
bark_decibels: 55,
tricks_known: 11,
name: "albert_einstein".to_string(),
};
let fishe = Goldfyshe {
swim_speed: 998,
name: "alec_guinness".to_string(),
};
let arc = Arc::new(vec![
Box::new(catte) as Box<dyn Pet+Sync+Send>,
Box::new(dogge1) as Box<dyn Pet+Sync+Send>,
Box::new(fishe) as Box<dyn Pet+Sync+Send>,
Box::new(dogge2) as Box<dyn Pet+Sync+Send>]);
let (tx1, rx1) = channel();
let arc1 = arc.clone();
let t1 = thread::spawn(move|| { check_legs(arc1); tx1.send(()); });
let (tx2, rx2) = channel();
let arc2 = arc.clone();
let t2 = thread::spawn(move|| { check_names(arc2); tx2.send(()); });
let (tx3, rx3) = channel();
let arc3 = arc.clone();
let t3 = thread::spawn(move|| { check_pedigree(arc3); tx3.send(()); });
rx1.recv();
rx2.recv();
rx3.recv();
t1.join();
t2.join();
t3.join();
}
fn check_legs(arc: Arc<Vec<Box<dyn Pet+Sync+Send>>>) {
let mut legs = 0;
for pet in arc.iter() {
legs += pet.num_legs();
}
assert!(legs == 12);
}
fn check_names(arc: Arc<Vec<Box<dyn Pet+Sync+Send>>>) {
for pet in arc.iter() {
pet.name(Box::new(|name| {
assert!(name.as_bytes()[0] == 'a' as u8 && name.as_bytes()[1] == 'l' as u8);
}))
}
}
fn check_pedigree(arc: Arc<Vec<Box<dyn Pet+Sync+Send>>>) {
for pet in arc.iter() {
assert!(pet.of_good_pedigree());
}
}