From 92c30832585130b3629dcc2258d6ceb2b3629e46 Mon Sep 17 00:00:00 2001 From: Pierre Chifflier Date: Mon, 9 Sep 2024 14:13:49 +0200 Subject: [PATCH] Add new helper `TlsRecordsParser` to manage record, with support for fragmentation --- assets/tls_record_ch_fragmented_1.bin | Bin 0 -> 15 bytes assets/tls_record_ch_fragmented_2.bin | Bin 0 -> 46 bytes src/lib.rs | 2 + src/tls_record.rs | 5 +- src/tls_records_parser.rs | 219 ++++++++++++++++++++++++++ 5 files changed, 225 insertions(+), 1 deletion(-) create mode 100644 assets/tls_record_ch_fragmented_1.bin create mode 100644 assets/tls_record_ch_fragmented_2.bin create mode 100644 src/tls_records_parser.rs diff --git a/assets/tls_record_ch_fragmented_1.bin b/assets/tls_record_ch_fragmented_1.bin new file mode 100644 index 0000000000000000000000000000000000000000..8576ee5ee10d3929f860eb86a9ac770904285135 GIT binary patch literal 15 TcmWe*W@g}GWMI$-(m((J1B?J& literal 0 HcmV?d00001 diff --git a/assets/tls_record_ch_fragmented_2.bin b/assets/tls_record_ch_fragmented_2.bin new file mode 100644 index 0000000000000000000000000000000000000000..050b7bc549ba540b12f7523eee858e35f3d33bb1 GIT binary patch literal 46 ZcmWe*W@gZ2zzdid^cfi#SQr`@7yu7I0Q>*| literal 0 HcmV?d00001 diff --git a/src/lib.rs b/src/lib.rs index 306bd10..76fa5d4 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -151,6 +151,7 @@ mod tls_extensions; mod tls_handshake; mod tls_message; mod tls_record; +mod tls_records_parser; mod tls_sign_hash; mod tls_states; @@ -164,6 +165,7 @@ pub use tls_extensions::*; pub use tls_handshake::*; pub use tls_message::*; pub use tls_record::*; +pub use tls_records_parser::*; pub use tls_sign_hash::*; pub use tls_states::*; diff --git a/src/tls_record.rs b/src/tls_record.rs index 3057fa7..fcdf684 100644 --- a/src/tls_record.rs +++ b/src/tls_record.rs @@ -92,6 +92,9 @@ pub fn parse_tls_record_header(i: &[u8]) -> IResult<&[u8], TlsRecordHeader> { /// /// Note that message length is checked (not required for parser safety, but for /// strict protocol conformance). +/// +/// This function will fail on fragmented records. To support fragmented records, use +/// [crate::TlsRecordsParser]]. #[rustfmt::skip] #[allow(clippy::trivially_copy_pass_by_ref)] // TlsRecordHeader is only 6 bytes, but we prefer not breaking current API pub fn parse_tls_record_with_header<'i>(i:&'i [u8], hdr:&TlsRecordHeader ) -> IResult<&'i [u8], Vec>> { @@ -133,7 +136,7 @@ pub fn parse_tls_encrypted(i: &[u8]) -> IResult<&[u8], TlsEncrypted> { /// /// This function is used to get the record type, and to make sure the record is /// complete (not fragmented). -/// After calling this function, use [`parse_tls_record_with_header`] to parse content. +/// After calling this function, use [`parse_tls_record_with_header`] or [crate::TlsRecordsParser] to parse content. pub fn parse_tls_raw_record(i: &[u8]) -> IResult<&[u8], TlsRawRecord> { let (i, hdr) = parse_tls_record_header(i)?; if hdr.len > MAX_RECORD_LEN { diff --git a/src/tls_records_parser.rs b/src/tls_records_parser.rs new file mode 100644 index 0000000..e914ec6 --- /dev/null +++ b/src/tls_records_parser.rs @@ -0,0 +1,219 @@ +use crate::{ + parse_tls_record_with_header, TlsMessage, TlsRawRecord, TlsRecordHeader, TlsRecordType, +}; +use alloc::vec::Vec; +use nom::{ + error::{Error, ErrorKind}, + Err, IResult, Needed, +}; + +pub const MAX_RECORD_DATA: usize = 10 * 1024 * 1024; + +/// Helper tool to defragment and parse TLS records +#[derive(Debug, Default)] +pub struct TlsRecordsParser { + record_defrag_buffer: Vec, + current_record_type: Option, +} + +impl TlsRecordsParser { + /// Reset the parser state (deleting all previous records) + pub fn reset(&mut self) { + *self = Self::default(); + } + + /// Returns `true` if defragmentation is in progress + pub fn defrag_in_progress(&self) -> bool { + self.current_record_type.is_some() + } + + /// Attempt to parse all messages from a single record + /// + /// Record types `ChangeCipherSpec` and `Alert` cannot be fragmented. + /// + /// This function does not defragment data, but guarantees that no data is copied. + /// + /// Returns + /// - the bytes remaining, and the list of messages. The remaining bytes should be empty. + /// - `Incomplete` if a message is fragmented. Caller should use function [Self::parse_record] instead + /// - `ErrorKind::NonEmpty` if defragmentation is already in progress + pub fn parse_record_nocopy<'a>( + &mut self, + record: TlsRawRecord<'a>, + ) -> IResult<&'a [u8], Vec>> { + if self.defrag_in_progress() { + return Err(Err::Failure(Error::new(&[], ErrorKind::NonEmpty))); + } + match parse_tls_record_with_header(record.data, &record.hdr) { + Err(Err::Error(e)) | Err(Err::Failure(e)) if e.code == ErrorKind::Complete => { + Err(Err::Incomplete(Needed::Unknown)) + } + other => other, + } + } + + /// Attempt to parse all messages from a record (iterative) + /// + /// Parse all messages from a record, keeping previous fragments (incrementally) if required. + /// If current record is fragmented, copy record data and return `Incomplete`. + /// + /// Record types `ChangeCipherSpec` and `Alert` cannot be fragmented. + /// + /// Record types cannot be interleaved. If defragmentation has started for a record type, other record types + /// will be rejected. + /// + /// Returns + /// - the bytes remaining, and the list of messages. The remaining bytes should be empty. + /// - `Incomplete` if a message is fragmented. Caller should get next record and call function again + /// - `ErrorKind::TooLarge` if record contents exceeds [`MAX_RECORD_DATA`] + /// - `ErrorKind::Tag` if the provided record does not have the same record type as the first record from the list + pub fn parse_record<'p, 'a: 'p>( + &'p mut self, + record: TlsRawRecord<'a>, + ) -> IResult<&'p [u8], Vec>> { + if !self.defrag_in_progress() { + // first fragment + + if record.hdr.record_type == TlsRecordType::Alert + || record.hdr.record_type == TlsRecordType::ChangeCipherSpec + { + return self.parse_record_nocopy(record); + } + + // before defragmenting, check that message is indeed fragmented + match parse_tls_record_with_header(record.data, &record.hdr) { + Ok(res) => return Ok(res), + Err(Err::Incomplete(_)) => (), + Err(Err::Error(e)) | Err(Err::Failure(e)) if e.code == ErrorKind::Complete => (), + Err(e) => return Err(e), + } + + // record is indeed fragmented: keep contents and return Incomplete + self.current_record_type = Some(record.hdr.record_type); + // replace previous buffer + self.record_defrag_buffer.clear(); + self.record_defrag_buffer.extend_from_slice(record.data); + return Err(Err::Incomplete(Needed::Unknown)); + } + + // record is not the first + debug_assert!(!self.record_defrag_buffer.is_empty()); + + let record_type = record.hdr.record_type; + if Some(record_type) != self.current_record_type { + return Err(Err::Error(Error::new(&[], ErrorKind::Tag))); + } + + if self + .record_defrag_buffer + .len() + .saturating_add(record.data.len()) + >= MAX_RECORD_DATA + { + return Err(Err::Error(Error::new(&[], ErrorKind::TooLarge))); + } + self.record_defrag_buffer.extend_from_slice(record.data); + + // create a pseudo-header with correct length + let header = TlsRecordHeader { + len: self.record_defrag_buffer.len() as u16, + ..record.hdr + }; + + match parse_tls_record_with_header(&self.record_defrag_buffer, &header) { + // we have a complete message list. Remove the parsed records and return + Ok(r) => { + // set current_record_type to None, but keep buffer (remaining bytes) + self.current_record_type = None; + Ok(r) + } + Err(Err::Error(e)) | Err(Err::Failure(e)) if e.code == ErrorKind::Complete => { + Err(Err::Incomplete(Needed::Unknown)) + } + // other errors + other => other, + } + } +} + +#[cfg(test)] +mod tests { + use crate::{parse_tls_raw_record, TlsMessageHandshake, TlsVersion}; + + use super::*; + + static REC_CH: &[u8] = include_bytes!("../assets/client_hello_dhe.bin"); + static REC_CH_FRAG_1: &[u8] = include_bytes!("../assets/tls_record_ch_fragmented_1.bin"); + static REC_CH_FRAG_2: &[u8] = include_bytes!("../assets/tls_record_ch_fragmented_2.bin"); + + #[test] + fn tls_records_parser_nocopy() { + let (_, record) = parse_tls_raw_record(REC_CH).expect("could not parse client_hello"); + let (_, record1) = parse_tls_raw_record(REC_CH_FRAG_1).expect("could not parse fragment 1"); + + // + // check that _nocopy parser works + let mut parser = TlsRecordsParser::default(); + let parser_result_nocopy = parser.parse_record_nocopy(record); + + assert!(parser_result_nocopy.is_ok()); + + // + // check that _nocopy parser fails with fragmented data + let mut parser = TlsRecordsParser::default(); + let parser_result_nocopy = parser.parse_record_nocopy(record1.clone()); + assert!(matches!(parser_result_nocopy, Err(Err::Incomplete(_)))); + } + + #[test] + fn tls_records_parser_fragmented() { + let (_, record) = parse_tls_raw_record(REC_CH).expect("could not parse client_hello"); + let (_, record1) = parse_tls_raw_record(REC_CH_FRAG_1).expect("could not parse fragment 1"); + let (_, record2) = parse_tls_raw_record(REC_CH_FRAG_2).expect("could not parse fragment 2"); + + // + // check that parser works with complete data + let mut parser = TlsRecordsParser::default(); + let (rem, messages) = parser.parse_record(record).expect("parsing failed"); + assert!(rem.is_empty()); + assert_eq!(messages.len(), 1); + assert!(!parser.defrag_in_progress()); + + // + // check that parser works with fragmented data + let mut parser = TlsRecordsParser::default(); + let parser_result1 = parser.parse_record(record1); + assert!(matches!(parser_result1, Err(Err::Incomplete(_)))); + let (rem, messages) = parser + .parse_record(record2) + .expect("defragmentation failed"); + assert!(rem.is_empty()); + assert_eq!(messages.len(), 1); + let ch = &messages[0]; + assert!(matches!( + ch, + TlsMessage::Handshake(TlsMessageHandshake::ClientHello(_)) + )); + + parser.reset(); + + // // does not compile (expected): remaining bytes borrow `parser` and cannot be used + // // after `parser` has been modified (here, mutably borrowed) + // assert!(!rem.is_empty()); + } + + #[test] + fn tls_records_parser_empty() { + let record = TlsRawRecord { + hdr: TlsRecordHeader { + record_type: TlsRecordType::Handshake, + version: TlsVersion::Tls12, + len: 0, + }, + data: &[], + }; + let mut parser = TlsRecordsParser::default(); + let parser_result = parser.parse_record(record); + assert!(parser_result.is_err()); + } +}