36 stable releases (4 major)

4.0.0 Oct 27, 2024
3.1.8 Oct 22, 2024
3.1.5 Sep 27, 2024
2.0.9 Aug 30, 2024
0.7.22 Jan 22, 2024

#25 in #tun

Download history 2597/week @ 2024-09-22 2864/week @ 2024-09-29 2269/week @ 2024-10-06 2426/week @ 2024-10-13 2968/week @ 2024-10-20 2592/week @ 2024-10-27 2662/week @ 2024-11-03 2333/week @ 2024-11-10 2187/week @ 2024-11-17 2869/week @ 2024-11-24 2747/week @ 2024-12-01 2358/week @ 2024-12-08 2323/week @ 2024-12-15 2211/week @ 2024-12-22 2001/week @ 2024-12-29 1462/week @ 2025-01-05

8,236 downloads per month
Used in 3 crates

WTFPL license

150KB
3K SLoC

TUN interfaces

Crates.io tun2 WTFPL

This crate allows the creation and usage of TUN interfaces, the aim is to make this cross-platform.

Now that I (@ssrlive) am a co-contributor of the tun crate, this crate is no longer maintained and all code is merged into the tun crate.

Usage

First, add the following to your Cargo.toml:

[dependencies]
tun2 = "4"

If you want to use the TUN interface with mio/tokio, you need to enable the async feature:

[dependencies]
tun2 = { version = "4", features = ["async"] }

Example

The following example creates and configures a TUN interface and starts reading packets from it.

use std::io::Read;

fn main() -> Result<(), Box<dyn std::error::Error + Send + Sync + 'static>> {
    let mut config = tun2::Configuration::default();
    config
        .address((10, 0, 0, 9))
        .netmask((255, 255, 255, 0))
        .destination((10, 0, 0, 1))
        .up();

    #[cfg(target_os = "linux")]
    config.platform_config(|config| {
        // requiring root privilege to acquire complete functions
        config.ensure_root_privileges(true);
    });

    let mut dev = tun2::create(&config)?;
    let mut buf = [0; 4096];

    loop {
        let amount = dev.read(&mut buf)?;
        println!("{:?}", &buf[0..amount]);
    }
}

Platforms

Supported Platforms

  • Windows
  • Linux
  • macOS
  • FreeBSD
  • Android
  • iOS

Linux

You will need the tun module to be loaded and root is required to create interfaces.

macOS & FreeBSD

tun2 will automatically set up a route according to the provided configuration, which does a similar thing like this:

sudo route -n add -net 10.0.0.0/24 10.0.0.1

iOS

You can pass the file descriptor of the TUN device to tun2 to create the interface.

Here is an example to create the TUN device on iOS and pass the fd to tun2:

// Swift
class PacketTunnelProvider: NEPacketTunnelProvider {
    override func startTunnel(options: [String : NSObject]?, completionHandler: @escaping (Error?) -> Void) {
        let tunnelNetworkSettings = createTunnelSettings() // Configure TUN address, DNS, mtu, routing...
        setTunnelNetworkSettings(tunnelNetworkSettings) { [weak self] error in
            // The tunnel of this tunFd is contains `Packet Information` prifix.
            let tunFd = self?.packetFlow.value(forKeyPath: "socket.fileDescriptor") as! Int32
            DispatchQueue.global(qos: .default).async {
                start_tun(tunFd)
            }
            completionHandler(nil)
        }
    }
}
#[no_mangle]
pub extern "C" fn start_tun(fd: std::os::raw::c_int) {
    let mut rt = tokio::runtime::Runtime::new().unwrap();
    rt.block_on(async {
        let mut cfg = tun2::Configuration::default();
        cfg.raw_fd(fd);
        #[cfg(target_os = "ios")]
        cfg.platform_config(|p_cfg| {
            p_cfg.packet_information(true);
        });
        let mut tun = tun2::create_as_async(&cfg).unwrap();
        let mut framed = tun.into_framed();
        while let Some(packet) = framed.next().await {
            ...
        }
    });
}

Windows

You need to copy the wintun.dll file which matches your architecture to the same directory as your executable and run your program as administrator.

Dependencies

~2–12MB
~137K SLoC