Skip to content

Commit 8342e4b

Browse files
authored
util: assert compatibility between LengthDelimitedCodec options (#6414)
1 parent 4c453e9 commit 8342e4b

File tree

2 files changed

+98
-1
lines changed

2 files changed

+98
-1
lines changed

tokio-util/src/codec/length_delimited.rs

+38-1
Original file line numberDiff line numberDiff line change
@@ -386,6 +386,10 @@ use std::{cmp, fmt, mem};
386386
/// `Builder` enables constructing configured length delimited codecs. Note
387387
/// that not all configuration settings apply to both encoding and decoding. See
388388
/// the documentation for specific methods for more detail.
389+
///
390+
/// Note that the if the value of [`Builder::max_frame_length`] becomes larger than
391+
/// what can actually fit in [`Builder::length_field_length`], it will be clipped to
392+
/// the maximum value that can fit.
389393
#[derive(Debug, Clone, Copy)]
390394
pub struct Builder {
391395
// Maximum frame length
@@ -935,8 +939,12 @@ impl Builder {
935939
/// # }
936940
/// ```
937941
pub fn new_codec(&self) -> LengthDelimitedCodec {
942+
let mut builder = *self;
943+
944+
builder.adjust_max_frame_len();
945+
938946
LengthDelimitedCodec {
939-
builder: *self,
947+
builder,
940948
state: DecodeState::Head,
941949
}
942950
}
@@ -1018,6 +1026,35 @@ impl Builder {
10181026
self.num_skip
10191027
.unwrap_or(self.length_field_offset + self.length_field_len)
10201028
}
1029+
1030+
fn adjust_max_frame_len(&mut self) {
1031+
// This function is basically `std::u64::saturating_add_signed`. Since it
1032+
// requires MSRV 1.66, its implementation is copied here.
1033+
//
1034+
// TODO: use the method from std when MSRV becomes >= 1.66
1035+
fn saturating_add_signed(num: u64, rhs: i64) -> u64 {
1036+
let (res, overflow) = num.overflowing_add(rhs as u64);
1037+
if overflow == (rhs < 0) {
1038+
res
1039+
} else if overflow {
1040+
u64::MAX
1041+
} else {
1042+
0
1043+
}
1044+
}
1045+
1046+
// Calculate the maximum number that can be represented using `length_field_len` bytes.
1047+
let max_number = match 1u64.checked_shl((8 * self.length_field_len) as u32) {
1048+
Some(shl) => shl - 1,
1049+
None => u64::MAX,
1050+
};
1051+
1052+
let max_allowed_len = saturating_add_signed(max_number, self.length_adjustment as i64);
1053+
1054+
if self.max_frame_len as u64 > max_allowed_len {
1055+
self.max_frame_len = usize::try_from(max_allowed_len).unwrap_or(usize::MAX);
1056+
}
1057+
}
10211058
}
10221059

10231060
impl Default for Builder {

tokio-util/tests/length_delimited.rs

+60
Original file line numberDiff line numberDiff line change
@@ -689,6 +689,66 @@ fn encode_overflow() {
689689
codec.encode(Bytes::from("hello"), &mut buf).unwrap();
690690
}
691691

692+
#[test]
693+
fn frame_does_not_fit() {
694+
let codec = LengthDelimitedCodec::builder()
695+
.length_field_length(1)
696+
.max_frame_length(256)
697+
.new_codec();
698+
699+
assert_eq!(codec.max_frame_length(), 255);
700+
}
701+
702+
#[test]
703+
fn neg_adjusted_frame_does_not_fit() {
704+
let codec = LengthDelimitedCodec::builder()
705+
.length_field_length(1)
706+
.length_adjustment(-1)
707+
.new_codec();
708+
709+
assert_eq!(codec.max_frame_length(), 254);
710+
}
711+
712+
#[test]
713+
fn pos_adjusted_frame_does_not_fit() {
714+
let codec = LengthDelimitedCodec::builder()
715+
.length_field_length(1)
716+
.length_adjustment(1)
717+
.new_codec();
718+
719+
assert_eq!(codec.max_frame_length(), 256);
720+
}
721+
722+
#[test]
723+
fn max_allowed_frame_fits() {
724+
let codec = LengthDelimitedCodec::builder()
725+
.length_field_length(std::mem::size_of::<usize>())
726+
.max_frame_length(usize::MAX)
727+
.new_codec();
728+
729+
assert_eq!(codec.max_frame_length(), usize::MAX);
730+
}
731+
732+
#[test]
733+
fn smaller_frame_len_not_adjusted() {
734+
let codec = LengthDelimitedCodec::builder()
735+
.max_frame_length(10)
736+
.length_field_length(std::mem::size_of::<usize>())
737+
.new_codec();
738+
739+
assert_eq!(codec.max_frame_length(), 10);
740+
}
741+
742+
#[test]
743+
fn max_allowed_length_field() {
744+
let codec = LengthDelimitedCodec::builder()
745+
.length_field_length(8)
746+
.max_frame_length(usize::MAX)
747+
.new_codec();
748+
749+
assert_eq!(codec.max_frame_length(), usize::MAX);
750+
}
751+
692752
// ===== Test utils =====
693753

694754
struct Mock {

0 commit comments

Comments
 (0)