|  | // Copyright (c) 2022 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 "quiche/quic/load_balancer/load_balancer_decoder.h" | 
|  |  | 
|  | #include <cstdint> | 
|  | #include <cstring> | 
|  | #include <optional> | 
|  |  | 
|  | #include "absl/types/span.h" | 
|  | #include "quiche/quic/core/quic_connection_id.h" | 
|  | #include "quiche/quic/load_balancer/load_balancer_config.h" | 
|  | #include "quiche/quic/load_balancer/load_balancer_server_id.h" | 
|  | #include "quiche/quic/platform/api/quic_bug_tracker.h" | 
|  |  | 
|  | namespace quic { | 
|  |  | 
|  | bool LoadBalancerDecoder::AddConfig(const LoadBalancerConfig& config) { | 
|  | if (config_[config.config_id()].has_value()) { | 
|  | return false; | 
|  | } | 
|  | config_[config.config_id()] = config; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void LoadBalancerDecoder::DeleteConfig(uint8_t config_id) { | 
|  | if (config_id >= kNumLoadBalancerConfigs) { | 
|  | QUIC_BUG(quic_bug_438896865_01) | 
|  | << "Decoder deleting config with invalid config_id " | 
|  | << static_cast<int>(config_id); | 
|  | return; | 
|  | } | 
|  | config_[config_id].reset(); | 
|  | } | 
|  |  | 
|  | // This is the core logic to extract a server ID given a valid config and | 
|  | // connection ID of sufficient length. | 
|  | bool LoadBalancerDecoder::GetServerId(const QuicConnectionId& connection_id, | 
|  | LoadBalancerServerId& server_id) const { | 
|  | std::optional<uint8_t> config_id = GetConfigId(connection_id); | 
|  | if (!config_id.has_value()) { | 
|  | return false; | 
|  | } | 
|  | std::optional<LoadBalancerConfig> config = config_[*config_id]; | 
|  | if (!config.has_value()) { | 
|  | return false; | 
|  | } | 
|  | // Benchmark tests show that minimizing the computation inside | 
|  | // LoadBalancerConfig saves CPU cycles. | 
|  | if (connection_id.length() < config->total_len()) { | 
|  | return false; | 
|  | } | 
|  | const uint8_t* data = | 
|  | reinterpret_cast<const uint8_t*>(connection_id.data()) + 1; | 
|  | uint8_t server_id_len = config->server_id_len(); | 
|  | server_id.set_length(server_id_len); | 
|  | if (!config->IsEncrypted()) { | 
|  | memcpy(server_id.mutable_data(), connection_id.data() + 1, server_id_len); | 
|  | return true; | 
|  | } | 
|  | if (config->plaintext_len() == kLoadBalancerBlockSize) { | 
|  | return config->BlockDecrypt(data, server_id.mutable_data()); | 
|  | } | 
|  | return config->FourPassDecrypt( | 
|  | absl::MakeConstSpan(data, connection_id.length() - 1), server_id); | 
|  | } | 
|  |  | 
|  | std::optional<uint8_t> LoadBalancerDecoder::GetConfigId( | 
|  | const QuicConnectionId& connection_id) { | 
|  | if (connection_id.IsEmpty()) { | 
|  | return std::optional<uint8_t>(); | 
|  | } | 
|  | return GetConfigId(*reinterpret_cast<const uint8_t*>(connection_id.data())); | 
|  | } | 
|  |  | 
|  | std::optional<uint8_t> LoadBalancerDecoder::GetConfigId( | 
|  | const uint8_t connection_id_first_byte) { | 
|  | uint8_t codepoint = (connection_id_first_byte >> kConnectionIdLengthBits); | 
|  | if (codepoint < kNumLoadBalancerConfigs) { | 
|  | return codepoint; | 
|  | } | 
|  | return std::optional<uint8_t>(); | 
|  | } | 
|  |  | 
|  | }  // namespace quic |