| // Copyright 2018 Google LLC |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // https://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| // A library that assigns a global numbering to the events in the trace. |
| |
| #ifndef THIRD_PARTY_QUIC_TRACE_LIB_ANALYSIS_TRACE_NUMBERING_H_ |
| #define THIRD_PARTY_QUIC_TRACE_LIB_ANALYSIS_TRACE_NUMBERING_H_ |
| |
| #include <glog/logging.h> |
| |
| #include <cstdint> |
| #include <unordered_map> |
| |
| #include "absl/container/flat_hash_map.h" |
| #include "quic_trace/quic_trace.pb.h" |
| |
| namespace quic_trace { |
| |
| // Trace offset of a packet in the connection trace. |
| using TraceOffset = uint64_t; |
| |
| struct Interval { |
| TraceOffset offset; |
| size_t size; |
| }; |
| |
| // Assigns a new connection offset to every sent packet, regardless of whether |
| // it's a retransmission or not. |
| // |
| // TODO(vasilvv): port QuicTcpLikeTraceConverter from QUIC for retransmission |
| // support. |
| class NumberingWithoutRetransmissions { |
| public: |
| Interval AssignTraceNumbering(const Event& event) { |
| size_t size = 0; |
| for (const Frame& frame : event.frames()) { |
| if (frame.has_stream_frame_info()) { |
| size += frame.stream_frame_info().length(); |
| } |
| if (frame.has_crypto_frame_info()) { |
| size += frame.crypto_frame_info().length(); |
| } |
| } |
| TraceOffset offset = current_offset_; |
| current_offset_ += size; |
| GetOffsets(event.encryption_level()) |
| ->emplace(event.packet_number(), Interval{offset, size}); |
| return {offset, size}; |
| } |
| |
| Interval GetTraceNumbering(uint64_t packet_number, |
| EncryptionLevel enc_level) { |
| auto offsets = GetOffsets(enc_level); |
| auto it = offsets->find(packet_number); |
| return it != offsets->end() ? it->second : Interval{0, 0}; |
| } |
| |
| private: |
| absl::flat_hash_map<uint64_t, Interval>* GetOffsets( |
| EncryptionLevel enc_level) { |
| switch (enc_level) { |
| case ENCRYPTION_INITIAL: |
| return &offsets_initial_; |
| case ENCRYPTION_HANDSHAKE: |
| return &offsets_handshake_; |
| case ENCRYPTION_0RTT: |
| case ENCRYPTION_1RTT: |
| return &offsets_1rtt_; |
| default: |
| LOG(FATAL) << "Unknown encryption level."; |
| } |
| } |
| |
| TraceOffset current_offset_ = 0; |
| absl::flat_hash_map<uint64_t, Interval> offsets_initial_; |
| absl::flat_hash_map<uint64_t, Interval> offsets_handshake_; |
| absl::flat_hash_map<uint64_t, Interval> offsets_1rtt_; |
| }; |
| |
| } // namespace quic_trace |
| |
| #endif // THIRD_PARTY_QUIC_TRACE_LIB_ANALYSIS_TRACE_NUMBERING_H_ |