summaryrefslogtreecommitdiff
path: root/test/sys/test_socket.rs
diff options
context:
space:
mode:
authorGleb Pomykalov <gleb@lancastr.com>2020-04-08 12:41:56 +0300
committerGleb Pomykalov <gleb@lancastr.com>2020-04-26 03:07:03 +0300
commit833369b2413a44ed0477cf8618e4088e523ca137 (patch)
treec246dfe95339d12f9d157312ce0b044fbda84cfe /test/sys/test_socket.rs
parent490e979518256ae38823fd5f4c6c89005d39da84 (diff)
downloadnix-833369b2413a44ed0477cf8618e4088e523ca137.zip
Support sendmmsg/recvmmsg
Diffstat (limited to 'test/sys/test_socket.rs')
-rw-r--r--test/sys/test_socket.rs134
1 files changed, 134 insertions, 0 deletions
diff --git a/test/sys/test_socket.rs b/test/sys/test_socket.rs
index 9194f4c5..ae5ac66c 100644
--- a/test/sys/test_socket.rs
+++ b/test/sys/test_socket.rs
@@ -304,6 +304,140 @@ mod recvfrom {
.expect("setsockopt UDP_GRO failed");
}
}
+
+ #[cfg(any(
+ target_os = "linux",
+ target_os = "android",
+ target_os = "freebsd",
+ target_os = "openbsd",
+ target_os = "netbsd",
+ ))]
+ #[test]
+ pub fn udp_sendmmsg() {
+ use nix::sys::uio::IoVec;
+
+ let std_sa = SocketAddr::from_str("127.0.0.1:6793").unwrap();
+ let std_sa2 = SocketAddr::from_str("127.0.0.1:6794").unwrap();
+ let inet_addr = InetAddr::from_std(&std_sa);
+ let inet_addr2 = InetAddr::from_std(&std_sa2);
+ let sock_addr = SockAddr::new_inet(inet_addr);
+ let sock_addr2 = SockAddr::new_inet(inet_addr2);
+
+ let rsock = socket(AddressFamily::Inet,
+ SockType::Datagram,
+ SockFlag::empty(),
+ None
+ ).unwrap();
+ bind(rsock, &sock_addr).unwrap();
+ let ssock = socket(
+ AddressFamily::Inet,
+ SockType::Datagram,
+ SockFlag::empty(),
+ None,
+ ).expect("send socket failed");
+
+ let from = sendrecv(rsock, ssock, move |s, m, flags| {
+ let iov = [IoVec::from_slice(m)];
+ let mut msgs = Vec::new();
+ msgs.push(
+ SendMmsgData {
+ iov: &iov,
+ cmsgs: &[],
+ addr: Some(sock_addr),
+ _lt: Default::default(),
+ });
+
+ let batch_size = 15;
+
+ for _ in 0..batch_size {
+ msgs.push(
+ SendMmsgData {
+ iov: &iov,
+ cmsgs: &[],
+ addr: Some(sock_addr2),
+ _lt: Default::default(),
+ }
+ );
+ }
+ sendmmsg(s, msgs.iter(), flags)
+ .map(move |sent_bytes| {
+ assert!(sent_bytes.len() >= 1);
+ for sent in &sent_bytes {
+ assert_eq!(*sent, m.len());
+ }
+ sent_bytes.len()
+ })
+ }, |_, _ | {});
+ // UDP sockets should set the from address
+ assert_eq!(AddressFamily::Inet, from.unwrap().family());
+ }
+
+ #[cfg(any(
+ target_os = "linux",
+ target_os = "android",
+ target_os = "freebsd",
+ target_os = "netbsd",
+ ))]
+ #[test]
+ pub fn udp_recvmmsg() {
+ use nix::sys::uio::IoVec;
+ use nix::sys::socket::{MsgFlags, recvmmsg};
+
+ const NUM_MESSAGES_SENT: usize = 2;
+ const DATA: [u8; 2] = [1,2];
+
+ let std_sa = SocketAddr::from_str("127.0.0.1:6798").unwrap();
+ let inet_addr = InetAddr::from_std(&std_sa);
+ let sock_addr = SockAddr::new_inet(inet_addr);
+
+ let rsock = socket(AddressFamily::Inet,
+ SockType::Datagram,
+ SockFlag::empty(),
+ None
+ ).unwrap();
+ bind(rsock, &sock_addr).unwrap();
+ let ssock = socket(
+ AddressFamily::Inet,
+ SockType::Datagram,
+ SockFlag::empty(),
+ None,
+ ).expect("send socket failed");
+
+ let send_thread = thread::spawn(move || {
+ for _ in 0..NUM_MESSAGES_SENT {
+ sendto(ssock, &DATA[..], &sock_addr, MsgFlags::empty()).unwrap();
+ }
+ });
+
+ let mut msgs = std::collections::LinkedList::new();
+
+ // Buffers to receive exactly `NUM_MESSAGES_SENT` messages
+ let mut receive_buffers = [[0u8; 32]; NUM_MESSAGES_SENT];
+ let iovs: Vec<_> = receive_buffers.iter_mut().map(|buf| {
+ [IoVec::from_mut_slice(&mut buf[..])]
+ }).collect();
+
+ for iov in &iovs {
+ msgs.push_back(RecvMmsgData {
+ iov: iov,
+ cmsg_buffer: None,
+ })
+ };
+
+ let res = recvmmsg(rsock, &mut msgs, MsgFlags::empty(), None).expect("recvmmsg");
+ assert_eq!(res.len(), DATA.len());
+
+ for RecvMsg { address, bytes, .. } in res.into_iter() {
+ assert_eq!(AddressFamily::Inet, address.unwrap().family());
+ assert_eq!(DATA.len(), bytes);
+ }
+
+ for buf in &receive_buffers {
+ assert_eq!(&buf[..DATA.len()], DATA);
+ }
+
+ send_thread.join().unwrap();
+ }
}
// Test error handling of our recvmsg wrapper