mirror of
https://github.com/domcyrus/rustnet.git
synced 2025-12-30 18:39:52 -06:00
* refactor: simplify QUIC DPI and unify SNI extraction helpers * refactor: reorganize platform code and improve code structure
224 lines
7.2 KiB
Rust
224 lines
7.2 KiB
Rust
use anyhow::Result;
|
|
use std::{env, fs::File, path::PathBuf};
|
|
|
|
fn main() -> Result<()> {
|
|
// Generate shell completions and manpage
|
|
generate_assets()?;
|
|
|
|
// Add library search paths for cross-compilation
|
|
setup_cross_compilation_libs();
|
|
|
|
// Compile eBPF programs on Linux when the feature is enabled
|
|
// Check TARGET environment variable, not cfg!, to handle cross-compilation correctly
|
|
let target = env::var("TARGET").unwrap_or_default();
|
|
if target.contains("linux") && env::var("CARGO_FEATURE_EBPF").is_ok() {
|
|
compile_ebpf_programs();
|
|
}
|
|
|
|
#[cfg(target_os = "windows")]
|
|
download_windows_npcap_sdk()?;
|
|
|
|
println!("cargo:rerun-if-changed=src/network/platform/linux/ebpf/programs/");
|
|
println!("cargo:rerun-if-changed=src/cli.rs");
|
|
println!("cargo:rerun-if-changed=resources/ebpf/vmlinux/");
|
|
|
|
Ok(())
|
|
}
|
|
|
|
include!("src/cli.rs");
|
|
|
|
fn setup_cross_compilation_libs() {
|
|
let target = env::var("TARGET").unwrap_or_default();
|
|
|
|
match target.as_str() {
|
|
"aarch64-unknown-linux-gnu" => {
|
|
println!("cargo:rustc-link-search=native=/usr/lib/aarch64-linux-gnu");
|
|
println!("cargo:rustc-link-lib=elf");
|
|
println!("cargo:rustc-link-lib=z");
|
|
}
|
|
"armv7-unknown-linux-gnueabihf" => {
|
|
println!("cargo:rustc-link-search=native=/usr/lib/arm-linux-gnueabihf");
|
|
println!("cargo:rustc-link-lib=elf");
|
|
println!("cargo:rustc-link-lib=z");
|
|
}
|
|
"x86_64-unknown-freebsd" => {
|
|
// FreeBSD uses libpcap from base system (in /usr/lib)
|
|
// When cross-compiling, the sysroot should provide these
|
|
println!("cargo:rustc-link-lib=pcap");
|
|
}
|
|
_ => {
|
|
// For other targets, including native builds, let pkg-config handle it
|
|
}
|
|
}
|
|
}
|
|
|
|
fn generate_assets() -> Result<()> {
|
|
use clap::ValueEnum;
|
|
use clap_complete::Shell;
|
|
use clap_mangen::Man;
|
|
|
|
let mut cmd = build_cli();
|
|
|
|
// build into `RUSTNET_ASSET_DIR` with a fallback to `OUT_DIR`
|
|
let asset_dir: PathBuf = env::var_os("RUSTNET_ASSET_DIR")
|
|
.or_else(|| env::var_os("OUT_DIR"))
|
|
.ok_or_else(|| anyhow::anyhow!("OUT_DIR is unset"))?
|
|
.into();
|
|
|
|
// completion
|
|
for &shell in Shell::value_variants() {
|
|
clap_complete::generate_to(shell, &mut cmd, "rustnet", &asset_dir)?;
|
|
}
|
|
|
|
// manpage
|
|
let mut manpage_out = File::create(asset_dir.join("rustnet.1"))?;
|
|
let manpage = Man::new(cmd);
|
|
manpage.render(&mut manpage_out)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[cfg(target_os = "windows")]
|
|
fn download_windows_npcap_sdk() -> Result<()> {
|
|
use std::{
|
|
fs,
|
|
io::{self, Write},
|
|
};
|
|
|
|
println!("cargo:rerun-if-changed=build.rs");
|
|
|
|
// get npcap SDK
|
|
const NPCAP_SDK: &str = "npcap-sdk-1.15.zip";
|
|
|
|
let npcap_sdk_download_url = format!("https://npcap.com/dist/{NPCAP_SDK}");
|
|
let cache_dir = PathBuf::from(env::var("CARGO_MANIFEST_DIR")?).join("target");
|
|
let npcap_sdk_cache_path = cache_dir.join(NPCAP_SDK);
|
|
|
|
let npcap_zip = match fs::read(&npcap_sdk_cache_path) {
|
|
// use cached
|
|
Ok(zip_data) => {
|
|
eprintln!("Found cached npcap SDK");
|
|
zip_data
|
|
}
|
|
// download SDK
|
|
Err(_) => {
|
|
eprintln!("Downloading npcap SDK");
|
|
|
|
// download
|
|
let mut zip_data = vec![];
|
|
let _res = http_req::request::get(npcap_sdk_download_url, &mut zip_data)?;
|
|
|
|
// write cache
|
|
fs::create_dir_all(cache_dir)?;
|
|
let mut cache = fs::File::create(npcap_sdk_cache_path)?;
|
|
cache.write_all(&zip_data)?;
|
|
|
|
zip_data
|
|
}
|
|
};
|
|
|
|
// extract libraries based on target architecture
|
|
let target = env::var("TARGET").unwrap_or_else(|_| "unknown".to_string());
|
|
let (packet_lib_path, wpcap_lib_path) = if target.contains("aarch64") {
|
|
("Lib/ARM64/Packet.lib", "Lib/ARM64/wpcap.lib")
|
|
} else if target.contains("x86_64") {
|
|
("Lib/x64/Packet.lib", "Lib/x64/wpcap.lib")
|
|
} else if target.contains("i686") || target.contains("i586") {
|
|
("Lib/Packet.lib", "Lib/wpcap.lib")
|
|
} else {
|
|
panic!("Unsupported target: {}", target)
|
|
};
|
|
|
|
let mut archive = zip::ZipArchive::new(io::Cursor::new(npcap_zip))?;
|
|
|
|
// Extract Packet.lib
|
|
let mut packet_lib = archive.by_name(packet_lib_path)?;
|
|
let lib_dir = PathBuf::from(env::var("OUT_DIR")?).join("npcap_sdk");
|
|
fs::create_dir_all(&lib_dir)?;
|
|
let packet_lib_dest = lib_dir.join("Packet.lib");
|
|
let mut packet_file = fs::File::create(packet_lib_dest)?;
|
|
io::copy(&mut packet_lib, &mut packet_file)?;
|
|
drop(packet_lib);
|
|
|
|
// Extract wpcap.lib
|
|
let mut wpcap_lib = archive.by_name(wpcap_lib_path)?;
|
|
let wpcap_lib_dest = lib_dir.join("wpcap.lib");
|
|
let mut wpcap_file = fs::File::create(wpcap_lib_dest)?;
|
|
io::copy(&mut wpcap_lib, &mut wpcap_file)?;
|
|
|
|
println!(
|
|
"cargo:rustc-link-search=native={}",
|
|
lib_dir
|
|
.to_str()
|
|
.ok_or_else(|| anyhow::anyhow!("{lib_dir:?} is not valid UTF-8"))?
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[cfg(all(target_os = "linux", feature = "ebpf"))]
|
|
fn get_vmlinux_header(arch: &str) -> Result<PathBuf> {
|
|
// Use bundled vmlinux.h from resources directory
|
|
let manifest_dir = PathBuf::from(env::var("CARGO_MANIFEST_DIR")?);
|
|
let bundled_dir = manifest_dir.join("resources/ebpf/vmlinux").join(arch);
|
|
let bundled_file = bundled_dir.join("vmlinux.h");
|
|
|
|
if bundled_file.exists() {
|
|
println!("cargo:warning=Using bundled vmlinux.h for {}", arch);
|
|
Ok(bundled_dir)
|
|
} else {
|
|
Err(anyhow::anyhow!(
|
|
"Bundled vmlinux.h not found for architecture '{}'. Expected at: {}",
|
|
arch,
|
|
bundled_file.display()
|
|
))
|
|
}
|
|
}
|
|
|
|
#[cfg(all(target_os = "linux", feature = "ebpf"))]
|
|
fn compile_ebpf_programs() {
|
|
use libbpf_cargo::SkeletonBuilder;
|
|
use std::ffi::OsStr;
|
|
use std::path::PathBuf;
|
|
|
|
let mut out = PathBuf::from(env::var("OUT_DIR").unwrap());
|
|
out.push("socket_tracker.skel.rs");
|
|
|
|
let src = "src/network/platform/linux/ebpf/programs/socket_tracker.bpf.c";
|
|
|
|
println!("cargo:warning=Building eBPF program using libbpf-cargo");
|
|
|
|
// Get target architecture for cross-compilation
|
|
let arch = env::var("CARGO_CFG_TARGET_ARCH")
|
|
.expect("CARGO_CFG_TARGET_ARCH must be set in build script");
|
|
|
|
// Map Rust arch names to eBPF target arch defines and vmlinux.h arch names
|
|
let (target_arch_define, vmlinux_arch) = match arch.as_str() {
|
|
"x86_64" => ("-D__TARGET_ARCH_x86", "x86"),
|
|
"aarch64" => ("-D__TARGET_ARCH_arm64", "aarch64"),
|
|
"arm" => ("-D__TARGET_ARCH_arm", "arm"),
|
|
_ => ("-D__TARGET_ARCH_x86", "x86"), // fallback
|
|
};
|
|
|
|
// Get bundled architecture-specific vmlinux.h
|
|
let vmlinux_include_path =
|
|
get_vmlinux_header(vmlinux_arch).expect("Failed to locate bundled vmlinux.h");
|
|
|
|
SkeletonBuilder::new()
|
|
.source(src)
|
|
.clang_args([
|
|
OsStr::new("-I"),
|
|
vmlinux_include_path.as_os_str(),
|
|
OsStr::new(target_arch_define),
|
|
])
|
|
.build_and_generate(&out)
|
|
.unwrap();
|
|
|
|
println!("cargo:rerun-if-changed={}", src);
|
|
}
|
|
|
|
#[cfg(not(all(target_os = "linux", feature = "ebpf")))]
|
|
fn compile_ebpf_programs() {
|
|
// No-op when not on Linux or eBPF feature is not enabled
|
|
}
|