1 | //! The BSD sockets API requires us to read the `ss_family` field before we can |
2 | //! interpret the rest of a `sockaddr` produced by the kernel. |
3 | #![allow (unsafe_code)] |
4 | |
5 | use crate::backend::c; |
6 | #[cfg (target_os = "linux" )] |
7 | use crate::net::xdp::SocketAddrXdp; |
8 | use crate::net::{SocketAddrAny, SocketAddrStorage, SocketAddrUnix, SocketAddrV4, SocketAddrV6}; |
9 | use core::mem::size_of; |
10 | |
11 | pub(crate) unsafe fn write_sockaddr( |
12 | addr: &SocketAddrAny, |
13 | storage: *mut SocketAddrStorage, |
14 | ) -> usize { |
15 | match addr { |
16 | SocketAddrAny::V4(v4: &SocketAddrV4) => write_sockaddr_v4(v4, storage), |
17 | SocketAddrAny::V6(v6: &SocketAddrV6) => write_sockaddr_v6(v6, storage), |
18 | SocketAddrAny::Unix(unix: &SocketAddrUnix) => write_sockaddr_unix(unix, storage), |
19 | #[cfg (target_os = "linux" )] |
20 | SocketAddrAny::Xdp(xdp: &SocketAddrXdp) => write_sockaddr_xdp(xdp, storage), |
21 | } |
22 | } |
23 | |
24 | pub(crate) fn encode_sockaddr_v4(v4: &SocketAddrV4) -> c::sockaddr_in { |
25 | c::sockaddr_in { |
26 | sin_family: c::AF_INET as _, |
27 | sin_port: u16::to_be(self:v4.port()), |
28 | sin_addr: c::in_addr { |
29 | s_addr: u32::from_ne_bytes(v4.ip().octets()), |
30 | }, |
31 | __pad: [0_u8; 8], |
32 | } |
33 | } |
34 | |
35 | unsafe fn write_sockaddr_v4(v4: &SocketAddrV4, storage: *mut SocketAddrStorage) -> usize { |
36 | let encoded: sockaddr_in = encode_sockaddr_v4(v4); |
37 | core::ptr::write(dst:storage.cast(), src:encoded); |
38 | size_of::<c::sockaddr_in>() |
39 | } |
40 | |
41 | pub(crate) fn encode_sockaddr_v6(v6: &SocketAddrV6) -> c::sockaddr_in6 { |
42 | c::sockaddr_in6 { |
43 | sin6_family: c::AF_INET6 as _, |
44 | sin6_port: u16::to_be(self:v6.port()), |
45 | sin6_flowinfo: u32::to_be(self:v6.flowinfo()), |
46 | sin6_addr: c::in6_addr { |
47 | in6_u: linux_raw_sys::net::in6_addr__bindgen_ty_1 { |
48 | u6_addr8: v6.ip().octets(), |
49 | }, |
50 | }, |
51 | sin6_scope_id: v6.scope_id(), |
52 | } |
53 | } |
54 | |
55 | unsafe fn write_sockaddr_v6(v6: &SocketAddrV6, storage: *mut SocketAddrStorage) -> usize { |
56 | let encoded: sockaddr_in6 = encode_sockaddr_v6(v6); |
57 | core::ptr::write(dst:storage.cast(), src:encoded); |
58 | size_of::<c::sockaddr_in6>() |
59 | } |
60 | |
61 | unsafe fn write_sockaddr_unix(unix: &SocketAddrUnix, storage: *mut SocketAddrStorage) -> usize { |
62 | core::ptr::write(dst:storage.cast(), src:unix.unix); |
63 | unix.len() |
64 | } |
65 | |
66 | #[cfg (target_os = "linux" )] |
67 | pub(crate) fn encode_sockaddr_xdp(xdp: &SocketAddrXdp) -> c::sockaddr_xdp { |
68 | c::sockaddr_xdp { |
69 | sxdp_family: c::AF_XDP as _, |
70 | sxdp_flags: xdp.flags().bits(), |
71 | sxdp_ifindex: xdp.interface_index(), |
72 | sxdp_queue_id: xdp.queue_id(), |
73 | sxdp_shared_umem_fd: xdp.shared_umem_fd(), |
74 | } |
75 | } |
76 | |
77 | #[cfg (target_os = "linux" )] |
78 | unsafe fn write_sockaddr_xdp(xdp: &SocketAddrXdp, storage: *mut SocketAddrStorage) -> usize { |
79 | let encoded: sockaddr_xdp = encode_sockaddr_xdp(xdp); |
80 | core::ptr::write(dst:storage.cast(), src:encoded); |
81 | size_of::<c::sockaddr_xdp>() |
82 | } |
83 | |