• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "net/quic/quic_packet_generator.h"
6 
7 #include "base/basictypes.h"
8 #include "base/logging.h"
9 #include "net/quic/quic_fec_group.h"
10 #include "net/quic/quic_utils.h"
11 
12 using base::StringPiece;
13 
14 namespace net {
15 
16 namespace {
17 
18 // We want to put some space between a protected packet and the FEC packet to
19 // avoid losing them both within the same loss episode. On the other hand,
20 // we expect to be able to recover from any loss in about an RTT.
21 // We resolve this tradeoff by sending an FEC packet atmost half an RTT,
22 // or equivalently, half a cwnd, after the first protected packet. Since we
23 // don't want to delay an FEC packet past half an RTT, we set the max FEC
24 // group size to be half the current congestion window.
25 const float kCongestionWindowMultiplierForFecGroupSize = 0.5;
26 
27 }  // namespace
28 
29 class QuicAckNotifier;
30 
QuicPacketGenerator(QuicConnectionId connection_id,QuicFramer * framer,QuicRandom * random_generator,DelegateInterface * delegate)31 QuicPacketGenerator::QuicPacketGenerator(QuicConnectionId connection_id,
32                                          QuicFramer* framer,
33                                          QuicRandom* random_generator,
34                                          DelegateInterface* delegate)
35     : delegate_(delegate),
36       debug_delegate_(NULL),
37       packet_creator_(connection_id, framer, random_generator),
38       batch_mode_(false),
39       should_fec_protect_(false),
40       should_send_ack_(false),
41       should_send_feedback_(false),
42       should_send_stop_waiting_(false) {
43 }
44 
~QuicPacketGenerator()45 QuicPacketGenerator::~QuicPacketGenerator() {
46   for (QuicFrames::iterator it = queued_control_frames_.begin();
47        it != queued_control_frames_.end(); ++it) {
48     switch (it->type) {
49       case PADDING_FRAME:
50         delete it->padding_frame;
51         break;
52       case STREAM_FRAME:
53         delete it->stream_frame;
54         break;
55       case ACK_FRAME:
56         delete it->ack_frame;
57         break;
58       case CONGESTION_FEEDBACK_FRAME:
59         delete it->congestion_feedback_frame;
60         break;
61       case RST_STREAM_FRAME:
62         delete it->rst_stream_frame;
63         break;
64       case CONNECTION_CLOSE_FRAME:
65         delete it->connection_close_frame;
66         break;
67       case GOAWAY_FRAME:
68         delete it->goaway_frame;
69         break;
70       case WINDOW_UPDATE_FRAME:
71         delete it->window_update_frame;
72         break;
73       case BLOCKED_FRAME:
74         delete it->blocked_frame;
75         break;
76       case STOP_WAITING_FRAME:
77         delete it->stop_waiting_frame;
78         break;
79       case PING_FRAME:
80         delete it->ping_frame;
81         break;
82       case NUM_FRAME_TYPES:
83         DCHECK(false) << "Cannot delete type: " << it->type;
84     }
85   }
86 }
87 
88 // NetworkChangeVisitor method.
OnCongestionWindowChange(QuicByteCount congestion_window)89 void QuicPacketGenerator::OnCongestionWindowChange(
90     QuicByteCount congestion_window) {
91   packet_creator_.set_max_packets_per_fec_group(
92       static_cast<size_t>(kCongestionWindowMultiplierForFecGroupSize *
93                           congestion_window / kDefaultTCPMSS));
94 }
95 
SetShouldSendAck(bool also_send_feedback,bool also_send_stop_waiting)96 void QuicPacketGenerator::SetShouldSendAck(bool also_send_feedback,
97                                            bool also_send_stop_waiting) {
98   should_send_ack_ = true;
99   should_send_feedback_ = also_send_feedback;
100   should_send_stop_waiting_ = also_send_stop_waiting;
101   SendQueuedFrames(false);
102 }
103 
SetShouldSendStopWaiting()104 void QuicPacketGenerator::SetShouldSendStopWaiting() {
105   should_send_stop_waiting_ = true;
106   SendQueuedFrames(false);
107 }
108 
AddControlFrame(const QuicFrame & frame)109 void QuicPacketGenerator::AddControlFrame(const QuicFrame& frame) {
110   queued_control_frames_.push_back(frame);
111   SendQueuedFrames(false);
112 }
113 
ConsumeData(QuicStreamId id,const IOVector & data_to_write,QuicStreamOffset offset,bool fin,FecProtection fec_protection,QuicAckNotifier * notifier)114 QuicConsumedData QuicPacketGenerator::ConsumeData(QuicStreamId id,
115                                                   const IOVector& data_to_write,
116                                                   QuicStreamOffset offset,
117                                                   bool fin,
118                                                   FecProtection fec_protection,
119                                                   QuicAckNotifier* notifier) {
120   IsHandshake handshake = id == kCryptoStreamId ? IS_HANDSHAKE : NOT_HANDSHAKE;
121   // To make reasoning about crypto frames easier, we don't combine them with
122   // other retransmittable frames in a single packet.
123   const bool flush = handshake == IS_HANDSHAKE &&
124       packet_creator_.HasPendingRetransmittableFrames();
125   SendQueuedFrames(flush);
126 
127   size_t total_bytes_consumed = 0;
128   bool fin_consumed = false;
129 
130   if (!packet_creator_.HasRoomForStreamFrame(id, offset)) {
131     SerializeAndSendPacket();
132   }
133 
134   if (fec_protection == MUST_FEC_PROTECT) {
135     MaybeStartFecProtection();
136   }
137 
138   IOVector data = data_to_write;
139   size_t data_size = data.TotalBufferSize();
140   while (delegate_->ShouldGeneratePacket(NOT_RETRANSMISSION,
141                                          HAS_RETRANSMITTABLE_DATA, handshake)) {
142     QuicFrame frame;
143     size_t bytes_consumed;
144     if (notifier != NULL) {
145       // We want to track which packet this stream frame ends up in.
146       bytes_consumed = packet_creator_.CreateStreamFrameWithNotifier(
147           id, data, offset + total_bytes_consumed, fin, notifier, &frame);
148     } else {
149       bytes_consumed = packet_creator_.CreateStreamFrame(
150           id, data, offset + total_bytes_consumed, fin, &frame);
151     }
152     if (!AddFrame(frame)) {
153       LOG(DFATAL) << "Failed to add stream frame.";
154       // Inability to add a STREAM frame creates an unrecoverable hole in a
155       // the stream, so it's best to close the connection.
156       delegate_->CloseConnection(QUIC_INTERNAL_ERROR, false);
157       return QuicConsumedData(0, false);
158     }
159 
160     total_bytes_consumed += bytes_consumed;
161     fin_consumed = fin && total_bytes_consumed == data_size;
162     data.Consume(bytes_consumed);
163     DCHECK(data.Empty() || packet_creator_.BytesFree() == 0u);
164 
165     // TODO(ianswett): Restore packet reordering.
166     if (!InBatchMode() || !packet_creator_.HasRoomForStreamFrame(id, offset)) {
167       SerializeAndSendPacket();
168     }
169 
170     if (data.Empty()) {
171       // We're done writing the data. Exit the loop.
172       // We don't make this a precondition because we could have 0 bytes of data
173       // if we're simply writing a fin.
174       if (fec_protection == MUST_FEC_PROTECT) {
175         // Turn off FEC protection when we're done writing protected data.
176         DVLOG(1) << "Turning FEC protection OFF";
177         should_fec_protect_ = false;
178       }
179       break;
180     }
181   }
182 
183   // Don't allow the handshake to be bundled with other retransmittable frames.
184   if (handshake == IS_HANDSHAKE) {
185     SendQueuedFrames(true);
186   }
187 
188   // Try to close FEC group since we've either run out of data to send or we're
189   // blocked. If not in batch mode, force close the group.
190   // TODO(jri): This method should be called with flush=false here
191   // once the timer-based FEC sending is done, to separate FEC sending from
192   // the end of batch operations.
193   MaybeSendFecPacketAndCloseGroup(!InBatchMode());
194 
195   DCHECK(InBatchMode() || !packet_creator_.HasPendingFrames());
196   return QuicConsumedData(total_bytes_consumed, fin_consumed);
197 }
198 
CanSendWithNextPendingFrameAddition() const199 bool QuicPacketGenerator::CanSendWithNextPendingFrameAddition() const {
200   DCHECK(HasPendingFrames());
201   HasRetransmittableData retransmittable =
202       (should_send_ack_ || should_send_feedback_ || should_send_stop_waiting_)
203       ? NO_RETRANSMITTABLE_DATA : HAS_RETRANSMITTABLE_DATA;
204   if (retransmittable == HAS_RETRANSMITTABLE_DATA) {
205       DCHECK(!queued_control_frames_.empty());  // These are retransmittable.
206   }
207   return delegate_->ShouldGeneratePacket(NOT_RETRANSMISSION, retransmittable,
208                                          NOT_HANDSHAKE);
209 }
210 
SendQueuedFrames(bool flush)211 void QuicPacketGenerator::SendQueuedFrames(bool flush) {
212   // Only add pending frames if we are SURE we can then send the whole packet.
213   while (HasPendingFrames() &&
214          (flush || CanSendWithNextPendingFrameAddition())) {
215     if (!AddNextPendingFrame()) {
216       // Packet was full, so serialize and send it.
217       SerializeAndSendPacket();
218     }
219   }
220 
221   if (!InBatchMode() || flush) {
222     if (packet_creator_.HasPendingFrames()) {
223       SerializeAndSendPacket();
224     }
225     // Ensure the FEC group is closed at the end of this method unless other
226     // writes are pending.
227     MaybeSendFecPacketAndCloseGroup(true);
228   }
229 }
230 
MaybeStartFecProtection()231 void QuicPacketGenerator::MaybeStartFecProtection() {
232   if (!packet_creator_.IsFecEnabled()) {
233     return;
234   }
235   DVLOG(1) << "Turning FEC protection ON";
236   should_fec_protect_ = true;
237   if (packet_creator_.IsFecProtected()) {
238     // Only start creator's FEC protection if not already on.
239     return;
240   }
241   if (HasQueuedFrames()) {
242     // TODO(jri): This currently requires that the generator flush out any
243     // pending frames when FEC protection is turned on. If current packet can be
244     // converted to an FEC protected packet, do it. This will require the
245     // generator to check if the resulting expansion still allows the incoming
246     // frame to be added to the packet.
247     SendQueuedFrames(true);
248   }
249   packet_creator_.StartFecProtectingPackets();
250   DCHECK(packet_creator_.IsFecProtected());
251 }
252 
MaybeSendFecPacketAndCloseGroup(bool force)253 void QuicPacketGenerator::MaybeSendFecPacketAndCloseGroup(bool force) {
254   if (!packet_creator_.IsFecProtected() ||
255       packet_creator_.HasPendingFrames() ||
256       !packet_creator_.ShouldSendFec(force)) {
257     return;
258   }
259   // TODO(jri): SerializeFec can return a NULL packet, and this should
260   // cause an early return, with a call to delegate_->OnPacketGenerationError.
261   SerializedPacket serialized_fec = packet_creator_.SerializeFec();
262   DCHECK(serialized_fec.packet);
263   delegate_->OnSerializedPacket(serialized_fec);
264   // Turn FEC protection off if creator's protection is on and the creator
265   // does not have an open FEC group.
266   // Note: We only wait until the frames queued in the creator are flushed;
267   // pending frames in the generator will not keep us from turning FEC off.
268   if (!should_fec_protect_ && !packet_creator_.IsFecGroupOpen()) {
269     packet_creator_.StopFecProtectingPackets();
270     DCHECK(!packet_creator_.IsFecProtected());
271   }
272 }
273 
InBatchMode()274 bool QuicPacketGenerator::InBatchMode() {
275   return batch_mode_;
276 }
277 
StartBatchOperations()278 void QuicPacketGenerator::StartBatchOperations() {
279   batch_mode_ = true;
280 }
281 
FinishBatchOperations()282 void QuicPacketGenerator::FinishBatchOperations() {
283   batch_mode_ = false;
284   SendQueuedFrames(false);
285 }
286 
FlushAllQueuedFrames()287 void QuicPacketGenerator::FlushAllQueuedFrames() {
288   SendQueuedFrames(true);
289 }
290 
HasQueuedFrames() const291 bool QuicPacketGenerator::HasQueuedFrames() const {
292   return packet_creator_.HasPendingFrames() || HasPendingFrames();
293 }
294 
HasPendingFrames() const295 bool QuicPacketGenerator::HasPendingFrames() const {
296   return should_send_ack_ || should_send_feedback_ ||
297       should_send_stop_waiting_ || !queued_control_frames_.empty();
298 }
299 
AddNextPendingFrame()300 bool QuicPacketGenerator::AddNextPendingFrame() {
301   if (should_send_ack_) {
302     pending_ack_frame_.reset(delegate_->CreateAckFrame());
303     // If we can't this add the frame now, then we still need to do so later.
304     should_send_ack_ = !AddFrame(QuicFrame(pending_ack_frame_.get()));
305     // Return success if we have cleared out this flag (i.e., added the frame).
306     // If we still need to send, then the frame is full, and we have failed.
307     return !should_send_ack_;
308   }
309 
310   if (should_send_feedback_) {
311     pending_feedback_frame_.reset(delegate_->CreateFeedbackFrame());
312     // If we can't this add the frame now, then we still need to do so later.
313     should_send_feedback_ = !AddFrame(QuicFrame(pending_feedback_frame_.get()));
314     // Return success if we have cleared out this flag (i.e., added the frame).
315     // If we still need to send, then the frame is full, and we have failed.
316     return !should_send_feedback_;
317   }
318 
319   if (should_send_stop_waiting_) {
320     pending_stop_waiting_frame_.reset(delegate_->CreateStopWaitingFrame());
321     // If we can't this add the frame now, then we still need to do so later.
322     should_send_stop_waiting_ =
323         !AddFrame(QuicFrame(pending_stop_waiting_frame_.get()));
324     // Return success if we have cleared out this flag (i.e., added the frame).
325     // If we still need to send, then the frame is full, and we have failed.
326     return !should_send_stop_waiting_;
327   }
328 
329   LOG_IF(DFATAL, queued_control_frames_.empty())
330       << "AddNextPendingFrame called with no queued control frames.";
331   if (!AddFrame(queued_control_frames_.back())) {
332     // Packet was full.
333     return false;
334   }
335   queued_control_frames_.pop_back();
336   return true;
337 }
338 
AddFrame(const QuicFrame & frame)339 bool QuicPacketGenerator::AddFrame(const QuicFrame& frame) {
340   bool success = packet_creator_.AddSavedFrame(frame);
341   if (success && debug_delegate_) {
342     debug_delegate_->OnFrameAddedToPacket(frame);
343   }
344   return success;
345 }
346 
SerializeAndSendPacket()347 void QuicPacketGenerator::SerializeAndSendPacket() {
348   SerializedPacket serialized_packet = packet_creator_.SerializePacket();
349   DCHECK(serialized_packet.packet);
350   delegate_->OnSerializedPacket(serialized_packet);
351   MaybeSendFecPacketAndCloseGroup(false);
352 }
353 
StopSendingVersion()354 void QuicPacketGenerator::StopSendingVersion() {
355   packet_creator_.StopSendingVersion();
356 }
357 
sequence_number() const358 QuicPacketSequenceNumber QuicPacketGenerator::sequence_number() const {
359   return packet_creator_.sequence_number();
360 }
361 
max_packet_length() const362 size_t QuicPacketGenerator::max_packet_length() const {
363   return packet_creator_.max_packet_length();
364 }
365 
set_max_packet_length(size_t length)366 void QuicPacketGenerator::set_max_packet_length(size_t length) {
367   packet_creator_.set_max_packet_length(length);
368 }
369 
SerializeVersionNegotiationPacket(const QuicVersionVector & supported_versions)370 QuicEncryptedPacket* QuicPacketGenerator::SerializeVersionNegotiationPacket(
371     const QuicVersionVector& supported_versions) {
372   return packet_creator_.SerializeVersionNegotiationPacket(supported_versions);
373 }
374 
ReserializeAllFrames(const QuicFrames & frames,QuicSequenceNumberLength original_length)375 SerializedPacket QuicPacketGenerator::ReserializeAllFrames(
376     const QuicFrames& frames,
377     QuicSequenceNumberLength original_length) {
378   return packet_creator_.ReserializeAllFrames(frames, original_length);
379 }
380 
UpdateSequenceNumberLength(QuicPacketSequenceNumber least_packet_awaited_by_peer,QuicByteCount congestion_window)381 void QuicPacketGenerator::UpdateSequenceNumberLength(
382       QuicPacketSequenceNumber least_packet_awaited_by_peer,
383       QuicByteCount congestion_window) {
384   return packet_creator_.UpdateSequenceNumberLength(
385       least_packet_awaited_by_peer, congestion_window);
386 }
387 
set_encryption_level(EncryptionLevel level)388 void QuicPacketGenerator::set_encryption_level(EncryptionLevel level) {
389   packet_creator_.set_encryption_level(level);
390 }
391 
392 }  // namespace net
393