#vpn #api-bindings #tun-tap #wireguard #wintun #tap #tun

wintun-bindings

Safe idiomatic bindings to the WinTun C library and more enhancements

33 releases

new 0.7.28 Jan 13, 2025
0.7.27 Dec 28, 2024
0.7.17 Nov 9, 2024
0.7.16 Oct 12, 2024
0.6.4 Aug 30, 2024

#243 in Network programming

Download history 1467/week @ 2024-09-24 1167/week @ 2024-10-01 1441/week @ 2024-10-08 990/week @ 2024-10-15 1202/week @ 2024-10-22 1059/week @ 2024-10-29 1044/week @ 2024-11-05 699/week @ 2024-11-12 1199/week @ 2024-11-19 1359/week @ 2024-11-26 1538/week @ 2024-12-03 1429/week @ 2024-12-10 1081/week @ 2024-12-17 1658/week @ 2024-12-24 1608/week @ 2024-12-31 1794/week @ 2025-01-07

6,420 downloads per month
Used in 12 crates (3 directly)

MIT license

1.5MB
2K SLoC

Contains (Windows DLL, 550KB) wintun/bin/x86/wintun.dll, (Windows DLL, 430KB) wintun/bin/amd64/wintun.dll, (Windows DLL, 365KB) wintun/bin/arm/wintun.dll, (Windows DLL, 225KB) wintun/bin/arm64/wintun.dll

wintun-bindings

Safe rust idiomatic bindings for the Wintun C library: https://wintun.net

All features of the Wintun library are wrapped using pure rust types and functions to make usage feel ergonomic.

Version Documentation Download License

Usage

Inside your code load the wintun.dll signed driver file, downloaded from https://wintun.net, using load, load_from_path or load_from_library.

Then either call Adapter::create or Adapter::open to obtain a wintun adapter. Start a session with Adapter::start_session.

Example


//Must be run as Administrator because we create network adapters
//Load the wintun dll file so that we can call the underlying C functions
//Unsafe because we are loading an arbitrary dll file
let wintun = unsafe { wintun_bindings::load_from_path("path/to/wintun.dll") }
    .expect("Failed to load wintun dll");

//Try to open an adapter with the name "Demo"
let adapter = match wintun_bindings::Adapter::open(&wintun, "Demo") {
    Ok(a) => a,
    Err(_) => {
        //If loading failed (most likely it didn't exist), create a new one
        wintun_bindings::Adapter::create(&wintun, "Demo", "Example", None)
            .expect("Failed to create wintun adapter!")
    }
};
//Specify the size of the ring buffer the wintun driver should use.
let session = adapter.start_session(wintun_bindings::MAX_RING_CAPACITY).unwrap();

//Get a 20 byte packet from the ring buffer
let mut packet = session.allocate_send_packet(20).unwrap();
let bytes: &mut [u8] = packet.bytes_mut();
//Write IPV4 version and header length
bytes[0] = 0x40;

//Finish writing IP header
bytes[9] = 0x69;
bytes[10] = 0x04;
bytes[11] = 0x20;
//...

//Send the packet to wintun virtual adapter for processing by the system
session.send_packet(packet);

//Stop any readers blocking for data on other threads
//Only needed when a blocking reader is preventing shutdown Ie. it holds an Arc to the
//session, blocking it from being dropped
let _ = session.shutdown();

//the session is stopped on drop
//drop(session);

//drop(adapter)
//And the adapter closes its resources when dropped

See examples/wireshark.rs for a more complete example that writes received packets to a pcap file.

Features

  • panic_on_unsent_packets: Panics if a send packet is dropped without being sent. Useful for debugging packet issues because unsent packets that are dropped without being sent hold up wintun's internal ring buffer.

  • verify_binary_signature: Verifies the signature of the wintun dll file before loading it.

  • async: Enables async support for the library. Just add async feature to your Cargo.toml:

    [dependencies]
    wintun-bindings = { version = "0.7", features = ["async"] }
    

    And simply transform your Session into an AsyncSession:

    // ...
    let session = adapter.start_session(MAX_RING_CAPACITY)?;
    let mut reader_session = AsyncSession::from(session.clone());
    let mut writer_session: AsyncSession = session.clone().into();
    // ...
    

License: MIT

Dependencies

~10–18MB
~237K SLoC