netlink_packet_route/address/
attribute.rsuse crate::address::{AddressError, AddressFlags, CacheInfo, CacheInfoBuffer};
use byteorder::{ByteOrder, NativeEndian};
use netlink_packet_utils::nla::{DefaultNla, Nla, NlaBuffer};
use netlink_packet_utils::parsers::{parse_string, parse_u32};
use netlink_packet_utils::{Emitable, Parseable};
use std::mem::size_of;
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr};
const IFA_ADDRESS: u16 = 1;
const IFA_LOCAL: u16 = 2;
const IFA_LABEL: u16 = 3;
const IFA_BROADCAST: u16 = 4;
const IFA_ANYCAST: u16 = 5;
const IFA_CACHEINFO: u16 = 6;
const IFA_MULTICAST: u16 = 7;
const IFA_FLAGS: u16 = 8;
const IPV4_ADDR_LEN: usize = 4;
const IPV6_ADDR_LEN: usize = 16;
#[derive(Debug, PartialEq, Eq, Clone)]
#[non_exhaustive]
pub enum AddressAttribute {
Address(IpAddr),
Local(IpAddr),
Label(String),
Broadcast(Ipv4Addr),
Anycast(Ipv6Addr),
CacheInfo(CacheInfo),
Multicast(Ipv6Addr),
Flags(AddressFlags),
Other(DefaultNla),
}
impl Nla for AddressAttribute {
fn value_len(&self) -> usize {
match *self {
Self::Broadcast(_) => IPV4_ADDR_LEN,
Self::Anycast(_) | Self::Multicast(_) => IPV6_ADDR_LEN,
Self::Address(ref addr) | Self::Local(ref addr) => {
if addr.is_ipv6() {
IPV6_ADDR_LEN
} else {
IPV4_ADDR_LEN
}
}
Self::Label(ref string) => string.as_bytes().len() + 1,
Self::Flags(_) => size_of::<u32>(),
Self::CacheInfo(ref attr) => attr.buffer_len(),
Self::Other(ref attr) => attr.value_len(),
}
}
fn emit_value(&self, buffer: &mut [u8]) {
match *self {
Self::Broadcast(ref addr) => buffer.copy_from_slice(&addr.octets()),
Self::Anycast(ref addr) | Self::Multicast(ref addr) => {
buffer.copy_from_slice(&addr.octets())
}
Self::Address(ref addr) | Self::Local(ref addr) => match addr {
IpAddr::V4(addr4) => buffer.copy_from_slice(&addr4.octets()),
IpAddr::V6(addr6) => buffer.copy_from_slice(&addr6.octets()),
},
Self::Label(ref string) => {
buffer[..string.len()].copy_from_slice(string.as_bytes());
buffer[string.len()] = 0;
}
Self::Flags(ref value) => NativeEndian::write_u32(buffer, value.bits()),
Self::CacheInfo(ref attr) => attr.emit(buffer),
Self::Other(ref attr) => attr.emit_value(buffer),
}
}
fn kind(&self) -> u16 {
match *self {
Self::Address(_) => IFA_ADDRESS,
Self::Local(_) => IFA_LOCAL,
Self::Label(_) => IFA_LABEL,
Self::Broadcast(_) => IFA_BROADCAST,
Self::Anycast(_) => IFA_ANYCAST,
Self::CacheInfo(_) => IFA_CACHEINFO,
Self::Multicast(_) => IFA_MULTICAST,
Self::Flags(_) => IFA_FLAGS,
Self::Other(ref nla) => nla.kind(),
}
}
}
impl<'a, T: AsRef<[u8]> + ?Sized> Parseable<NlaBuffer<&'a T>> for AddressAttribute {
type Error = AddressError;
fn parse(buf: &NlaBuffer<&'a T>) -> Result<Self, AddressError> {
let payload = buf.value();
Ok(match buf.kind() {
IFA_ADDRESS => {
if payload.len() == IPV4_ADDR_LEN {
let mut data = [0u8; IPV4_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV4_ADDR_LEN]);
Self::Address(IpAddr::from(data))
} else if payload.len() == IPV6_ADDR_LEN {
let mut data = [0u8; IPV6_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV6_ADDR_LEN]);
Self::Address(IpAddr::from(data))
} else {
return Err(AddressError::ParseAttributeInvalidPayload {
kind: "IFA_ADDRESS",
payload_length: payload.len(),
});
}
}
IFA_LOCAL => {
if payload.len() == IPV4_ADDR_LEN {
let mut data = [0u8; IPV4_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV4_ADDR_LEN]);
Self::Local(IpAddr::from(data))
} else if payload.len() == IPV6_ADDR_LEN {
let mut data = [0u8; IPV6_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV6_ADDR_LEN]);
Self::Local(IpAddr::from(data))
} else {
return Err(AddressError::ParseAttributeInvalidPayload {
kind: "IFA_LOCAL",
payload_length: payload.len(),
});
}
}
IFA_LABEL => Self::Label(
parse_string(payload)
.map_err(|err| AddressError::ParseAttribute { kind: "IFA_LABEL", err })?,
),
IFA_BROADCAST => {
if payload.len() == IPV4_ADDR_LEN {
let mut data = [0u8; IPV4_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV4_ADDR_LEN]);
Self::Broadcast(Ipv4Addr::from(data))
} else {
return Err(AddressError::ParseAttributeInvalidPayload {
kind: "IFA_BROADCAST",
payload_length: payload.len(),
});
}
}
IFA_ANYCAST => {
if payload.len() == IPV6_ADDR_LEN {
let mut data = [0u8; IPV6_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV6_ADDR_LEN]);
Self::Anycast(Ipv6Addr::from(data))
} else {
return Err(AddressError::ParseAttributeInvalidPayload {
kind: "IFA_ANYCAST",
payload_length: payload.len(),
});
}
}
IFA_CACHEINFO => Self::CacheInfo(
CacheInfo::parse(&CacheInfoBuffer::new(payload))
.map_err(|err| AddressError::ParseAttribute { kind: "IFA_CACHEINFO", err })?,
),
IFA_MULTICAST => {
if payload.len() == IPV6_ADDR_LEN {
let mut data = [0u8; IPV6_ADDR_LEN];
data.copy_from_slice(&payload[0..IPV6_ADDR_LEN]);
Self::Multicast(Ipv6Addr::from(data))
} else {
return Err(AddressError::ParseAttributeInvalidPayload {
kind: "IFA_MULTICAST",
payload_length: payload.len(),
});
}
}
IFA_FLAGS => Self::Flags(AddressFlags::from_bits_retain(
parse_u32(payload)
.map_err(|err| AddressError::ParseAttribute { kind: "IFA_FLAGS", err })?,
)),
kind => Self::Other(
DefaultNla::parse(buf)
.map_err(|err| AddressError::ParseUnknownNLA { kind, err })?,
),
})
}
}