-
-
Save sfackler/d614e6c130f3462f443e6c0c6255383a to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
extern crate libc; | |
#[macro_use] | |
extern crate foreign_types; | |
use foreign_types::{ForeignType, ForeignTypeRef}; | |
use std::io; | |
use std::ptr; | |
use std::ffi::CStr; | |
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr}; | |
foreign_type! { | |
type CType = libc::ifaddrs; | |
fn drop = libc::freeifaddrs; | |
pub struct IfAddrs; | |
pub struct IfAddrsRef; | |
} | |
impl IfAddrs { | |
pub fn get() -> io::Result<IfAddrs> { | |
unsafe { | |
let mut ifaddrs = ptr::null_mut(); | |
let r = libc::getifaddrs(&mut ifaddrs); | |
if r == 0 { | |
Ok(IfAddrs::from_ptr(ifaddrs)) | |
} else { | |
Err(io::Error::last_os_error()) | |
} | |
} | |
} | |
} | |
impl IfAddrsRef { | |
pub fn next(&self) -> Option<&IfAddrsRef> { | |
unsafe { | |
let next = (*self.as_ptr()).ifa_next; | |
if next.is_null() { | |
None | |
} else { | |
Some(IfAddrsRef::from_ptr(next)) | |
} | |
} | |
} | |
pub fn name(&self) -> &str { | |
unsafe { | |
let s = CStr::from_ptr((*self.as_ptr()).ifa_name); | |
s.to_str().unwrap() | |
} | |
} | |
pub fn addr(&self) -> Option<IpAddr> { | |
unsafe { | |
let addr = (*self.as_ptr()).ifa_addr; | |
if addr.is_null() { | |
return None; | |
} | |
match (*addr).sa_family as _ { | |
libc::AF_INET => { | |
let addr = addr as *mut libc::sockaddr_in; | |
// It seems like this to_be shouldn't be needed? | |
let addr = Ipv4Addr::from((*addr).sin_addr.s_addr.to_be()); | |
Some(IpAddr::V4(addr)) | |
} | |
libc::AF_INET6 => { | |
let addr = addr as *mut libc::sockaddr_in6; | |
let addr = Ipv6Addr::from((*addr).sin6_addr.s6_addr); | |
Some(IpAddr::V6(addr)) | |
} | |
_ => None, | |
} | |
} | |
} | |
pub fn iter<'a>(&'a self) -> Iter<'a> { | |
Iter(Some(self)) | |
} | |
} | |
impl<'a> IntoIterator for &'a IfAddrs { | |
type Item = &'a IfAddrsRef; | |
type IntoIter = Iter<'a>; | |
fn into_iter(self) -> Iter<'a> { | |
self.iter() | |
} | |
} | |
impl<'a> IntoIterator for &'a IfAddrsRef { | |
type Item = &'a IfAddrsRef; | |
type IntoIter = Iter<'a>; | |
fn into_iter(self) -> Iter<'a> { | |
self.iter() | |
} | |
} | |
pub struct Iter<'a>(Option<&'a IfAddrsRef>); | |
impl<'a> Iterator for Iter<'a> { | |
type Item = &'a IfAddrsRef; | |
fn next(&mut self) -> Option<&'a IfAddrsRef> { | |
let cur = match self.0 { | |
Some(cur) => cur, | |
None => return None, | |
}; | |
self.0 = cur.next(); | |
Some(cur) | |
} | |
} | |
#[cfg(test)] | |
mod test { | |
use super::*; | |
#[test] | |
fn basic() { | |
let addrs = IfAddrs::get().unwrap(); | |
println!( | |
"{:?}", | |
addrs | |
.iter() | |
.map(|a| (a.name(), a.addr())) | |
.collect::<Vec<_>>() | |
); | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment