// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/quic/quic_packet_generator.h"
#include "base/basictypes.h"
#include "base/logging.h"
#include "net/quic/quic_fec_group.h"
#include "net/quic/quic_utils.h"
using base::StringPiece;
namespace net {
namespace {
// We want to put some space between a protected packet and the FEC packet to
// avoid losing them both within the same loss episode. On the other hand,
// we expect to be able to recover from any loss in about an RTT.
// We resolve this tradeoff by sending an FEC packet atmost half an RTT,
// or equivalently, half a cwnd, after the first protected packet. Since we
// don't want to delay an FEC packet past half an RTT, we set the max FEC
// group size to be half the current congestion window.
const float kCongestionWindowMultiplierForFecGroupSize = 0.5;
} // namespace
class QuicAckNotifier;
QuicPacketGenerator::QuicPacketGenerator(QuicConnectionId connection_id,
QuicFramer* framer,
QuicRandom* random_generator,
DelegateInterface* delegate)
: delegate_(delegate),
debug_delegate_(NULL),
packet_creator_(connection_id, framer, random_generator),
batch_mode_(false),
should_fec_protect_(false),
should_send_ack_(false),
should_send_feedback_(false),
should_send_stop_waiting_(false) {
}
QuicPacketGenerator::~QuicPacketGenerator() {
for (QuicFrames::iterator it = queued_control_frames_.begin();
it != queued_control_frames_.end(); ++it) {
switch (it->type) {
case PADDING_FRAME:
delete it->padding_frame;
break;
case STREAM_FRAME:
delete it->stream_frame;
break;
case ACK_FRAME:
delete it->ack_frame;
break;
case CONGESTION_FEEDBACK_FRAME:
delete it->congestion_feedback_frame;
break;
case RST_STREAM_FRAME:
delete it->rst_stream_frame;
break;
case CONNECTION_CLOSE_FRAME:
delete it->connection_close_frame;
break;
case GOAWAY_FRAME:
delete it->goaway_frame;
break;
case WINDOW_UPDATE_FRAME:
delete it->window_update_frame;
break;
case BLOCKED_FRAME:
delete it->blocked_frame;
break;
case STOP_WAITING_FRAME:
delete it->stop_waiting_frame;
break;
case PING_FRAME:
delete it->ping_frame;
break;
case NUM_FRAME_TYPES:
DCHECK(false) << "Cannot delete type: " << it->type;
}
}
}
// NetworkChangeVisitor method.
void QuicPacketGenerator::OnCongestionWindowChange(
QuicByteCount congestion_window) {
packet_creator_.set_max_packets_per_fec_group(
static_cast<size_t>(kCongestionWindowMultiplierForFecGroupSize *
congestion_window / kDefaultTCPMSS));
}
void QuicPacketGenerator::SetShouldSendAck(bool also_send_feedback,
bool also_send_stop_waiting) {
should_send_ack_ = true;
should_send_feedback_ = also_send_feedback;
should_send_stop_waiting_ = also_send_stop_waiting;
SendQueuedFrames(false);
}
void QuicPacketGenerator::SetShouldSendStopWaiting() {
should_send_stop_waiting_ = true;
SendQueuedFrames(false);
}
void QuicPacketGenerator::AddControlFrame(const QuicFrame& frame) {
queued_control_frames_.push_back(frame);
SendQueuedFrames(false);
}
QuicConsumedData QuicPacketGenerator::ConsumeData(QuicStreamId id,
const IOVector& data_to_write,
QuicStreamOffset offset,
bool fin,
FecProtection fec_protection,
QuicAckNotifier* notifier) {
IsHandshake handshake = id == kCryptoStreamId ? IS_HANDSHAKE : NOT_HANDSHAKE;
// To make reasoning about crypto frames easier, we don't combine them with
// other retransmittable frames in a single packet.
const bool flush = handshake == IS_HANDSHAKE &&
packet_creator_.HasPendingRetransmittableFrames();
SendQueuedFrames(flush);
size_t total_bytes_consumed = 0;
bool fin_consumed = false;
if (!packet_creator_.HasRoomForStreamFrame(id, offset)) {
SerializeAndSendPacket();
}
if (fec_protection == MUST_FEC_PROTECT) {
MaybeStartFecProtection();
}
IOVector data = data_to_write;
size_t data_size = data.TotalBufferSize();
while (delegate_->ShouldGeneratePacket(NOT_RETRANSMISSION,
HAS_RETRANSMITTABLE_DATA, handshake)) {
QuicFrame frame;
size_t bytes_consumed;
if (notifier != NULL) {
// We want to track which packet this stream frame ends up in.
bytes_consumed = packet_creator_.CreateStreamFrameWithNotifier(
id, data, offset + total_bytes_consumed, fin, notifier, &frame);
} else {
bytes_consumed = packet_creator_.CreateStreamFrame(
id, data, offset + total_bytes_consumed, fin, &frame);
}
if (!AddFrame(frame)) {
LOG(DFATAL) << "Failed to add stream frame.";
// Inability to add a STREAM frame creates an unrecoverable hole in a
// the stream, so it's best to close the connection.
delegate_->CloseConnection(QUIC_INTERNAL_ERROR, false);
return QuicConsumedData(0, false);
}
total_bytes_consumed += bytes_consumed;
fin_consumed = fin && total_bytes_consumed == data_size;
data.Consume(bytes_consumed);
DCHECK(data.Empty() || packet_creator_.BytesFree() == 0u);
// TODO(ianswett): Restore packet reordering.
if (!InBatchMode() || !packet_creator_.HasRoomForStreamFrame(id, offset)) {
SerializeAndSendPacket();
}
if (data.Empty()) {
// We're done writing the data. Exit the loop.
// We don't make this a precondition because we could have 0 bytes of data
// if we're simply writing a fin.
if (fec_protection == MUST_FEC_PROTECT) {
// Turn off FEC protection when we're done writing protected data.
DVLOG(1) << "Turning FEC protection OFF";
should_fec_protect_ = false;
}
break;
}
}
// Don't allow the handshake to be bundled with other retransmittable frames.
if (handshake == IS_HANDSHAKE) {
SendQueuedFrames(true);
}
// Try to close FEC group since we've either run out of data to send or we're
// blocked. If not in batch mode, force close the group.
// TODO(jri): This method should be called with flush=false here
// once the timer-based FEC sending is done, to separate FEC sending from
// the end of batch operations.
MaybeSendFecPacketAndCloseGroup(!InBatchMode());
DCHECK(InBatchMode() || !packet_creator_.HasPendingFrames());
return QuicConsumedData(total_bytes_consumed, fin_consumed);
}
bool QuicPacketGenerator::CanSendWithNextPendingFrameAddition() const {
DCHECK(HasPendingFrames());
HasRetransmittableData retransmittable =
(should_send_ack_ || should_send_feedback_ || should_send_stop_waiting_)
? NO_RETRANSMITTABLE_DATA : HAS_RETRANSMITTABLE_DATA;
if (retransmittable == HAS_RETRANSMITTABLE_DATA) {
DCHECK(!queued_control_frames_.empty()); // These are retransmittable.
}
return delegate_->ShouldGeneratePacket(NOT_RETRANSMISSION, retransmittable,
NOT_HANDSHAKE);
}
void QuicPacketGenerator::SendQueuedFrames(bool flush) {
// Only add pending frames if we are SURE we can then send the whole packet.
while (HasPendingFrames() &&
(flush || CanSendWithNextPendingFrameAddition())) {
if (!AddNextPendingFrame()) {
// Packet was full, so serialize and send it.
SerializeAndSendPacket();
}
}
if (!InBatchMode() || flush) {
if (packet_creator_.HasPendingFrames()) {
SerializeAndSendPacket();
}
// Ensure the FEC group is closed at the end of this method unless other
// writes are pending.
MaybeSendFecPacketAndCloseGroup(true);
}
}
void QuicPacketGenerator::MaybeStartFecProtection() {
if (!packet_creator_.IsFecEnabled()) {
return;
}
DVLOG(1) << "Turning FEC protection ON";
should_fec_protect_ = true;
if (packet_creator_.IsFecProtected()) {
// Only start creator's FEC protection if not already on.
return;
}
if (HasQueuedFrames()) {
// TODO(jri): This currently requires that the generator flush out any
// pending frames when FEC protection is turned on. If current packet can be
// converted to an FEC protected packet, do it. This will require the
// generator to check if the resulting expansion still allows the incoming
// frame to be added to the packet.
SendQueuedFrames(true);
}
packet_creator_.StartFecProtectingPackets();
DCHECK(packet_creator_.IsFecProtected());
}
void QuicPacketGenerator::MaybeSendFecPacketAndCloseGroup(bool force) {
if (!packet_creator_.IsFecProtected() ||
packet_creator_.HasPendingFrames() ||
!packet_creator_.ShouldSendFec(force)) {
return;
}
// TODO(jri): SerializeFec can return a NULL packet, and this should
// cause an early return, with a call to delegate_->OnPacketGenerationError.
SerializedPacket serialized_fec = packet_creator_.SerializeFec();
DCHECK(serialized_fec.packet);
delegate_->OnSerializedPacket(serialized_fec);
// Turn FEC protection off if creator's protection is on and the creator
// does not have an open FEC group.
// Note: We only wait until the frames queued in the creator are flushed;
// pending frames in the generator will not keep us from turning FEC off.
if (!should_fec_protect_ && !packet_creator_.IsFecGroupOpen()) {
packet_creator_.StopFecProtectingPackets();
DCHECK(!packet_creator_.IsFecProtected());
}
}
bool QuicPacketGenerator::InBatchMode() {
return batch_mode_;
}
void QuicPacketGenerator::StartBatchOperations() {
batch_mode_ = true;
}
void QuicPacketGenerator::FinishBatchOperations() {
batch_mode_ = false;
SendQueuedFrames(false);
}
void QuicPacketGenerator::FlushAllQueuedFrames() {
SendQueuedFrames(true);
}
bool QuicPacketGenerator::HasQueuedFrames() const {
return packet_creator_.HasPendingFrames() || HasPendingFrames();
}
bool QuicPacketGenerator::HasPendingFrames() const {
return should_send_ack_ || should_send_feedback_ ||
should_send_stop_waiting_ || !queued_control_frames_.empty();
}
bool QuicPacketGenerator::AddNextPendingFrame() {
if (should_send_ack_) {
pending_ack_frame_.reset(delegate_->CreateAckFrame());
// If we can't this add the frame now, then we still need to do so later.
should_send_ack_ = !AddFrame(QuicFrame(pending_ack_frame_.get()));
// Return success if we have cleared out this flag (i.e., added the frame).
// If we still need to send, then the frame is full, and we have failed.
return !should_send_ack_;
}
if (should_send_feedback_) {
pending_feedback_frame_.reset(delegate_->CreateFeedbackFrame());
// If we can't this add the frame now, then we still need to do so later.
should_send_feedback_ = !AddFrame(QuicFrame(pending_feedback_frame_.get()));
// Return success if we have cleared out this flag (i.e., added the frame).
// If we still need to send, then the frame is full, and we have failed.
return !should_send_feedback_;
}
if (should_send_stop_waiting_) {
pending_stop_waiting_frame_.reset(delegate_->CreateStopWaitingFrame());
// If we can't this add the frame now, then we still need to do so later.
should_send_stop_waiting_ =
!AddFrame(QuicFrame(pending_stop_waiting_frame_.get()));
// Return success if we have cleared out this flag (i.e., added the frame).
// If we still need to send, then the frame is full, and we have failed.
return !should_send_stop_waiting_;
}
LOG_IF(DFATAL, queued_control_frames_.empty())
<< "AddNextPendingFrame called with no queued control frames.";
if (!AddFrame(queued_control_frames_.back())) {
// Packet was full.
return false;
}
queued_control_frames_.pop_back();
return true;
}
bool QuicPacketGenerator::AddFrame(const QuicFrame& frame) {
bool success = packet_creator_.AddSavedFrame(frame);
if (success && debug_delegate_) {
debug_delegate_->OnFrameAddedToPacket(frame);
}
return success;
}
void QuicPacketGenerator::SerializeAndSendPacket() {
SerializedPacket serialized_packet = packet_creator_.SerializePacket();
DCHECK(serialized_packet.packet);
delegate_->OnSerializedPacket(serialized_packet);
MaybeSendFecPacketAndCloseGroup(false);
}
void QuicPacketGenerator::StopSendingVersion() {
packet_creator_.StopSendingVersion();
}
QuicPacketSequenceNumber QuicPacketGenerator::sequence_number() const {
return packet_creator_.sequence_number();
}
size_t QuicPacketGenerator::max_packet_length() const {
return packet_creator_.max_packet_length();
}
void QuicPacketGenerator::set_max_packet_length(size_t length) {
packet_creator_.set_max_packet_length(length);
}
QuicEncryptedPacket* QuicPacketGenerator::SerializeVersionNegotiationPacket(
const QuicVersionVector& supported_versions) {
return packet_creator_.SerializeVersionNegotiationPacket(supported_versions);
}
SerializedPacket QuicPacketGenerator::ReserializeAllFrames(
const QuicFrames& frames,
QuicSequenceNumberLength original_length) {
return packet_creator_.ReserializeAllFrames(frames, original_length);
}
void QuicPacketGenerator::UpdateSequenceNumberLength(
QuicPacketSequenceNumber least_packet_awaited_by_peer,
QuicByteCount congestion_window) {
return packet_creator_.UpdateSequenceNumberLength(
least_packet_awaited_by_peer, congestion_window);
}
void QuicPacketGenerator::set_encryption_level(EncryptionLevel level) {
packet_creator_.set_encryption_level(level);
}
} // namespace net