1

Finish packet rewrite

This commit is contained in:
Evan Pratten 2023-07-18 16:06:53 -04:00
parent 1542492e61
commit 94fdf28012
16 changed files with 838 additions and 499 deletions

View File

@ -1,163 +1,163 @@
/// Quickly convert a byte slice into an ICMP packet // /// Quickly convert a byte slice into an ICMP packet
#[macro_export] // #[macro_export]
macro_rules! into_icmp { // macro_rules! into_icmp {
($bytes:expr) => { // ($bytes:expr) => {
pnet_packet::icmp::IcmpPacket::owned($bytes).ok_or_else(|| { // pnet_packet::icmp::IcmpPacket::owned($bytes).ok_or_else(|| {
crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len()) // crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len())
}) // })
}; // };
} // }
/// Quickly convert a byte slice into an ICMPv6 packet // /// Quickly convert a byte slice into an ICMPv6 packet
#[macro_export] // #[macro_export]
macro_rules! into_icmpv6 { // macro_rules! into_icmpv6 {
($bytes:expr) => { // ($bytes:expr) => {
pnet_packet::icmpv6::Icmpv6Packet::owned($bytes).ok_or_else(|| { // pnet_packet::icmpv6::Icmpv6Packet::owned($bytes).ok_or_else(|| {
crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len()) // crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len())
}) // })
}; // };
} // }
/// Quickly convert a byte slice into a UDP packet // /// Quickly convert a byte slice into a UDP packet
#[macro_export] // #[macro_export]
macro_rules! into_udp { // macro_rules! into_udp {
($bytes:expr) => { // ($bytes:expr) => {
pnet_packet::udp::UdpPacket::owned($bytes).ok_or_else(|| { // pnet_packet::udp::UdpPacket::owned($bytes).ok_or_else(|| {
crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len()) // crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len())
}) // })
}; // };
} // }
/// Quickly convert a byte slice into a TCP packet // /// Quickly convert a byte slice into a TCP packet
#[macro_export] // #[macro_export]
macro_rules! into_tcp { // macro_rules! into_tcp {
($bytes:expr) => { // ($bytes:expr) => {
pnet_packet::tcp::TcpPacket::owned($bytes).ok_or_else(|| { // pnet_packet::tcp::TcpPacket::owned($bytes).ok_or_else(|| {
crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len()) // crate::nat::xlat::PacketTranslationError::InputPacketTooShort($bytes.len())
}) // })
}; // };
} // }
/// Quickly construct an IPv6 packet with the given parameters // /// Quickly construct an IPv6 packet with the given parameters
#[macro_export] // #[macro_export]
macro_rules! ipv6_packet { // macro_rules! ipv6_packet {
($source:expr, $destination:expr, $next_header:expr, $hop_limit:expr, $payload:expr) => { // ($source:expr, $destination:expr, $next_header:expr, $hop_limit:expr, $payload:expr) => {
ipv6_packet!( // ipv6_packet!(
$source, // $source,
$destination, // $destination,
0, // 0,
0, // 0,
$next_header, // $next_header,
$hop_limit, // $hop_limit,
$payload // $payload
) // )
}; // };
($source:expr, $destination:expr, $traffic_class:expr, $flow_label:expr, $next_header:expr, $hop_limit:expr, $payload:expr) => {{ // ($source:expr, $destination:expr, $traffic_class:expr, $flow_label:expr, $next_header:expr, $hop_limit:expr, $payload:expr) => {{
let mut output = // let mut output =
pnet_packet::ipv6::MutableIpv6Packet::owned(vec![0u8; 40 + $payload.len()]).unwrap(); // pnet_packet::ipv6::MutableIpv6Packet::owned(vec![0u8; 40 + $payload.len()]).unwrap();
output.set_version(6); // output.set_version(6);
output.set_traffic_class($traffic_class); // output.set_traffic_class($traffic_class);
output.set_flow_label($flow_label); // output.set_flow_label($flow_label);
output.set_next_header($next_header); // output.set_next_header($next_header);
output.set_hop_limit($hop_limit); // output.set_hop_limit($hop_limit);
output.set_source($source); // output.set_source($source);
output.set_destination($destination); // output.set_destination($destination);
output.set_payload_length($payload.len() as u16); // output.set_payload_length($payload.len() as u16);
output.set_payload($payload); // output.set_payload($payload);
pnet_packet::ipv6::Ipv6Packet::owned(output.to_immutable().packet().to_vec()).unwrap() // pnet_packet::ipv6::Ipv6Packet::owned(output.to_immutable().packet().to_vec()).unwrap()
}}; // }};
} // }
/// Quickly construct an IPv4 packet with the given parameters // /// Quickly construct an IPv4 packet with the given parameters
#[macro_export] // #[macro_export]
macro_rules! ipv4_packet { // macro_rules! ipv4_packet {
($source:expr, $destination:expr, $next_level_protocol:expr, $ttl:expr, $payload:expr) => { // ($source:expr, $destination:expr, $next_level_protocol:expr, $ttl:expr, $payload:expr) => {
ipv4_packet!( // ipv4_packet!(
$source, // $source,
$destination, // $destination,
0, // 0,
0, // 0,
0, // 0,
0, // 0,
0, // 0,
$ttl, // $ttl,
$next_level_protocol, // $next_level_protocol,
// &[], // // &[],
$payload // $payload
) // )
}; // };
// NOTE: Temporarily disabled options, since we aren't using them // // NOTE: Temporarily disabled options, since we aren't using them
// ($source:expr, $destination:expr, $dscp:expr, $ecn:expr, $identification:expr, $flags:expr, $fragment_offset:expr, $ttl:expr, $next_level_protocol:expr, $options:expr, $payload:expr) => {{ // // ($source:expr, $destination:expr, $dscp:expr, $ecn:expr, $identification:expr, $flags:expr, $fragment_offset:expr, $ttl:expr, $next_level_protocol:expr, $options:expr, $payload:expr) => {{
($source:expr, $destination:expr, $dscp:expr, $ecn:expr, $identification:expr, $flags:expr, $fragment_offset:expr, $ttl:expr, $next_level_protocol:expr, $payload:expr) => {{ // ($source:expr, $destination:expr, $dscp:expr, $ecn:expr, $identification:expr, $flags:expr, $fragment_offset:expr, $ttl:expr, $next_level_protocol:expr, $payload:expr) => {{
// let total_option_length = $options // // let total_option_length = $options
// .iter() // // .iter()
// .map(|o: pnet_packet::ipv4::Ipv4Option| pnet_packet::Packet::payload(o).len()) // // .map(|o: pnet_packet::ipv4::Ipv4Option| pnet_packet::Packet::payload(o).len())
// .sum::<usize>(); // // .sum::<usize>();
let total_option_length: usize = 0; // let total_option_length: usize = 0;
let mut output = pnet_packet::ipv4::MutableIpv4Packet::owned(vec![ // let mut output = pnet_packet::ipv4::MutableIpv4Packet::owned(vec![
0u8; // 0u8;
20 + total_option_length // 20 + total_option_length
+ $payload.len() // + $payload.len()
]) // ])
.unwrap(); // .unwrap();
output.set_version(4); // output.set_version(4);
output.set_header_length(((20 + total_option_length) / (32 / 8)) as u8); // Dynamic header length :( // output.set_header_length(((20 + total_option_length) / (32 / 8)) as u8); // Dynamic header length :(
output.set_dscp($dscp); // output.set_dscp($dscp);
output.set_ecn($ecn); // output.set_ecn($ecn);
output.set_total_length((20 + total_option_length + $payload.len()) as u16); // output.set_total_length((20 + total_option_length + $payload.len()) as u16);
output.set_identification($identification); // output.set_identification($identification);
output.set_flags($flags); // output.set_flags($flags);
output.set_fragment_offset($fragment_offset); // output.set_fragment_offset($fragment_offset);
output.set_ttl($ttl); // output.set_ttl($ttl);
output.set_next_level_protocol($next_level_protocol); // output.set_next_level_protocol($next_level_protocol);
output.set_source($source); // output.set_source($source);
output.set_destination($destination); // output.set_destination($destination);
// output.set_options($options); // // output.set_options($options);
output.set_payload($payload); // output.set_payload($payload);
output.set_checksum(0); // output.set_checksum(0);
output.set_checksum(pnet_packet::ipv4::checksum(&output.to_immutable())); // output.set_checksum(pnet_packet::ipv4::checksum(&output.to_immutable()));
pnet_packet::ipv4::Ipv4Packet::owned(output.to_immutable().packet().to_vec()).unwrap() // pnet_packet::ipv4::Ipv4Packet::owned(output.to_immutable().packet().to_vec()).unwrap()
}}; // }};
} // }
/// Quickly construct an ICMPv6 packet with the given parameters // /// Quickly construct an ICMPv6 packet with the given parameters
#[macro_export] // #[macro_export]
macro_rules! icmpv6_packet { // macro_rules! icmpv6_packet {
($source:expr, $destination:expr, $message_type:expr, $code:expr) => { // ($source:expr, $destination:expr, $message_type:expr, $code:expr) => {
icmpv6_packet!($source, $destination, $message_type, $code, &[0u8; 0]) // icmpv6_packet!($source, $destination, $message_type, $code, &[0u8; 0])
}; // };
($source:expr, $destination:expr, $message_type:expr, $code:expr, $payload:expr) => {{ // ($source:expr, $destination:expr, $message_type:expr, $code:expr, $payload:expr) => {{
let mut output = // let mut output =
pnet_packet::icmpv6::MutableIcmpv6Packet::owned(vec![0u8; 4 + $payload.len()]).unwrap(); // pnet_packet::icmpv6::MutableIcmpv6Packet::owned(vec![0u8; 4 + $payload.len()]).unwrap();
output.set_icmpv6_type($message_type); // output.set_icmpv6_type($message_type);
output.set_icmpv6_code($code); // output.set_icmpv6_code($code);
output.set_payload($payload); // output.set_payload($payload);
output.set_checksum(0); // output.set_checksum(0);
output.set_checksum(pnet_packet::icmpv6::checksum( // output.set_checksum(pnet_packet::icmpv6::checksum(
&output.to_immutable(), // &output.to_immutable(),
&$source, // &$source,
&$destination, // &$destination,
)); // ));
pnet_packet::icmpv6::Icmpv6Packet::owned(output.to_immutable().packet().to_vec()).unwrap() // pnet_packet::icmpv6::Icmpv6Packet::owned(output.to_immutable().packet().to_vec()).unwrap()
}}; // }};
} // }
/// Quickly construct an ICMP packet with the given parameters // /// Quickly construct an ICMP packet with the given parameters
#[macro_export] // #[macro_export]
macro_rules! icmp_packet { // macro_rules! icmp_packet {
($message_type:expr, $code:expr) => { // ($message_type:expr, $code:expr) => {
icmp_packet!($message_type, $code, &[0u8; 0]) // icmp_packet!($message_type, $code, &[0u8; 0])
}; // };
($message_type:expr, $code:expr, $payload:expr) => {{ // ($message_type:expr, $code:expr, $payload:expr) => {{
let mut output = // let mut output =
pnet_packet::icmp::MutableIcmpPacket::owned(vec![0u8; 4 + $payload.len()]).unwrap(); // pnet_packet::icmp::MutableIcmpPacket::owned(vec![0u8; 4 + $payload.len()]).unwrap();
output.set_icmp_type($message_type); // output.set_icmp_type($message_type);
output.set_icmp_code($code); // output.set_icmp_code($code);
output.set_payload($payload); // output.set_payload($payload);
output.set_checksum(0); // output.set_checksum(0);
output.set_checksum(pnet_packet::icmp::checksum(&output.to_immutable())); // output.set_checksum(pnet_packet::icmp::checksum(&output.to_immutable()));
pnet_packet::icmp::IcmpPacket::owned(output.to_immutable().packet().to_vec()).unwrap() // pnet_packet::icmp::IcmpPacket::owned(output.to_immutable().packet().to_vec()).unwrap()
}}; // }};
} // }

