浏览代码

Add emit to IPSec

thegreathir 1 年之前
父节点
当前提交
450b68648d
共有 2 个文件被更改,包括 42 次插入4 次删除
  1. 39 4
      src/wire/ipsec_ah.rs
  2. 3 0
      src/wire/mod.rs

+ 39 - 4
src/wire/ipsec_ah.rs

@@ -9,6 +9,19 @@ pub struct Packet<T: AsRef<[u8]>> {
     buffer: T,
     buffer: T,
 }
 }
 
 
+//     0                   1                   2                   3
+//     0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+//   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+//   | Next Header   |  Payload Len  |          RESERVED             |
+//   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+//   |                 Security Parameters Index (SPI)               |
+//   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+//   |                    Sequence Number Field                      |
+//   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+//   |                                                               |
+//   +                Integrity Check Value-ICV (variable)           |
+//   |                                                               |
+//   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 mod field {
 mod field {
     #![allow(non_snake_case)]
     #![allow(non_snake_case)]
 
 
@@ -121,6 +134,12 @@ impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
         data[field::PAYLOAD_LEN] = value
         data[field::PAYLOAD_LEN] = value
     }
     }
 
 
+    /// Clear reserved field
+    fn clear_reserved(&mut self) {
+        let data = self.buffer.as_mut();
+        data[field::RESERVED].fill(0)
+    }
+
     /// Set security parameters index field
     /// Set security parameters index field
     fn set_security_parameters_index(&mut self, value: u32) {
     fn set_security_parameters_index(&mut self, value: u32) {
         let data = self.buffer.as_mut();
         let data = self.buffer.as_mut();
@@ -143,8 +162,7 @@ impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
 }
 }
 
 
 pub struct Repr<'a> {
 pub struct Repr<'a> {
-    next_header: u8,
-    payload_len: u8,
+    next_header: IpProtocol,
     security_parameters_index: u32,
     security_parameters_index: u32,
     sequence_number: u32,
     sequence_number: u32,
     integrity_check_value: &'a [u8],
     integrity_check_value: &'a [u8],
@@ -154,13 +172,30 @@ impl<'a> Repr<'a> {
     /// Parse an IPSec Authentication Header packet and return a high-level representation,
     /// Parse an IPSec Authentication Header packet and return a high-level representation,
     pub fn parse<T: AsRef<[u8]>>(packet: &Packet<&'a T>) -> Result<Repr<'a>> {
     pub fn parse<T: AsRef<[u8]>>(packet: &Packet<&'a T>) -> Result<Repr<'a>> {
         Ok(Repr {
         Ok(Repr {
-            next_header: packet.next_header().into(),
-            payload_len: packet.payload_len(),
+            next_header: packet.next_header(),
             security_parameters_index: packet.security_parameters_index(),
             security_parameters_index: packet.security_parameters_index(),
             sequence_number: packet.sequence_number(),
             sequence_number: packet.sequence_number(),
             integrity_check_value: packet.integrity_check_value(),
             integrity_check_value: packet.integrity_check_value(),
         })
         })
     }
     }
+
+    /// Return the length of a packet that will be emitted from this high-level representation.
+    pub const fn buffer_len(&self) -> usize {
+        self.integrity_check_value.len() + field::SEQUENCE_NUMBER.end
+    }
+
+    /// Emit a high-level representation into an IPSec Authentication Header.
+    pub fn emit<T: AsRef<[u8]> + AsMut<[u8]> + ?Sized>(&self, packet: &mut Packet<&'a mut T>) {
+        packet.set_next_header(self.next_header);
+        
+        let payload_len = (field::SEQUENCE_NUMBER.end + self.integrity_check_value.len() - 2) / 4;
+        packet.set_payload_len(payload_len as u8);
+
+        packet.clear_reserved();
+        packet.set_security_parameters_index(self.security_parameters_index);
+        packet.set_sequence_number(self.sequence_number);
+        packet.integrity_check_value_mut().copy_from_slice(self.integrity_check_value);
+    }
 }
 }
 
 
 // This sample can be used for test: https://www.cloudshark.org/captures/4d1561a5935f
 // This sample can be used for test: https://www.cloudshark.org/captures/4d1561a5935f

+ 3 - 0
src/wire/mod.rs

@@ -275,6 +275,9 @@ pub use self::dns::{
     Repr as DnsRepr, Type as DnsQueryType,
     Repr as DnsRepr, Type as DnsQueryType,
 };
 };
 
 
+#[cfg(feature = "proto-ipsec-ah")]
+pub use self::ipsec_ah::{Packet as IPSecAuthHeaderPacket, Repr as IPSecAuthHeaderRepr};
+
 /// Parsing a packet failed.
 /// Parsing a packet failed.
 ///
 ///
 /// Either it is malformed, or it is not supported by smoltcp.
 /// Either it is malformed, or it is not supported by smoltcp.