From fca597a194fb0ba9ae060089220c76d21aea6b8e Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Wed, 2 Aug 2023 14:30:11 -0400 Subject: [PATCH] Remove protomask-tun in favor of easy-tun --- protomask-tun/Cargo.toml | 23 ---- protomask-tun/README.md | 3 - protomask-tun/src/error.rs | 10 -- protomask-tun/src/lib.rs | 5 - protomask-tun/src/tun.rs | 226 ------------------------------------- 5 files changed, 267 deletions(-) delete mode 100644 protomask-tun/Cargo.toml delete mode 100644 protomask-tun/README.md delete mode 100644 protomask-tun/src/error.rs delete mode 100644 protomask-tun/src/lib.rs delete mode 100644 protomask-tun/src/tun.rs diff --git a/protomask-tun/Cargo.toml b/protomask-tun/Cargo.toml deleted file mode 100644 index a2a24aa..0000000 --- a/protomask-tun/Cargo.toml +++ /dev/null @@ -1,23 +0,0 @@ -[package] -name = "protomask-tun" -version = "0.1.0" -authors = ["Evan Pratten "] -edition = "2021" -description = "An async interface to Linux TUN devices" -readme = "README.md" -homepage = "https://github.com/ewpratten/protomask/protomask-tun" -documentation = "https://docs.rs/protomask-tun" -repository = "https://github.com/ewpratten/protomask" -license = "GPL-3.0" -keywords = [] -categories = [] - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html -[dependencies] -tokio = { version = "1.29.1", features = ["sync", "rt"] } -log = "0.4.19" -thiserror = "1.0.43" -tun-tap = "0.1.3" -rtnetlink = "0.13.0" -futures = "0.3.28" -ipnet = "^2.8.0" diff --git a/protomask-tun/README.md b/protomask-tun/README.md deleted file mode 100644 index 2f77949..0000000 --- a/protomask-tun/README.md +++ /dev/null @@ -1,3 +0,0 @@ -# protomask-tun - -An async interface to Linux TUN devices. Support library for `protomask`. diff --git a/protomask-tun/src/error.rs b/protomask-tun/src/error.rs deleted file mode 100644 index ed62a8e..0000000 --- a/protomask-tun/src/error.rs +++ /dev/null @@ -1,10 +0,0 @@ -#[derive(Debug, thiserror::Error)] -pub enum Error { - #[error(transparent)] - IoError(#[from] std::io::Error), - - #[error(transparent)] - NetlinkError(#[from] rtnetlink::Error), -} - -pub type Result = std::result::Result; \ No newline at end of file diff --git a/protomask-tun/src/lib.rs b/protomask-tun/src/lib.rs deleted file mode 100644 index b75b6df..0000000 --- a/protomask-tun/src/lib.rs +++ /dev/null @@ -1,5 +0,0 @@ -mod error; -mod tun; - -pub use error::{Error, Result}; -pub use tun::TunDevice; diff --git a/protomask-tun/src/tun.rs b/protomask-tun/src/tun.rs deleted file mode 100644 index 79511b5..0000000 --- a/protomask-tun/src/tun.rs +++ /dev/null @@ -1,226 +0,0 @@ -use std::{ - io::{Read, Write}, - net::IpAddr, - os::fd::{AsRawFd, FromRawFd}, -}; - -use futures::TryStreamExt; -use ipnet::IpNet; -use tokio::{ - sync::{broadcast, mpsc}, - task, -}; -use tun_tap::Mode; - -use crate::Result; - -#[derive(Debug)] -pub struct TunDevice { - device: tun_tap::Iface, - rt_handle: rtnetlink::Handle, - link_index: u32, - mtu: usize, -} - -impl TunDevice { - /// Create and bring up a new TUN device - /// - /// ## Name format - /// - /// The name field can be any string. If `%d` is present in the string, - /// it will be replaced with a unique number. - pub async fn new(name: &str) -> Result { - // Bring up an rtnetlink connection - let (rt_connection, rt_handle, _) = rtnetlink::new_connection().map_err(|err| { - log::error!("Failed to open rtnetlink connection"); - log::error!("{}", err); - err - })?; - tokio::spawn(rt_connection); - - // Create the TUN device - let tun_device = tun_tap::Iface::without_packet_info(name, Mode::Tun)?; - log::debug!("Created new TUN device: {}", tun_device.name()); - - // Get access to the link through rtnetlink - // NOTE: I don't think there is any way this can fail, so `except` is probably OK - let tun_link = rt_handle - .link() - .get() - .match_name(tun_device.name().to_owned()) - .execute() - .try_next() - .await? - .expect("Failed to access newly created TUN device"); - - // Bring the link up - rt_handle - .link() - .set(tun_link.header.index) - .up() - .execute() - .await - .map_err(|err| { - log::error!("Failed to bring up link"); - log::error!("{}", err); - err - })?; - log::debug!("Brought {} up", tun_device.name()); - - // Read the link MTU - let mtu: usize = - std::fs::read_to_string(format!("/sys/class/net/{}/mtu", tun_device.name())) - .expect("Failed to read link MTU") - .strip_suffix("\n") - .unwrap() - .parse() - .unwrap(); - - Ok(Self { - device: tun_device, - rt_handle, - link_index: tun_link.header.index, - mtu, - }) - } - - /// Add an IP address to this device - pub async fn add_address(&mut self, ip_address: IpAddr, prefix_len: u8) -> Result<()> { - self.rt_handle - .address() - .add(self.link_index, ip_address, prefix_len) - .execute() - .await - .map_err(|err| { - log::error!("Failed to add address {} to link", ip_address); - log::error!("{}", err); - err - })?; - - Ok(()) - } - - /// Remove an IP address from this device - pub async fn remove_address(&mut self, ip_address: IpAddr, prefix_len: u8) -> Result<()> { - // Find the address message that matches the given address - if let Some(address_message) = self - .rt_handle - .address() - .get() - .set_link_index_filter(self.link_index) - .set_address_filter(ip_address) - .set_prefix_length_filter(prefix_len) - .execute() - .try_next() - .await - .map_err(|err| { - log::error!("Failed to find address {} on link", ip_address); - log::error!("{}", err); - err - })? - { - // Delete the address - self.rt_handle - .address() - .del(address_message) - .execute() - .await - .map_err(|err| { - log::error!("Failed to remove address {} from link", ip_address); - log::error!("{}", err); - err - })?; - } - - Ok(()) - } - - /// Add a route to this device - pub async fn add_route(&mut self, destination: IpNet) -> Result<()> { - match destination { - IpNet::V4(destination) => { - self.rt_handle - .route() - .add() - .v4() - .output_interface(self.link_index) - .destination_prefix(destination.addr(), destination.prefix_len()) - .execute() - .await - .map_err(|err| { - log::error!("Failed to add route {} to link", destination); - log::error!("{}", err); - err - })?; - } - IpNet::V6(destination) => { - self.rt_handle - .route() - .add() - .v6() - .output_interface(self.link_index) - .destination_prefix(destination.addr(), destination.prefix_len()) - .execute() - .await - .map_err(|err| { - log::error!("Failed to add route {} to link", destination); - log::error!("{}", err); - err - })?; - } - } - - Ok(()) - } - - /// Spawns worker threads, and returns a tx/rx pair for the caller to interact with them - pub async fn spawn_worker(&self) -> (mpsc::Sender>, broadcast::Receiver>) { - // Create a channel for packets to be sent to the caller - let (tx_to_caller, rx_from_worker) = broadcast::channel(65535); - - // Create a channel for packets being received from the caller - let (tx_to_worker, mut rx_from_caller) = mpsc::channel(65535); - - // Clone some values for use in worker threads - let mtu = self.mtu; - let device_fd = self.device.as_raw_fd(); - - // Create a task that broadcasts all incoming packets - let _rx_task = task::spawn_blocking(move || { - // Build a buffer to read packets into - let mut buffer = vec![0u8; mtu]; - - // Create a file to access the TUN device - let mut device = unsafe { std::fs::File::from_raw_fd(device_fd) }; - - loop { - // Read a packet from the TUN device - let packet_len = device.read(&mut buffer[..]).unwrap(); - let packet = buffer[..packet_len].to_vec(); - - // Broadcast the packet to all listeners - tx_to_caller.send(packet).unwrap(); - } - }); - - // Create a task that sends all outgoing packets - let _tx_task = task::spawn(async move { - // Create a file to access the TUN device - let mut device = unsafe { std::fs::File::from_raw_fd(device_fd) }; - - loop { - // Wait for a packet to be sent - let packet: Vec = rx_from_caller.recv().await.unwrap(); - - // Write the packet to the TUN device - device.write_all(&packet[..]).unwrap(); - } - }); - - // Create a task that sends all outgoing packets - let _tx_task = task::spawn_blocking(|| {}); - - // Return an rx/tx pair for the caller to interact with the workers - (tx_to_worker, rx_from_worker) - } -}