1#[cfg(test)]
2mod tests;
3
4use crate::ffi::{c_int, c_void};
5use crate::io::{self, BorrowedCursor, ErrorKind, IoSlice, IoSliceMut};
6use crate::net::{Ipv4Addr, Ipv6Addr, Shutdown, SocketAddr, SocketAddrV4, SocketAddrV6};
7use crate::sys::common::small_c_string::run_with_cstr;
8use crate::sys_common::{AsInner, FromInner};
9use crate::time::Duration;
10use crate::{cmp, fmt, mem, ptr};
11
12cfg_if::cfg_if! {
13 if #[cfg(target_os = "hermit")] {
14 mod hermit;
15 pub use hermit::*;
16 } else if #[cfg(target_os = "solid_asp3")] {
17 mod solid;
18 pub use solid::*;
19 } else if #[cfg(target_family = "unix")] {
20 mod unix;
21 pub use unix::*;
22 } else if #[cfg(all(target_os = "wasi", target_env = "p2"))] {
23 mod wasip2;
24 pub use wasip2::*;
25 } else if #[cfg(target_os = "windows")] {
26 mod windows;
27 pub use windows::*;
28 }
29}
30
31use netc as c;
32
33cfg_if::cfg_if! {
34 if #[cfg(any(
35 target_os = "dragonfly",
36 target_os = "freebsd",
37 target_os = "openbsd",
38 target_os = "netbsd",
39 target_os = "illumos",
40 target_os = "solaris",
41 target_os = "haiku",
42 target_os = "l4re",
43 target_os = "nto",
44 target_os = "nuttx",
45 target_vendor = "apple",
46 ))] {
47 use c::IPV6_JOIN_GROUP as IPV6_ADD_MEMBERSHIP;
48 use c::IPV6_LEAVE_GROUP as IPV6_DROP_MEMBERSHIP;
49 } else {
50 use c::IPV6_ADD_MEMBERSHIP;
51 use c::IPV6_DROP_MEMBERSHIP;
52 }
53}
54
55cfg_if::cfg_if! {
56 if #[cfg(any(
57 target_os = "linux", target_os = "android",
58 target_os = "hurd",
59 target_os = "dragonfly", target_os = "freebsd",
60 target_os = "openbsd", target_os = "netbsd",
61 target_os = "solaris", target_os = "illumos",
62 target_os = "haiku", target_os = "nto"))] {
63 use libc::MSG_NOSIGNAL;
64 } else {
65 const MSG_NOSIGNAL: c_int = 0x0;
66 }
67}
68
69cfg_if::cfg_if! {
70 if #[cfg(any(
71 target_os = "dragonfly", target_os = "freebsd",
72 target_os = "openbsd", target_os = "netbsd",
73 target_os = "solaris", target_os = "illumos",
74 target_os = "nto"))] {
75 use crate::ffi::c_uchar;
76 type IpV4MultiCastType = c_uchar;
77 } else {
78 type IpV4MultiCastType = c_int;
79 }
80}
81
82fn ip_v4_addr_to_c(addr: &Ipv4Addr) -> c::in_addr {
87 c::in_addr { s_addr: u32::from_ne_bytes(addr.octets()) }
90}
91
92fn ip_v6_addr_to_c(addr: &Ipv6Addr) -> c::in6_addr {
93 c::in6_addr { s6_addr: addr.octets() }
94}
95
96fn ip_v4_addr_from_c(addr: c::in_addr) -> Ipv4Addr {
97 Ipv4Addr::from(addr.s_addr.to_ne_bytes())
98}
99
100fn ip_v6_addr_from_c(addr: c::in6_addr) -> Ipv6Addr {
101 Ipv6Addr::from(addr.s6_addr)
102}
103
104fn socket_addr_v4_to_c(addr: &SocketAddrV4) -> c::sockaddr_in {
105 c::sockaddr_in {
106 sin_family: c::AF_INET as c::sa_family_t,
107 sin_port: addr.port().to_be(),
108 sin_addr: ip_v4_addr_to_c(addr.ip()),
109 ..unsafe { mem::zeroed() }
110 }
111}
112
113fn socket_addr_v6_to_c(addr: &SocketAddrV6) -> c::sockaddr_in6 {
114 c::sockaddr_in6 {
115 sin6_family: c::AF_INET6 as c::sa_family_t,
116 sin6_port: addr.port().to_be(),
117 sin6_addr: ip_v6_addr_to_c(addr.ip()),
118 sin6_flowinfo: addr.flowinfo(),
119 sin6_scope_id: addr.scope_id(),
120 ..unsafe { mem::zeroed() }
121 }
122}
123
124fn socket_addr_v4_from_c(addr: c::sockaddr_in) -> SocketAddrV4 {
125 SocketAddrV4::new(ip_v4_addr_from_c(addr.sin_addr), u16::from_be(addr.sin_port))
126}
127
128fn socket_addr_v6_from_c(addr: c::sockaddr_in6) -> SocketAddrV6 {
129 SocketAddrV6::new(
130 ip_v6_addr_from_c(addr.sin6_addr),
131 u16::from_be(addr.sin6_port),
132 addr.sin6_flowinfo,
133 addr.sin6_scope_id,
134 )
135}
136
137#[repr(C)]
142union SocketAddrCRepr {
143 v4: c::sockaddr_in,
144 v6: c::sockaddr_in6,
145}
146
147impl SocketAddrCRepr {
148 fn as_ptr(&self) -> *const c::sockaddr {
149 self as *const _ as *const c::sockaddr
150 }
151}
152
153fn socket_addr_to_c(addr: &SocketAddr) -> (SocketAddrCRepr, c::socklen_t) {
154 match addr {
155 SocketAddr::V4(a) => {
156 let sockaddr = SocketAddrCRepr { v4: socket_addr_v4_to_c(a) };
157 (sockaddr, mem::size_of::<c::sockaddr_in>() as c::socklen_t)
158 }
159 SocketAddr::V6(a) => {
160 let sockaddr = SocketAddrCRepr { v6: socket_addr_v6_to_c(a) };
161 (sockaddr, mem::size_of::<c::sockaddr_in6>() as c::socklen_t)
162 }
163 }
164}
165
166unsafe fn socket_addr_from_c(
167 storage: *const c::sockaddr_storage,
168 len: usize,
169) -> io::Result<SocketAddr> {
170 match (*storage).ss_family as c_int {
171 c::AF_INET => {
172 assert!(len >= mem::size_of::<c::sockaddr_in>());
173 Ok(SocketAddr::V4(socket_addr_v4_from_c(unsafe {
174 *(storage as *const _ as *const c::sockaddr_in)
175 })))
176 }
177 c::AF_INET6 => {
178 assert!(len >= mem::size_of::<c::sockaddr_in6>());
179 Ok(SocketAddr::V6(socket_addr_v6_from_c(unsafe {
180 *(storage as *const _ as *const c::sockaddr_in6)
181 })))
182 }
183 _ => Err(io::const_error!(ErrorKind::InvalidInput, "invalid argument")),
184 }
185}
186
187pub fn setsockopt<T>(
192 sock: &Socket,
193 level: c_int,
194 option_name: c_int,
195 option_value: T,
196) -> io::Result<()> {
197 unsafe {
198 cvt(c::setsockopt(
199 sock.as_raw(),
200 level,
201 option_name,
202 (&raw const option_value) as *const _,
203 mem::size_of::<T>() as c::socklen_t,
204 ))?;
205 Ok(())
206 }
207}
208
209pub fn getsockopt<T: Copy>(sock: &Socket, level: c_int, option_name: c_int) -> io::Result<T> {
210 unsafe {
211 let mut option_value: T = mem::zeroed();
212 let mut option_len = mem::size_of::<T>() as c::socklen_t;
213 cvt(c::getsockopt(
214 sock.as_raw(),
215 level,
216 option_name,
217 (&raw mut option_value) as *mut _,
218 &mut option_len,
219 ))?;
220 Ok(option_value)
221 }
222}
223
224fn sockname<F>(f: F) -> io::Result<SocketAddr>
225where
226 F: FnOnce(*mut c::sockaddr, *mut c::socklen_t) -> c_int,
227{
228 unsafe {
229 let mut storage: c::sockaddr_storage = mem::zeroed();
230 let mut len = mem::size_of_val(&storage) as c::socklen_t;
231 cvt(f((&raw mut storage) as *mut _, &mut len))?;
232 socket_addr_from_c(&storage, len as usize)
233 }
234}
235
236#[cfg(target_os = "android")]
237fn to_ipv6mr_interface(value: u32) -> c_int {
238 value as c_int
239}
240
241#[cfg(not(target_os = "android"))]
242fn to_ipv6mr_interface(value: u32) -> crate::ffi::c_uint {
243 value as crate::ffi::c_uint
244}
245
246pub struct LookupHost {
251 original: *mut c::addrinfo,
252 cur: *mut c::addrinfo,
253 port: u16,
254}
255
256impl LookupHost {
257 pub fn port(&self) -> u16 {
258 self.port
259 }
260}
261
262impl Iterator for LookupHost {
263 type Item = SocketAddr;
264 fn next(&mut self) -> Option<SocketAddr> {
265 loop {
266 unsafe {
267 let cur = self.cur.as_ref()?;
268 self.cur = cur.ai_next;
269 match socket_addr_from_c(cur.ai_addr.cast(), cur.ai_addrlen as usize) {
270 Ok(addr) => return Some(addr),
271 Err(_) => continue,
272 }
273 }
274 }
275 }
276}
277
278unsafe impl Sync for LookupHost {}
279unsafe impl Send for LookupHost {}
280
281impl Drop for LookupHost {
282 fn drop(&mut self) {
283 unsafe { c::freeaddrinfo(self.original) }
284 }
285}
286
287impl TryFrom<&str> for LookupHost {
288 type Error = io::Error;
289
290 fn try_from(s: &str) -> io::Result<LookupHost> {
291 macro_rules! try_opt {
292 ($e:expr, $msg:expr) => {
293 match $e {
294 Some(r) => r,
295 None => return Err(io::const_error!(io::ErrorKind::InvalidInput, $msg)),
296 }
297 };
298 }
299
300 let (host, port_str) = try_opt!(s.rsplit_once(':'), "invalid socket address");
302 let port: u16 = try_opt!(port_str.parse().ok(), "invalid port value");
303 (host, port).try_into()
304 }
305}
306
307impl<'a> TryFrom<(&'a str, u16)> for LookupHost {
308 type Error = io::Error;
309
310 fn try_from((host, port): (&'a str, u16)) -> io::Result<LookupHost> {
311 init();
312
313 run_with_cstr(host.as_bytes(), &|c_host| {
314 let mut hints: c::addrinfo = unsafe { mem::zeroed() };
315 hints.ai_socktype = c::SOCK_STREAM;
316 let mut res = ptr::null_mut();
317 unsafe {
318 cvt_gai(c::getaddrinfo(c_host.as_ptr(), ptr::null(), &hints, &mut res))
319 .map(|_| LookupHost { original: res, cur: res, port })
320 }
321 })
322 }
323}
324
325pub struct TcpStream {
330 inner: Socket,
331}
332
333impl TcpStream {
334 pub fn connect(addr: io::Result<&SocketAddr>) -> io::Result<TcpStream> {
335 let addr = addr?;
336
337 init();
338
339 let sock = Socket::new(addr, c::SOCK_STREAM)?;
340 sock.connect(addr)?;
341 Ok(TcpStream { inner: sock })
342 }
343
344 pub fn connect_timeout(addr: &SocketAddr, timeout: Duration) -> io::Result<TcpStream> {
345 init();
346
347 let sock = Socket::new(addr, c::SOCK_STREAM)?;
348 sock.connect_timeout(addr, timeout)?;
349 Ok(TcpStream { inner: sock })
350 }
351
352 #[inline]
353 pub fn socket(&self) -> &Socket {
354 &self.inner
355 }
356
357 pub fn into_socket(self) -> Socket {
358 self.inner
359 }
360
361 pub fn set_read_timeout(&self, dur: Option<Duration>) -> io::Result<()> {
362 self.inner.set_timeout(dur, c::SO_RCVTIMEO)
363 }
364
365 pub fn set_write_timeout(&self, dur: Option<Duration>) -> io::Result<()> {
366 self.inner.set_timeout(dur, c::SO_SNDTIMEO)
367 }
368
369 pub fn read_timeout(&self) -> io::Result<Option<Duration>> {
370 self.inner.timeout(c::SO_RCVTIMEO)
371 }
372
373 pub fn write_timeout(&self) -> io::Result<Option<Duration>> {
374 self.inner.timeout(c::SO_SNDTIMEO)
375 }
376
377 pub fn peek(&self, buf: &mut [u8]) -> io::Result<usize> {
378 self.inner.peek(buf)
379 }
380
381 pub fn read(&self, buf: &mut [u8]) -> io::Result<usize> {
382 self.inner.read(buf)
383 }
384
385 pub fn read_buf(&self, buf: BorrowedCursor<'_>) -> io::Result<()> {
386 self.inner.read_buf(buf)
387 }
388
389 pub fn read_vectored(&self, bufs: &mut [IoSliceMut<'_>]) -> io::Result<usize> {
390 self.inner.read_vectored(bufs)
391 }
392
393 #[inline]
394 pub fn is_read_vectored(&self) -> bool {
395 self.inner.is_read_vectored()
396 }
397
398 pub fn write(&self, buf: &[u8]) -> io::Result<usize> {
399 let len = cmp::min(buf.len(), <wrlen_t>::MAX as usize) as wrlen_t;
400 let ret = cvt(unsafe {
401 c::send(self.inner.as_raw(), buf.as_ptr() as *const c_void, len, MSG_NOSIGNAL)
402 })?;
403 Ok(ret as usize)
404 }
405
406 pub fn write_vectored(&self, bufs: &[IoSlice<'_>]) -> io::Result<usize> {
407 self.inner.write_vectored(bufs)
408 }
409
410 #[inline]
411 pub fn is_write_vectored(&self) -> bool {
412 self.inner.is_write_vectored()
413 }
414
415 pub fn peer_addr(&self) -> io::Result<SocketAddr> {
416 sockname(|buf, len| unsafe { c::getpeername(self.inner.as_raw(), buf, len) })
417 }
418
419 pub fn socket_addr(&self) -> io::Result<SocketAddr> {
420 sockname(|buf, len| unsafe { c::getsockname(self.inner.as_raw(), buf, len) })
421 }
422
423 pub fn shutdown(&self, how: Shutdown) -> io::Result<()> {
424 self.inner.shutdown(how)
425 }
426
427 pub fn duplicate(&self) -> io::Result<TcpStream> {
428 self.inner.duplicate().map(|s| TcpStream { inner: s })
429 }
430
431 pub fn set_linger(&self, linger: Option<Duration>) -> io::Result<()> {
432 self.inner.set_linger(linger)
433 }
434
435 pub fn linger(&self) -> io::Result<Option<Duration>> {
436 self.inner.linger()
437 }
438
439 pub fn set_nodelay(&self, nodelay: bool) -> io::Result<()> {
440 self.inner.set_nodelay(nodelay)
441 }
442
443 pub fn nodelay(&self) -> io::Result<bool> {
444 self.inner.nodelay()
445 }
446
447 pub fn set_ttl(&self, ttl: u32) -> io::Result<()> {
448 setsockopt(&self.inner, c::IPPROTO_IP, c::IP_TTL, ttl as c_int)
449 }
450
451 pub fn ttl(&self) -> io::Result<u32> {
452 let raw: c_int = getsockopt(&self.inner, c::IPPROTO_IP, c::IP_TTL)?;
453 Ok(raw as u32)
454 }
455
456 pub fn take_error(&self) -> io::Result<Option<io::Error>> {
457 self.inner.take_error()
458 }
459
460 pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> {
461 self.inner.set_nonblocking(nonblocking)
462 }
463}
464
465impl AsInner<Socket> for TcpStream {
466 #[inline]
467 fn as_inner(&self) -> &Socket {
468 &self.inner
469 }
470}
471
472impl FromInner<Socket> for TcpStream {
473 fn from_inner(socket: Socket) -> TcpStream {
474 TcpStream { inner: socket }
475 }
476}
477
478impl fmt::Debug for TcpStream {
479 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
480 let mut res = f.debug_struct("TcpStream");
481
482 if let Ok(addr) = self.socket_addr() {
483 res.field("addr", &addr);
484 }
485
486 if let Ok(peer) = self.peer_addr() {
487 res.field("peer", &peer);
488 }
489
490 let name = if cfg!(windows) { "socket" } else { "fd" };
491 res.field(name, &self.inner.as_raw()).finish()
492 }
493}
494
495pub struct TcpListener {
500 inner: Socket,
501}
502
503impl TcpListener {
504 pub fn bind(addr: io::Result<&SocketAddr>) -> io::Result<TcpListener> {
505 let addr = addr?;
506
507 init();
508
509 let sock = Socket::new(addr, c::SOCK_STREAM)?;
510
511 #[cfg(not(windows))]
519 setsockopt(&sock, c::SOL_SOCKET, c::SO_REUSEADDR, 1 as c_int)?;
520
521 let (addr, len) = socket_addr_to_c(addr);
523 cvt(unsafe { c::bind(sock.as_raw(), addr.as_ptr(), len as _) })?;
524
525 cfg_if::cfg_if! {
526 if #[cfg(target_os = "horizon")] {
527 let backlog = 20;
531 } else if #[cfg(target_os = "haiku")] {
532 let backlog = 32;
535 } else {
536 let backlog = 128;
538 }
539 }
540
541 cvt(unsafe { c::listen(sock.as_raw(), backlog) })?;
543 Ok(TcpListener { inner: sock })
544 }
545
546 #[inline]
547 pub fn socket(&self) -> &Socket {
548 &self.inner
549 }
550
551 pub fn into_socket(self) -> Socket {
552 self.inner
553 }
554
555 pub fn socket_addr(&self) -> io::Result<SocketAddr> {
556 sockname(|buf, len| unsafe { c::getsockname(self.inner.as_raw(), buf, len) })
557 }
558
559 pub fn accept(&self) -> io::Result<(TcpStream, SocketAddr)> {
560 let mut storage: c::sockaddr_storage = unsafe { mem::zeroed() };
561 let mut len = mem::size_of_val(&storage) as c::socklen_t;
562 let sock = self.inner.accept((&raw mut storage) as *mut _, &mut len)?;
563 let addr = unsafe { socket_addr_from_c(&storage, len as usize)? };
564 Ok((TcpStream { inner: sock }, addr))
565 }
566
567 pub fn duplicate(&self) -> io::Result<TcpListener> {
568 self.inner.duplicate().map(|s| TcpListener { inner: s })
569 }
570
571 pub fn set_ttl(&self, ttl: u32) -> io::Result<()> {
572 setsockopt(&self.inner, c::IPPROTO_IP, c::IP_TTL, ttl as c_int)
573 }
574
575 pub fn ttl(&self) -> io::Result<u32> {
576 let raw: c_int = getsockopt(&self.inner, c::IPPROTO_IP, c::IP_TTL)?;
577 Ok(raw as u32)
578 }
579
580 pub fn set_only_v6(&self, only_v6: bool) -> io::Result<()> {
581 setsockopt(&self.inner, c::IPPROTO_IPV6, c::IPV6_V6ONLY, only_v6 as c_int)
582 }
583
584 pub fn only_v6(&self) -> io::Result<bool> {
585 let raw: c_int = getsockopt(&self.inner, c::IPPROTO_IPV6, c::IPV6_V6ONLY)?;
586 Ok(raw != 0)
587 }
588
589 pub fn take_error(&self) -> io::Result<Option<io::Error>> {
590 self.inner.take_error()
591 }
592
593 pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> {
594 self.inner.set_nonblocking(nonblocking)
595 }
596}
597
598impl FromInner<Socket> for TcpListener {
599 fn from_inner(socket: Socket) -> TcpListener {
600 TcpListener { inner: socket }
601 }
602}
603
604impl fmt::Debug for TcpListener {
605 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
606 let mut res = f.debug_struct("TcpListener");
607
608 if let Ok(addr) = self.socket_addr() {
609 res.field("addr", &addr);
610 }
611
612 let name = if cfg!(windows) { "socket" } else { "fd" };
613 res.field(name, &self.inner.as_raw()).finish()
614 }
615}
616
617pub struct UdpSocket {
622 inner: Socket,
623}
624
625impl UdpSocket {
626 pub fn bind(addr: io::Result<&SocketAddr>) -> io::Result<UdpSocket> {
627 let addr = addr?;
628
629 init();
630
631 let sock = Socket::new(addr, c::SOCK_DGRAM)?;
632 let (addr, len) = socket_addr_to_c(addr);
633 cvt(unsafe { c::bind(sock.as_raw(), addr.as_ptr(), len as _) })?;
634 Ok(UdpSocket { inner: sock })
635 }
636
637 #[inline]
638 pub fn socket(&self) -> &Socket {
639 &self.inner
640 }
641
642 pub fn into_socket(self) -> Socket {
643 self.inner
644 }
645
646 pub fn peer_addr(&self) -> io::Result<SocketAddr> {
647 sockname(|buf, len| unsafe { c::getpeername(self.inner.as_raw(), buf, len) })
648 }
649
650 pub fn socket_addr(&self) -> io::Result<SocketAddr> {
651 sockname(|buf, len| unsafe { c::getsockname(self.inner.as_raw(), buf, len) })
652 }
653
654 pub fn recv_from(&self, buf: &mut [u8]) -> io::Result<(usize, SocketAddr)> {
655 self.inner.recv_from(buf)
656 }
657
658 pub fn peek_from(&self, buf: &mut [u8]) -> io::Result<(usize, SocketAddr)> {
659 self.inner.peek_from(buf)
660 }
661
662 pub fn send_to(&self, buf: &[u8], dst: &SocketAddr) -> io::Result<usize> {
663 let len = cmp::min(buf.len(), <wrlen_t>::MAX as usize) as wrlen_t;
664 let (dst, dstlen) = socket_addr_to_c(dst);
665 let ret = cvt(unsafe {
666 c::sendto(
667 self.inner.as_raw(),
668 buf.as_ptr() as *const c_void,
669 len,
670 MSG_NOSIGNAL,
671 dst.as_ptr(),
672 dstlen,
673 )
674 })?;
675 Ok(ret as usize)
676 }
677
678 pub fn duplicate(&self) -> io::Result<UdpSocket> {
679 self.inner.duplicate().map(|s| UdpSocket { inner: s })
680 }
681
682 pub fn set_read_timeout(&self, dur: Option<Duration>) -> io::Result<()> {
683 self.inner.set_timeout(dur, c::SO_RCVTIMEO)
684 }
685
686 pub fn set_write_timeout(&self, dur: Option<Duration>) -> io::Result<()> {
687 self.inner.set_timeout(dur, c::SO_SNDTIMEO)
688 }
689
690 pub fn read_timeout(&self) -> io::Result<Option<Duration>> {
691 self.inner.timeout(c::SO_RCVTIMEO)
692 }
693
694 pub fn write_timeout(&self) -> io::Result<Option<Duration>> {
695 self.inner.timeout(c::SO_SNDTIMEO)
696 }
697
698 pub fn set_broadcast(&self, broadcast: bool) -> io::Result<()> {
699 setsockopt(&self.inner, c::SOL_SOCKET, c::SO_BROADCAST, broadcast as c_int)
700 }
701
702 pub fn broadcast(&self) -> io::Result<bool> {
703 let raw: c_int = getsockopt(&self.inner, c::SOL_SOCKET, c::SO_BROADCAST)?;
704 Ok(raw != 0)
705 }
706
707 pub fn set_multicast_loop_v4(&self, multicast_loop_v4: bool) -> io::Result<()> {
708 setsockopt(
709 &self.inner,
710 c::IPPROTO_IP,
711 c::IP_MULTICAST_LOOP,
712 multicast_loop_v4 as IpV4MultiCastType,
713 )
714 }
715
716 pub fn multicast_loop_v4(&self) -> io::Result<bool> {
717 let raw: IpV4MultiCastType = getsockopt(&self.inner, c::IPPROTO_IP, c::IP_MULTICAST_LOOP)?;
718 Ok(raw != 0)
719 }
720
721 pub fn set_multicast_ttl_v4(&self, multicast_ttl_v4: u32) -> io::Result<()> {
722 setsockopt(
723 &self.inner,
724 c::IPPROTO_IP,
725 c::IP_MULTICAST_TTL,
726 multicast_ttl_v4 as IpV4MultiCastType,
727 )
728 }
729
730 pub fn multicast_ttl_v4(&self) -> io::Result<u32> {
731 let raw: IpV4MultiCastType = getsockopt(&self.inner, c::IPPROTO_IP, c::IP_MULTICAST_TTL)?;
732 Ok(raw as u32)
733 }
734
735 pub fn set_multicast_loop_v6(&self, multicast_loop_v6: bool) -> io::Result<()> {
736 setsockopt(&self.inner, c::IPPROTO_IPV6, c::IPV6_MULTICAST_LOOP, multicast_loop_v6 as c_int)
737 }
738
739 pub fn multicast_loop_v6(&self) -> io::Result<bool> {
740 let raw: c_int = getsockopt(&self.inner, c::IPPROTO_IPV6, c::IPV6_MULTICAST_LOOP)?;
741 Ok(raw != 0)
742 }
743
744 pub fn join_multicast_v4(&self, multiaddr: &Ipv4Addr, interface: &Ipv4Addr) -> io::Result<()> {
745 let mreq = c::ip_mreq {
746 imr_multiaddr: ip_v4_addr_to_c(multiaddr),
747 imr_interface: ip_v4_addr_to_c(interface),
748 };
749 setsockopt(&self.inner, c::IPPROTO_IP, c::IP_ADD_MEMBERSHIP, mreq)
750 }
751
752 pub fn join_multicast_v6(&self, multiaddr: &Ipv6Addr, interface: u32) -> io::Result<()> {
753 let mreq = c::ipv6_mreq {
754 ipv6mr_multiaddr: ip_v6_addr_to_c(multiaddr),
755 ipv6mr_interface: to_ipv6mr_interface(interface),
756 };
757 setsockopt(&self.inner, c::IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, mreq)
758 }
759
760 pub fn leave_multicast_v4(&self, multiaddr: &Ipv4Addr, interface: &Ipv4Addr) -> io::Result<()> {
761 let mreq = c::ip_mreq {
762 imr_multiaddr: ip_v4_addr_to_c(multiaddr),
763 imr_interface: ip_v4_addr_to_c(interface),
764 };
765 setsockopt(&self.inner, c::IPPROTO_IP, c::IP_DROP_MEMBERSHIP, mreq)
766 }
767
768 pub fn leave_multicast_v6(&self, multiaddr: &Ipv6Addr, interface: u32) -> io::Result<()> {
769 let mreq = c::ipv6_mreq {
770 ipv6mr_multiaddr: ip_v6_addr_to_c(multiaddr),
771 ipv6mr_interface: to_ipv6mr_interface(interface),
772 };
773 setsockopt(&self.inner, c::IPPROTO_IPV6, IPV6_DROP_MEMBERSHIP, mreq)
774 }
775
776 pub fn set_ttl(&self, ttl: u32) -> io::Result<()> {
777 setsockopt(&self.inner, c::IPPROTO_IP, c::IP_TTL, ttl as c_int)
778 }
779
780 pub fn ttl(&self) -> io::Result<u32> {
781 let raw: c_int = getsockopt(&self.inner, c::IPPROTO_IP, c::IP_TTL)?;
782 Ok(raw as u32)
783 }
784
785 pub fn take_error(&self) -> io::Result<Option<io::Error>> {
786 self.inner.take_error()
787 }
788
789 pub fn set_nonblocking(&self, nonblocking: bool) -> io::Result<()> {
790 self.inner.set_nonblocking(nonblocking)
791 }
792
793 pub fn recv(&self, buf: &mut [u8]) -> io::Result<usize> {
794 self.inner.read(buf)
795 }
796
797 pub fn peek(&self, buf: &mut [u8]) -> io::Result<usize> {
798 self.inner.peek(buf)
799 }
800
801 pub fn send(&self, buf: &[u8]) -> io::Result<usize> {
802 let len = cmp::min(buf.len(), <wrlen_t>::MAX as usize) as wrlen_t;
803 let ret = cvt(unsafe {
804 c::send(self.inner.as_raw(), buf.as_ptr() as *const c_void, len, MSG_NOSIGNAL)
805 })?;
806 Ok(ret as usize)
807 }
808
809 pub fn connect(&self, addr: io::Result<&SocketAddr>) -> io::Result<()> {
810 let (addr, len) = socket_addr_to_c(addr?);
811 cvt_r(|| unsafe { c::connect(self.inner.as_raw(), addr.as_ptr(), len) }).map(drop)
812 }
813}
814
815impl FromInner<Socket> for UdpSocket {
816 fn from_inner(socket: Socket) -> UdpSocket {
817 UdpSocket { inner: socket }
818 }
819}
820
821impl fmt::Debug for UdpSocket {
822 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
823 let mut res = f.debug_struct("UdpSocket");
824
825 if let Ok(addr) = self.socket_addr() {
826 res.field("addr", &addr);
827 }
828
829 let name = if cfg!(windows) { "socket" } else { "fd" };
830 res.field(name, &self.inner.as_raw()).finish()
831 }
832}