|  | // Copyright (c) 2013 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/quic/quic_connection_logger.h" | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <limits> | 
|  | #include <memory> | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/callback.h" | 
|  | #include "base/metrics/histogram_base.h" | 
|  | #include "base/metrics/histogram_functions.h" | 
|  | #include "base/metrics/histogram_macros.h" | 
|  | #include "base/strings/string_number_conversions.h" | 
|  | #include "base/values.h" | 
|  | #include "net/base/ip_address.h" | 
|  | #include "net/cert/x509_certificate.h" | 
|  | #include "net/log/net_log.h" | 
|  | #include "net/log/net_log_capture_mode.h" | 
|  | #include "net/log/net_log_event_type.h" | 
|  | #include "net/quic/quic_address_mismatch.h" | 
|  | #include "net/third_party/quic/core/crypto/crypto_handshake_message.h" | 
|  | #include "net/third_party/quic/core/crypto/crypto_protocol.h" | 
|  | #include "net/third_party/quic/core/quic_connection_id.h" | 
|  | #include "net/third_party/quic/core/quic_packets.h" | 
|  | #include "net/third_party/quic/core/quic_socket_address_coder.h" | 
|  | #include "net/third_party/quic/core/quic_time.h" | 
|  | #include "net/third_party/quic/core/quic_utils.h" | 
|  | #include "net/third_party/quic/platform/api/quic_string_piece.h" | 
|  |  | 
|  | using std::string; | 
|  |  | 
|  | namespace net { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicPacketCallback( | 
|  | const IPEndPoint* self_address, | 
|  | const IPEndPoint* peer_address, | 
|  | size_t packet_size, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetString("self_address", self_address->ToString()); | 
|  | dict->SetString("peer_address", peer_address->ToString()); | 
|  | dict->SetInteger("size", packet_size); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicPacketSentCallback( | 
|  | const quic::SerializedPacket& serialized_packet, | 
|  | quic::TransmissionType transmission_type, | 
|  | quic::QuicTime sent_time, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("transmission_type", transmission_type); | 
|  | dict->SetKey("packet_number", | 
|  | NetLogNumberValue(serialized_packet.packet_number.ToUint64())); | 
|  | dict->SetInteger("size", serialized_packet.encrypted_length); | 
|  | dict->SetKey("sent_time_us", NetLogNumberValue(sent_time.ToDebuggingValue())); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicPacketRetransmittedCallback( | 
|  | quic::QuicPacketNumber old_packet_number, | 
|  | quic::QuicPacketNumber new_packet_number, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetKey("old_packet_number", | 
|  | NetLogNumberValue(old_packet_number.ToUint64())); | 
|  | dict->SetKey("new_packet_number", | 
|  | NetLogNumberValue(new_packet_number.ToUint64())); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicPacketLostCallback( | 
|  | quic::QuicPacketNumber packet_number, | 
|  | quic::TransmissionType transmission_type, | 
|  | quic::QuicTime detection_time, | 
|  | NetLogCaptureMode /*capture_mode*/) { | 
|  | auto dict = std::make_unique<base::DictionaryValue>(); | 
|  | dict->SetInteger("transmission_type", transmission_type); | 
|  | dict->SetKey("packet_number", NetLogNumberValue(packet_number.ToUint64())); | 
|  | dict->SetKey("detection_time_us", | 
|  | NetLogNumberValue(detection_time.ToDebuggingValue())); | 
|  | #if defined(STARBOARD) | 
|  | // Copy ellision is not a standard C++11 feature. | 
|  | return std::move(dict); | 
|  | #else | 
|  | return dict; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicDuplicatePacketCallback( | 
|  | quic::QuicPacketNumber packet_number, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetKey("packet_number", NetLogNumberValue(packet_number.ToUint64())); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicPacketHeaderCallback( | 
|  | const quic::QuicPacketHeader* header, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetString("connection_id", | 
|  | header->destination_connection_id.ToString()); | 
|  | dict->SetInteger("reset_flag", header->reset_flag); | 
|  | dict->SetInteger("version_flag", header->version_flag); | 
|  | dict->SetKey("packet_number", | 
|  | NetLogNumberValue(header->packet_number.ToUint64())); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicStreamFrameCallback( | 
|  | const quic::QuicStreamFrame& frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("stream_id", frame.stream_id); | 
|  | dict->SetBoolean("fin", frame.fin); | 
|  | dict->SetKey("offset", NetLogNumberValue(frame.offset)); | 
|  | dict->SetInteger("length", frame.data_length); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicAckFrameCallback( | 
|  | const quic::QuicAckFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | auto dict = std::make_unique<base::DictionaryValue>(); | 
|  | dict->SetKey("largest_observed", | 
|  | NetLogNumberValue(frame->largest_acked.ToUint64())); | 
|  | dict->SetKey("delta_time_largest_observed_us", | 
|  | NetLogNumberValue(frame->ack_delay_time.ToMicroseconds())); | 
|  |  | 
|  | auto missing = std::make_unique<base::ListValue>(); | 
|  | if (!frame->packets.Empty()) { | 
|  | // V34 and above express acked packets, but only print | 
|  | // missing packets, because it's typically a shorter list. | 
|  | for (quic::QuicPacketNumber packet = frame->packets.Min(); | 
|  | packet < frame->largest_acked; ++packet) { | 
|  | if (!frame->packets.Contains(packet)) { | 
|  | missing->GetList().push_back(NetLogNumberValue(packet.ToUint64())); | 
|  | } | 
|  | } | 
|  | } | 
|  | dict->Set("missing_packets", std::move(missing)); | 
|  |  | 
|  | auto received = std::make_unique<base::ListValue>(); | 
|  | const quic::PacketTimeVector& received_times = frame->received_packet_times; | 
|  | for (auto it = received_times.begin(); it != received_times.end(); ++it) { | 
|  | auto info = std::make_unique<base::DictionaryValue>(); | 
|  | info->SetKey("packet_number", NetLogNumberValue(it->first.ToUint64())); | 
|  | info->SetKey("received", NetLogNumberValue(it->second.ToDebuggingValue())); | 
|  | received->Append(std::move(info)); | 
|  | } | 
|  | dict->Set("received_packet_times", std::move(received)); | 
|  |  | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicRstStreamFrameCallback( | 
|  | const quic::QuicRstStreamFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("stream_id", frame->stream_id); | 
|  | dict->SetInteger("quic_rst_stream_error", frame->error_code); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicConnectionCloseFrameCallback( | 
|  | const quic::QuicConnectionCloseFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("quic_error", frame->error_code); | 
|  | dict->SetString("details", frame->error_details); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicWindowUpdateFrameCallback( | 
|  | const quic::QuicWindowUpdateFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("stream_id", frame->stream_id); | 
|  | dict->SetKey("byte_offset", NetLogNumberValue(frame->byte_offset)); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicBlockedFrameCallback( | 
|  | const quic::QuicBlockedFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("stream_id", frame->stream_id); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicGoAwayFrameCallback( | 
|  | const quic::QuicGoAwayFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("quic_error", frame->error_code); | 
|  | dict->SetInteger("last_good_stream_id", frame->last_good_stream_id); | 
|  | dict->SetString("reason_phrase", frame->reason_phrase); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicStopWaitingFrameCallback( | 
|  | const quic::QuicStopWaitingFrame* frame, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | auto dict = std::make_unique<base::DictionaryValue>(); | 
|  | auto sent_info = std::make_unique<base::DictionaryValue>(); | 
|  | sent_info->SetKey("least_unacked", | 
|  | NetLogNumberValue(frame->least_unacked.ToUint64())); | 
|  | dict->Set("sent_info", std::move(sent_info)); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicVersionNegotiationPacketCallback( | 
|  | const quic::QuicVersionNegotiationPacket* packet, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | auto dict = std::make_unique<base::DictionaryValue>(); | 
|  | auto versions = std::make_unique<base::ListValue>(); | 
|  | for (auto it = packet->versions.begin(); it != packet->versions.end(); ++it) { | 
|  | versions->AppendString(ParsedQuicVersionToString(*it)); | 
|  | } | 
|  | dict->Set("versions", std::move(versions)); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicPublicResetPacketCallback( | 
|  | const IPEndPoint* server_hello_address, | 
|  | const IPEndPoint* public_reset_address, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetString("server_hello_address", server_hello_address->ToString()); | 
|  | dict->SetString("public_reset_address", public_reset_address->ToString()); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicCryptoHandshakeMessageCallback( | 
|  | const quic::CryptoHandshakeMessage* message, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetString("quic_crypto_handshake_message", message->DebugString()); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicOnConnectionClosedCallback( | 
|  | quic::QuicErrorCode error, | 
|  | string error_details, | 
|  | quic::ConnectionCloseSource source, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | std::unique_ptr<base::DictionaryValue> dict(new base::DictionaryValue()); | 
|  | dict->SetInteger("quic_error", error); | 
|  | dict->SetString("details", error_details); | 
|  | dict->SetBoolean("from_peer", source == quic::ConnectionCloseSource::FROM_PEER | 
|  | ? true | 
|  | : false); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<base::Value> NetLogQuicCertificateVerifiedCallback( | 
|  | scoped_refptr<X509Certificate> cert, | 
|  | NetLogCaptureMode /* capture_mode */) { | 
|  | // Only the subjects are logged so that we can investigate connection pooling. | 
|  | // More fields could be logged in the future. | 
|  | std::vector<std::string> dns_names; | 
|  | cert->GetSubjectAltName(&dns_names, nullptr); | 
|  | auto dict = std::make_unique<base::DictionaryValue>(); | 
|  | auto subjects = std::make_unique<base::ListValue>(); | 
|  | for (auto& dns_name : dns_names) { | 
|  | subjects->GetList().emplace_back(std::move(dns_name)); | 
|  | } | 
|  | dict->Set("subjects", std::move(subjects)); | 
|  | return std::move(dict); | 
|  | } | 
|  |  | 
|  | void UpdatePublicResetAddressMismatchHistogram( | 
|  | const IPEndPoint& server_hello_address, | 
|  | const IPEndPoint& public_reset_address) { | 
|  | int sample = GetAddressMismatch(server_hello_address, public_reset_address); | 
|  | // We are seemingly talking to an older server that does not support the | 
|  | // feature, so we can't report the results in the histogram. | 
|  | if (sample < 0) { | 
|  | return; | 
|  | } | 
|  | UMA_HISTOGRAM_ENUMERATION("Net.QuicSession.PublicResetAddressMismatch2", | 
|  | static_cast<QuicAddressMismatch>(sample), | 
|  | QUIC_ADDRESS_MISMATCH_MAX); | 
|  | } | 
|  |  | 
|  | // If |address| is an IPv4-mapped IPv6 address, returns ADDRESS_FAMILY_IPV4 | 
|  | // instead of ADDRESS_FAMILY_IPV6. Othewise, behaves like GetAddressFamily(). | 
|  | AddressFamily GetRealAddressFamily(const IPAddress& address) { | 
|  | return address.IsIPv4MappedIPv6() ? ADDRESS_FAMILY_IPV4 | 
|  | : GetAddressFamily(address); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | QuicConnectionLogger::QuicConnectionLogger( | 
|  | quic::QuicSpdySession* session, | 
|  | const char* const connection_description, | 
|  | std::unique_ptr<SocketPerformanceWatcher> socket_performance_watcher, | 
|  | const NetLogWithSource& net_log) | 
|  | : net_log_(net_log), | 
|  | session_(session), | 
|  | last_received_packet_size_(0), | 
|  | no_packet_received_after_ping_(false), | 
|  | previous_received_packet_size_(0), | 
|  | num_out_of_order_received_packets_(0), | 
|  | num_out_of_order_large_received_packets_(0), | 
|  | num_packets_received_(0), | 
|  | num_frames_received_(0), | 
|  | num_duplicate_frames_received_(0), | 
|  | num_incorrect_connection_ids_(0), | 
|  | num_undecryptable_packets_(0), | 
|  | num_duplicate_packets_(0), | 
|  | num_blocked_frames_received_(0), | 
|  | num_blocked_frames_sent_(0), | 
|  | connection_description_(connection_description), | 
|  | socket_performance_watcher_(std::move(socket_performance_watcher)), | 
|  | net_log_is_capturing_(net_log_.IsCapturing()) { | 
|  | timer_.Start(FROM_HERE, base::TimeDelta::FromSeconds(1), this, | 
|  | &QuicConnectionLogger::UpdateIsCapturing); | 
|  | } | 
|  |  | 
|  | QuicConnectionLogger::~QuicConnectionLogger() { | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.OutOfOrderPacketsReceived", | 
|  | num_out_of_order_received_packets_); | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.OutOfOrderLargePacketsReceived", | 
|  | num_out_of_order_large_received_packets_); | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.IncorrectConnectionIDsReceived", | 
|  | num_incorrect_connection_ids_); | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.UndecryptablePacketsReceived", | 
|  | num_undecryptable_packets_); | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.DuplicatePacketsReceived", | 
|  | num_duplicate_packets_); | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.BlockedFrames.Received", | 
|  | num_blocked_frames_received_); | 
|  | UMA_HISTOGRAM_COUNTS_1M("Net.QuicSession.BlockedFrames.Sent", | 
|  | num_blocked_frames_sent_); | 
|  |  | 
|  | const quic::QuicConnectionStats& stats = session_->connection()->GetStats(); | 
|  | UMA_HISTOGRAM_TIMES("Net.QuicSession.MinRTT", | 
|  | base::TimeDelta::FromMicroseconds(stats.min_rtt_us)); | 
|  | UMA_HISTOGRAM_TIMES("Net.QuicSession.SmoothedRTT", | 
|  | base::TimeDelta::FromMicroseconds(stats.srtt_us)); | 
|  |  | 
|  | if (num_frames_received_ > 0) { | 
|  | int duplicate_stream_frame_per_thousand = | 
|  | num_duplicate_frames_received_ * 1000 / num_frames_received_; | 
|  | if (num_packets_received_ < 100) { | 
|  | UMA_HISTOGRAM_CUSTOM_COUNTS( | 
|  | "Net.QuicSession.StreamFrameDuplicatedShortConnection", | 
|  | duplicate_stream_frame_per_thousand, 1, 1000, 75); | 
|  | } else { | 
|  | UMA_HISTOGRAM_CUSTOM_COUNTS( | 
|  | "Net.QuicSession.StreamFrameDuplicatedLongConnection", | 
|  | duplicate_stream_frame_per_thousand, 1, 1000, 75); | 
|  | } | 
|  | } | 
|  |  | 
|  | RecordAggregatePacketLossRate(); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnFrameAddedToPacket(const quic::QuicFrame& frame) { | 
|  | switch (frame.type) { | 
|  | case quic::PADDING_FRAME: | 
|  | break; | 
|  | case quic::STREAM_FRAME: | 
|  | break; | 
|  | case quic::ACK_FRAME: { | 
|  | break; | 
|  | } | 
|  | case quic::RST_STREAM_FRAME: | 
|  | base::UmaHistogramSparse("Net.QuicSession.RstStreamErrorCodeClient", | 
|  | frame.rst_stream_frame->error_code); | 
|  | break; | 
|  | case quic::CONNECTION_CLOSE_FRAME: | 
|  | break; | 
|  | case quic::GOAWAY_FRAME: | 
|  | break; | 
|  | case quic::WINDOW_UPDATE_FRAME: | 
|  | break; | 
|  | case quic::BLOCKED_FRAME: | 
|  | ++num_blocked_frames_sent_; | 
|  | break; | 
|  | case quic::STOP_WAITING_FRAME: | 
|  | break; | 
|  | case quic::PING_FRAME: | 
|  | UMA_HISTOGRAM_BOOLEAN("Net.QuicSession.ConnectionFlowControlBlocked", | 
|  | session_->IsConnectionFlowControlBlocked()); | 
|  | UMA_HISTOGRAM_BOOLEAN("Net.QuicSession.StreamFlowControlBlocked", | 
|  | session_->IsStreamFlowControlBlocked()); | 
|  | break; | 
|  | case quic::MTU_DISCOVERY_FRAME: | 
|  | break; | 
|  | case quic::APPLICATION_CLOSE_FRAME: | 
|  | break; | 
|  | case quic::NEW_CONNECTION_ID_FRAME: | 
|  | break; | 
|  | case quic::MAX_STREAM_ID_FRAME: | 
|  | break; | 
|  | case quic::STREAM_ID_BLOCKED_FRAME: | 
|  | break; | 
|  | case quic::PATH_RESPONSE_FRAME: | 
|  | break; | 
|  | case quic::PATH_CHALLENGE_FRAME: | 
|  | break; | 
|  | case quic::STOP_SENDING_FRAME: | 
|  | break; | 
|  | case quic::MESSAGE_FRAME: | 
|  | break; | 
|  | case quic::CRYPTO_FRAME: | 
|  | break; | 
|  | case quic::NEW_TOKEN_FRAME: | 
|  | break; | 
|  | case quic::RETIRE_CONNECTION_ID_FRAME: | 
|  | break; | 
|  | default: | 
|  | DCHECK(false) << "Illegal frame type: " << frame.type; | 
|  | } | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | switch (frame.type) { | 
|  | case quic::PADDING_FRAME: | 
|  | break; | 
|  | case quic::STREAM_FRAME: | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_STREAM_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicStreamFrameCallback, frame.stream_frame)); | 
|  | break; | 
|  | case quic::ACK_FRAME: { | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_ACK_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicAckFrameCallback, frame.ack_frame)); | 
|  | break; | 
|  | } | 
|  | case quic::RST_STREAM_FRAME: | 
|  | base::UmaHistogramSparse("Net.QuicSession.RstStreamErrorCodeClient", | 
|  | frame.rst_stream_frame->error_code); | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_RST_STREAM_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicRstStreamFrameCallback, | 
|  | frame.rst_stream_frame)); | 
|  | break; | 
|  | case quic::CONNECTION_CLOSE_FRAME: | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_CONNECTION_CLOSE_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicConnectionCloseFrameCallback, | 
|  | frame.connection_close_frame)); | 
|  | break; | 
|  | case quic::GOAWAY_FRAME: | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_GOAWAY_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicGoAwayFrameCallback, frame.goaway_frame)); | 
|  | break; | 
|  | case quic::WINDOW_UPDATE_FRAME: | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_WINDOW_UPDATE_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicWindowUpdateFrameCallback, | 
|  | frame.window_update_frame)); | 
|  | break; | 
|  | case quic::BLOCKED_FRAME: | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_BLOCKED_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicBlockedFrameCallback, frame.blocked_frame)); | 
|  | break; | 
|  | case quic::STOP_WAITING_FRAME: | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_STOP_WAITING_FRAME_SENT, | 
|  | base::Bind(&NetLogQuicStopWaitingFrameCallback, | 
|  | &frame.stop_waiting_frame)); | 
|  | break; | 
|  | case quic::PING_FRAME: | 
|  | UMA_HISTOGRAM_BOOLEAN("Net.QuicSession.ConnectionFlowControlBlocked", | 
|  | session_->IsConnectionFlowControlBlocked()); | 
|  | UMA_HISTOGRAM_BOOLEAN("Net.QuicSession.StreamFlowControlBlocked", | 
|  | session_->IsStreamFlowControlBlocked()); | 
|  | // PingFrame has no contents to log, so just record that it was sent. | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_PING_FRAME_SENT); | 
|  | break; | 
|  | case quic::MTU_DISCOVERY_FRAME: | 
|  | // MtuDiscoveryFrame is PingFrame on wire, it does not have any payload. | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_MTU_DISCOVERY_FRAME_SENT); | 
|  | break; | 
|  | case quic::APPLICATION_CLOSE_FRAME: | 
|  | break; | 
|  | case quic::NEW_CONNECTION_ID_FRAME: | 
|  | break; | 
|  | case quic::MAX_STREAM_ID_FRAME: | 
|  | break; | 
|  | case quic::STREAM_ID_BLOCKED_FRAME: | 
|  | break; | 
|  | case quic::PATH_RESPONSE_FRAME: | 
|  | break; | 
|  | case quic::PATH_CHALLENGE_FRAME: | 
|  | break; | 
|  | case quic::STOP_SENDING_FRAME: | 
|  | break; | 
|  | case quic::MESSAGE_FRAME: | 
|  | break; | 
|  | case quic::CRYPTO_FRAME: | 
|  | break; | 
|  | case quic::NEW_TOKEN_FRAME: | 
|  | break; | 
|  | case quic::RETIRE_CONNECTION_ID_FRAME: | 
|  | break; | 
|  | default: | 
|  | DCHECK(false) << "Illegal frame type: " << frame.type; | 
|  | } | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPacketSent( | 
|  | const quic::SerializedPacket& serialized_packet, | 
|  | quic::QuicPacketNumber original_packet_number, | 
|  | quic::TransmissionType transmission_type, | 
|  | quic::QuicTime sent_time) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | if (!original_packet_number.IsInitialized()) { | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_PACKET_SENT, | 
|  | base::Bind(&NetLogQuicPacketSentCallback, serialized_packet, | 
|  | transmission_type, sent_time)); | 
|  | } else { | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_PACKET_RETRANSMITTED, | 
|  | base::Bind(&NetLogQuicPacketRetransmittedCallback, | 
|  | original_packet_number, serialized_packet.packet_number)); | 
|  | } | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPacketLoss( | 
|  | quic::QuicPacketNumber lost_packet_number, | 
|  | quic::TransmissionType transmission_type, | 
|  | quic::QuicTime detection_time) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_PACKET_LOST, | 
|  | base::Bind(&NetLogQuicPacketLostCallback, lost_packet_number, | 
|  | transmission_type, detection_time)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPingSent() { | 
|  | no_packet_received_after_ping_ = true; | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPacketReceived( | 
|  | const quic::QuicSocketAddress& self_address, | 
|  | const quic::QuicSocketAddress& peer_address, | 
|  | const quic::QuicEncryptedPacket& packet) { | 
|  | if (local_address_from_self_.GetFamily() == ADDRESS_FAMILY_UNSPECIFIED) { | 
|  | local_address_from_self_ = self_address.impl().socket_address(); | 
|  | UMA_HISTOGRAM_ENUMERATION( | 
|  | "Net.QuicSession.ConnectionTypeFromSelf", | 
|  | GetRealAddressFamily(self_address.impl().socket_address().address()), | 
|  | ADDRESS_FAMILY_LAST); | 
|  | } | 
|  |  | 
|  | previous_received_packet_size_ = last_received_packet_size_; | 
|  | last_received_packet_size_ = packet.length(); | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_PACKET_RECEIVED, | 
|  | base::Bind(&NetLogQuicPacketCallback, | 
|  | &self_address.impl().socket_address(), | 
|  | &peer_address.impl().socket_address(), packet.length())); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnUnauthenticatedHeader( | 
|  | const quic::QuicPacketHeader& header) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_UNAUTHENTICATED_PACKET_HEADER_RECEIVED, | 
|  | base::Bind(&NetLogQuicPacketHeaderCallback, &header)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnIncorrectConnectionId( | 
|  | quic::QuicConnectionId connection_id) { | 
|  | ++num_incorrect_connection_ids_; | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnUndecryptablePacket() { | 
|  | ++num_undecryptable_packets_; | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnDuplicatePacket( | 
|  | quic::QuicPacketNumber packet_number) { | 
|  | ++num_duplicate_packets_; | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_DUPLICATE_PACKET_RECEIVED, | 
|  | base::Bind(&NetLogQuicDuplicatePacketCallback, packet_number)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnProtocolVersionMismatch( | 
|  | quic::ParsedQuicVersion received_version) { | 
|  | // TODO(rtenneti): Add logging. | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPacketHeader( | 
|  | const quic::QuicPacketHeader& header) { | 
|  | if (!first_received_packet_number_.IsInitialized()) { | 
|  | first_received_packet_number_ = header.packet_number; | 
|  | } else if (header.packet_number < first_received_packet_number_) { | 
|  | // Ignore packets with packet numbers less than | 
|  | // first_received_packet_number_. | 
|  | return; | 
|  | } | 
|  | ++num_packets_received_; | 
|  | if (!largest_received_packet_number_.IsInitialized()) { | 
|  | largest_received_packet_number_ = header.packet_number; | 
|  | } else if (largest_received_packet_number_ < header.packet_number) { | 
|  | uint64_t delta = header.packet_number - largest_received_packet_number_; | 
|  | if (delta > 1) { | 
|  | // There is a gap between the largest packet previously received and | 
|  | // the current packet.  This indicates either loss, or out-of-order | 
|  | // delivery. | 
|  | UMA_HISTOGRAM_COUNTS_1M( | 
|  | "Net.QuicSession.PacketGapReceived", | 
|  | static_cast<base::HistogramBase::Sample>(delta - 1)); | 
|  | } | 
|  | largest_received_packet_number_ = header.packet_number; | 
|  | } | 
|  | if (header.packet_number - first_received_packet_number_ < | 
|  | received_packets_.size()) { | 
|  | received_packets_[header.packet_number - first_received_packet_number_] = | 
|  | true; | 
|  | } | 
|  | if (last_received_packet_number_.IsInitialized() && | 
|  | header.packet_number < last_received_packet_number_) { | 
|  | ++num_out_of_order_received_packets_; | 
|  | if (previous_received_packet_size_ < last_received_packet_size_) | 
|  | ++num_out_of_order_large_received_packets_; | 
|  | UMA_HISTOGRAM_COUNTS_1M( | 
|  | "Net.QuicSession.OutOfOrderGapReceived", | 
|  | static_cast<base::HistogramBase::Sample>(last_received_packet_number_ - | 
|  | header.packet_number)); | 
|  | } else if (no_packet_received_after_ping_) { | 
|  | if (last_received_packet_number_.IsInitialized()) { | 
|  | UMA_HISTOGRAM_COUNTS_1M( | 
|  | "Net.QuicSession.PacketGapReceivedNearPing", | 
|  | static_cast<base::HistogramBase::Sample>( | 
|  | header.packet_number - last_received_packet_number_)); | 
|  | } | 
|  | no_packet_received_after_ping_ = false; | 
|  | } | 
|  | last_received_packet_number_ = header.packet_number; | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_PACKET_AUTHENTICATED); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnStreamFrame(const quic::QuicStreamFrame& frame) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_STREAM_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicStreamFrameCallback, frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnIncomingAck( | 
|  | const quic::QuicAckFrame& frame, | 
|  | quic::QuicTime ack_receive_time, | 
|  | quic::QuicPacketNumber largest_observed, | 
|  | bool rtt_updated, | 
|  | quic::QuicPacketNumber least_unacked_sent_packet) { | 
|  | const size_t kApproximateLargestSoloAckBytes = 100; | 
|  | if (last_received_packet_number_ - first_received_packet_number_ < | 
|  | received_acks_.size() && | 
|  | last_received_packet_size_ < kApproximateLargestSoloAckBytes) { | 
|  | received_acks_[last_received_packet_number_ - | 
|  | first_received_packet_number_] = true; | 
|  | } | 
|  |  | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_ACK_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicAckFrameCallback, &frame)); | 
|  |  | 
|  | // TODO(rch, rtenneti) sort out histograms for QUIC_VERSION_34 and above. | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnStopWaitingFrame( | 
|  | const quic::QuicStopWaitingFrame& frame) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_STOP_WAITING_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicStopWaitingFrameCallback, &frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnRstStreamFrame( | 
|  | const quic::QuicRstStreamFrame& frame) { | 
|  | base::UmaHistogramSparse("Net.QuicSession.RstStreamErrorCodeServer", | 
|  | frame.error_code); | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_RST_STREAM_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicRstStreamFrameCallback, &frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnConnectionCloseFrame( | 
|  | const quic::QuicConnectionCloseFrame& frame) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_CONNECTION_CLOSE_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicConnectionCloseFrameCallback, &frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnWindowUpdateFrame( | 
|  | const quic::QuicWindowUpdateFrame& frame, | 
|  | const quic::QuicTime& receive_time) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_WINDOW_UPDATE_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicWindowUpdateFrameCallback, &frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnBlockedFrame(const quic::QuicBlockedFrame& frame) { | 
|  | ++num_blocked_frames_received_; | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_BLOCKED_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicBlockedFrameCallback, &frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnGoAwayFrame(const quic::QuicGoAwayFrame& frame) { | 
|  | UMA_HISTOGRAM_BOOLEAN("Net.QuicSession.GoAwayReceivedForConnectionMigration", | 
|  | frame.error_code == quic::QUIC_ERROR_MIGRATING_PORT); | 
|  |  | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_GOAWAY_FRAME_RECEIVED, | 
|  | base::Bind(&NetLogQuicGoAwayFrameCallback, &frame)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPingFrame(const quic::QuicPingFrame& frame) { | 
|  | // PingFrame has no contents to log, so just record that it was received. | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_PING_FRAME_RECEIVED); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnPublicResetPacket( | 
|  | const quic::QuicPublicResetPacket& packet) { | 
|  | UpdatePublicResetAddressMismatchHistogram( | 
|  | local_address_from_shlo_, packet.client_address.impl().socket_address()); | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_PUBLIC_RESET_PACKET_RECEIVED, | 
|  | base::Bind(&NetLogQuicPublicResetPacketCallback, | 
|  | &local_address_from_shlo_, | 
|  | &packet.client_address.impl().socket_address())); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnVersionNegotiationPacket( | 
|  | const quic::QuicVersionNegotiationPacket& packet) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_VERSION_NEGOTIATION_PACKET_RECEIVED, | 
|  | base::Bind(&NetLogQuicVersionNegotiationPacketCallback, &packet)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnCryptoHandshakeMessageReceived( | 
|  | const quic::CryptoHandshakeMessage& message) { | 
|  | if (message.tag() == quic::kSHLO) { | 
|  | quic::QuicStringPiece address; | 
|  | quic::QuicSocketAddressCoder decoder; | 
|  | if (message.GetStringPiece(quic::kCADR, &address) && | 
|  | decoder.Decode(address.data(), address.size())) { | 
|  | local_address_from_shlo_ = | 
|  | IPEndPoint(decoder.ip().impl().ip_address(), decoder.port()); | 
|  | UMA_HISTOGRAM_ENUMERATION( | 
|  | "Net.QuicSession.ConnectionTypeFromPeer", | 
|  | GetRealAddressFamily(local_address_from_shlo_.address()), | 
|  | ADDRESS_FAMILY_LAST); | 
|  | } | 
|  | } | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_CRYPTO_HANDSHAKE_MESSAGE_RECEIVED, | 
|  | base::Bind(&NetLogQuicCryptoHandshakeMessageCallback, &message)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnCryptoHandshakeMessageSent( | 
|  | const quic::CryptoHandshakeMessage& message) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_CRYPTO_HANDSHAKE_MESSAGE_SENT, | 
|  | base::Bind(&NetLogQuicCryptoHandshakeMessageCallback, &message)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnConnectionClosed( | 
|  | quic::QuicErrorCode error, | 
|  | const string& error_details, | 
|  | quic::ConnectionCloseSource source) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_CLOSED, | 
|  | base::Bind(&NetLogQuicOnConnectionClosedCallback, error, | 
|  | error_details, source)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnSuccessfulVersionNegotiation( | 
|  | const quic::ParsedQuicVersion& version) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | string quic_version = QuicVersionToString(version.transport_version); | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_VERSION_NEGOTIATED, | 
|  | NetLog::StringCallback("version", &quic_version)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::UpdateReceivedFrameCounts( | 
|  | quic::QuicStreamId stream_id, | 
|  | int num_frames_received, | 
|  | int num_duplicate_frames_received) { | 
|  | if (stream_id != quic::QuicUtils::GetCryptoStreamId( | 
|  | session_->connection()->transport_version())) { | 
|  | num_frames_received_ += num_frames_received; | 
|  | num_duplicate_frames_received_ += num_duplicate_frames_received; | 
|  | } | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnCertificateVerified( | 
|  | const CertVerifyResult& result) { | 
|  | if (!net_log_is_capturing_) | 
|  | return; | 
|  | if (result.cert_status == CERT_STATUS_INVALID) { | 
|  | net_log_.AddEvent(NetLogEventType::QUIC_SESSION_CERTIFICATE_VERIFY_FAILED); | 
|  | return; | 
|  | } | 
|  | net_log_.AddEvent( | 
|  | NetLogEventType::QUIC_SESSION_CERTIFICATE_VERIFIED, | 
|  | base::Bind(&NetLogQuicCertificateVerifiedCallback, result.verified_cert)); | 
|  | } | 
|  |  | 
|  | base::HistogramBase* QuicConnectionLogger::Get6PacketHistogram( | 
|  | const char* which_6) const { | 
|  | // This histogram takes a binary encoding of the 6 consecutive packets | 
|  | // received.  As a result, there are 64 possible sample-patterns. | 
|  | string prefix("Net.QuicSession.6PacketsPatternsReceived_"); | 
|  | return base::LinearHistogram::FactoryGet( | 
|  | prefix + which_6 + connection_description_, 1, 64, 65, | 
|  | base::HistogramBase::kUmaTargetedHistogramFlag); | 
|  | } | 
|  |  | 
|  | float QuicConnectionLogger::ReceivedPacketLossRate() const { | 
|  | if (!largest_received_packet_number_.IsInitialized()) | 
|  | return 0.0f; | 
|  | float num_packets = | 
|  | largest_received_packet_number_ - first_received_packet_number_ + 1; | 
|  | float num_missing = num_packets - num_packets_received_; | 
|  | return num_missing / num_packets; | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::OnRttChanged(quic::QuicTime::Delta rtt) const { | 
|  | // Notify socket performance watcher of the updated RTT value. | 
|  | if (!socket_performance_watcher_) | 
|  | return; | 
|  |  | 
|  | int64_t microseconds = rtt.ToMicroseconds(); | 
|  | if (microseconds != 0 && | 
|  | socket_performance_watcher_->ShouldNotifyUpdatedRTT()) { | 
|  | socket_performance_watcher_->OnUpdatedRTTAvailable( | 
|  | base::TimeDelta::FromMicroseconds(rtt.ToMicroseconds())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::RecordAggregatePacketLossRate() const { | 
|  | // For short connections under 22 packets in length, we'll rely on the | 
|  | // Net.QuicSession.21CumulativePacketsReceived_* histogram to indicate packet | 
|  | // loss rates.  This way we avoid tremendously anomalous contributions to our | 
|  | // histogram.  (e.g., if we only got 5 packets, but lost 1, we'd otherwise | 
|  | // record a 20% loss in this histogram!). We may still get some strange data | 
|  | // (1 loss in 22 is still high :-/). | 
|  | if (!largest_received_packet_number_.IsInitialized() || | 
|  | largest_received_packet_number_ - first_received_packet_number_ < 22) | 
|  | return; | 
|  |  | 
|  | string prefix("Net.QuicSession.PacketLossRate_"); | 
|  | base::HistogramBase* histogram = base::Histogram::FactoryGet( | 
|  | prefix + connection_description_, 1, 1000, 75, | 
|  | base::HistogramBase::kUmaTargetedHistogramFlag); | 
|  | histogram->Add(static_cast<base::HistogramBase::Sample>( | 
|  | ReceivedPacketLossRate() * 1000)); | 
|  | } | 
|  |  | 
|  | void QuicConnectionLogger::UpdateIsCapturing() { | 
|  | net_log_is_capturing_ = net_log_.IsCapturing(); | 
|  | } | 
|  |  | 
|  | }  // namespace net |