blob: d2d1699253763a2302ba6efd404e678464cf8b55 [file] [log] [blame] [edit]
// Copyright (c) 2018 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/third_party/quiche/src/quic/core/http/http_encoder.h"
#include "net/third_party/quiche/src/quic/core/quic_data_writer.h"
#include "net/third_party/quiche/src/quic/platform/api/quic_bug_tracker.h"
#include "net/third_party/quiche/src/quic/platform/api/quic_fallthrough.h"
#include "net/third_party/quiche/src/quic/platform/api/quic_logging.h"
#include "net/third_party/quiche/src/quic/platform/api/quic_string.h"
namespace quic {
namespace {
// Set the first byte of a PRIORITY frame according to its fields.
uint8_t SetPriorityFields(uint8_t num,
PriorityElementType type,
bool prioritized) {
switch (type) {
case REQUEST_STREAM:
return num;
case PUSH_STREAM:
if (prioritized) {
return num | (1 << 6);
}
return num | (1 << 4);
case PLACEHOLDER:
if (prioritized) {
return num | (1 << 7);
}
return num | (1 << 5);
case ROOT_OF_TREE:
if (prioritized) {
num = num | (1 << 6);
return num | (1 << 7);
}
num = num | (1 << 4);
return num | (1 << 5);
default:
QUIC_NOTREACHED();
return num;
}
}
// Length of the type field of a frame.
static const size_t kFrameTypeLength = 1;
// Length of the weight field of a priority frame.
static const size_t kPriorityWeightLength = 1;
// Length of a priority frame's first byte.
static const size_t kPriorityFirstByteLength = 1;
// Length of a key in the map of a settings frame.
static const size_t kSettingsMapKeyLength = 2;
} // namespace
HttpEncoder::HttpEncoder() {}
HttpEncoder::~HttpEncoder() {}
QuicByteCount HttpEncoder::SerializeDataFrameHeader(
QuicByteCount payload_length,
std::unique_ptr<char[]>* output) {
DCHECK_NE(0u, payload_length);
QuicByteCount header_length =
QuicDataWriter::GetVarInt62Len(payload_length) + kFrameTypeLength;
output->reset(new char[header_length]);
QuicDataWriter writer(header_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::DATA, &writer)) {
return header_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializeHeadersFrameHeader(
QuicByteCount payload_length,
std::unique_ptr<char[]>* output) {
DCHECK_NE(0u, payload_length);
QuicByteCount header_length =
QuicDataWriter::GetVarInt62Len(payload_length) + kFrameTypeLength;
output->reset(new char[header_length]);
QuicDataWriter writer(header_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::HEADERS, &writer)) {
return header_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializePriorityFrame(
const PriorityFrame& priority,
std::unique_ptr<char[]>* output) {
QuicByteCount payload_length =
kPriorityFirstByteLength +
QuicDataWriter::GetVarInt62Len(priority.prioritized_element_id) +
QuicDataWriter::GetVarInt62Len(priority.element_dependency_id) +
kPriorityWeightLength;
QuicByteCount total_length = GetTotalLength(payload_length);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (!WriteFrameHeader(payload_length, HttpFrameType::PRIORITY, &writer)) {
return 0;
}
// Set the first byte of the payload.
uint8_t bits = 0;
bits = SetPriorityFields(bits, priority.prioritized_type, true);
bits = SetPriorityFields(bits, priority.dependency_type, false);
if (priority.exclusive) {
bits |= 1;
}
if (writer.WriteUInt8(bits) &&
writer.WriteVarInt62(priority.prioritized_element_id) &&
writer.WriteVarInt62(priority.element_dependency_id) &&
writer.WriteUInt8(priority.weight)) {
return total_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializeCancelPushFrame(
const CancelPushFrame& cancel_push,
std::unique_ptr<char[]>* output) {
QuicByteCount payload_length =
QuicDataWriter::GetVarInt62Len(cancel_push.push_id);
QuicByteCount total_length = GetTotalLength(payload_length);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::CANCEL_PUSH, &writer) &&
writer.WriteVarInt62(cancel_push.push_id)) {
return total_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializeSettingsFrame(
const SettingsFrame& settings,
std::unique_ptr<char[]>* output) {
// Calculate the key sizes.
QuicByteCount payload_length = settings.values.size() * kSettingsMapKeyLength;
// Calculate the value sizes.
for (auto it = settings.values.begin(); it != settings.values.end(); ++it) {
payload_length += QuicDataWriter::GetVarInt62Len(it->second);
}
QuicByteCount total_length = GetTotalLength(payload_length);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (!WriteFrameHeader(payload_length, HttpFrameType::SETTINGS, &writer)) {
return 0;
}
for (auto it = settings.values.begin(); it != settings.values.end(); ++it) {
if (!writer.WriteUInt16(it->first) || !writer.WriteVarInt62(it->second)) {
return 0;
}
}
return total_length;
}
QuicByteCount HttpEncoder::SerializePushPromiseFrameWithOnlyPushId(
const PushPromiseFrame& push_promise,
std::unique_ptr<char[]>* output) {
QuicByteCount payload_length =
QuicDataWriter::GetVarInt62Len(push_promise.push_id) +
push_promise.headers.length();
// GetTotalLength() is not used because headers will not be serialized.
QuicByteCount total_length =
QuicDataWriter::GetVarInt62Len(payload_length) + kFrameTypeLength +
QuicDataWriter::GetVarInt62Len(push_promise.push_id);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::PUSH_PROMISE, &writer) &&
writer.WriteVarInt62(push_promise.push_id)) {
return total_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializeGoAwayFrame(
const GoAwayFrame& goaway,
std::unique_ptr<char[]>* output) {
QuicByteCount payload_length =
QuicDataWriter::GetVarInt62Len(goaway.stream_id);
QuicByteCount total_length = GetTotalLength(payload_length);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::GOAWAY, &writer) &&
writer.WriteVarInt62(goaway.stream_id)) {
return total_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializeMaxPushIdFrame(
const MaxPushIdFrame& max_push_id,
std::unique_ptr<char[]>* output) {
QuicByteCount payload_length =
QuicDataWriter::GetVarInt62Len(max_push_id.push_id);
QuicByteCount total_length = GetTotalLength(payload_length);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::MAX_PUSH_ID, &writer) &&
writer.WriteVarInt62(max_push_id.push_id)) {
return total_length;
}
return 0;
}
QuicByteCount HttpEncoder::SerializeDuplicatePushFrame(
const DuplicatePushFrame& duplicate_push,
std::unique_ptr<char[]>* output) {
QuicByteCount payload_length =
QuicDataWriter::GetVarInt62Len(duplicate_push.push_id);
QuicByteCount total_length = GetTotalLength(payload_length);
output->reset(new char[total_length]);
QuicDataWriter writer(total_length, output->get());
if (WriteFrameHeader(payload_length, HttpFrameType::DUPLICATE_PUSH,
&writer) &&
writer.WriteVarInt62(duplicate_push.push_id)) {
return total_length;
}
return 0;
}
bool HttpEncoder::WriteFrameHeader(QuicByteCount length,
HttpFrameType type,
QuicDataWriter* writer) {
return writer->WriteVarInt62(length) &&
writer->WriteUInt8(static_cast<uint8_t>(type));
}
QuicByteCount HttpEncoder::GetTotalLength(QuicByteCount payload_length) {
return QuicDataWriter::GetVarInt62Len(payload_length) + kFrameTypeLength +
payload_length;
}
} // namespace quic