2012-07-24 00:14:43 -05:00
|
|
|
// Compare bounded and unbounded protocol performance.
|
|
|
|
|
|
|
|
// xfail-pretty
|
|
|
|
|
|
|
|
use std;
|
|
|
|
|
2012-09-05 14:32:05 -05:00
|
|
|
use pipes::{spawn_service, recv};
|
|
|
|
use std::time::precise_time_s;
|
2012-07-24 00:14:43 -05:00
|
|
|
|
2012-08-22 20:10:48 -05:00
|
|
|
proto! pingpong (
|
2012-07-24 00:14:43 -05:00
|
|
|
ping: send {
|
|
|
|
ping -> pong
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
|
2012-07-24 00:14:43 -05:00
|
|
|
pong: recv {
|
|
|
|
pong -> ping
|
|
|
|
}
|
2012-08-22 20:10:48 -05:00
|
|
|
)
|
2012-07-24 00:14:43 -05:00
|
|
|
|
2012-08-22 20:10:48 -05:00
|
|
|
proto! pingpong_unbounded (
|
2012-07-24 00:14:43 -05:00
|
|
|
ping: send {
|
|
|
|
ping -> pong
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
|
2012-07-24 00:14:43 -05:00
|
|
|
pong: recv {
|
|
|
|
pong -> ping
|
|
|
|
}
|
|
|
|
|
|
|
|
you_will_never_catch_me: send {
|
|
|
|
never_ever_ever -> you_will_never_catch_me
|
|
|
|
}
|
2012-08-22 20:10:48 -05:00
|
|
|
)
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
// This stuff should go in libcore::pipes
|
2012-08-22 19:47:11 -05:00
|
|
|
macro_rules! move_it (
|
2012-08-06 15:17:35 -05:00
|
|
|
{ $x:expr } => { let t <- *ptr::addr_of($x); t }
|
2012-08-22 19:47:11 -05:00
|
|
|
)
|
2012-07-24 00:14:43 -05:00
|
|
|
|
2012-08-22 19:47:11 -05:00
|
|
|
macro_rules! follow (
|
2012-08-22 19:24:52 -05:00
|
|
|
{
|
2012-07-24 00:14:43 -05:00
|
|
|
$($message:path($($x: ident),+) -> $next:ident $e:expr)+
|
|
|
|
} => (
|
2012-08-06 14:34:08 -05:00
|
|
|
|m| match move m {
|
2012-08-20 14:23:37 -05:00
|
|
|
$(Some($message($($x,)* next)) => {
|
2012-08-06 15:17:35 -05:00
|
|
|
// FIXME (#2329) use regular move here once move out of
|
|
|
|
// enums is supported.
|
|
|
|
let $next = unsafe { move_it!(next) };
|
|
|
|
$e })+
|
|
|
|
_ => { fail }
|
2012-07-24 00:14:43 -05:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
{
|
2012-07-24 00:14:43 -05:00
|
|
|
$($message:path -> $next:ident $e:expr)+
|
|
|
|
} => (
|
2012-08-06 14:34:08 -05:00
|
|
|
|m| match move m {
|
2012-08-20 14:23:37 -05:00
|
|
|
$(Some($message(next)) => {
|
2012-08-06 15:17:35 -05:00
|
|
|
// FIXME (#2329) use regular move here once move out of
|
|
|
|
// enums is supported.
|
|
|
|
let $next = unsafe { move_it!(next) };
|
2012-07-24 00:14:43 -05:00
|
|
|
$e })+
|
2012-08-06 15:17:35 -05:00
|
|
|
_ => { fail }
|
|
|
|
}
|
2012-07-24 00:14:43 -05:00
|
|
|
)
|
2012-08-22 19:47:11 -05:00
|
|
|
)
|
2012-07-24 00:14:43 -05:00
|
|
|
|
2012-09-07 16:52:28 -05:00
|
|
|
fn switch<T: Send, Tb: Send, U>(+endp: pipes::RecvPacketBuffered<T, Tb>,
|
2012-08-20 14:23:37 -05:00
|
|
|
f: fn(+Option<T>) -> U) -> U {
|
2012-07-24 00:14:43 -05:00
|
|
|
f(pipes::try_recv(endp))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Here's the benchmark
|
|
|
|
|
|
|
|
fn bounded(count: uint) {
|
2012-09-07 20:08:21 -05:00
|
|
|
use pingpong::*;
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
let mut ch = do spawn_service(init) |ch| {
|
|
|
|
let mut count = count;
|
|
|
|
let mut ch = ch;
|
|
|
|
while count > 0 {
|
2012-08-22 19:24:52 -05:00
|
|
|
ch = switch(ch, follow! (
|
2012-07-24 00:14:43 -05:00
|
|
|
ping -> next { server::pong(next) }
|
2012-08-22 19:24:52 -05:00
|
|
|
));
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
count -= 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut count = count;
|
|
|
|
while count > 0 {
|
|
|
|
let ch_ = client::ping(ch);
|
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
ch = switch(ch_, follow! (
|
2012-07-24 00:14:43 -05:00
|
|
|
pong -> next { next }
|
2012-08-22 19:24:52 -05:00
|
|
|
));
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
count -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn unbounded(count: uint) {
|
2012-09-07 20:08:21 -05:00
|
|
|
use pingpong_unbounded::*;
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
let mut ch = do spawn_service(init) |ch| {
|
|
|
|
let mut count = count;
|
|
|
|
let mut ch = ch;
|
|
|
|
while count > 0 {
|
2012-08-22 19:24:52 -05:00
|
|
|
ch = switch(ch, follow! (
|
2012-07-24 00:14:43 -05:00
|
|
|
ping -> next { server::pong(next) }
|
2012-08-22 19:24:52 -05:00
|
|
|
));
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
count -= 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut count = count;
|
|
|
|
while count > 0 {
|
|
|
|
let ch_ = client::ping(ch);
|
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
ch = switch(ch_, follow! (
|
2012-07-24 00:14:43 -05:00
|
|
|
pong -> next { next }
|
2012-08-22 19:24:52 -05:00
|
|
|
));
|
2012-07-24 00:14:43 -05:00
|
|
|
|
|
|
|
count -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn timeit(f: fn()) -> float {
|
|
|
|
let start = precise_time_s();
|
|
|
|
f();
|
|
|
|
let stop = precise_time_s();
|
|
|
|
stop - start
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
2012-08-06 15:17:35 -05:00
|
|
|
let count = if os::getenv(~"RUST_BENCH").is_some() {
|
|
|
|
250000
|
|
|
|
} else {
|
|
|
|
100
|
|
|
|
};
|
2012-07-24 00:14:43 -05:00
|
|
|
let bounded = do timeit { bounded(count) };
|
|
|
|
let unbounded = do timeit { unbounded(count) };
|
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
io::println(fmt!("count: %?\n", count));
|
|
|
|
io::println(fmt!("bounded: %? s\t(%? μs/message)",
|
|
|
|
bounded, bounded * 1000000. / (count as float)));
|
|
|
|
io::println(fmt!("unbounded: %? s\t(%? μs/message)",
|
|
|
|
unbounded, unbounded * 1000000. / (count as float)));
|
2012-07-24 00:14:43 -05:00
|
|
|
|
2012-08-22 19:24:52 -05:00
|
|
|
io::println(fmt!("\n\
|
2012-07-24 00:14:43 -05:00
|
|
|
bounded is %?%% faster",
|
2012-08-22 19:24:52 -05:00
|
|
|
(unbounded - bounded) / bounded * 100.));
|
2012-07-24 00:14:43 -05:00
|
|
|
}
|