use std::ffi::c_void;
use std::mem::size_of;
use std::net::{Ipv4Addr, Ipv6Addr};
use std::ptr::null_mut;
use std::slice::from_raw_parts;
use libc::{free, malloc, wchar_t, wcslen};
use winapi::{
ctypes::c_ulong,
shared::{
ws2def::{AF_UNSPEC, SOCKADDR_IN},
ws2ipdef::SOCKADDR_IN6,
netioapi::{ConvertLengthToIpv4Mask, ConvertInterfaceLuidToIndex},
ntdef::ULONG,
ifdef::IF_LUID,
winerror,
},
um::{
iptypes::{IP_ADAPTER_ADDRESSES_LH, IP_ADAPTER_UNICAST_ADDRESS_LH},
iphlpapi::GetAdaptersAddresses,
},
};
use crate::utils::hex::HexSlice;
use crate::utils::ffialloc::FFIAlloc;
use crate::{Error, NetworkInterface, NetworkInterfaceConfig, Result};
use crate::interface::Netmask;
type AdapterAddress = IP_ADAPTER_ADDRESSES_LH;
const AF_INET: u16 = winapi::shared::ws2def::AF_INET as u16;
const AF_INET6: u16 = winapi::shared::ws2def::AF_INET6 as u16;
const GET_ADAPTERS_ADDRESSES_FAMILY: u32 = AF_UNSPEC as u32;
const GET_ADAPTERS_ADDRESSES_FLAGS: ULONG = winapi::um::iptypes::GAA_FLAG_INCLUDE_PREFIX;
impl NetworkInterfaceConfig for NetworkInterface {
fn show() -> Result<Vec<NetworkInterface>> {
let mut buffer_size: u32 = 15000;
const MAX_TRIES: i32 = 10;
let mut try_no = 1;
let adapter_address = loop {
let adapter_address = FFIAlloc::alloc(buffer_size as usize).ok_or_else(|| {
Error::GetIfAddrsError(String::from("GetAdaptersAddresses"), 1)
})?;
let res = unsafe {
GetAdaptersAddresses(
GET_ADAPTERS_ADDRESSES_FAMILY,
GET_ADAPTERS_ADDRESSES_FLAGS,
null_mut(),
adapter_address.as_mut_ptr(),
&mut buffer_size,
)
};
match res {
winerror::ERROR_SUCCESS => {
break Ok(adapter_address);
}
winerror::ERROR_BUFFER_OVERFLOW => {
if try_no == MAX_TRIES {
break Err(Error::GetIfAddrsError(
"GetAdapterAddresses: alloc error".to_string(),
res as i32,
));
}
try_no += 1;
}
_ => {
break Err(Error::GetIfAddrsError(
"GetAdapterAddresses".to_string(),
res as i32,
));
}
}
}?;
let mut adapter_address_it = adapter_address.as_ptr();
let mut network_interfaces: Vec<NetworkInterface> = Vec::new();
while !adapter_address_it.is_null() {
let address_name = make_adapter_address_name(adapter_address_it)?;
let index = get_adapter_address_index(adapter_address_it)?;
let mac_addr_len = unsafe { (*adapter_address_it).PhysicalAddressLength } as _;
let mac_addr = match mac_addr_len {
0 => None,
len => Some(format!(
"{}",
HexSlice::new(unsafe { &(*adapter_address_it).PhysicalAddress[..len] })
)),
};
let mut current_unicast_address = unsafe { (*adapter_address_it).FirstUnicastAddress };
while !current_unicast_address.is_null() {
let address = unsafe { (*current_unicast_address).Address };
match unsafe { (*address.lpSockaddr).sa_family } {
AF_INET => {
let sockaddr: *mut SOCKADDR_IN = address.lpSockaddr as *mut SOCKADDR_IN;
let addr = make_ipv4_addr(&sockaddr);
let netmask = make_ipv4_netmask(¤t_unicast_address);
let network_interface = NetworkInterface::new_afinet(
&address_name,
addr,
netmask,
lookup_ipv4_broadcast_addr(adapter_address_it, &sockaddr),
index,
)
.with_mac_addr(mac_addr.clone());
network_interfaces.push(network_interface);
}
AF_INET6 => {
let sockaddr: *mut SOCKADDR_IN6 = address.lpSockaddr as *mut SOCKADDR_IN6;
let addr = make_ipv6_addr(&sockaddr)?;
let netmask = make_ipv6_netmask(&sockaddr);
let network_interface = NetworkInterface::new_afinet6(
&address_name,
addr,
netmask,
None,
index,
)
.with_mac_addr(mac_addr.clone());
network_interfaces.push(network_interface);
}
_ => {}
}
if !current_unicast_address.is_null() {
current_unicast_address = unsafe { (*current_unicast_address).Next };
}
}
adapter_address_it = unsafe { (*adapter_address_it).Next };
}
Ok(network_interfaces)
}
}
fn lookup_ipv4_broadcast_addr(
adapter_address: *const IP_ADAPTER_ADDRESSES_LH,
unicast_ip: &*mut SOCKADDR_IN,
) -> Option<Ipv4Addr> {
let mut prefix_address = unsafe { (*adapter_address).FirstPrefix };
let mut prefix_index_v4 = 0;
let mut broadcast_index: Option<i32> = None;
while !prefix_address.is_null() {
let address = unsafe { (*prefix_address).Address };
if unsafe { (*address.lpSockaddr).sa_family } == AF_INET {
if let Some(broadcast_index) = broadcast_index {
if prefix_index_v4 == broadcast_index {
let sockaddr: *mut SOCKADDR_IN = address.lpSockaddr as *mut SOCKADDR_IN;
return Some(make_ipv4_addr(&sockaddr));
}
} else if prefix_index_v4 % 3 == 1
&& ipv4_addr_equal(&(address.lpSockaddr as *mut SOCKADDR_IN), unicast_ip)
{
broadcast_index = Some(prefix_index_v4 + 1);
}
prefix_index_v4 += 1;
}
prefix_address = unsafe { (*prefix_address).Next };
}
None
}
fn make_adapter_address_name(adapter_address: *const AdapterAddress) -> Result<String> {
let address_name = unsafe { (*adapter_address).FriendlyName };
let address_name_length = unsafe { wcslen(address_name as *const wchar_t) };
let byte_slice = unsafe { from_raw_parts(address_name, address_name_length) };
let string = String::from_utf16(byte_slice).map_err(Error::from)?;
Ok(string)
}
fn make_ipv6_addr(sockaddr: &*mut SOCKADDR_IN6) -> Result<Ipv6Addr> {
let address_bytes = unsafe { (*(*sockaddr)).sin6_addr.u.Byte() };
let ip = Ipv6Addr::from(*address_bytes);
Ok(ip)
}
fn make_ipv4_addr(sockaddr: &*mut SOCKADDR_IN) -> Ipv4Addr {
let address = unsafe { (*(*sockaddr)).sin_addr.S_un.S_addr() };
if cfg!(target_endian = "little") {
return Ipv4Addr::from(address.swap_bytes());
}
Ipv4Addr::from(*address)
}
fn ipv4_addr_equal(sockaddr1: &*mut SOCKADDR_IN, sockaddr2: &*mut SOCKADDR_IN) -> bool {
let address1 = unsafe { (*(*sockaddr1)).sin_addr.S_un.S_addr() };
let address2 = unsafe { (*(*sockaddr2)).sin_addr.S_un.S_addr() };
address1 == address2
}
fn make_ipv4_netmask(unicast_address: &*mut IP_ADAPTER_UNICAST_ADDRESS_LH) -> Netmask<Ipv4Addr> {
let mut mask: c_ulong = 0;
let on_link_prefix_length = unsafe { (*(*unicast_address)).OnLinkPrefixLength };
unsafe {
ConvertLengthToIpv4Mask(on_link_prefix_length as u32, &mut mask as *mut c_ulong);
}
if cfg!(target_endian = "little") {
return Some(Ipv4Addr::from(mask.swap_bytes()));
}
Some(Ipv4Addr::from(mask))
}
fn make_ipv6_netmask(_sockaddr: &*mut SOCKADDR_IN6) -> Netmask<Ipv6Addr> {
None
}
fn get_adapter_address_index(adapter_address: *const AdapterAddress) -> Result<u32> {
let adapter_luid = &unsafe { (*adapter_address).Luid } as *const IF_LUID;
let index = &mut 0u32 as *mut u32;
match unsafe { ConvertInterfaceLuidToIndex(adapter_luid, index) } {
0 => Ok(unsafe { *index }),
e => Err(crate::error::Error::GetIfNameError(
"ConvertInterfaceLuidToIndex".to_string(),
e,
)),
}
}
#[cfg(test)]
mod tests {
use std::{process::Command, cmp::min};
use crate::{NetworkInterface, NetworkInterfaceConfig, Addr};
#[test]
fn test_mac_addr() {
const MAC_ADDR_LEN: usize = "00:22:48:03:ED:76".len();
let output = Command::new("getmac").arg("/nh").output().unwrap().stdout;
let output_string = String::from_utf8(output).unwrap();
let mac_addr_list: Vec<_> = output_string
.lines()
.filter_map(|line| {
let line = line.trim();
let line = &line[..min(MAC_ADDR_LEN, line.len())];
match line.split('-').count() {
6 => Some(line.replace('-', ":")),
_ => None,
}
})
.collect();
assert!(!mac_addr_list.is_empty());
let interfaces = NetworkInterface::show().unwrap();
for mac_addr in mac_addr_list {
assert!(interfaces
.iter()
.any(|int| int.mac_addr.as_ref() == Some(&mac_addr)));
}
}
#[test]
fn test_ipv4_broadcast() {
let interfaces = NetworkInterface::show().unwrap();
for ipv4 in interfaces.iter().flat_map(|i| &i.addr).filter_map(|addr| {
if let Addr::V4(ipv4) = addr {
Some(ipv4)
} else {
None
}
}) {
let Some(bc_addr) = ipv4.broadcast else {
continue;
};
let ip_bytes = ipv4.ip.octets();
let mask_bytes = ipv4.netmask.unwrap().octets();
let bc_bytes = bc_addr.octets();
for i in 0..4 {
assert_eq!(ip_bytes[i] & mask_bytes[i], bc_bytes[i] & mask_bytes[i]);
assert_eq!(bc_bytes[i] | mask_bytes[i], 255);
}
}
}
}