mod.rs 3.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. #![allow(unsafe_code)]
  2. use std::{mem, ptr, io};
  3. use std::os::unix::io::RawFd;
  4. use crate::time::Duration;
  5. #[cfg(target_os = "linux")]
  6. #[path = "linux.rs"]
  7. mod imp;
  8. #[cfg(all(feature = "phy-raw_socket", target_os = "linux"))]
  9. pub mod raw_socket;
  10. #[cfg(all(feature = "phy-raw_socket", not(target_os = "linux"), unix))]
  11. pub mod bpf;
  12. #[cfg(all(feature = "phy-tap_interface", target_os = "linux"))]
  13. pub mod tap_interface;
  14. #[cfg(all(feature = "phy-raw_socket", target_os = "linux"))]
  15. pub use self::raw_socket::RawSocketDesc;
  16. #[cfg(all(feature = "phy-raw_socket", not(target_os = "linux"), unix))]
  17. pub use self::bpf::BpfDevice as RawSocketDesc;
  18. #[cfg(all(feature = "phy-tap_interface", target_os = "linux"))]
  19. pub use self::tap_interface::TapInterfaceDesc;
  20. /// Wait until given file descriptor becomes readable, but no longer than given timeout.
  21. pub fn wait(fd: RawFd, duration: Option<Duration>) -> io::Result<()> {
  22. unsafe {
  23. let mut readfds = {
  24. let mut readfds = mem::MaybeUninit::<libc::fd_set>::uninit();
  25. libc::FD_ZERO(readfds.as_mut_ptr());
  26. libc::FD_SET(fd, readfds.as_mut_ptr());
  27. readfds.assume_init()
  28. };
  29. let mut writefds = {
  30. let mut writefds = mem::MaybeUninit::<libc::fd_set>::uninit();
  31. libc::FD_ZERO(writefds.as_mut_ptr());
  32. writefds.assume_init()
  33. };
  34. let mut exceptfds = {
  35. let mut exceptfds = mem::MaybeUninit::<libc::fd_set>::uninit();
  36. libc::FD_ZERO(exceptfds.as_mut_ptr());
  37. exceptfds.assume_init()
  38. };
  39. let mut timeout = libc::timeval { tv_sec: 0, tv_usec: 0 };
  40. let timeout_ptr =
  41. if let Some(duration) = duration {
  42. timeout.tv_usec = (duration.total_millis() * 1_000) as libc::suseconds_t;
  43. &mut timeout as *mut _
  44. } else {
  45. ptr::null_mut()
  46. };
  47. let res = libc::select(fd + 1, &mut readfds, &mut writefds, &mut exceptfds, timeout_ptr);
  48. if res == -1 { return Err(io::Error::last_os_error()) }
  49. Ok(())
  50. }
  51. }
  52. #[cfg(all(any(feature = "phy-tap_interface", feature = "phy-raw_socket"), unix))]
  53. #[repr(C)]
  54. #[derive(Debug)]
  55. struct ifreq {
  56. ifr_name: [libc::c_char; libc::IF_NAMESIZE],
  57. ifr_data: libc::c_int /* ifr_ifindex or ifr_mtu */
  58. }
  59. #[cfg(all(any(feature = "phy-tap_interface", feature = "phy-raw_socket"), unix))]
  60. fn ifreq_for(name: &str) -> ifreq {
  61. let mut ifreq = ifreq {
  62. ifr_name: [0; libc::IF_NAMESIZE],
  63. ifr_data: 0
  64. };
  65. for (i, byte) in name.as_bytes().iter().enumerate() {
  66. ifreq.ifr_name[i] = *byte as libc::c_char
  67. }
  68. ifreq
  69. }
  70. #[cfg(all(target_os = "linux", any(feature = "phy-tap_interface", feature = "phy-raw_socket")))]
  71. fn ifreq_ioctl(lower: libc::c_int, ifreq: &mut ifreq,
  72. cmd: libc::c_ulong) -> io::Result<libc::c_int> {
  73. unsafe {
  74. let res = libc::ioctl(lower, cmd as _, ifreq as *mut ifreq);
  75. if res == -1 { return Err(io::Error::last_os_error()) }
  76. }
  77. Ok(ifreq.ifr_data)
  78. }