From 8e7ed8ae21e448c664d42b48091472d86bbbc2c7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Dawid=20Ci=C4=99=C5=BCarkiewicz?= Date: Fri, 4 Nov 2016 21:48:34 -0700 Subject: Add some benchmarks. To be moved into separate repo. --- benches/latency.rs | 105 +++++++++++++++++++++++++++++++++++++++++++++++++++++ benches/mio-ops.rs | 57 +++++++++++++++++++++++++++++ 2 files changed, 162 insertions(+) create mode 100644 benches/latency.rs create mode 100644 benches/mio-ops.rs (limited to 'benches') diff --git a/benches/latency.rs b/benches/latency.rs new file mode 100644 index 00000000..a4b91bb5 --- /dev/null +++ b/benches/latency.rs @@ -0,0 +1,105 @@ +#![feature(test)] + +extern crate test; +extern crate futures; +#[macro_use] +extern crate tokio_core; + +use std::io; +use std::sync::Arc; +use std::sync::atomic::{AtomicBool, Ordering}; +use std::net::SocketAddr; + +use futures::{Future, Poll}; +use tokio_core::net::UdpSocket; +use tokio_core::reactor::Core; + +use test::Bencher; +use std::thread; +use std::time::Duration; + +/// UDP echo server +struct Server { + socket : UdpSocket, + buf : Vec, + to_send : Option<(usize, SocketAddr)>, + stop : Arc, +} + +impl Server { + fn new(s : UdpSocket, stop : Arc) -> Self { + Server { + socket: s, + to_send: None, + buf: vec![0u8; 1600], + stop: stop, + } + } +} + +impl Future for Server { + type Item = (); + type Error = io::Error; + + fn poll(&mut self) -> Poll<(), io::Error> { + loop { + if self.stop.load(Ordering::SeqCst) { + return Ok(futures::Async::Ready(())) + } + + if let Some((size, peer)) = self.to_send.take() { + match self.socket.send_to(&self.buf[..size], &peer) { + Err(e) => { + self.to_send = Some((size, peer)); + return try_nb!(Err(e)); + }, + Ok(_) => { } + } + } + + self.to_send = Some( + try_nb!(self.socket.recv_from(&mut self.buf)) + ); + } + } +} +#[bench] +fn udp_echo_latency(b: &mut Bencher) { + let server_addr= "127.0.0.1:7398".to_string(); + let server_addr = server_addr.parse::().unwrap(); + let client_addr= "127.0.0.1:7399".to_string(); + let client_addr = client_addr.parse::().unwrap(); + + let stop = Arc::new(AtomicBool::new(false)); + let stop2 = stop.clone(); + + let child = thread::spawn(move || { + let mut l = Core::new().unwrap(); + let handle = l.handle(); + + let socket = tokio_core::net::UdpSocket::bind(&server_addr, &handle).unwrap(); + + let server = Server::new(socket, stop); + + l.run(server).unwrap(); + }); + + // TODO: More reliable way to bind server socket and start server + // first + thread::sleep(Duration::from_millis(100)); + + let client = std::net::UdpSocket::bind(client_addr).unwrap(); + + let mut buf = [0u8; 1000]; + b.iter(|| { + client.send_to(&buf, &server_addr).unwrap(); + let _ = client.recv_from(&mut buf).unwrap(); + }); + + // Stop the server; TODO: Use better method + stop2.store(true, Ordering::SeqCst); + thread::sleep(Duration::from_millis(1)); + client.send_to(&buf, &server_addr).unwrap(); + + child.join().unwrap(); +} diff --git a/benches/mio-ops.rs b/benches/mio-ops.rs new file mode 100644 index 00000000..43de5f77 --- /dev/null +++ b/benches/mio-ops.rs @@ -0,0 +1,57 @@ +// Measure cost of different operations +// to get a sense of performance tradeoffs +#![feature(test)] + +extern crate test; +extern crate mio; + +use test::Bencher; + +use mio::tcp::TcpListener; +use mio::{Token, Ready, PollOpt}; + + +#[bench] +fn mio_register_deregister(b: &mut Bencher) { + let addr = "127.0.0.1:13265".parse().unwrap(); + // Setup the server socket + let sock = TcpListener::bind(&addr).unwrap(); + let poll = mio::Poll::new().unwrap(); + + const CLIENT: Token = Token(1); + + b.iter(|| { + poll.register(&sock, CLIENT, Ready::readable(), + PollOpt::edge()).unwrap(); + poll.deregister(&sock).unwrap(); + }); +} + +#[bench] +fn mio_reregister(b: &mut Bencher) { + let addr = "127.0.0.1:13265".parse().unwrap(); + // Setup the server socket + let sock = TcpListener::bind(&addr).unwrap(); + let poll = mio::Poll::new().unwrap(); + + const CLIENT: Token = Token(1); + poll.register(&sock, CLIENT, Ready::readable(), + PollOpt::edge()).unwrap(); + + b.iter(|| { + poll.reregister(&sock, CLIENT, Ready::readable(), + PollOpt::edge()).unwrap(); + }); + poll.deregister(&sock).unwrap(); +} + +#[bench] +fn mio_poll(b: &mut Bencher) { + let poll = mio::Poll::new().unwrap(); + let timeout = std::time::Duration::new(0, 0); + let mut events = mio::Events::with_capacity(1024); + + b.iter(|| { + poll.poll(&mut events, Some(timeout)).unwrap(); + }); +} -- cgit v1.2.3