raw_socket.rs 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. use super::*;
  2. use crate::phy::Medium;
  3. use std::os::unix::io::{AsRawFd, RawFd};
  4. use std::{io, mem};
  5. #[derive(Debug)]
  6. pub struct RawSocketDesc {
  7. protocol: libc::c_short,
  8. lower: libc::c_int,
  9. ifreq: ifreq,
  10. }
  11. impl AsRawFd for RawSocketDesc {
  12. fn as_raw_fd(&self) -> RawFd {
  13. self.lower
  14. }
  15. }
  16. impl RawSocketDesc {
  17. pub fn new(name: &str, medium: Medium) -> io::Result<RawSocketDesc> {
  18. let protocol = match medium {
  19. #[cfg(feature = "medium-ethernet")]
  20. Medium::Ethernet => imp::ETH_P_ALL,
  21. #[cfg(feature = "medium-ip")]
  22. Medium::Ip => imp::ETH_P_ALL,
  23. #[cfg(feature = "medium-ieee802154")]
  24. Medium::Ieee802154 => imp::ETH_P_IEEE802154,
  25. };
  26. let lower = unsafe {
  27. let lower = libc::socket(
  28. libc::AF_PACKET,
  29. libc::SOCK_RAW | libc::SOCK_NONBLOCK,
  30. protocol.to_be() as i32,
  31. );
  32. if lower == -1 {
  33. return Err(io::Error::last_os_error());
  34. }
  35. lower
  36. };
  37. Ok(RawSocketDesc {
  38. protocol,
  39. lower,
  40. ifreq: ifreq_for(name),
  41. })
  42. }
  43. pub fn interface_mtu(&mut self) -> io::Result<usize> {
  44. ifreq_ioctl(self.lower, &mut self.ifreq, imp::SIOCGIFMTU).map(|mtu| mtu as usize)
  45. }
  46. pub fn bind_interface(&mut self) -> io::Result<()> {
  47. let sockaddr = libc::sockaddr_ll {
  48. sll_family: libc::AF_PACKET as u16,
  49. sll_protocol: self.protocol.to_be() as u16,
  50. sll_ifindex: ifreq_ioctl(self.lower, &mut self.ifreq, imp::SIOCGIFINDEX)?,
  51. sll_hatype: 1,
  52. sll_pkttype: 0,
  53. sll_halen: 6,
  54. sll_addr: [0; 8],
  55. };
  56. unsafe {
  57. let res = libc::bind(
  58. self.lower,
  59. &sockaddr as *const libc::sockaddr_ll as *const libc::sockaddr,
  60. mem::size_of::<libc::sockaddr_ll>() as libc::socklen_t,
  61. );
  62. if res == -1 {
  63. return Err(io::Error::last_os_error());
  64. }
  65. }
  66. Ok(())
  67. }
  68. pub fn recv(&mut self, buffer: &mut [u8]) -> io::Result<usize> {
  69. unsafe {
  70. let len = libc::recv(
  71. self.lower,
  72. buffer.as_mut_ptr() as *mut libc::c_void,
  73. buffer.len(),
  74. 0,
  75. );
  76. if len == -1 {
  77. return Err(io::Error::last_os_error());
  78. }
  79. Ok(len as usize)
  80. }
  81. }
  82. pub fn send(&mut self, buffer: &[u8]) -> io::Result<usize> {
  83. unsafe {
  84. let len = libc::send(
  85. self.lower,
  86. buffer.as_ptr() as *const libc::c_void,
  87. buffer.len(),
  88. 0,
  89. );
  90. if len == -1 {
  91. return Err(io::Error::last_os_error());
  92. }
  93. Ok(len as usize)
  94. }
  95. }
  96. }
  97. impl Drop for RawSocketDesc {
  98. fn drop(&mut self) {
  99. unsafe {
  100. libc::close(self.lower);
  101. }
  102. }
  103. }