Fix some TCP bugs
This commit is contained in:
parent
deaa9e87c5
commit
10d7bf520f
@ -31,7 +31,11 @@ impl TunDevice {
|
|||||||
/// it will be replaced with a unique number.
|
/// it will be replaced with a unique number.
|
||||||
pub async fn new(name: &str) -> Result<Self> {
|
pub async fn new(name: &str) -> Result<Self> {
|
||||||
// Bring up an rtnetlink connection
|
// Bring up an rtnetlink connection
|
||||||
let (rt_connection, rt_handle, _) = rtnetlink::new_connection()?;
|
let (rt_connection, rt_handle, _) = rtnetlink::new_connection().map_err(|err| {
|
||||||
|
log::error!("Failed to open rtnetlink connection");
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?;
|
||||||
tokio::spawn(rt_connection);
|
tokio::spawn(rt_connection);
|
||||||
|
|
||||||
// Create the TUN device
|
// Create the TUN device
|
||||||
@ -55,7 +59,12 @@ impl TunDevice {
|
|||||||
.set(tun_link.header.index)
|
.set(tun_link.header.index)
|
||||||
.up()
|
.up()
|
||||||
.execute()
|
.execute()
|
||||||
.await?;
|
.await
|
||||||
|
.map_err(|err| {
|
||||||
|
log::error!("Failed to bring up link");
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?;
|
||||||
log::debug!("Brought {} up", tun_device.name());
|
log::debug!("Brought {} up", tun_device.name());
|
||||||
|
|
||||||
// Read the link MTU
|
// Read the link MTU
|
||||||
@ -81,7 +90,12 @@ impl TunDevice {
|
|||||||
.address()
|
.address()
|
||||||
.add(self.link_index, ip_address, prefix_len)
|
.add(self.link_index, ip_address, prefix_len)
|
||||||
.execute()
|
.execute()
|
||||||
.await?;
|
.await
|
||||||
|
.map_err(|err| {
|
||||||
|
log::error!("Failed to add address {} to link", ip_address);
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -98,14 +112,24 @@ impl TunDevice {
|
|||||||
.set_prefix_length_filter(prefix_len)
|
.set_prefix_length_filter(prefix_len)
|
||||||
.execute()
|
.execute()
|
||||||
.try_next()
|
.try_next()
|
||||||
.await?
|
.await
|
||||||
|
.map_err(|err| {
|
||||||
|
log::error!("Failed to find address {} on link", ip_address);
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?
|
||||||
{
|
{
|
||||||
// Delete the address
|
// Delete the address
|
||||||
self.rt_handle
|
self.rt_handle
|
||||||
.address()
|
.address()
|
||||||
.del(address_message)
|
.del(address_message)
|
||||||
.execute()
|
.execute()
|
||||||
.await?;
|
.await
|
||||||
|
.map_err(|err| {
|
||||||
|
log::error!("Failed to remove address {} from link", ip_address);
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -119,18 +143,30 @@ impl TunDevice {
|
|||||||
.route()
|
.route()
|
||||||
.add()
|
.add()
|
||||||
.v4()
|
.v4()
|
||||||
|
.output_interface(self.link_index)
|
||||||
.destination_prefix(destination.addr(), destination.prefix_len())
|
.destination_prefix(destination.addr(), destination.prefix_len())
|
||||||
.execute()
|
.execute()
|
||||||
.await?;
|
.await
|
||||||
|
.map_err(|err| {
|
||||||
|
log::error!("Failed to add route {} to link", destination);
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?;
|
||||||
}
|
}
|
||||||
IpNet::V6(destination) => {
|
IpNet::V6(destination) => {
|
||||||
self.rt_handle
|
self.rt_handle
|
||||||
.route()
|
.route()
|
||||||
.add()
|
.add()
|
||||||
.v6()
|
.v6()
|
||||||
|
.output_interface(self.link_index)
|
||||||
.destination_prefix(destination.addr(), destination.prefix_len())
|
.destination_prefix(destination.addr(), destination.prefix_len())
|
||||||
.execute()
|
.execute()
|
||||||
.await?;
|
.await
|
||||||
|
.map_err(|err| {
|
||||||
|
log::error!("Failed to add route {} to link", destination);
|
||||||
|
log::error!("{}", err);
|
||||||
|
err
|
||||||
|
})?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,10 +176,10 @@ impl TunDevice {
|
|||||||
/// Spawns worker threads, and returns a tx/rx pair for the caller to interact with them
|
/// Spawns worker threads, and returns a tx/rx pair for the caller to interact with them
|
||||||
pub async fn spawn_worker(&self) -> (mpsc::Sender<Vec<u8>>, broadcast::Receiver<Vec<u8>>) {
|
pub async fn spawn_worker(&self) -> (mpsc::Sender<Vec<u8>>, broadcast::Receiver<Vec<u8>>) {
|
||||||
// Create a channel for packets to be sent to the caller
|
// Create a channel for packets to be sent to the caller
|
||||||
let (tx_to_caller, rx_from_worker) = broadcast::channel(32);
|
let (tx_to_caller, rx_from_worker) = broadcast::channel(65535);
|
||||||
|
|
||||||
// Create a channel for packets being received from the caller
|
// Create a channel for packets being received from the caller
|
||||||
let (tx_to_worker, mut rx_from_caller) = mpsc::channel(32);
|
let (tx_to_worker, mut rx_from_caller) = mpsc::channel(65535);
|
||||||
|
|
||||||
// Clone some values for use in worker threads
|
// Clone some values for use in worker threads
|
||||||
let mtu = self.mtu;
|
let mtu = self.mtu;
|
||||||
|
@ -82,52 +82,67 @@ impl Nat64 {
|
|||||||
// Process packets in a loop
|
// Process packets in a loop
|
||||||
loop {
|
loop {
|
||||||
// Try to read a packet
|
// Try to read a packet
|
||||||
let packet = rx.recv().await?;
|
match rx.recv().await {
|
||||||
|
Ok(packet) => {
|
||||||
|
// Clone the TX so the worker can respond with data
|
||||||
|
let tx = tx.clone();
|
||||||
|
|
||||||
// Clone the TX so the worker can respond with data
|
// Separate logic is needed for handling IPv4 vs IPv6 packets, so a check must be done here
|
||||||
let tx = tx.clone();
|
match packet[0] >> 4 {
|
||||||
|
4 => {
|
||||||
|
// Parse the packet
|
||||||
|
let packet: Ipv4Packet<Vec<u8>> = packet.try_into()?;
|
||||||
|
|
||||||
// Separate logic is needed for handling IPv4 vs IPv6 packets, so a check must be done here
|
// Drop packets that aren't destined for a destination the table knows about
|
||||||
match packet[0] >> 4 {
|
if !self.table.contains(&IpAddr::V4(packet.destination_address)) {
|
||||||
4 => {
|
continue;
|
||||||
// Parse the packet
|
}
|
||||||
let packet: Ipv4Packet<Vec<u8>> = packet.try_into()?;
|
|
||||||
|
|
||||||
// Drop packets that aren't destined for a destination the table knows about
|
// Get the new source and dest addresses
|
||||||
if !self.table.contains(&IpAddr::V4(packet.destination_address)) {
|
let new_source =
|
||||||
continue;
|
embed_address(packet.source_address, self.ipv6_nat_prefix);
|
||||||
|
let new_destination =
|
||||||
|
self.table.get_reverse(packet.destination_address)?;
|
||||||
|
|
||||||
|
// Spawn a task to process the packet
|
||||||
|
tokio::spawn(async move {
|
||||||
|
let output =
|
||||||
|
translate_ipv4_to_ipv6(packet, new_source, new_destination)
|
||||||
|
.unwrap();
|
||||||
|
tx.send(output.into()).await.unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
6 => {
|
||||||
|
// Parse the packet
|
||||||
|
let packet: Ipv6Packet<Vec<u8>> = packet.try_into()?;
|
||||||
|
|
||||||
|
// Get the new source and dest addresses
|
||||||
|
let new_source =
|
||||||
|
self.table.get_or_assign_ipv4(packet.source_address)?;
|
||||||
|
let new_destination = extract_address(packet.destination_address);
|
||||||
|
|
||||||
|
// Spawn a task to process the packet
|
||||||
|
tokio::spawn(async move {
|
||||||
|
let output =
|
||||||
|
translate_ipv6_to_ipv4(packet, new_source, new_destination)
|
||||||
|
.unwrap();
|
||||||
|
tx.send(output.into()).await.unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
n => {
|
||||||
|
log::warn!("Unknown IP version: {}", n);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Ok(())
|
||||||
// Get the new source and dest addresses
|
|
||||||
let new_source = embed_address(packet.source_address, self.ipv6_nat_prefix);
|
|
||||||
let new_destination = self.table.get_reverse(packet.destination_address)?;
|
|
||||||
|
|
||||||
// Spawn a task to process the packet
|
|
||||||
tokio::spawn(async move {
|
|
||||||
let output =
|
|
||||||
translate_ipv4_to_ipv6(packet, new_source, new_destination).unwrap();
|
|
||||||
tx.send(output.into()).await.unwrap();
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
6 => {
|
Err(error) => match error {
|
||||||
// Parse the packet
|
broadcast::error::RecvError::Lagged(count) => {
|
||||||
let packet: Ipv6Packet<Vec<u8>> = packet.try_into()?;
|
log::warn!("Translator running behind! Dropping {} packets", count);
|
||||||
|
Ok(())
|
||||||
// Get the new source and dest addresses
|
}
|
||||||
let new_source = self.table.get_or_assign_ipv4(packet.source_address)?;
|
error => Err(error),
|
||||||
let new_destination = extract_address(packet.destination_address);
|
},
|
||||||
|
}?;
|
||||||
// Spawn a task to process the packet
|
|
||||||
tokio::spawn(async move {
|
|
||||||
let output =
|
|
||||||
translate_ipv6_to_ipv4(packet, new_source, new_destination).unwrap();
|
|
||||||
tx.send(output.into()).await.unwrap();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
n => {
|
|
||||||
log::warn!("Unknown IP version: {}", n);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -100,12 +100,11 @@ impl<T> TcpPacket<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Get the length of the options in words
|
/// Get the length of the options in words
|
||||||
fn options_length_words(&self) -> u8 {
|
fn options_length(&self) -> usize {
|
||||||
self.options
|
self.options
|
||||||
.iter()
|
.iter()
|
||||||
.map(|option| TcpOptionPacket::packet_size(option) as u8)
|
.map(|option| TcpOptionPacket::packet_size(option))
|
||||||
.sum::<u8>()
|
.sum::<usize>()
|
||||||
/ 4
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -182,18 +181,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>> ,
|
T: Into<Vec<u8>>,
|
||||||
{
|
{
|
||||||
fn into(self) -> Vec<u8> {
|
fn into(self) -> Vec<u8> {
|
||||||
// Get the options length in words
|
// Get the options length in words
|
||||||
let options_length_words = self.options_length_words();
|
let options_length = self.options_length();
|
||||||
|
|
||||||
// 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()
|
||||||
+ (options_length_words as usize * 4)
|
+ options_length
|
||||||
+ 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();
|
||||||
@ -206,12 +205,12 @@ where
|
|||||||
output.set_sequence(self.sequence);
|
output.set_sequence(self.sequence);
|
||||||
output.set_acknowledgement(self.ack_number);
|
output.set_acknowledgement(self.ack_number);
|
||||||
|
|
||||||
|
// Write the offset
|
||||||
|
output.set_data_offset(5 + (options_length / 4) as u8);
|
||||||
|
|
||||||
// Write the options
|
// Write the options
|
||||||
output.set_options(&self.options);
|
output.set_options(&self.options);
|
||||||
|
|
||||||
// Write the offset
|
|
||||||
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());
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user