|  | // Copyright 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/qpack/qpack_constants.h" | 
|  |  | 
|  | #include <limits> | 
|  |  | 
|  | #include "net/third_party/quiche/src/quic/platform/api/quic_logging.h" | 
|  |  | 
|  | namespace quic { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Validate that | 
|  | //  * in each instruction, the bits of |value| that are zero in |mask| are zero; | 
|  | //  * every byte matches exactly one opcode. | 
|  | void ValidateLangague(const QpackLanguage* language) { | 
|  | #ifndef NDEBUG | 
|  | for (const auto* instruction : *language) { | 
|  | DCHECK_EQ(0, instruction->opcode.value & ~instruction->opcode.mask); | 
|  | } | 
|  |  | 
|  | for (uint8_t byte = 0; byte < std::numeric_limits<uint8_t>::max(); ++byte) { | 
|  | size_t match_count = 0; | 
|  | for (const auto* instruction : *language) { | 
|  | if ((byte & instruction->opcode.mask) == instruction->opcode.value) { | 
|  | ++match_count; | 
|  | } | 
|  | } | 
|  | DCHECK_EQ(1u, match_count) << static_cast<int>(byte); | 
|  | } | 
|  | #else | 
|  | (void)language; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | bool operator==(const QpackInstructionOpcode& a, | 
|  | const QpackInstructionOpcode& b) { | 
|  | return std::tie(a.value, a.mask) == std::tie(b.value, b.mask); | 
|  | } | 
|  |  | 
|  | const QpackInstruction* InsertWithNameReferenceInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b10000000, 0b10000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kSbit, 0b01000000}, | 
|  | {QpackInstructionFieldType::kVarint, 6}, | 
|  | {QpackInstructionFieldType::kValue, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* InsertWithoutNameReferenceInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b01000000, 0b11000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kName, 5}, | 
|  | {QpackInstructionFieldType::kValue, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* DuplicateInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00000000, 0b11100000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, {{QpackInstructionFieldType::kVarint, 5}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* SetDynamicTableCapacityInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00100000, 0b11100000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, {{QpackInstructionFieldType::kVarint, 5}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackLanguage* QpackEncoderStreamLanguage() { | 
|  | static const QpackLanguage* const language = new QpackLanguage{ | 
|  | InsertWithNameReferenceInstruction(), | 
|  | InsertWithoutNameReferenceInstruction(), DuplicateInstruction(), | 
|  | SetDynamicTableCapacityInstruction()}; | 
|  | ValidateLangague(language); | 
|  | return language; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* InsertCountIncrementInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00000000, 0b11000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, {{QpackInstructionFieldType::kVarint, 6}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* HeaderAcknowledgementInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b10000000, 0b10000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, {{QpackInstructionFieldType::kVarint, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* StreamCancellationInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b01000000, 0b11000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, {{QpackInstructionFieldType::kVarint, 6}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackLanguage* QpackDecoderStreamLanguage() { | 
|  | static const QpackLanguage* const language = new QpackLanguage{ | 
|  | InsertCountIncrementInstruction(), HeaderAcknowledgementInstruction(), | 
|  | StreamCancellationInstruction()}; | 
|  | ValidateLangague(language); | 
|  | return language; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* QpackPrefixInstruction() { | 
|  | // This opcode matches every input. | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00000000, 0b00000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kVarint, 8}, | 
|  | {QpackInstructionFieldType::kSbit, 0b10000000}, | 
|  | {QpackInstructionFieldType::kVarint2, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackLanguage* QpackPrefixLanguage() { | 
|  | static const QpackLanguage* const language = | 
|  | new QpackLanguage{QpackPrefixInstruction()}; | 
|  | ValidateLangague(language); | 
|  | return language; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* QpackIndexedHeaderFieldInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b10000000, 0b10000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kSbit, 0b01000000}, | 
|  | {QpackInstructionFieldType::kVarint, 6}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* QpackIndexedHeaderFieldPostBaseInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00010000, 0b11110000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, {{QpackInstructionFieldType::kVarint, 4}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* QpackLiteralHeaderFieldNameReferenceInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b01000000, 0b11000000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kSbit, 0b00010000}, | 
|  | {QpackInstructionFieldType::kVarint, 4}, | 
|  | {QpackInstructionFieldType::kValue, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* QpackLiteralHeaderFieldPostBaseInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00000000, 0b11110000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kVarint, 3}, | 
|  | {QpackInstructionFieldType::kValue, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackInstruction* QpackLiteralHeaderFieldInstruction() { | 
|  | static const QpackInstructionOpcode* const opcode = | 
|  | new QpackInstructionOpcode{0b00100000, 0b11100000}; | 
|  | static const QpackInstruction* const instruction = | 
|  | new QpackInstruction{*opcode, | 
|  | {{QpackInstructionFieldType::kName, 3}, | 
|  | {QpackInstructionFieldType::kValue, 7}}}; | 
|  | return instruction; | 
|  | } | 
|  |  | 
|  | const QpackLanguage* QpackRequestStreamLanguage() { | 
|  | static const QpackLanguage* const language = | 
|  | new QpackLanguage{QpackIndexedHeaderFieldInstruction(), | 
|  | QpackIndexedHeaderFieldPostBaseInstruction(), | 
|  | QpackLiteralHeaderFieldNameReferenceInstruction(), | 
|  | QpackLiteralHeaderFieldPostBaseInstruction(), | 
|  | QpackLiteralHeaderFieldInstruction()}; | 
|  | ValidateLangague(language); | 
|  | return language; | 
|  | } | 
|  |  | 
|  | }  // namespace quic |