| // 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 "quic/core/qpack/qpack_instructions.h" | 
 |  | 
 | #include <limits> | 
 |  | 
 | #include "absl/strings/string_view.h" | 
 | #include "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) { | 
 |     QUICHE_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; | 
 |       } | 
 |     } | 
 |     QUICHE_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; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::InsertWithNameReference( | 
 |     bool is_static, | 
 |     uint64_t name_index, | 
 |     absl::string_view value) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = InsertWithNameReferenceInstruction(); | 
 |   instruction_with_values.s_bit_ = is_static; | 
 |   instruction_with_values.varint_ = name_index; | 
 |   instruction_with_values.value_ = value; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues | 
 | QpackInstructionWithValues::InsertWithoutNameReference( | 
 |     absl::string_view name, | 
 |     absl::string_view value) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = | 
 |       InsertWithoutNameReferenceInstruction(); | 
 |   instruction_with_values.name_ = name; | 
 |   instruction_with_values.value_ = value; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::Duplicate( | 
 |     uint64_t index) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = DuplicateInstruction(); | 
 |   instruction_with_values.varint_ = index; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::SetDynamicTableCapacity( | 
 |     uint64_t capacity) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = SetDynamicTableCapacityInstruction(); | 
 |   instruction_with_values.varint_ = capacity; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::InsertCountIncrement( | 
 |     uint64_t increment) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = InsertCountIncrementInstruction(); | 
 |   instruction_with_values.varint_ = increment; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::HeaderAcknowledgement( | 
 |     uint64_t stream_id) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = HeaderAcknowledgementInstruction(); | 
 |   instruction_with_values.varint_ = stream_id; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::StreamCancellation( | 
 |     uint64_t stream_id) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = StreamCancellationInstruction(); | 
 |   instruction_with_values.varint_ = stream_id; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::Prefix( | 
 |     uint64_t required_insert_count) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = QpackPrefixInstruction(); | 
 |   instruction_with_values.varint_ = required_insert_count; | 
 |   instruction_with_values.varint2_ = 0;    // Delta Base. | 
 |   instruction_with_values.s_bit_ = false;  // Delta Base sign. | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::IndexedHeaderField( | 
 |     bool is_static, | 
 |     uint64_t index) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = QpackIndexedHeaderFieldInstruction(); | 
 |   instruction_with_values.s_bit_ = is_static; | 
 |   instruction_with_values.varint_ = index; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues | 
 | QpackInstructionWithValues::LiteralHeaderFieldNameReference( | 
 |     bool is_static, | 
 |     uint64_t index, | 
 |     absl::string_view value) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = | 
 |       QpackLiteralHeaderFieldNameReferenceInstruction(); | 
 |   instruction_with_values.s_bit_ = is_static; | 
 |   instruction_with_values.varint_ = index; | 
 |   instruction_with_values.value_ = value; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | // static | 
 | QpackInstructionWithValues QpackInstructionWithValues::LiteralHeaderField( | 
 |     absl::string_view name, | 
 |     absl::string_view value) { | 
 |   QpackInstructionWithValues instruction_with_values; | 
 |   instruction_with_values.instruction_ = QpackLiteralHeaderFieldInstruction(); | 
 |   instruction_with_values.name_ = name; | 
 |   instruction_with_values.value_ = value; | 
 |  | 
 |   return instruction_with_values; | 
 | } | 
 |  | 
 | }  // namespace quic |