mirror of
https://mirror.suhoan.cn/https://github.com/EasyTier/EasyTier.git
synced 2025-12-14 05:37:23 +08:00
471 lines
15 KiB
Rust
471 lines
15 KiB
Rust
use bytes::Bytes;
|
|
use bytes::BytesMut;
|
|
use zerocopy::byteorder::*;
|
|
use zerocopy::AsBytes;
|
|
use zerocopy::FromBytes;
|
|
use zerocopy::FromZeroes;
|
|
|
|
type DefaultEndian = LittleEndian;
|
|
|
|
// TCP TunnelHeader
|
|
#[repr(C, packed)]
|
|
#[derive(AsBytes, FromBytes, FromZeroes, Clone, Debug, Default)]
|
|
pub struct TCPTunnelHeader {
|
|
pub len: U32<DefaultEndian>,
|
|
}
|
|
pub const TCP_TUNNEL_HEADER_SIZE: usize = std::mem::size_of::<TCPTunnelHeader>();
|
|
|
|
#[derive(AsBytes, FromZeroes, Clone, Debug)]
|
|
#[repr(u8)]
|
|
pub enum UdpPacketType {
|
|
Invalid = 0,
|
|
Syn = 1,
|
|
Sack = 2,
|
|
Data = 3,
|
|
Fin = 4,
|
|
HolePunch = 5,
|
|
}
|
|
|
|
#[repr(C, packed)]
|
|
#[derive(AsBytes, FromBytes, FromZeroes, Clone, Debug, Default)]
|
|
pub struct UDPTunnelHeader {
|
|
pub conn_id: U32<DefaultEndian>,
|
|
pub msg_type: u8,
|
|
pub padding: u8,
|
|
pub len: U16<DefaultEndian>,
|
|
}
|
|
pub const UDP_TUNNEL_HEADER_SIZE: usize = std::mem::size_of::<UDPTunnelHeader>();
|
|
|
|
#[repr(C, packed)]
|
|
#[derive(AsBytes, FromBytes, FromZeroes, Clone, Debug, Default)]
|
|
pub struct WGTunnelHeader {
|
|
pub ipv4_header: [u8; 20],
|
|
}
|
|
pub const WG_TUNNEL_HEADER_SIZE: usize = std::mem::size_of::<WGTunnelHeader>();
|
|
|
|
#[derive(AsBytes, FromZeroes, Clone, Debug)]
|
|
#[repr(u8)]
|
|
pub enum PacketType {
|
|
Invalid = 0,
|
|
Data = 1,
|
|
HandShake = 2,
|
|
RoutePacket = 3,
|
|
Ping = 4,
|
|
Pong = 5,
|
|
TaRpc = 6,
|
|
Route = 7,
|
|
}
|
|
|
|
bitflags::bitflags! {
|
|
struct PeerManagerHeaderFlags: u8 {
|
|
const ENCRYPTED = 0b0000_0001;
|
|
}
|
|
}
|
|
|
|
#[repr(C, packed)]
|
|
#[derive(AsBytes, FromBytes, FromZeroes, Clone, Debug, Default)]
|
|
pub struct PeerManagerHeader {
|
|
pub from_peer_id: U32<DefaultEndian>,
|
|
pub to_peer_id: U32<DefaultEndian>,
|
|
pub packet_type: u8,
|
|
pub flags: u8,
|
|
reserved: U16<DefaultEndian>,
|
|
pub len: U32<DefaultEndian>,
|
|
}
|
|
pub const PEER_MANAGER_HEADER_SIZE: usize = std::mem::size_of::<PeerManagerHeader>();
|
|
|
|
impl PeerManagerHeader {
|
|
pub fn is_encrypted(&self) -> bool {
|
|
PeerManagerHeaderFlags::from_bits(self.flags)
|
|
.unwrap()
|
|
.contains(PeerManagerHeaderFlags::ENCRYPTED)
|
|
}
|
|
|
|
pub fn set_encrypted(&mut self, encrypted: bool) {
|
|
let mut flags = PeerManagerHeaderFlags::from_bits(self.flags).unwrap();
|
|
if encrypted {
|
|
flags.insert(PeerManagerHeaderFlags::ENCRYPTED);
|
|
} else {
|
|
flags.remove(PeerManagerHeaderFlags::ENCRYPTED);
|
|
}
|
|
self.flags = flags.bits();
|
|
}
|
|
}
|
|
|
|
// reserve the space for aes tag and nonce
|
|
#[repr(C, packed)]
|
|
#[derive(AsBytes, FromBytes, FromZeroes, Clone, Debug, Default)]
|
|
pub struct AesGcmTail {
|
|
pub tag: [u8; 16],
|
|
pub nonce: [u8; 12],
|
|
}
|
|
pub const AES_GCM_ENCRYPTION_RESERVED: usize = std::mem::size_of::<AesGcmTail>();
|
|
|
|
pub const TAIL_RESERVED_SIZE: usize = AES_GCM_ENCRYPTION_RESERVED;
|
|
|
|
const fn max(a: usize, b: usize) -> usize {
|
|
[a, b][(a < b) as usize]
|
|
}
|
|
|
|
#[derive(Default, Debug)]
|
|
pub struct ZCPacketOffsets {
|
|
pub payload_offset: usize,
|
|
pub peer_manager_header_offset: usize,
|
|
pub tcp_tunnel_header_offset: usize,
|
|
pub udp_tunnel_header_offset: usize,
|
|
pub wg_tunnel_header_offset: usize,
|
|
pub dummy_tunnel_header_offset: usize,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
|
pub enum ZCPacketType {
|
|
// received from peer tcp connection
|
|
TCP,
|
|
// received from peer udp connection
|
|
UDP,
|
|
// received from peer wireguard connection
|
|
WG,
|
|
// received from local tun device, should reserve header space for tcp or udp tunnel
|
|
NIC,
|
|
// tunnel without header
|
|
DummyTunnel,
|
|
}
|
|
|
|
const PAYLOAD_OFFSET_FOR_NIC_PACKET: usize = max(
|
|
max(TCP_TUNNEL_HEADER_SIZE, UDP_TUNNEL_HEADER_SIZE),
|
|
WG_TUNNEL_HEADER_SIZE,
|
|
) + PEER_MANAGER_HEADER_SIZE;
|
|
|
|
const INVALID_OFFSET: usize = usize::MAX;
|
|
|
|
const fn get_converted_offset(old_hdr_size: usize, new_hdr_size: usize) -> usize {
|
|
if old_hdr_size < new_hdr_size {
|
|
INVALID_OFFSET
|
|
} else {
|
|
old_hdr_size - new_hdr_size
|
|
}
|
|
}
|
|
|
|
impl ZCPacketType {
|
|
pub fn get_packet_offsets(&self) -> ZCPacketOffsets {
|
|
match self {
|
|
ZCPacketType::TCP => ZCPacketOffsets {
|
|
payload_offset: TCP_TUNNEL_HEADER_SIZE + PEER_MANAGER_HEADER_SIZE,
|
|
peer_manager_header_offset: TCP_TUNNEL_HEADER_SIZE,
|
|
tcp_tunnel_header_offset: 0,
|
|
udp_tunnel_header_offset: get_converted_offset(
|
|
TCP_TUNNEL_HEADER_SIZE,
|
|
UDP_TUNNEL_HEADER_SIZE,
|
|
),
|
|
wg_tunnel_header_offset: get_converted_offset(
|
|
TCP_TUNNEL_HEADER_SIZE,
|
|
WG_TUNNEL_HEADER_SIZE,
|
|
),
|
|
dummy_tunnel_header_offset: get_converted_offset(TCP_TUNNEL_HEADER_SIZE, 0),
|
|
},
|
|
ZCPacketType::UDP => ZCPacketOffsets {
|
|
payload_offset: UDP_TUNNEL_HEADER_SIZE + PEER_MANAGER_HEADER_SIZE,
|
|
peer_manager_header_offset: UDP_TUNNEL_HEADER_SIZE,
|
|
tcp_tunnel_header_offset: get_converted_offset(
|
|
UDP_TUNNEL_HEADER_SIZE,
|
|
TCP_TUNNEL_HEADER_SIZE,
|
|
),
|
|
udp_tunnel_header_offset: 0,
|
|
wg_tunnel_header_offset: get_converted_offset(
|
|
UDP_TUNNEL_HEADER_SIZE,
|
|
WG_TUNNEL_HEADER_SIZE,
|
|
),
|
|
dummy_tunnel_header_offset: get_converted_offset(UDP_TUNNEL_HEADER_SIZE, 0),
|
|
},
|
|
ZCPacketType::WG => ZCPacketOffsets {
|
|
payload_offset: WG_TUNNEL_HEADER_SIZE + PEER_MANAGER_HEADER_SIZE,
|
|
peer_manager_header_offset: WG_TUNNEL_HEADER_SIZE,
|
|
tcp_tunnel_header_offset: get_converted_offset(
|
|
WG_TUNNEL_HEADER_SIZE,
|
|
TCP_TUNNEL_HEADER_SIZE,
|
|
),
|
|
udp_tunnel_header_offset: get_converted_offset(
|
|
WG_TUNNEL_HEADER_SIZE,
|
|
UDP_TUNNEL_HEADER_SIZE,
|
|
),
|
|
wg_tunnel_header_offset: 0,
|
|
dummy_tunnel_header_offset: get_converted_offset(WG_TUNNEL_HEADER_SIZE, 0),
|
|
},
|
|
ZCPacketType::NIC => ZCPacketOffsets {
|
|
payload_offset: PAYLOAD_OFFSET_FOR_NIC_PACKET,
|
|
peer_manager_header_offset: PAYLOAD_OFFSET_FOR_NIC_PACKET
|
|
- PEER_MANAGER_HEADER_SIZE,
|
|
tcp_tunnel_header_offset: PAYLOAD_OFFSET_FOR_NIC_PACKET
|
|
- PEER_MANAGER_HEADER_SIZE
|
|
- TCP_TUNNEL_HEADER_SIZE,
|
|
udp_tunnel_header_offset: PAYLOAD_OFFSET_FOR_NIC_PACKET
|
|
- PEER_MANAGER_HEADER_SIZE
|
|
- UDP_TUNNEL_HEADER_SIZE,
|
|
wg_tunnel_header_offset: PAYLOAD_OFFSET_FOR_NIC_PACKET
|
|
- PEER_MANAGER_HEADER_SIZE
|
|
- WG_TUNNEL_HEADER_SIZE,
|
|
dummy_tunnel_header_offset: PAYLOAD_OFFSET_FOR_NIC_PACKET
|
|
- PEER_MANAGER_HEADER_SIZE,
|
|
},
|
|
ZCPacketType::DummyTunnel => ZCPacketOffsets {
|
|
payload_offset: PEER_MANAGER_HEADER_SIZE,
|
|
peer_manager_header_offset: 0,
|
|
tcp_tunnel_header_offset: get_converted_offset(0, TCP_TUNNEL_HEADER_SIZE),
|
|
udp_tunnel_header_offset: get_converted_offset(0, UDP_TUNNEL_HEADER_SIZE),
|
|
wg_tunnel_header_offset: get_converted_offset(0, WG_TUNNEL_HEADER_SIZE),
|
|
dummy_tunnel_header_offset: 0,
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct ZCPacket {
|
|
inner: BytesMut,
|
|
packet_type: ZCPacketType,
|
|
}
|
|
|
|
impl ZCPacket {
|
|
pub fn new_nic_packet() -> Self {
|
|
Self {
|
|
inner: BytesMut::new(),
|
|
packet_type: ZCPacketType::NIC,
|
|
}
|
|
}
|
|
|
|
pub fn new_from_buf(buf: BytesMut, packet_type: ZCPacketType) -> Self {
|
|
Self {
|
|
inner: buf,
|
|
packet_type,
|
|
}
|
|
}
|
|
|
|
pub fn new_with_payload(payload: &[u8]) -> Self {
|
|
let mut ret = Self::new_nic_packet();
|
|
let payload_off = ret.packet_type.get_packet_offsets().payload_offset;
|
|
let total_len = payload_off + payload.len();
|
|
ret.inner.reserve(total_len);
|
|
unsafe { ret.inner.set_len(total_len) };
|
|
ret.mut_payload()[..payload.len()].copy_from_slice(&payload);
|
|
ret
|
|
}
|
|
|
|
pub fn new_for_tun(cap: usize, packet_info_len: usize) -> Self {
|
|
let mut ret = Self::new_nic_packet();
|
|
ret.inner.reserve(cap);
|
|
let total_len = ret.packet_type.get_packet_offsets().payload_offset - packet_info_len;
|
|
unsafe { ret.inner.set_len(total_len) };
|
|
ret
|
|
}
|
|
|
|
pub fn packet_type(&self) -> ZCPacketType {
|
|
self.packet_type
|
|
}
|
|
|
|
pub fn payload_offset(&self) -> usize {
|
|
self.packet_type.get_packet_offsets().payload_offset
|
|
}
|
|
|
|
pub fn mut_payload(&mut self) -> &mut [u8] {
|
|
let offset = self.payload_offset();
|
|
&mut self.inner[offset..]
|
|
}
|
|
|
|
pub fn mut_peer_manager_header(&mut self) -> Option<&mut PeerManagerHeader> {
|
|
PeerManagerHeader::mut_from_prefix(
|
|
&mut self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.peer_manager_header_offset..],
|
|
)
|
|
}
|
|
|
|
pub fn mut_tcp_tunnel_header(&mut self) -> Option<&mut TCPTunnelHeader> {
|
|
TCPTunnelHeader::mut_from_prefix(
|
|
&mut self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.tcp_tunnel_header_offset..],
|
|
)
|
|
}
|
|
|
|
pub fn mut_udp_tunnel_header(&mut self) -> Option<&mut UDPTunnelHeader> {
|
|
UDPTunnelHeader::mut_from_prefix(
|
|
&mut self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.udp_tunnel_header_offset..],
|
|
)
|
|
}
|
|
|
|
pub fn mut_wg_tunnel_header(&mut self) -> Option<&mut WGTunnelHeader> {
|
|
WGTunnelHeader::mut_from_prefix(
|
|
&mut self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.wg_tunnel_header_offset..],
|
|
)
|
|
}
|
|
|
|
// ref versions
|
|
pub fn payload(&self) -> &[u8] {
|
|
&self.inner[self.payload_offset()..]
|
|
}
|
|
|
|
pub fn peer_manager_header(&self) -> Option<&PeerManagerHeader> {
|
|
PeerManagerHeader::ref_from_prefix(
|
|
&self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.peer_manager_header_offset..],
|
|
)
|
|
}
|
|
|
|
pub fn tcp_tunnel_header(&self) -> Option<&TCPTunnelHeader> {
|
|
TCPTunnelHeader::ref_from_prefix(
|
|
&self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.tcp_tunnel_header_offset..],
|
|
)
|
|
}
|
|
|
|
pub fn udp_tunnel_header(&self) -> Option<&UDPTunnelHeader> {
|
|
UDPTunnelHeader::ref_from_prefix(
|
|
&self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.udp_tunnel_header_offset..],
|
|
)
|
|
}
|
|
|
|
pub fn udp_payload(&self) -> &[u8] {
|
|
&self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.udp_tunnel_header_offset
|
|
+ UDP_TUNNEL_HEADER_SIZE..]
|
|
}
|
|
|
|
pub fn payload_len(&self) -> usize {
|
|
self.inner.len() - self.payload_offset()
|
|
}
|
|
|
|
pub fn buf_len(&self) -> usize {
|
|
self.inner.len()
|
|
}
|
|
|
|
pub fn fill_peer_manager_hdr(&mut self, from_peer_id: u32, to_peer_id: u32, packet_type: u8) {
|
|
let payload_len = self.payload_len();
|
|
let hdr = self.mut_peer_manager_header().unwrap();
|
|
hdr.from_peer_id.set(from_peer_id);
|
|
hdr.to_peer_id.set(to_peer_id);
|
|
hdr.packet_type = packet_type;
|
|
hdr.flags = 0;
|
|
hdr.len.set(payload_len as u32);
|
|
}
|
|
|
|
pub fn tunnel_payload(&self) -> &[u8] {
|
|
&self.inner[self
|
|
.packet_type
|
|
.get_packet_offsets()
|
|
.peer_manager_header_offset..]
|
|
}
|
|
|
|
pub fn tunnel_payload_bytes(mut self) -> BytesMut {
|
|
self.inner.split_off(
|
|
self.packet_type
|
|
.get_packet_offsets()
|
|
.peer_manager_header_offset,
|
|
)
|
|
}
|
|
|
|
pub fn convert_type(mut self, target_packet_type: ZCPacketType) -> Self {
|
|
if target_packet_type == self.packet_type {
|
|
return self;
|
|
}
|
|
|
|
let new_offset = match target_packet_type {
|
|
ZCPacketType::TCP => {
|
|
self.packet_type
|
|
.get_packet_offsets()
|
|
.tcp_tunnel_header_offset
|
|
}
|
|
ZCPacketType::UDP => {
|
|
self.packet_type
|
|
.get_packet_offsets()
|
|
.udp_tunnel_header_offset
|
|
}
|
|
ZCPacketType::WG => {
|
|
self.packet_type
|
|
.get_packet_offsets()
|
|
.wg_tunnel_header_offset
|
|
}
|
|
ZCPacketType::DummyTunnel => {
|
|
self.packet_type
|
|
.get_packet_offsets()
|
|
.dummy_tunnel_header_offset
|
|
}
|
|
ZCPacketType::NIC => unreachable!(),
|
|
};
|
|
|
|
tracing::debug!(?self.packet_type, ?target_packet_type, ?new_offset, "convert zc packet type");
|
|
|
|
if new_offset == INVALID_OFFSET {
|
|
// copy peer manager header and payload to new buffer
|
|
let tunnel_payload = self.tunnel_payload();
|
|
let new_pm_offset = target_packet_type
|
|
.get_packet_offsets()
|
|
.peer_manager_header_offset;
|
|
let mut buf = BytesMut::with_capacity(new_pm_offset + tunnel_payload.len());
|
|
unsafe { buf.set_len(new_pm_offset) };
|
|
buf.extend_from_slice(tunnel_payload);
|
|
return Self::new_from_buf(buf, target_packet_type);
|
|
}
|
|
|
|
return Self::new_from_buf(self.inner.split_off(new_offset), target_packet_type);
|
|
}
|
|
|
|
pub fn into_bytes(self) -> Bytes {
|
|
self.inner.freeze()
|
|
}
|
|
|
|
pub fn inner(self) -> BytesMut {
|
|
self.inner
|
|
}
|
|
|
|
pub fn mut_inner(&mut self) -> &mut BytesMut {
|
|
&mut self.inner
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn test_zc_packet() {
|
|
let payload = b"hello world";
|
|
let mut packet = ZCPacket::new_with_payload(payload);
|
|
let peer_manager_header = packet.mut_peer_manager_header().unwrap();
|
|
peer_manager_header.packet_type = PacketType::Data as u8;
|
|
peer_manager_header.len.set(payload.len() as u32);
|
|
|
|
let tcp_tunnel_header = packet.mut_tcp_tunnel_header().unwrap();
|
|
tcp_tunnel_header.len.set(payload.len() as u32);
|
|
|
|
// let udp_tunnel_header = packet.mut_udp_tunnel_header().unwrap();
|
|
// udp_tunnel_header.conn_id = 1;
|
|
// udp_tunnel_header.msg_type = 2;
|
|
// udp_tunnel_header.len = payload.len() as u32;
|
|
|
|
assert_eq!(packet.payload(), b"hello world");
|
|
assert_eq!(packet.payload_len(), 11);
|
|
println!("{:?}", packet.inner);
|
|
|
|
let tcp_packet = packet.convert_type(ZCPacketType::TCP).into_bytes();
|
|
assert_eq!(&tcp_packet[..1], b"\x0b");
|
|
println!("{:?}", tcp_packet);
|
|
}
|
|
}
|