use std::net::Ipv6Addr;
use xenet_core::mac::MacAddr;
use xenet_packet::ethernet::MAC_ADDR_LEN;
use xenet_packet::icmpv6::{self, Icmpv6Type, MutableIcmpv6Packet};
use xenet_packet::icmpv6::ndp::{NDP_SOL_PACKET_LEN, NDP_OPT_PACKET_LEN};
use xenet_packet::icmpv6::ndp::{MutableNdpOptionPacket, MutableNeighborSolicitPacket, NdpOptionTypes};
fn octets_len(len: usize) -> u8 {
(len.next_power_of_two() >> 3).try_into().unwrap()
}
#[derive(Clone, Debug)]
pub struct NdpPacketBuilder {
pub src_mac: MacAddr,
pub dst_mac: MacAddr,
pub src_ip: Ipv6Addr,
pub dst_ip: Ipv6Addr,
}
impl NdpPacketBuilder {
pub fn new(src_mac: MacAddr, src_ip: Ipv6Addr, dst_ip: Ipv6Addr) -> NdpPacketBuilder {
NdpPacketBuilder {
src_mac: src_mac,
dst_mac: MacAddr::broadcast(),
src_ip: src_ip,
dst_ip: dst_ip,
}
}
pub fn build(&self) -> Vec<u8> {
let mut buffer = [0u8; NDP_SOL_PACKET_LEN + NDP_OPT_PACKET_LEN + MAC_ADDR_LEN];
let mut ndp_packet =
MutableNeighborSolicitPacket::new(&mut buffer).unwrap();
ndp_packet.set_target_addr(self.dst_ip);
ndp_packet.set_icmpv6_type(Icmpv6Type::NeighborSolicitation);
ndp_packet.set_checksum(0x3131);
let mut opt_packet = MutableNdpOptionPacket::new(ndp_packet.get_options_raw_mut()).unwrap();
opt_packet.set_option_type(NdpOptionTypes::SourceLLAddr);
opt_packet.set_length(octets_len(MAC_ADDR_LEN));
opt_packet.set_data(&self.src_mac.octets());
let mut icmpv6_packet = MutableIcmpv6Packet::new(&mut buffer).unwrap();
icmpv6_packet.set_checksum(icmpv6::checksum(
&icmpv6_packet.to_immutable(),
&self.src_ip,
&self.dst_ip,
));
buffer.to_vec()
}
}