Skip to content

Commit

Permalink
Ensure ack delay is correctly set
Browse files Browse the repository at this point in the history
Sponsored by Stormshield
  • Loading branch information
aochagavia committed Jul 6, 2023
1 parent 92cb99e commit 16b5822
Show file tree
Hide file tree
Showing 4 changed files with 64 additions and 2 deletions.
37 changes: 37 additions & 0 deletions quinn-proto/src/connection/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3293,6 +3293,43 @@ impl Connection {
self.spaces[self.highest_space].immediate_ack_pending = true;
}

/// Decodes a packet, returning its decrypted payload, so it can be inspected in tests
#[cfg(test)]
pub(crate) fn decode_packet(&self, event: &ConnectionEvent) -> Option<Vec<u8>> {
let (first_decode, remaining) = match &event.0 {
ConnectionEventInner::Datagram {
first_decode,
remaining,
..
} => (first_decode, remaining),
_ => return None,
};

if remaining.is_some() {
panic!("Packets should never be coalesced in tests");
}

let decrypted_header = packet_crypto::unprotect_header(
first_decode.clone(),
&self.spaces,
self.zero_rtt_crypto.as_ref(),
self.peer_params.stateless_reset_token,
)?;

let mut packet = decrypted_header.packet?;
packet_crypto::decrypt_packet_body(
&mut packet,
&self.spaces,
self.zero_rtt_crypto.as_ref(),
self.key_phase,
self.prev_crypto.as_ref(),
self.next_crypto.as_ref(),
)
.ok()?;

Some(packet.payload.to_vec())
}

/// The number of bytes of packets containing retransmittable frames that have not been
/// acknowledged or declared lost.
#[cfg(test)]
Expand Down
6 changes: 4 additions & 2 deletions quinn-proto/src/packet.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ use crate::{
// to inspect the version and packet type (which depends on the version).
// This information allows us to fully decode and decrypt the packet.
#[allow(unreachable_pub)] // fuzzing only
#[cfg_attr(test, derive(Clone))]
#[derive(Debug)]
pub struct PartialDecode {
plain_header: PlainHeader,
Expand Down Expand Up @@ -234,7 +235,8 @@ impl Packet {
}
}

#[derive(Debug, Clone)]
#[cfg_attr(test, derive(Clone))]
#[derive(Debug)]
pub(crate) enum Header {
Initial {
dst_cid: ConnectionId,
Expand Down Expand Up @@ -477,7 +479,7 @@ impl PartialEncode {
}
}

#[derive(Debug)]
#[derive(Clone, Debug)]
pub(crate) enum PlainHeader {
Initial {
dst_cid: ConnectionId,
Expand Down
14 changes: 14 additions & 0 deletions quinn-proto/src/tests/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2237,6 +2237,7 @@ fn single_ack_eliciting_packet_triggers_ack_after_delay() {
0
);

pair.client.capture_inbound_packets = true;
pair.drive();
let stats_after_drive = pair.client_conn_mut(client_ch).stats();
assert_eq!(
Expand All @@ -2251,6 +2252,19 @@ fn single_ack_eliciting_packet_triggers_ack_after_delay() {
start + Duration::from_millis(default_max_ack_delay_ms)
);

// The ACK delay is properly calculated
assert_eq!(pair.client.captured_packets.len(), 1);
let mut frames =
frame::Iter::new(pair.client.captured_packets.remove(0).into()).collect::<Vec<_>>();
assert_eq!(frames.len(), 1);
if let Frame::Ack(ack) = frames.remove(0) {
let ack_delay_exp = TransportParameters::default().ack_delay_exponent;
let delay = ack.delay << ack_delay_exp.into_inner();
assert_eq!(delay, default_max_ack_delay_ms * 1_000);
} else {
panic!("Expected ACK frame");
}

// Sanity check: no loss probe was sent, because the delayed ACK was received on time
assert_eq!(
stats_after_drive.frame_tx.ping - stats_after_connect.frame_tx.ping,
Expand Down
9 changes: 9 additions & 0 deletions quinn-proto/src/tests/util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -276,6 +276,8 @@ pub(super) struct TestEndpoint {
accepted: Option<ConnectionHandle>,
pub(super) connections: HashMap<ConnectionHandle, Connection>,
conn_events: HashMap<ConnectionHandle, VecDeque<ConnectionEvent>>,
pub(super) captured_packets: Vec<Vec<u8>>,
pub(super) capture_inbound_packets: bool,
}

impl TestEndpoint {
Expand All @@ -300,6 +302,8 @@ impl TestEndpoint {
accepted: None,
connections: HashMap::default(),
conn_events: HashMap::default(),
captured_packets: Vec::new(),
capture_inbound_packets: false,
}
}

Expand All @@ -322,6 +326,11 @@ impl TestEndpoint {
self.accepted = Some(ch);
}
DatagramEvent::ConnectionEvent(ch, event) => {
if self.capture_inbound_packets {
let packet = self.connections[&ch].decode_packet(&event);
self.captured_packets.extend(packet);
}

self.conn_events
.entry(ch)
.or_insert_with(VecDeque::new)
Expand Down

0 comments on commit 16b5822

Please sign in to comment.