1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
use std::net::{IpAddr, SocketAddr}; use std::time::Duration; use rand::random; use socket2::{Domain, Protocol, Socket, Type}; use crate::errors::Error; use crate::packet::{EchoReply, EchoRequest, IcmpV4, IcmpV6, IpV4Packet, ICMP_HEADER_SIZE}; const TOKEN_SIZE: usize = 24; const ECHO_REQUEST_BUFFER_SIZE: usize = ICMP_HEADER_SIZE + TOKEN_SIZE; type Token = [u8; TOKEN_SIZE]; pub fn ping( addr: IpAddr, timeout: Option<Duration>, ttl: Option<u32>, ident: Option<u16>, seq_cnt: Option<u16>, payload: Option<&Token>, ) -> Result<(), Error> { let timeout = match timeout { Some(timeout) => Some(timeout), None => Some(Duration::from_secs(4)), }; let dest = SocketAddr::new(addr, 0); let mut buffer = [0; ECHO_REQUEST_BUFFER_SIZE]; let default_payload: &Token = &random(); let request = EchoRequest { ident: ident.unwrap_or_else(random), seq_cnt: seq_cnt.unwrap_or(1), payload: payload.unwrap_or(default_payload), }; let socket = if dest.is_ipv4() { if request.encode::<IcmpV4>(&mut buffer[..]).is_err() { return Err(Error::Internal); } Socket::new(Domain::ipv4(), Type::raw(), Some(Protocol::icmpv4()))? } else { if request.encode::<IcmpV6>(&mut buffer[..]).is_err() { return Err(Error::Internal); } Socket::new(Domain::ipv6(), Type::raw(), Some(Protocol::icmpv6()))? }; socket.set_ttl(ttl.unwrap_or(64))?; socket.set_write_timeout(timeout)?; socket.send_to(&buffer, &dest.into())?; socket.set_read_timeout(timeout)?; let mut buffer: [u8; 2048] = [0; 2048]; socket.recv_from(&mut buffer)?; let _reply = if dest.is_ipv4() { let ipv4_packet = match IpV4Packet::decode(&buffer) { Ok(packet) => packet, Err(_) => return Err(Error::Internal), }; match EchoReply::decode::<IcmpV4>(ipv4_packet.data) { Ok(reply) => reply, Err(_) => return Err(Error::Internal), } } else { match EchoReply::decode::<IcmpV6>(&buffer) { Ok(reply) => reply, Err(_) => return Err(Error::Internal), } }; Ok(()) }