View File

@ -1,6 +1,6 @@
use crate::packet::protocols::{ use crate::packet::{
icmp::IcmpPacket, icmpv6::Icmpv6Packet, ipv4::Ipv4Packet, ipv6::Ipv6Packet, raw::RawBytes, protocols::{ipv4::Ipv4Packet, ipv6::Ipv6Packet},
tcp::TcpPacket, udp::UdpPacket, xlat::ip::{translate_ipv4_to_ipv6, translate_ipv6_to_ipv4},
}; };
use self::{ use self::{
@ -8,7 +8,6 @@ use self::{
utils::{embed_address, extract_address}, utils::{embed_address, extract_address},
}; };
use ipnet::{Ipv4Net, Ipv6Net}; use ipnet::{Ipv4Net, Ipv6Net};
use pnet_packet::ip::IpNextHeaderProtocols;
use protomask_tun::TunDevice; use protomask_tun::TunDevice;
use std::{ use std::{
net::{IpAddr, Ipv4Addr, Ipv6Addr}, net::{IpAddr, Ipv4Addr, Ipv6Addr},
@ -31,12 +30,14 @@ pub enum Nat64Error {
TunError(#[from] protomask_tun::Error), TunError(#[from] protomask_tun::Error),
#[error(transparent)] #[error(transparent)]
IoError(#[from] std::io::Error), IoError(#[from] std::io::Error),
#[error(transparent)] // #[error(transparent)]
XlatError(#[from] xlat::PacketTranslationError), // XlatError(#[from] xlat::PacketTranslationError),
#[error(transparent)] #[error(transparent)]
PacketHandlingError(#[from] crate::packet::error::PacketError), PacketHandlingError(#[from] crate::packet::error::PacketError),
#[error(transparent)] #[error(transparent)]
PacketReceiveError(#[from] broadcast::error::RecvError), PacketReceiveError(#[from] broadcast::error::RecvError),
#[error(transparent)]
PacketSendError(#[from] mpsc::error::SendError<Vec<u8>>),
} }
pub struct Nat64 { pub struct Nat64 {
@ -88,7 +89,7 @@ impl Nat64 {
let packet = rx.recv().await?; let packet = rx.recv().await?;
// Clone the TX so the worker can respond with data // Clone the TX so the worker can respond with data
let mut tx = tx.clone(); let tx = tx.clone();
// Separate logic is needed for handling IPv4 vs IPv6 packets, so a check must be done here // Separate logic is needed for handling IPv4 vs IPv6 packets, so a check must be done here
match packet[0] >> 4 { match packet[0] >> 4 {
@ -107,9 +108,9 @@ impl Nat64 {
// Spawn a task to process the packet // Spawn a task to process the packet
tokio::spawn(async move { tokio::spawn(async move {
process_inbound_ipv4(packet, new_source, new_destination, &mut tx) let output =
.await translate_ipv4_to_ipv6(packet, new_source, new_destination).unwrap();
.unwrap(); tx.send(output.into()).await.unwrap();
}); });
} }
6 => { 6 => {
@ -122,9 +123,9 @@ impl Nat64 {
// Spawn a task to process the packet // Spawn a task to process the packet
tokio::spawn(async move { tokio::spawn(async move {
process_inbound_ipv6(packet, new_source, new_destination, &mut tx) let output =
.await translate_ipv6_to_ipv4(packet, new_source, new_destination).unwrap();
.unwrap(); tx.send(output.into()).await.unwrap();
}); });
} }
n => { n => {
@ -132,85 +133,5 @@ impl Nat64 {
} }
} }
} }
Ok(())
} }
} }
/// Process an inbound IPv4 packet
async fn process_inbound_ipv4(
packet: Ipv4Packet<Vec<u8>>,
new_source: Ipv6Addr,
new_destination: Ipv6Addr,
tx: &mut mpsc::Sender<Vec<u8>>,
) -> Result<(), Nat64Error> {
// Handle each possible embedded packet type
match packet.protocol {
IpNextHeaderProtocols::Icmp => {
let icmp_packet: IcmpPacket<RawBytes> = packet.payload.try_into()?;
todo!()
}
IpNextHeaderProtocols::Udp => {
let udp_packet: UdpPacket<RawBytes> = UdpPacket::new_from_bytes_raw_payload(
&packet.payload,
IpAddr::V4(packet.source_address),
IpAddr::V4(packet.destination_address),
)?;
todo!()
}
IpNextHeaderProtocols::Tcp => {
let tcp_packet: TcpPacket<RawBytes> = TcpPacket::new_from_bytes_raw_payload(
&packet.payload,
IpAddr::V4(packet.source_address),
IpAddr::V4(packet.destination_address),
)?;
todo!()
}
_ => {
log::warn!("Unsupported next level protocol: {}", packet.protocol);
}
}
Ok(())
}
/// Process an inbound IPv6 packet
async fn process_inbound_ipv6(
packet: Ipv6Packet<Vec<u8>>,
new_source: Ipv4Addr,
new_destination: Ipv4Addr,
tx: &mut mpsc::Sender<Vec<u8>>,
) -> Result<(), Nat64Error> {
// Handle each possible embedded packet type
match packet.next_header {
IpNextHeaderProtocols::Icmpv6 => {
let icmpv6_packet: Icmpv6Packet<RawBytes> = Icmpv6Packet::new_from_bytes_raw_payload(
&packet.payload,
packet.source_address,
packet.destination_address,
)?;
todo!()
}
IpNextHeaderProtocols::Udp => {
let udp_packet: UdpPacket<RawBytes> = UdpPacket::new_from_bytes_raw_payload(
&packet.payload,
IpAddr::V6(packet.source_address),
IpAddr::V6(packet.destination_address),
)?;
todo!()
}
IpNextHeaderProtocols::Tcp => {
let tcp_packet: TcpPacket<RawBytes> = TcpPacket::new_from_bytes_raw_payload(
&packet.payload,
IpAddr::V6(packet.source_address),
IpAddr::V6(packet.destination_address),
)?;
todo!()
}
_ => {
log::warn!("Unsupported next level protocol: {}", packet.next_header);
}
}
Ok(())
}

View File

@ -15,108 +15,108 @@ use crate::{icmp_packet, icmpv6_packet, ipv4_packet, ipv6_packet};
use super::PacketTranslationError; use super::PacketTranslationError;
/// Best effort translation from an ICMP type and code to an ICMPv6 type and code // /// Best effort translation from an ICMP type and code to an ICMPv6 type and code
fn translate_type_and_code_4_to_6( // fn translate_type_and_code_4_to_6(
icmp_type: IcmpType, // icmp_type: IcmpType,
icmp_code: IcmpCode, // icmp_code: IcmpCode,
) -> Option<(Icmpv6Type, Icmpv6Code)> { // ) -> Option<(Icmpv6Type, Icmpv6Code)> {
match (icmp_type, icmp_code) { // match (icmp_type, icmp_code) {
// Echo Request // // Echo Request
(IcmpTypes::EchoRequest, _) => Some((Icmpv6Types::EchoRequest, Icmpv6Code(0))), // (IcmpTypes::EchoRequest, _) => Some((Icmpv6Types::EchoRequest, Icmpv6Code(0))),
// Echo Reply // // Echo Reply
(IcmpTypes::EchoReply, _) => Some((Icmpv6Types::EchoReply, Icmpv6Code(0))), // (IcmpTypes::EchoReply, _) => Some((Icmpv6Types::EchoReply, Icmpv6Code(0))),
// Packet Too Big // // Packet Too Big
( // (
IcmpTypes::DestinationUnreachable, // IcmpTypes::DestinationUnreachable,
destination_unreachable::IcmpCodes::FragmentationRequiredAndDFFlagSet, // destination_unreachable::IcmpCodes::FragmentationRequiredAndDFFlagSet,
) => Some((Icmpv6Types::PacketTooBig, Icmpv6Code(0))), // ) => Some((Icmpv6Types::PacketTooBig, Icmpv6Code(0))),
// Destination Unreachable // // Destination Unreachable
(IcmpTypes::DestinationUnreachable, icmp_code) => Some(( // (IcmpTypes::DestinationUnreachable, icmp_code) => Some((
Icmpv6Types::DestinationUnreachable, // Icmpv6Types::DestinationUnreachable,
#[cfg_attr(rustfmt, rustfmt_skip)] // #[cfg_attr(rustfmt, rustfmt_skip)]
Icmpv6Code(match icmp_code { // Icmpv6Code(match icmp_code {
destination_unreachable::IcmpCodes::DestinationHostUnreachable => 3, // destination_unreachable::IcmpCodes::DestinationHostUnreachable => 3,
destination_unreachable::IcmpCodes::DestinationProtocolUnreachable => 4, // destination_unreachable::IcmpCodes::DestinationProtocolUnreachable => 4,
destination_unreachable::IcmpCodes::DestinationPortUnreachable => 4, // destination_unreachable::IcmpCodes::DestinationPortUnreachable => 4,
destination_unreachable::IcmpCodes::SourceRouteFailed => 5, // destination_unreachable::IcmpCodes::SourceRouteFailed => 5,
destination_unreachable::IcmpCodes::SourceHostIsolated => 2, // destination_unreachable::IcmpCodes::SourceHostIsolated => 2,
destination_unreachable::IcmpCodes::NetworkAdministrativelyProhibited => 1, // destination_unreachable::IcmpCodes::NetworkAdministrativelyProhibited => 1,
destination_unreachable::IcmpCodes::HostAdministrativelyProhibited => 1, // destination_unreachable::IcmpCodes::HostAdministrativelyProhibited => 1,
destination_unreachable::IcmpCodes::CommunicationAdministrativelyProhibited => 1, // destination_unreachable::IcmpCodes::CommunicationAdministrativelyProhibited => 1,
// Default to No Route to Destination // // Default to No Route to Destination
_ => 0, // _ => 0,
}), // }),
)), // )),
// Time Exceeded // // Time Exceeded
(IcmpTypes::TimeExceeded, icmp_code) => { // (IcmpTypes::TimeExceeded, icmp_code) => {
Some((Icmpv6Types::TimeExceeded, Icmpv6Code(icmp_code.0))) // Some((Icmpv6Types::TimeExceeded, Icmpv6Code(icmp_code.0)))
} // }
// Default unsupported // // Default unsupported
_ => { // _ => {
log::warn!( // log::warn!(
"Unsupported ICMP code and type: {:?}, {:?}", // "Unsupported ICMP code and type: {:?}, {:?}",
icmp_type, // icmp_type,
icmp_code // icmp_code
); // );
None // None
} // }
} // }
} // }
/// Best effort translation from an ICMPv6 type and code to an ICMP type and code // /// Best effort translation from an ICMPv6 type and code to an ICMP type and code
fn translate_type_and_code_6_to_4( // fn translate_type_and_code_6_to_4(
icmp_type: Icmpv6Type, // icmp_type: Icmpv6Type,
icmp_code: Icmpv6Code, // icmp_code: Icmpv6Code,
) -> Option<(IcmpType, IcmpCode)> { // ) -> Option<(IcmpType, IcmpCode)> {
match (icmp_type, icmp_code) { // match (icmp_type, icmp_code) {
// Echo Request // // Echo Request
(Icmpv6Types::EchoRequest, _) => Some((IcmpTypes::EchoRequest, IcmpCode(0))), // (Icmpv6Types::EchoRequest, _) => Some((IcmpTypes::EchoRequest, IcmpCode(0))),
// Echo Reply // // Echo Reply
(Icmpv6Types::EchoReply, _) => Some((IcmpTypes::EchoReply, IcmpCode(0))), // (Icmpv6Types::EchoReply, _) => Some((IcmpTypes::EchoReply, IcmpCode(0))),
// Packet Too Big // // Packet Too Big
(Icmpv6Types::PacketTooBig, _) => Some(( // (Icmpv6Types::PacketTooBig, _) => Some((
IcmpTypes::DestinationUnreachable, // IcmpTypes::DestinationUnreachable,
destination_unreachable::IcmpCodes::FragmentationRequiredAndDFFlagSet, // destination_unreachable::IcmpCodes::FragmentationRequiredAndDFFlagSet,
)), // )),
// Destination Unreachable // // Destination Unreachable
(Icmpv6Types::DestinationUnreachable, icmp_code) => Some(( // (Icmpv6Types::DestinationUnreachable, icmp_code) => Some((
IcmpTypes::DestinationUnreachable, // IcmpTypes::DestinationUnreachable,
#[cfg_attr(rustfmt, rustfmt_skip)] // #[cfg_attr(rustfmt, rustfmt_skip)]
match icmp_code.0 { // match icmp_code.0 {
1 => destination_unreachable::IcmpCodes::CommunicationAdministrativelyProhibited, // 1 => destination_unreachable::IcmpCodes::CommunicationAdministrativelyProhibited,
2 => destination_unreachable::IcmpCodes::SourceHostIsolated, // 2 => destination_unreachable::IcmpCodes::SourceHostIsolated,
3 => destination_unreachable::IcmpCodes::DestinationHostUnreachable, // 3 => destination_unreachable::IcmpCodes::DestinationHostUnreachable,
4 => destination_unreachable::IcmpCodes::DestinationPortUnreachable, // 4 => destination_unreachable::IcmpCodes::DestinationPortUnreachable,
5 => destination_unreachable::IcmpCodes::SourceRouteFailed, // 5 => destination_unreachable::IcmpCodes::SourceRouteFailed,
_ => destination_unreachable::IcmpCodes::DestinationNetworkUnreachable, // _ => destination_unreachable::IcmpCodes::DestinationNetworkUnreachable,
}, // },
)), // )),
// Time Exceeded // // Time Exceeded
(Icmpv6Types::TimeExceeded, icmp_code) => { // (Icmpv6Types::TimeExceeded, icmp_code) => {
Some((IcmpTypes::TimeExceeded, IcmpCode(icmp_code.0))) // Some((IcmpTypes::TimeExceeded, IcmpCode(icmp_code.0)))
} // }
// Default unsupported // // Default unsupported
_ => { // _ => {
log::warn!( // log::warn!(
"Unsupported ICMPv6 code and type: {:?}, {:?}", // "Unsupported ICMPv6 code and type: {:?}, {:?}",
icmp_type, // icmp_type,
icmp_code // icmp_code
); // );
None // None
} // }
} // }
} // }
/// Translate an ICMP packet into an ICMPv6 packet /// Translate an ICMP packet into an ICMPv6 packet
pub fn translate_icmp_4_to_6( pub fn translate_icmp_4_to_6(

View File

@ -1,17 +1,17 @@
//! Packet type translation functionality // //! Packet type translation functionality
mod icmp; // mod icmp;
mod tcp; // mod tcp;
mod udp; // mod udp;
pub use icmp::{translate_icmp_4_to_6, translate_icmp_6_to_4}; // pub use icmp::{translate_icmp_4_to_6, translate_icmp_6_to_4};
pub use tcp::{translate_tcp_4_to_6, translate_tcp_6_to_4}; // // pub use tcp::{translate_tcp_4_to_6, translate_tcp_6_to_4};
pub use udp::{translate_udp_4_to_6, translate_udp_6_to_4}; // // pub use udp::{translate_udp_4_to_6, translate_udp_6_to_4};
#[derive(Debug, thiserror::Error)] // #[derive(Debug, thiserror::Error)]
pub enum PacketTranslationError { // pub enum PacketTranslationError {
#[error("Input packet too short. Got {0} bytes")] // #[error("Input packet too short. Got {0} bytes")]
InputPacketTooShort(usize), // InputPacketTooShort(usize),
#[error("Embedded packet too short. Got {0} bytes")] // #[error("Embedded packet too short. Got {0} bytes")]
EmbeddedPacketTooShort(usize), // EmbeddedPacketTooShort(usize),
} // }

View File

@ -1,52 +1,52 @@
use std::net::{Ipv4Addr, Ipv6Addr}; // use std::net::{Ipv4Addr, Ipv6Addr};
use pnet_packet::{ // use pnet_packet::{
tcp::{self, MutableTcpPacket, TcpPacket}, // tcp::{self, MutableTcpPacket, TcpPacket},
Packet, // Packet,
}; // };
use super::PacketTranslationError; // use super::PacketTranslationError;
/// Translate an IPv4 TCP packet into an IPv6 TCP packet (aka: recalculate checksum) // /// Translate an IPv4 TCP packet into an IPv6 TCP packet (aka: recalculate checksum)
pub fn translate_tcp_4_to_6( // pub fn translate_tcp_4_to_6(
ipv4_tcp: TcpPacket, // ipv4_tcp: TcpPacket,
new_source: Ipv6Addr, // new_source: Ipv6Addr,
new_dest: Ipv6Addr, // new_dest: Ipv6Addr,
) -> Result<TcpPacket, PacketTranslationError> { // ) -> Result<TcpPacket, PacketTranslationError> {
// Create a mutable clone of the IPv4 TCP packet, so it can be adapted for use in IPv6 // // Create a mutable clone of the IPv4 TCP packet, so it can be adapted for use in IPv6
let mut ipv6_tcp = MutableTcpPacket::owned(ipv4_tcp.packet().to_vec()) // let mut ipv6_tcp = MutableTcpPacket::owned(ipv4_tcp.packet().to_vec())
.ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv4_tcp.packet().len()))?; // .ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv4_tcp.packet().len()))?;
// Rewrite the checksum for use in an IPv6 packet // // Rewrite the checksum for use in an IPv6 packet
ipv6_tcp.set_checksum(0); // ipv6_tcp.set_checksum(0);
ipv6_tcp.set_checksum(tcp::ipv6_checksum( // ipv6_tcp.set_checksum(tcp::ipv6_checksum(
&ipv4_tcp.to_immutable(), // &ipv4_tcp.to_immutable(),
&new_source, // &new_source,
&new_dest, // &new_dest,
)); // ));
// Return the translated packet // // Return the translated packet
Ok(TcpPacket::owned(ipv6_tcp.packet().to_vec()).unwrap()) // Ok(TcpPacket::owned(ipv6_tcp.packet().to_vec()).unwrap())
} // }
/// Translate an IPv6 TCP packet into an IPv4 TCP packet (aka: recalculate checksum) // /// Translate an IPv6 TCP packet into an IPv4 TCP packet (aka: recalculate checksum)
pub fn translate_tcp_6_to_4( // pub fn translate_tcp_6_to_4(
ipv6_tcp: TcpPacket, // ipv6_tcp: TcpPacket,
new_source: Ipv4Addr, // new_source: Ipv4Addr,
new_dest: Ipv4Addr, // new_dest: Ipv4Addr,
) -> Result<TcpPacket, PacketTranslationError> { // ) -> Result<TcpPacket, PacketTranslationError> {
// Create a mutable clone of the IPv6 TCP packet, so it can be adapted for use in IPv4 // // Create a mutable clone of the IPv6 TCP packet, so it can be adapted for use in IPv4
let mut ipv4_tcp = MutableTcpPacket::owned(ipv6_tcp.packet().to_vec()) // let mut ipv4_tcp = MutableTcpPacket::owned(ipv6_tcp.packet().to_vec())
.ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv6_tcp.packet().len()))?; // .ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv6_tcp.packet().len()))?;
// Rewrite the checksum for use in an IPv4 packet // // Rewrite the checksum for use in an IPv4 packet
ipv4_tcp.set_checksum(0); // ipv4_tcp.set_checksum(0);
ipv4_tcp.set_checksum(tcp::ipv4_checksum( // ipv4_tcp.set_checksum(tcp::ipv4_checksum(
&ipv6_tcp.to_immutable(), // &ipv6_tcp.to_immutable(),
&new_source, // &new_source,
&new_dest, // &new_dest,
)); // ));
// Return the translated packet // // Return the translated packet
Ok(TcpPacket::owned(ipv4_tcp.packet().to_vec()).unwrap()) // Ok(TcpPacket::owned(ipv4_tcp.packet().to_vec()).unwrap())
} // }

View File

@ -1,117 +1,117 @@
use super::PacketTranslationError; // use super::PacketTranslationError;
use pnet_packet::{ // use pnet_packet::{
udp::{self, MutableUdpPacket, UdpPacket}, // udp::{self, MutableUdpPacket, UdpPacket},
Packet, // Packet,
}; // };
use std::net::{Ipv4Addr, Ipv6Addr}; // use std::net::{Ipv4Addr, Ipv6Addr};
/// Translate an IPv4 UDP packet into an IPv6 UDP packet (aka: recalculate checksum) // /// Translate an IPv4 UDP packet into an IPv6 UDP packet (aka: recalculate checksum)
pub fn translate_udp_4_to_6( // pub fn translate_udp_4_to_6(
ipv4_udp: UdpPacket, // ipv4_udp: UdpPacket,
new_source: Ipv6Addr, // new_source: Ipv6Addr,
new_dest: Ipv6Addr, // new_dest: Ipv6Addr,
) -> Result<UdpPacket, PacketTranslationError> { // ) -> Result<UdpPacket, PacketTranslationError> {
// Create a mutable clone of the IPv4 UDP packet, so it can be adapted for use in IPv6 // // Create a mutable clone of the IPv4 UDP packet, so it can be adapted for use in IPv6
let mut ipv6_udp = MutableUdpPacket::owned(ipv4_udp.packet().to_vec()) // let mut ipv6_udp = MutableUdpPacket::owned(ipv4_udp.packet().to_vec())
.ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv4_udp.packet().len()))?; // .ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv4_udp.packet().len()))?;
// Rewrite the checksum for use in an IPv6 packet // // Rewrite the checksum for use in an IPv6 packet
ipv6_udp.set_checksum(0); // ipv6_udp.set_checksum(0);
ipv6_udp.set_checksum(udp::ipv6_checksum( // ipv6_udp.set_checksum(udp::ipv6_checksum(
&ipv4_udp.to_immutable(), // &ipv4_udp.to_immutable(),
&new_source, // &new_source,
&new_dest, // &new_dest,
)); // ));
// Return the translated packet // // Return the translated packet
Ok(UdpPacket::owned(ipv6_udp.packet().to_vec()).unwrap()) // Ok(UdpPacket::owned(ipv6_udp.packet().to_vec()).unwrap())
} // }
/// Translate an IPv6 UDP packet into an IPv4 UDP packet (aka: recalculate checksum) // /// Translate an IPv6 UDP packet into an IPv4 UDP packet (aka: recalculate checksum)
pub fn translate_udp_6_to_4( // pub fn translate_udp_6_to_4(
ipv6_udp: UdpPacket, // ipv6_udp: UdpPacket,
new_source: Ipv4Addr, // new_source: Ipv4Addr,
new_dest: Ipv4Addr, // new_dest: Ipv4Addr,
) -> Result<UdpPacket, PacketTranslationError> { // ) -> Result<UdpPacket, PacketTranslationError> {
// Create a mutable clone of the IPv6 UDP packet, so it can be adapted for use in IPv4 // // Create a mutable clone of the IPv6 UDP packet, so it can be adapted for use in IPv4
let mut ipv4_udp = MutableUdpPacket::owned(ipv6_udp.packet().to_vec()) // let mut ipv4_udp = MutableUdpPacket::owned(ipv6_udp.packet().to_vec())
.ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv6_udp.packet().len()))?; // .ok_or_else(|| PacketTranslationError::InputPacketTooShort(ipv6_udp.packet().len()))?;
// Rewrite the checksum for use in an IPv4 packet // // Rewrite the checksum for use in an IPv4 packet
ipv4_udp.set_checksum(0); // ipv4_udp.set_checksum(0);
ipv4_udp.set_checksum(udp::ipv4_checksum( // ipv4_udp.set_checksum(udp::ipv4_checksum(
&ipv6_udp.to_immutable(), // &ipv6_udp.to_immutable(),
&new_source, // &new_source,
&new_dest, // &new_dest,
)); // ));
// Return the translated packet // // Return the translated packet
Ok(UdpPacket::owned(ipv4_udp.packet().to_vec()).unwrap()) // Ok(UdpPacket::owned(ipv4_udp.packet().to_vec()).unwrap())
} // }
#[cfg(test)] // #[cfg(test)]
mod tests { // mod tests {
use crate::into_udp; // use crate::into_udp;
use super::*; // use super::*;
#[test] // #[test]
fn test_udp_4_to_6() { // fn test_udp_4_to_6() {
// Build an example UDP packet // // Build an example UDP packet
let input = into_udp!(vec![ // let input = into_udp!(vec![
0, 255, // Source port // 0, 255, // Source port
0, 128, // Destination port // 0, 128, // Destination port
0, 4, // Length // 0, 4, // Length
0, 0, // Checksum (doesn't matter) // 0, 0, // Checksum (doesn't matter)
1, 2, 3, 4 // Data // 1, 2, 3, 4 // Data
]) // ])
.unwrap(); // .unwrap();
// Translate to IPv6 // // Translate to IPv6
let output = translate_udp_4_to_6( // let output = translate_udp_4_to_6(
input, // input,
"2001:db8::1".parse().unwrap(), // "2001:db8::1".parse().unwrap(),
"2001:db8::2".parse().unwrap(), // "2001:db8::2".parse().unwrap(),
); // );
// Check the output // // Check the output
assert!(output.is_ok()); // assert!(output.is_ok());
let output = output.unwrap(); // let output = output.unwrap();
// Check the output's contents // // Check the output's contents
assert_eq!(output.get_source(), 255); // assert_eq!(output.get_source(), 255);
assert_eq!(output.get_destination(), 128); // assert_eq!(output.get_destination(), 128);
assert_eq!(output.get_length(), 4); // assert_eq!(output.get_length(), 4);
assert_eq!(output.payload(), &[1, 2, 3, 4]); // assert_eq!(output.payload(), &[1, 2, 3, 4]);
} // }
#[test] // #[test]
fn test_udp_6_to_4() { // fn test_udp_6_to_4() {
// Build an example UDP packet // // Build an example UDP packet
let input = into_udp!(vec![ // let input = into_udp!(vec![
0, 255, // Source port // 0, 255, // Source port
0, 128, // Destination port // 0, 128, // Destination port
0, 4, // Length // 0, 4, // Length
0, 0, // Checksum (doesn't matter) // 0, 0, // Checksum (doesn't matter)
1, 2, 3, 4 // Data // 1, 2, 3, 4 // Data
]) // ])
.unwrap(); // .unwrap();
// Translate to IPv4 // // Translate to IPv4
let output = translate_udp_6_to_4( // let output = translate_udp_6_to_4(
input, // input,
"192.0.2.1".parse().unwrap(), // "192.0.2.1".parse().unwrap(),
"192.0.2.2".parse().unwrap(), // "192.0.2.2".parse().unwrap(),
); // );
// Check the output // // Check the output
assert!(output.is_ok()); // assert!(output.is_ok());
let output = output.unwrap(); // let output = output.unwrap();
// Check the output's contents // // Check the output's contents
assert_eq!(output.get_source(), 255); // assert_eq!(output.get_source(), 255);
assert_eq!(output.get_destination(), 128); // assert_eq!(output.get_destination(), 128);
assert_eq!(output.get_length(), 4); // assert_eq!(output.get_length(), 4);
assert_eq!(output.payload(), &[1, 2, 3, 4]); // assert_eq!(output.payload(), &[1, 2, 3, 4]);
} // }
} // }

