OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "net/quic/quic_sent_packet_manager.h" | 5 #include "net/quic/quic_sent_packet_manager.h" |
6 | 6 |
7 #include <algorithm> | 7 #include <algorithm> |
8 | 8 |
9 #include "base/logging.h" | 9 #include "base/logging.h" |
10 #include "base/stl_util.h" | 10 #include "base/stl_util.h" |
11 #include "net/quic/congestion_control/pacing_sender.h" | 11 #include "net/quic/congestion_control/pacing_sender.h" |
12 #include "net/quic/crypto/crypto_protocol.h" | 12 #include "net/quic/crypto/crypto_protocol.h" |
13 #include "net/quic/quic_ack_notifier_manager.h" | 13 #include "net/quic/quic_ack_notifier_manager.h" |
14 #include "net/quic/quic_connection_stats.h" | 14 #include "net/quic/quic_connection_stats.h" |
15 #include "net/quic/quic_flags.h" | 15 #include "net/quic/quic_flags.h" |
16 #include "net/quic/quic_utils_chromium.h" | 16 #include "net/quic/quic_utils_chromium.h" |
17 | 17 |
18 using std::make_pair; | 18 using std::make_pair; |
19 using std::max; | 19 using std::max; |
20 using std::min; | 20 using std::min; |
21 | 21 |
22 namespace net { | 22 namespace net { |
| 23 |
| 24 // The length of the recent min rtt window in seconds. Windowing is disabled for |
| 25 // values less than or equal to 0. |
| 26 int32 FLAGS_quic_recent_min_rtt_window_s = 60; |
| 27 |
23 namespace { | 28 namespace { |
24 static const int kDefaultRetransmissionTimeMs = 500; | 29 static const int kDefaultRetransmissionTimeMs = 500; |
25 // TCP RFC calls for 1 second RTO however Linux differs from this default and | 30 // TCP RFC calls for 1 second RTO however Linux differs from this default and |
26 // define the minimum RTO to 200ms, we will use the same until we have data to | 31 // define the minimum RTO to 200ms, we will use the same until we have data to |
27 // support a higher or lower value. | 32 // support a higher or lower value. |
28 static const int kMinRetransmissionTimeMs = 200; | 33 static const int kMinRetransmissionTimeMs = 200; |
29 static const int kMaxRetransmissionTimeMs = 60000; | 34 static const int kMaxRetransmissionTimeMs = 60000; |
30 static const size_t kMaxRetransmissions = 10; | 35 static const size_t kMaxRetransmissions = 10; |
31 | 36 |
32 // Only exponentially back off the handshake timer 5 times due to a timeout. | 37 // Only exponentially back off the handshake timer 5 times due to a timeout. |
33 static const size_t kMaxHandshakeRetransmissionBackoffs = 5; | 38 static const size_t kMaxHandshakeRetransmissionBackoffs = 5; |
34 static const size_t kMinHandshakeTimeoutMs = 10; | 39 static const size_t kMinHandshakeTimeoutMs = 10; |
35 | 40 |
36 // Sends up to two tail loss probes before firing an RTO, | 41 // Sends up to two tail loss probes before firing an RTO, |
37 // per draft RFC draft-dukkipati-tcpm-tcp-loss-probe. | 42 // per draft RFC draft-dukkipati-tcpm-tcp-loss-probe. |
38 static const size_t kDefaultMaxTailLossProbes = 2; | 43 static const size_t kDefaultMaxTailLossProbes = 2; |
39 static const int64 kMinTailLossProbeTimeoutMs = 10; | 44 static const int64 kMinTailLossProbeTimeoutMs = 10; |
40 | 45 |
41 // Number of samples before we force a new recent min rtt to be captured. | 46 // Number of samples before we force a new recent min rtt to be captured. |
42 static const size_t kNumMinRttSamplesAfterQuiescence = 2; | 47 static const size_t kNumMinRttSamplesAfterQuiescence = 2; |
43 | 48 |
44 // Number of unpaced packets to send after quiescence. | 49 // Number of unpaced packets to send after quiescence. |
45 static const size_t kInitialUnpacedBurst = 10; | 50 static const size_t kInitialUnpacedBurst = 10; |
46 | 51 |
| 52 // Use a 1 minute window for Recent Min RTT with BBR. |
| 53 |
47 bool HasCryptoHandshake(const TransmissionInfo& transmission_info) { | 54 bool HasCryptoHandshake(const TransmissionInfo& transmission_info) { |
48 if (transmission_info.retransmittable_frames == NULL) { | 55 if (transmission_info.retransmittable_frames == NULL) { |
49 return false; | 56 return false; |
50 } | 57 } |
51 return transmission_info.retransmittable_frames->HasCryptoHandshake() == | 58 return transmission_info.retransmittable_frames->HasCryptoHandshake() == |
52 IS_HANDSHAKE; | 59 IS_HANDSHAKE; |
53 } | 60 } |
54 | 61 |
55 } // namespace | 62 } // namespace |
56 | 63 |
(...skipping 27 matching lines...) Expand all Loading... |
84 | 91 |
85 void QuicSentPacketManager::SetFromConfig(const QuicConfig& config) { | 92 void QuicSentPacketManager::SetFromConfig(const QuicConfig& config) { |
86 if (config.HasReceivedInitialRoundTripTimeUs() && | 93 if (config.HasReceivedInitialRoundTripTimeUs() && |
87 config.ReceivedInitialRoundTripTimeUs() > 0) { | 94 config.ReceivedInitialRoundTripTimeUs() > 0) { |
88 rtt_stats_.set_initial_rtt_us(min(kMaxInitialRoundTripTimeUs, | 95 rtt_stats_.set_initial_rtt_us(min(kMaxInitialRoundTripTimeUs, |
89 config.ReceivedInitialRoundTripTimeUs())); | 96 config.ReceivedInitialRoundTripTimeUs())); |
90 } | 97 } |
91 // TODO(ianswett): BBR is currently a server only feature. | 98 // TODO(ianswett): BBR is currently a server only feature. |
92 if (config.HasReceivedConnectionOptions() && | 99 if (config.HasReceivedConnectionOptions() && |
93 ContainsQuicTag(config.ReceivedConnectionOptions(), kTBBR)) { | 100 ContainsQuicTag(config.ReceivedConnectionOptions(), kTBBR)) { |
| 101 if (FLAGS_quic_recent_min_rtt_window_s > 0) { |
| 102 rtt_stats_.set_recent_min_rtt_window( |
| 103 QuicTime::Delta::FromSeconds(FLAGS_quic_recent_min_rtt_window_s)); |
| 104 } |
94 send_algorithm_.reset( | 105 send_algorithm_.reset( |
95 SendAlgorithmInterface::Create(clock_, &rtt_stats_, kTCPBBR, stats_)); | 106 SendAlgorithmInterface::Create(clock_, &rtt_stats_, kTCPBBR, stats_)); |
96 } | 107 } |
97 if (config.congestion_feedback() == kPACE || | 108 if (config.congestion_feedback() == kPACE || |
98 (config.HasReceivedConnectionOptions() && | 109 (config.HasReceivedConnectionOptions() && |
99 ContainsQuicTag(config.ReceivedConnectionOptions(), kPACE))) { | 110 ContainsQuicTag(config.ReceivedConnectionOptions(), kPACE))) { |
100 MaybeEnablePacing(); | 111 MaybeEnablePacing(); |
101 } | 112 } |
102 // TODO(ianswett): Remove the "HasReceivedLossDetection" branch once | 113 // TODO(ianswett): Remove the "HasReceivedLossDetection" branch once |
103 // the ConnectionOptions code is live everywhere. | 114 // the ConnectionOptions code is live everywhere. |
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
137 } | 148 } |
138 | 149 |
139 // A notifier may be waiting to hear about ACKs for the original sequence | 150 // A notifier may be waiting to hear about ACKs for the original sequence |
140 // number. Inform them that the sequence number has changed. | 151 // number. Inform them that the sequence number has changed. |
141 ack_notifier_manager_.UpdateSequenceNumber(old_sequence_number, | 152 ack_notifier_manager_.UpdateSequenceNumber(old_sequence_number, |
142 new_sequence_number); | 153 new_sequence_number); |
143 | 154 |
144 unacked_packets_.OnRetransmittedPacket(old_sequence_number, | 155 unacked_packets_.OnRetransmittedPacket(old_sequence_number, |
145 new_sequence_number, | 156 new_sequence_number, |
146 transmission_type); | 157 transmission_type); |
| 158 |
| 159 if (debug_delegate_ != NULL) { |
| 160 debug_delegate_->OnRetransmittedPacket(old_sequence_number, |
| 161 new_sequence_number, |
| 162 transmission_type, |
| 163 clock_->ApproximateNow()); |
| 164 } |
147 } | 165 } |
148 | 166 |
149 void QuicSentPacketManager::OnIncomingAck( | 167 void QuicSentPacketManager::OnIncomingAck( |
150 const ReceivedPacketInfo& received_info, | 168 const ReceivedPacketInfo& received_info, |
151 QuicTime ack_receive_time) { | 169 QuicTime ack_receive_time) { |
152 QuicByteCount bytes_in_flight = unacked_packets_.bytes_in_flight(); | 170 QuicByteCount bytes_in_flight = unacked_packets_.bytes_in_flight(); |
153 | 171 |
154 // We rely on delta_time_largest_observed to compute an RTT estimate, so | 172 // We rely on delta_time_largest_observed to compute an RTT estimate, so |
155 // we only update rtt when the largest observed gets acked. | 173 // we only update rtt when the largest observed gets acked. |
156 bool largest_observed_acked = MaybeUpdateRTT(received_info, ack_receive_time); | 174 bool largest_observed_acked = MaybeUpdateRTT(received_info, ack_receive_time); |
(...skipping 13 matching lines...) Expand all Loading... |
170 } | 188 } |
171 | 189 |
172 // Anytime we are making forward progress and have a new RTT estimate, reset | 190 // Anytime we are making forward progress and have a new RTT estimate, reset |
173 // the backoff counters. | 191 // the backoff counters. |
174 if (largest_observed_acked) { | 192 if (largest_observed_acked) { |
175 // Reset all retransmit counters any time a new packet is acked. | 193 // Reset all retransmit counters any time a new packet is acked. |
176 consecutive_rto_count_ = 0; | 194 consecutive_rto_count_ = 0; |
177 consecutive_tlp_count_ = 0; | 195 consecutive_tlp_count_ = 0; |
178 consecutive_crypto_retransmission_count_ = 0; | 196 consecutive_crypto_retransmission_count_ = 0; |
179 } | 197 } |
| 198 |
| 199 if (debug_delegate_ != NULL) { |
| 200 debug_delegate_->OnIncomingAck(received_info, |
| 201 ack_receive_time, |
| 202 largest_observed_, |
| 203 largest_observed_acked, |
| 204 GetLeastUnackedSentPacket()); |
| 205 } |
180 } | 206 } |
181 | 207 |
182 void QuicSentPacketManager::MaybeInvokeCongestionEvent( | 208 void QuicSentPacketManager::MaybeInvokeCongestionEvent( |
183 bool rtt_updated, QuicByteCount bytes_in_flight) { | 209 bool rtt_updated, QuicByteCount bytes_in_flight) { |
184 if (rtt_updated || !packets_acked_.empty() || | 210 if (rtt_updated || !packets_acked_.empty() || |
185 !packets_lost_.empty()) { | 211 !packets_lost_.empty()) { |
186 send_algorithm_->OnCongestionEvent( | 212 send_algorithm_->OnCongestionEvent( |
187 rtt_updated, bytes_in_flight, packets_acked_, packets_lost_); | 213 rtt_updated, bytes_in_flight, packets_acked_, packets_lost_); |
188 packets_acked_.clear(); | 214 packets_acked_.clear(); |
189 packets_lost_.clear(); | 215 packets_lost_.clear(); |
(...skipping 290 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
480 | 506 |
481 // Only track packets as in flight that the send algorithm wants us to track. | 507 // Only track packets as in flight that the send algorithm wants us to track. |
482 const bool in_flight = | 508 const bool in_flight = |
483 send_algorithm_->OnPacketSent(sent_time, | 509 send_algorithm_->OnPacketSent(sent_time, |
484 unacked_packets_.bytes_in_flight(), | 510 unacked_packets_.bytes_in_flight(), |
485 sequence_number, | 511 sequence_number, |
486 bytes, | 512 bytes, |
487 has_retransmittable_data); | 513 has_retransmittable_data); |
488 unacked_packets_.SetSent(sequence_number, sent_time, bytes, in_flight); | 514 unacked_packets_.SetSent(sequence_number, sent_time, bytes, in_flight); |
489 | 515 |
| 516 if (debug_delegate_ != NULL) { |
| 517 debug_delegate_->OnSentPacket(sequence_number, sent_time, bytes); |
| 518 } |
| 519 |
490 // Reset the retransmission timer anytime a pending packet is sent. | 520 // Reset the retransmission timer anytime a pending packet is sent. |
491 return in_flight; | 521 return in_flight; |
492 } | 522 } |
493 | 523 |
494 void QuicSentPacketManager::OnRetransmissionTimeout() { | 524 void QuicSentPacketManager::OnRetransmissionTimeout() { |
495 DCHECK(unacked_packets_.HasInFlightPackets()); | 525 DCHECK(unacked_packets_.HasInFlightPackets()); |
496 DCHECK(!pending_tlp_transmission_); | 526 DCHECK(!pending_tlp_transmission_); |
497 // Handshake retransmission, timer based loss detection, TLP, and RTO are | 527 // Handshake retransmission, timer based loss detection, TLP, and RTO are |
498 // implemented with a single alarm. The handshake alarm is set when the | 528 // implemented with a single alarm. The handshake alarm is set when the |
499 // handshake has not completed, the loss alarm is set when the loss detection | 529 // handshake has not completed, the loss alarm is set when the loss detection |
(...skipping 291 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
791 } | 821 } |
792 | 822 |
793 bool QuicSentPacketManager::HasReliableBandwidthEstimate() const { | 823 bool QuicSentPacketManager::HasReliableBandwidthEstimate() const { |
794 return send_algorithm_->HasReliableBandwidthEstimate(); | 824 return send_algorithm_->HasReliableBandwidthEstimate(); |
795 } | 825 } |
796 | 826 |
797 QuicByteCount QuicSentPacketManager::GetCongestionWindow() const { | 827 QuicByteCount QuicSentPacketManager::GetCongestionWindow() const { |
798 return send_algorithm_->GetCongestionWindow(); | 828 return send_algorithm_->GetCongestionWindow(); |
799 } | 829 } |
800 | 830 |
| 831 QuicByteCount QuicSentPacketManager::GetSlowStartThreshold() const { |
| 832 return send_algorithm_->GetSlowStartThreshold(); |
| 833 } |
| 834 |
801 void QuicSentPacketManager::MaybeEnablePacing() { | 835 void QuicSentPacketManager::MaybeEnablePacing() { |
802 if (!FLAGS_enable_quic_pacing) { | 836 if (!FLAGS_enable_quic_pacing) { |
803 return; | 837 return; |
804 } | 838 } |
805 | 839 |
806 if (using_pacing_) { | 840 if (using_pacing_) { |
807 return; | 841 return; |
808 } | 842 } |
809 | 843 |
810 // Set up a pacing sender with a 5 millisecond alarm granularity. | 844 // Set up a pacing sender with a 5 millisecond alarm granularity. |
811 using_pacing_ = true; | 845 using_pacing_ = true; |
812 send_algorithm_.reset( | 846 send_algorithm_.reset( |
813 new PacingSender(send_algorithm_.release(), | 847 new PacingSender(send_algorithm_.release(), |
814 QuicTime::Delta::FromMilliseconds(5), | 848 QuicTime::Delta::FromMilliseconds(5), |
815 kInitialUnpacedBurst)); | 849 kInitialUnpacedBurst)); |
816 } | 850 } |
817 | 851 |
818 } // namespace net | 852 } // namespace net |
OLD | NEW |