1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
// SPDX-License-Identifier: MIT
use netlink_packet_utils::{
nla::{NlaBuffer, NlasIterator},
traits::{Emitable, Parseable},
DecodeError,
};
use crate::{
link::{link_flag::VecLinkFlag, LinkFlag, LinkLayerType},
AddressFamily,
};
const LINK_HEADER_LEN: usize = 16;
buffer!(LinkMessageBuffer(LINK_HEADER_LEN) {
interface_family: (u8, 0),
reserved_1: (u8, 1),
link_layer_type: (u16, 2..4),
link_index: (u32, 4..8),
flags: (u32, 8..12),
change_mask: (u32, 12..LINK_HEADER_LEN),
payload: (slice, LINK_HEADER_LEN..),
});
impl<'a, T: AsRef<[u8]> + ?Sized> LinkMessageBuffer<&'a T> {
pub fn attributes(
&self,
) -> impl Iterator<Item = Result<NlaBuffer<&'a [u8]>, DecodeError>> {
NlasIterator::new(self.payload())
}
}
/// High level representation of `RTM_GETLINK`, `RTM_SETLINK`, `RTM_NEWLINK` and
/// `RTM_DELLINK` messages headers.
///
/// These headers have the following structure:
///
/// ```no_rust
/// 0 8 16 24 32
/// +----------------+----------------+----------------+----------------+
/// |interface family| reserved | link layer type |
/// +----------------+----------------+----------------+----------------+
/// | link index |
/// +----------------+----------------+----------------+----------------+
/// | flags |
/// +----------------+----------------+----------------+----------------+
/// | change mask |
/// +----------------+----------------+----------------+----------------+
/// ```
///
/// `LinkHeader` exposes all these fields except for the "reserved" one.
#[derive(Debug, PartialEq, Eq, Clone, Default)]
pub struct LinkHeader {
/// Address family: one of the `AF_*` constants.
/// The [AddressFamily] has `From<u8>` and `From<AddressFamily> for u8`
/// implemented.
pub interface_family: AddressFamily,
/// Link index.
pub index: u32,
/// Link type. It should be set to one of the `ARPHRD_*`
/// constants. The most common value is [LinkLayerType::Ether] for
/// Ethernet.
/// The LinkLayerType has `From<u16>` and `From<LinkLayerType> for u16`
/// implemented.
pub link_layer_type: LinkLayerType,
/// State of the link, described by a combinations of `IFF_*`
/// constants, for instance `vec![LinkFlag::Up, LinkFlag::LowerUp]`.
/// To convert `Vec<LinkFlag>` into `u32`, you may:
/// `u32::from(&VecLinkFlag(Vec<LinkFlag>)`
/// To convert `u32` to `Vec<LinkFlag>`, you may:
/// `VecLinkFlag::from(u32).0`
pub flags: Vec<LinkFlag>,
/// Change mask for the `flags` field.
pub change_mask: Vec<LinkFlag>,
}
impl Emitable for LinkHeader {
fn buffer_len(&self) -> usize {
LINK_HEADER_LEN
}
fn emit(&self, buffer: &mut [u8]) {
let mut packet = LinkMessageBuffer::new(buffer);
packet.set_interface_family(u8::from(self.interface_family));
packet.set_link_index(self.index);
packet.set_change_mask(u32::from(&VecLinkFlag(
self.change_mask.to_vec(),
)));
packet.set_link_layer_type(u16::from(self.link_layer_type));
packet.set_flags(u32::from(&VecLinkFlag(self.flags.to_vec())));
}
}
impl<T: AsRef<[u8]>> Parseable<LinkMessageBuffer<T>> for LinkHeader {
fn parse(buf: &LinkMessageBuffer<T>) -> Result<Self, DecodeError> {
Ok(Self {
interface_family: buf.interface_family().into(),
link_layer_type: buf.link_layer_type().into(),
index: buf.link_index(),
change_mask: VecLinkFlag::from(buf.change_mask()).0,
flags: VecLinkFlag::from(buf.flags()).0,
})
}
}