bpf.rs 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. use std::io;
  2. use std::os::unix::io::{AsRawFd, RawFd};
  3. use libc;
  4. use super::{ifreq, ifreq_for};
  5. /// set interface
  6. #[cfg(target_os = "macos")]
  7. const BIOCSETIF: libc::c_ulong = 0x8020426c;
  8. /// get buffer length
  9. #[cfg(target_os = "macos")]
  10. const BIOCGBLEN: libc::c_ulong = 0x40044266;
  11. /// set immediate/nonblocking read
  12. #[cfg(target_os = "macos")]
  13. const BIOCIMMEDIATE: libc::c_ulong = 0x80044270;
  14. // TODO: check if this is same for OSes other than macos
  15. #[cfg(target_os = "macos")]
  16. const BPF_HDRLEN: usize = 18;
  17. macro_rules! try_ioctl {
  18. ($fd:expr,$cmd:expr,$req:expr) => {
  19. unsafe {
  20. if libc::ioctl($fd, $cmd, $req) == -1 {
  21. return Err(io::Error::last_os_error());
  22. }
  23. }
  24. };
  25. }
  26. #[derive(Debug)]
  27. pub struct BpfDevice {
  28. fd: libc::c_int,
  29. ifreq: ifreq,
  30. }
  31. impl AsRawFd for BpfDevice {
  32. fn as_raw_fd(&self) -> RawFd {
  33. self.fd
  34. }
  35. }
  36. fn open_device() -> io::Result<libc::c_int> {
  37. unsafe {
  38. for i in 0..256 {
  39. let dev = format!("/dev/bpf{}", i).as_ptr() as *const libc::c_char;
  40. match libc::open(dev, libc::O_RDWR) {
  41. -1 => continue,
  42. fd => return Ok(fd),
  43. };
  44. }
  45. }
  46. // at this point, all 256 BPF devices were busy and we weren't able to open any
  47. Err(io::Error::last_os_error())
  48. }
  49. impl BpfDevice {
  50. pub fn new(name: &str) -> io::Result<BpfDevice> {
  51. Ok(BpfDevice {
  52. fd: open_device()?,
  53. ifreq: ifreq_for(name),
  54. })
  55. }
  56. pub fn bind_interface(&mut self) -> io::Result<()> {
  57. try_ioctl!(self.fd, BIOCSETIF, &mut self.ifreq);
  58. Ok(())
  59. }
  60. /// This in fact does not return the interface's mtu,
  61. /// but it returns the size of the buffer that the app needs to allocate
  62. /// for the BPF device
  63. ///
  64. /// The `SIOGIFMTU` cant be called on a BPF descriptor. There is a workaround
  65. /// to get the actual interface mtu, but this should work better
  66. ///
  67. /// To get the interface MTU, you would need to create a raw socket first,
  68. /// and then call `SIOGIFMTU` for the same interface your BPF device is "bound" to.
  69. /// This MTU that you would get would not include the length of `struct bpf_hdr`
  70. /// which gets prepended to every packet by BPF,
  71. /// and your packet will be truncated if it has the length of the MTU.
  72. ///
  73. /// The buffer size for BPF is usually 4096 bytes, MTU is typically 1500 bytes.
  74. /// You could do something like `mtu += BPF_HDRLEN`,
  75. /// but you must change the buffer size the BPF device expects using `BIOCSBLEN` accordingly,
  76. /// and you must set it before setting the interface with the `BIOCSETIF` ioctl.
  77. ///
  78. /// The reason I said this should work better is because you might see some unexpected behavior,
  79. /// truncated/unaligned packets, I/O errors on read()
  80. /// if you change the buffer size to the actual MTU of the interface.
  81. pub fn interface_mtu(&mut self) -> io::Result<usize> {
  82. let mut bufsize: libc::c_int = 1;
  83. try_ioctl!(self.fd, BIOCIMMEDIATE, &mut bufsize as *mut libc::c_int);
  84. try_ioctl!(self.fd, BIOCGBLEN, &mut bufsize as *mut libc::c_int);
  85. Ok(bufsize as usize)
  86. }
  87. pub fn recv(&mut self, buffer: &mut [u8]) -> io::Result<usize> {
  88. unsafe {
  89. let len = libc::read(
  90. self.fd,
  91. buffer.as_mut_ptr() as *mut libc::c_void,
  92. buffer.len(),
  93. );
  94. if len == -1 || len < BPF_HDRLEN as isize {
  95. return Err(io::Error::last_os_error());
  96. }
  97. let len = len as usize;
  98. libc::memmove(
  99. buffer.as_mut_ptr() as *mut libc::c_void,
  100. &buffer[BPF_HDRLEN] as *const u8 as *const libc::c_void,
  101. len - BPF_HDRLEN,
  102. );
  103. Ok(len)
  104. }
  105. }
  106. pub fn send(&mut self, buffer: &[u8]) -> io::Result<usize> {
  107. unsafe {
  108. let len = libc::write(
  109. self.fd,
  110. buffer.as_ptr() as *const libc::c_void,
  111. buffer.len(),
  112. );
  113. if len == -1 {
  114. Err(io::Error::last_os_error()).unwrap()
  115. }
  116. Ok(len as usize)
  117. }
  118. }
  119. }
  120. impl Drop for BpfDevice {
  121. fn drop(&mut self) {
  122. unsafe {
  123. libc::close(self.fd);
  124. }
  125. }
  126. }