#![allow(non_camel_case_types)]
use std::{
ffi::{self, CStr, c_char},
sync::{LazyLock, Once},
};
use tracing::level_filters::LevelFilter;
mod config;
mod keys;
mod net_types;
mod tcp;
mod udp;
mod util;
pub use net_types::{
AF_INET, AF_INET6, in_addr_t, in6_addr_t, sa_family_t, sockaddr, sockaddr_data, sockaddr_in,
sockaddr_in6,
};
pub use tcp::{
tcp_listener, tcp_stream, ts_tcp_close, ts_tcp_close_listener, ts_tcp_connect, ts_tcp_listen,
ts_tcp_listener_local_addr, ts_tcp_local_addr, ts_tcp_recv, ts_tcp_remote_addr, ts_tcp_send,
};
pub use udp::{ts_udp_bind, ts_udp_close, ts_udp_recvfrom, ts_udp_sendto, udp_socket};
static TOKIO_RUNTIME: LazyLock<tokio::runtime::Runtime> = LazyLock::new(|| {
let rt = tokio::runtime::Builder::new_multi_thread()
.enable_all()
.build()
.unwrap();
tracing::info!("started tokio runtime");
rt
});
pub struct device(tailscale::Device);
static TRACING_ONCE: Once = Once::new();
#[unsafe(no_mangle)]
pub extern "C" fn ts_init_tracing() {
TRACING_ONCE.call_once(|| {
tracing_subscriber::fmt()
.with_env_filter(
tracing_subscriber::EnvFilter::builder()
.with_default_directive(LevelFilter::INFO.into())
.from_env_lossy(),
)
.init();
});
}
#[unsafe(no_mangle)]
pub unsafe extern "C" fn ts_init(
config: Option<&config::config>,
auth_token: *const c_char,
) -> Option<Box<device>> {
ts_init_tracing();
let config = match config {
Some(cfg) => unsafe { cfg.to_ts_config() },
None => Default::default(),
};
let auth_token = if auth_token.is_null() {
None
} else {
unsafe { util::str(auth_token).map(ToOwned::to_owned) }
};
match TOKIO_RUNTIME.block_on(tailscale::Device::new(&config, auth_token)) {
Ok(dev) => Some(Box::new(device(dev))),
Err(e) => {
tracing::error!(err = %e, "ts_init failed");
None
}
}
}
#[unsafe(no_mangle)]
pub unsafe extern "C" fn ts_init_from_key_file(
key_file: *const c_char,
auth_token: *const c_char,
) -> Option<Box<device>> {
let mut state = keys::persisted_key_state::default();
if unsafe { keys::ts_load_key_file(key_file, false, &mut state) } < 0 {
return None;
}
let config = config::config {
key_state: Some(&mut state),
..Default::default()
};
unsafe { ts_init(Some(&config), auth_token) }
}
#[unsafe(no_mangle)]
pub extern "C" fn ts_deinit(dev: Box<device>) {
drop(dev)
}
#[unsafe(no_mangle)]
pub extern "C" fn ts_ipv4_addr(dev: &device, dst: &mut in_addr_t) -> ffi::c_int {
let addr = match TOKIO_RUNTIME.block_on(dev.0.ipv4_addr()) {
Ok(addr) => addr,
Err(e) => {
tracing::error!(error = %e, "getting ipv4");
return -1;
}
};
dst.0 = addr.octets();
0
}
#[unsafe(no_mangle)]
pub extern "C" fn ts_ipv6_addr(dev: &device, dst: &mut in6_addr_t) -> ffi::c_int {
let addr = match TOKIO_RUNTIME.block_on(dev.0.ipv6_addr()) {
Ok(addr) => addr,
Err(e) => {
tracing::error!(error = %e, "getting ipv6");
return -1;
}
};
dst.0 = addr.segments();
0
}
#[unsafe(no_mangle)]
pub unsafe extern "C" fn ts_peer_ipv4_addr(
dev: &device,
peer_name: *const c_char,
addr: &mut in_addr_t,
) -> ffi::c_int {
unsafe {
_peer_by_addr(dev, peer_name, |n| {
*addr = n.tailnet_address.ipv4.addr().into();
})
}
}
#[unsafe(no_mangle)]
pub unsafe extern "C" fn ts_peer_ipv6_addr(
dev: &device,
peer_name: *const c_char,
addr: &mut in6_addr_t,
) -> ffi::c_int {
unsafe {
_peer_by_addr(dev, peer_name, |n| {
*addr = n.tailnet_address.ipv6.addr().into();
})
}
}
unsafe fn _peer_by_addr(
dev: &device,
peer_name: *const c_char,
on_node_info: impl FnOnce(&tailscale::NodeInfo),
) -> ffi::c_int {
let name = unsafe { CStr::from_ptr(peer_name) };
let Ok(name) = name.to_str() else {
tracing::error!("peer name: invalid utf-8");
return -1;
};
match TOKIO_RUNTIME.block_on(dev.0.peer_by_name(name)) {
Ok(Some(node)) => {
on_node_info(&node);
1
}
Ok(None) => 0,
Err(e) => {
tracing::error!(error = %e, "looking up peer");
-1
}
}
}