renet/src/phy/tuntap_interface.rs

110 lines
2.9 KiB
Rust
Raw Normal View History

2016-12-12 15:19:53 +08:00
use std::cell::RefCell;
use std::io;
2021-06-27 15:31:59 +08:00
use std::os::unix::io::{AsRawFd, RawFd};
use std::rc::Rc;
use std::vec::Vec;
2016-12-12 20:30:35 +08:00
2021-06-27 15:31:59 +08:00
use crate::phy::{self, sys, Device, DeviceCapabilities, Medium};
2020-12-27 07:11:30 +08:00
use crate::time::Instant;
2021-06-27 15:31:59 +08:00
use crate::Result;
2016-12-11 07:15:26 +08:00
2021-03-25 01:04:42 +08:00
/// A virtual TUN (IP) or TAP (Ethernet) interface.
2016-12-11 07:15:26 +08:00
#[derive(Debug)]
2021-03-25 01:04:42 +08:00
pub struct TunTapInterface {
2021-06-27 15:31:59 +08:00
lower: Rc<RefCell<sys::TunTapInterfaceDesc>>,
mtu: usize,
2021-03-25 01:04:42 +08:00
medium: Medium,
2016-12-11 07:15:26 +08:00
}
2021-03-25 01:04:42 +08:00
impl AsRawFd for TunTapInterface {
fn as_raw_fd(&self) -> RawFd {
self.lower.borrow().as_raw_fd()
}
}
2021-03-25 01:04:42 +08:00
impl TunTapInterface {
/// Attaches to a TUN/TAP interface called `name`, or creates it if it does not exist.
2016-12-11 07:15:26 +08:00
///
/// If `name` is a persistent interface configured with UID of the current user,
/// no special privileges are needed. Otherwise, this requires superuser privileges
/// or a corresponding capability set on the executable.
2021-03-25 01:04:42 +08:00
pub fn new(name: &str, medium: Medium) -> io::Result<TunTapInterface> {
let mut lower = sys::TunTapInterfaceDesc::new(name, medium)?;
2017-06-25 00:34:32 +08:00
lower.attach_interface()?;
let mtu = lower.interface_mtu()?;
2021-03-25 01:04:42 +08:00
Ok(TunTapInterface {
2016-12-12 20:30:35 +08:00
lower: Rc::new(RefCell::new(lower)),
2021-03-25 01:04:42 +08:00
mtu,
medium,
2016-12-11 07:15:26 +08:00
})
}
}
2021-03-25 01:04:42 +08:00
impl<'a> Device<'a> for TunTapInterface {
type RxToken = RxToken;
type TxToken = TxToken;
2016-12-12 20:30:35 +08:00
fn capabilities(&self) -> DeviceCapabilities {
DeviceCapabilities {
max_transmission_unit: self.mtu,
2021-03-25 01:04:42 +08:00
medium: self.medium,
..DeviceCapabilities::default()
}
}
2016-12-12 20:30:35 +08:00
fn receive(&'a mut self) -> Option<(Self::RxToken, Self::TxToken)> {
let mut lower = self.lower.borrow_mut();
let mut buffer = vec![0; self.mtu];
match lower.recv(&mut buffer[..]) {
Ok(size) => {
buffer.resize(size, 0);
let rx = RxToken { buffer };
2021-06-27 15:31:59 +08:00
let tx = TxToken {
lower: self.lower.clone(),
};
Some((rx, tx))
}
2021-06-27 15:31:59 +08:00
Err(ref err) if err.kind() == io::ErrorKind::WouldBlock => None,
Err(err) => panic!("{}", err),
}
}
fn transmit(&'a mut self) -> Option<Self::TxToken> {
Some(TxToken {
lower: self.lower.clone(),
})
}
}
#[doc(hidden)]
pub struct RxToken {
2021-06-27 15:31:59 +08:00
buffer: Vec<u8>,
}
impl phy::RxToken for RxToken {
fn consume<R, F>(mut self, _timestamp: Instant, f: F) -> Result<R>
2021-06-27 15:31:59 +08:00
where
F: FnOnce(&mut [u8]) -> Result<R>,
{
f(&mut self.buffer[..])
2016-12-12 15:19:53 +08:00
}
2016-12-12 20:30:35 +08:00
}
#[doc(hidden)]
pub struct TxToken {
2021-03-25 01:04:42 +08:00
lower: Rc<RefCell<sys::TunTapInterfaceDesc>>,
2016-12-12 20:30:35 +08:00
}
2016-12-11 07:15:26 +08:00
impl phy::TxToken for TxToken {
fn consume<R, F>(self, _timestamp: Instant, len: usize, f: F) -> Result<R>
2021-06-27 15:31:59 +08:00
where
F: FnOnce(&mut [u8]) -> Result<R>,
{
2016-12-12 20:30:35 +08:00
let mut lower = self.lower.borrow_mut();
let mut buffer = vec![0; len];
let result = f(&mut buffer);
lower.send(&buffer[..]).unwrap();
result
2016-12-11 07:15:26 +08:00
}
}