blob: 0bbb8041752f0b442f7f6f05341a25a448e015d5 [file] [log] [blame]
// 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.
LoadBalancerServerId LoadBalancerDecoder::GetServerId(
const QuicConnectionId& connection_id) const {
std::optional<uint8_t> config_id = GetConfigId(connection_id);
if (!config_id.has_value()) {
return LoadBalancerServerId();
}
std::optional<LoadBalancerConfig> config = config_[*config_id];
if (!config.has_value()) {
return LoadBalancerServerId();
}
// Benchmark tests show that minimizing the computation inside
// LoadBalancerConfig saves CPU cycles.
if (connection_id.length() < config->total_len()) {
return LoadBalancerServerId();
}
const uint8_t* data =
reinterpret_cast<const uint8_t*>(connection_id.data()) + 1;
uint8_t server_id_len = config->server_id_len();
if (!config->IsEncrypted()) {
return LoadBalancerServerId(absl::Span<const uint8_t>(data, server_id_len));
}
if (config->plaintext_len() == kLoadBalancerBlockSize) {
uint8_t scratch[kLoadBalancerBlockSize];
if (!config->BlockDecrypt(data, scratch)) {
return LoadBalancerServerId();
}
return LoadBalancerServerId(absl::Span<uint8_t>(scratch, server_id_len));
}
return config->FourPassDecrypt(
absl::MakeConstSpan(data, connection_id.length() - 1));
}
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