View File

@ -6,5 +6,9 @@ pub enum PacketError {
MismatchedAddressFamily(IpAddr, IpAddr), MismatchedAddressFamily(IpAddr, IpAddr),
#[error("Packet too short: {0}")] #[error("Packet too short: {0}")]
TooShort(usize), TooShort(usize),
#[error("Unsupported ICMP type: {0}")]
UnsupportedIcmpType(u8),
#[error("Unsupported ICMPv6 type: {0}")]
UnsupportedIcmpv6Type(u8),
} }

View File

@ -4,3 +4,4 @@
pub mod error; pub mod error;
pub mod protocols; pub mod protocols;
pub mod xlat;

View File

@ -10,3 +10,9 @@ impl TryFrom<Vec<u8>> for RawBytes {
Ok(Self(bytes)) Ok(Self(bytes))
} }
} }
impl Into<Vec<u8>> for RawBytes {
fn into(self) -> Vec<u8> {
self.0
}
}

View File

@ -182,15 +182,18 @@ impl TcpPacket<RawBytes> {
impl<T> Into<Vec<u8>> for TcpPacket<T> impl<T> Into<Vec<u8>> for TcpPacket<T>
where where
T: Into<Vec<u8>> + Copy, T: Into<Vec<u8>> ,
{ {
fn into(self) -> Vec<u8> { fn into(self) -> Vec<u8> {
// Get the options length in words
let options_length_words = self.options_length_words();
// Convert the payload into raw bytes // Convert the payload into raw bytes
let payload: Vec<u8> = self.payload.into(); let payload: Vec<u8> = self.payload.into();
// Allocate a mutable packet to write into // Allocate a mutable packet to write into
let total_length = pnet_packet::tcp::MutableTcpPacket::minimum_packet_size() let total_length = pnet_packet::tcp::MutableTcpPacket::minimum_packet_size()
+ (self.options_length_words() as usize * 4) + (options_length_words as usize * 4)
+ payload.len(); + payload.len();
let mut output = let mut output =
pnet_packet::tcp::MutableTcpPacket::owned(vec![0u8; total_length]).unwrap(); pnet_packet::tcp::MutableTcpPacket::owned(vec![0u8; total_length]).unwrap();
@ -207,7 +210,7 @@ where
output.set_options(&self.options); output.set_options(&self.options);
// Write the offset // Write the offset
output.set_data_offset(5 + self.options_length_words()); output.set_data_offset(5 + options_length_words);
// Write the flags // Write the flags
output.set_flags(self.flags.into()); output.set_flags(self.flags.into());

View File

@ -0,0 +1,92 @@
use std::net::{Ipv4Addr, Ipv6Addr};
use pnet_packet::{icmp::IcmpTypes, icmpv6::Icmpv6Types};
use crate::packet::{
error::PacketError,
protocols::{icmp::IcmpPacket, icmpv6::Icmpv6Packet, raw::RawBytes},
};
use super::ip::{translate_ipv4_to_ipv6, translate_ipv6_to_ipv4};
mod type_code;
/// Translates an ICMP packet to an ICMPv6 packet
pub fn translate_icmp_to_icmpv6(
input: IcmpPacket<RawBytes>,
new_source: Ipv6Addr,
new_destination: Ipv6Addr,
) -> Result<Icmpv6Packet<RawBytes>, PacketError> {
// Translate the type and code
let (icmpv6_type, icmpv6_code) =
type_code::translate_type_and_code_4_to_6(input.icmp_type, input.icmp_code)?;
// Some ICMP types require special payload edits
let payload = match icmpv6_type {
Icmpv6Types::TimeExceeded => {
// In this case, the current payload looks like: 4bytes + Ipv4(Data)
// This needs to be translated to: 4bytes + Ipv6(Data)
let inner_payload = input.payload.0[4..].to_vec();
// Translate
let inner_payload =
translate_ipv4_to_ipv6(inner_payload.try_into()?, new_source, new_destination)?;
let inner_payload: Vec<u8> = inner_payload.into();
// Build the new payload
RawBytes({
let mut buffer = Vec::with_capacity(4 + inner_payload.len());
buffer.extend_from_slice(&input.payload.0[..4]);
buffer.extend_from_slice(&inner_payload);
buffer
})
}
_ => input.payload,
};
// Build output packet
Ok(Icmpv6Packet::new(
new_source,
new_destination,
icmpv6_type,
icmpv6_code,
payload,
))
}
/// Translates an ICMPv6 packet to an ICMP packet
pub fn translate_icmpv6_to_icmp(
input: Icmpv6Packet<RawBytes>,
new_source: Ipv4Addr,
new_destination: Ipv4Addr,
) -> Result<IcmpPacket<RawBytes>, PacketError> {
// Translate the type and code
let (icmp_type, icmp_code) =
type_code::translate_type_and_code_6_to_4(input.icmp_type, input.icmp_code)?;
// Some ICMP types require special payload edits
let payload = match icmp_type {
IcmpTypes::TimeExceeded => {
// In this case, the current payload looks like: 4bytes + Ipv6(Data)
// This needs to be translated to: 4bytes + Ipv4(Data)
let inner_payload = input.payload.0[4..].to_vec();
// Translate
let inner_payload =
translate_ipv6_to_ipv4(inner_payload.try_into()?, new_source, new_destination)?;
let inner_payload: Vec<u8> = inner_payload.into();
// Build the new payload
RawBytes({
let mut buffer = Vec::with_capacity(4 + inner_payload.len());
buffer.extend_from_slice(&input.payload.0[..4]);
buffer.extend_from_slice(&inner_payload);
buffer
})
},
_ => input.payload,
};
// Build output packet
Ok(IcmpPacket::new(icmp_type, icmp_code, payload))
}

View File

@ -0,0 +1,97 @@
//! Functions to map between ICMP and ICMPv6 types/codes
use pnet_packet::{
icmp::{destination_unreachable, IcmpCode, IcmpType, IcmpTypes},
icmpv6::{Icmpv6Code, Icmpv6Type, Icmpv6Types},
};
use crate::packet::error::PacketError;
/// Best effort translation from an ICMP type and code to an ICMPv6 type and code
pub fn translate_type_and_code_4_to_6(
icmp_type: IcmpType,
icmp_code: IcmpCode,
) -> Result<(Icmpv6Type, Icmpv6Code), PacketError> {
match (icmp_type, icmp_code) {
// Echo Request
(IcmpTypes::EchoRequest, _) => Ok((Icmpv6Types::EchoRequest, Icmpv6Code(0))),
// Echo Reply
(IcmpTypes::EchoReply, _) => Ok((Icmpv6Types::EchoReply, Icmpv6Code(0))),
// Packet Too Big
(
IcmpTypes::DestinationUnreachable,
destination_unreachable::IcmpCodes::FragmentationRequiredAndDFFlagSet,
) => Ok((Icmpv6Types::PacketTooBig, Icmpv6Code(0))),
// Destination Unreachable
(IcmpTypes::DestinationUnreachable, icmp_code) => Ok((
Icmpv6Types::DestinationUnreachable,
#[cfg_attr(rustfmt, rustfmt_skip)]
Icmpv6Code(match icmp_code {
destination_unreachable::IcmpCodes::DestinationHostUnreachable => 3,
destination_unreachable::IcmpCodes::DestinationProtocolUnreachable => 4,
destination_unreachable::IcmpCodes::DestinationPortUnreachable => 4,
destination_unreachable::IcmpCodes::SourceRouteFailed => 5,
destination_unreachable::IcmpCodes::SourceHostIsolated => 2,
destination_unreachable::IcmpCodes::NetworkAdministrativelyProhibited => 1,
destination_unreachable::IcmpCodes::HostAdministrativelyProhibited => 1,
destination_unreachable::IcmpCodes::CommunicationAdministrativelyProhibited => 1,
// Default to No Route to Destination
_ => 0,
}),
)),
// Time Exceeded
(IcmpTypes::TimeExceeded, icmp_code) => {
Ok((Icmpv6Types::TimeExceeded, Icmpv6Code(icmp_code.0)))
}
// Default unsupported
(icmp_type, _) => Err(PacketError::UnsupportedIcmpType(icmp_type.0)),
}
}
/// Best effort translation from an ICMPv6 type and code to an ICMP type and code
pub fn translate_type_and_code_6_to_4(
icmp_type: Icmpv6Type,
icmp_code: Icmpv6Code,
) -> Result<(IcmpType, IcmpCode), PacketError> {
match (icmp_type, icmp_code) {
// Echo Request
(Icmpv6Types::EchoRequest, _) => Ok((IcmpTypes::EchoRequest, IcmpCode(0))),
// Echo Reply
(Icmpv6Types::EchoReply, _) => Ok((IcmpTypes::EchoReply, IcmpCode(0))),
// Packet Too Big
(Icmpv6Types::PacketTooBig, _) => Ok((
IcmpTypes::DestinationUnreachable,
destination_unreachable::IcmpCodes::FragmentationRequiredAndDFFlagSet,
)),
// Destination Unreachable
(Icmpv6Types::DestinationUnreachable, icmp_code) => Ok((
IcmpTypes::DestinationUnreachable,
#[cfg_attr(rustfmt, rustfmt_skip)]
match icmp_code.0 {
1 => destination_unreachable::IcmpCodes::CommunicationAdministrativelyProhibited,
2 => destination_unreachable::IcmpCodes::SourceHostIsolated,
3 => destination_unreachable::IcmpCodes::DestinationHostUnreachable,
4 => destination_unreachable::IcmpCodes::DestinationPortUnreachable,
5 => destination_unreachable::IcmpCodes::SourceRouteFailed,
_ => destination_unreachable::IcmpCodes::DestinationNetworkUnreachable,
},
)),
// Time Exceeded
(Icmpv6Types::TimeExceeded, icmp_code) => {
Ok((IcmpTypes::TimeExceeded, IcmpCode(icmp_code.0)))
}
// Default unsupported
(icmp_type, _) => Err(PacketError::UnsupportedIcmpv6Type(icmp_type.0)),
}
}

129
src/packet/xlat/ip.rs Normal file
View File

@ -0,0 +1,129 @@
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr};
use pnet_packet::ip::IpNextHeaderProtocols;
use crate::{
packet::protocols::{icmp::IcmpPacket, tcp::TcpPacket, udp::UdpPacket},
packet::{
error::PacketError,
protocols::{icmpv6::Icmpv6Packet, ipv4::Ipv4Packet, ipv6::Ipv6Packet, raw::RawBytes},
},
};
use super::{
icmp::{translate_icmp_to_icmpv6, translate_icmpv6_to_icmp},
tcp::{translate_tcp4_to_tcp6, translate_tcp6_to_tcp4},
udp::{translate_udp4_to_udp6, translate_udp6_to_udp4},
};
/// Translates an IPv4 packet to an IPv6 packet
pub fn translate_ipv4_to_ipv6(
input: Ipv4Packet<Vec<u8>>,
new_source: Ipv6Addr,
new_destination: Ipv6Addr,
) -> Result<Ipv6Packet<Vec<u8>>, PacketError> {
// Perform recursive translation to determine the new payload
let new_payload = match input.protocol {
IpNextHeaderProtocols::Icmp => {
let icmp_input: IcmpPacket<RawBytes> = input.payload.try_into()?;
translate_icmp_to_icmpv6(icmp_input, new_source, new_destination)?.into()
}
IpNextHeaderProtocols::Udp => {
let udp_input: UdpPacket<RawBytes> = UdpPacket::new_from_bytes_raw_payload(
&input.payload,
IpAddr::V4(input.source_address),
IpAddr::V4(input.destination_address),
)?;
translate_udp4_to_udp6(udp_input, new_source, new_destination)?.into()
}
IpNextHeaderProtocols::Tcp => {
let tcp_input: TcpPacket<RawBytes> = TcpPacket::new_from_bytes_raw_payload(
&input.payload,
IpAddr::V4(input.source_address),
IpAddr::V4(input.destination_address),
)?;
translate_tcp4_to_tcp6(tcp_input, new_source, new_destination)?.into()
}
_ => {
log::warn!("Unsupported next level protocol: {}", input.protocol);
input.payload
}
};
// Build the output IPv6 packet
let output = Ipv6Packet::new(
0,
0,
match input.protocol {
IpNextHeaderProtocols::Icmp => IpNextHeaderProtocols::Icmpv6,
proto => proto,
},
input.ttl,
new_source,
new_destination,
new_payload,
);
// Return the output
Ok(output)
}
/// Translates an IPv6 packet to an IPv4 packet
pub fn translate_ipv6_to_ipv4(
input: Ipv6Packet<Vec<u8>>,
new_source: Ipv4Addr,
new_destination: Ipv4Addr,
) -> Result<Ipv4Packet<Vec<u8>>, PacketError> {
// Perform recursive translation to determine the new payload
let new_payload = match input.next_header {
IpNextHeaderProtocols::Icmpv6 => {
let icmpv6_input: Icmpv6Packet<RawBytes> = Icmpv6Packet::new_from_bytes_raw_payload(
&input.payload,
input.source_address,
input.destination_address,
)?;
Some(translate_icmpv6_to_icmp(icmpv6_input, new_source, new_destination)?.into())
}
IpNextHeaderProtocols::Udp => {
let udp_input: UdpPacket<RawBytes> = UdpPacket::new_from_bytes_raw_payload(
&input.payload,
IpAddr::V6(input.source_address),
IpAddr::V6(input.destination_address),
)?;
Some(translate_udp6_to_udp4(udp_input, new_source, new_destination)?.into())
}
IpNextHeaderProtocols::Tcp => {
let tcp_input: TcpPacket<RawBytes> = TcpPacket::new_from_bytes_raw_payload(
&input.payload,
IpAddr::V6(input.source_address),
IpAddr::V6(input.destination_address),
)?;
Some(translate_tcp6_to_tcp4(tcp_input, new_source, new_destination)?.into())
}
_ => {
log::warn!("Unsupported next level protocol: {}", input.next_header);
None
}
};
// Build the output IPv4 packet
let output = Ipv4Packet::new(
0,
0,
0,
0,
0,
input.hop_limit,
match input.next_header {
IpNextHeaderProtocols::Icmpv6 => IpNextHeaderProtocols::Icmp,
proto => proto,
},
new_source,
new_destination,
vec![],
new_payload.unwrap_or_else(Vec::new),
);
// Return the output
Ok(output)
}

6
src/packet/xlat/mod.rs Normal file
View File

@ -0,0 +1,6 @@
//! Protocol translation logic
pub mod icmp;
pub mod ip;
pub mod tcp;
pub mod udp;

46
src/packet/xlat/tcp.rs Normal file
View File

@ -0,0 +1,46 @@
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr, SocketAddr};
use crate::packet::{
error::PacketError,
protocols::{raw::RawBytes, tcp::TcpPacket},
};
/// Translates an IPv4 TCP packet to an IPv6 TCP packet
pub fn translate_tcp4_to_tcp6(
input: TcpPacket<RawBytes>,
new_source_addr: Ipv6Addr,
new_destination_addr: Ipv6Addr,
) -> Result<TcpPacket<RawBytes>, PacketError> {
// Build the packet
Ok(TcpPacket::new(
SocketAddr::new(IpAddr::V6(new_source_addr), input.source().port()),
SocketAddr::new(IpAddr::V6(new_destination_addr), input.destination().port()),
input.sequence,
input.ack_number,
input.flags,
input.window_size,
input.urgent_pointer,
input.options,
input.payload,
)?)
}
/// Translates an IPv6 TCP packet to an IPv4 TCP packet
pub fn translate_tcp6_to_tcp4(
input: TcpPacket<RawBytes>,
new_source_addr: Ipv4Addr,
new_destination_addr: Ipv4Addr,
) -> Result<TcpPacket<RawBytes>, PacketError> {
// Build the packet
Ok(TcpPacket::new(
SocketAddr::new(IpAddr::V4(new_source_addr), input.source().port()),
SocketAddr::new(IpAddr::V4(new_destination_addr), input.destination().port()),
input.sequence,
input.ack_number,
input.flags,
input.window_size,
input.urgent_pointer,
input.options,
input.payload,
)?)
}

34
src/packet/xlat/udp.rs Normal file
View File

@ -0,0 +1,34 @@
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr, SocketAddr};
use crate::packet::{
error::PacketError,
protocols::{raw::RawBytes, udp::UdpPacket},
};
/// Translates an IPv4 UDP packet to an IPv6 UDP packet
pub fn translate_udp4_to_udp6(
input: UdpPacket<RawBytes>,
new_source_addr: Ipv6Addr,
new_destination_addr: Ipv6Addr,
) -> Result<UdpPacket<RawBytes>, PacketError> {
// Build the packet
Ok(UdpPacket::new(
SocketAddr::new(IpAddr::V6(new_source_addr), input.source().port()),
SocketAddr::new(IpAddr::V6(new_destination_addr), input.destination().port()),
input.payload,
)?)
}
/// Translates an IPv6 UDP packet to an IPv4 UDP packet
pub fn translate_udp6_to_udp4(
input: UdpPacket<RawBytes>,
new_source_addr: Ipv4Addr,
new_destination_addr: Ipv4Addr,
) -> Result<UdpPacket<RawBytes>, PacketError> {
// Build the packet
Ok(UdpPacket::new(
SocketAddr::new(IpAddr::V4(new_source_addr), input.source().port()),
SocketAddr::new(IpAddr::V4(new_destination_addr), input.destination().port()),
input.payload,
)?)
}