| // Copyright (c) 2010 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/socket/client_socket.h" |
| |
| #include "base/metrics/field_trial.h" |
| #include "base/metrics/histogram.h" |
| #include "base/string_number_conversions.h" |
| #include "base/values.h" |
| |
| namespace net { |
| |
| namespace { |
| |
| // Parameters for SOCKET_BYTES_RECEIVED and SOCKET_BYTES_SENT events. |
| // Includes bytes transferred and, if |bytes| is not NULL, the bytes themselves. |
| class NetLogBytesTransferredParameter : public NetLog::EventParameters { |
| public: |
| NetLogBytesTransferredParameter(int byte_count, const char* bytes); |
| |
| virtual Value* ToValue() const; |
| |
| private: |
| const int byte_count_; |
| std::string hex_encoded_bytes_; |
| bool has_bytes_; |
| }; |
| |
| NetLogBytesTransferredParameter::NetLogBytesTransferredParameter( |
| int byte_count, const char* transferred_bytes) |
| : byte_count_(byte_count), |
| has_bytes_(false) { |
| if (transferred_bytes) { |
| hex_encoded_bytes_ = base::HexEncode(transferred_bytes, byte_count); |
| has_bytes_ = true; |
| } |
| } |
| |
| Value* NetLogBytesTransferredParameter::ToValue() const { |
| DictionaryValue* dict = new DictionaryValue(); |
| dict->SetInteger("byte_count", byte_count_); |
| if (has_bytes_) |
| dict->SetString("hex_encoded_bytes", hex_encoded_bytes_); |
| return dict; |
| } |
| |
| } // namespace |
| |
| ClientSocket::UseHistory::UseHistory() |
| : was_ever_connected_(false), |
| was_used_to_convey_data_(false), |
| omnibox_speculation_(false), |
| subresource_speculation_(false) { |
| } |
| |
| ClientSocket::UseHistory::~UseHistory() { |
| EmitPreconnectionHistograms(); |
| } |
| |
| void ClientSocket::UseHistory::Reset() { |
| EmitPreconnectionHistograms(); |
| was_ever_connected_ = false; |
| was_used_to_convey_data_ = false; |
| // omnibox_speculation_ and subresource_speculation_ values |
| // are intentionally preserved. |
| } |
| |
| void ClientSocket::UseHistory::set_was_ever_connected() { |
| DCHECK(!was_used_to_convey_data_); |
| was_ever_connected_ = true; |
| } |
| |
| void ClientSocket::UseHistory::set_was_used_to_convey_data() { |
| DCHECK(was_ever_connected_); |
| was_used_to_convey_data_ = true; |
| } |
| |
| |
| void ClientSocket::UseHistory::set_subresource_speculation() { |
| DCHECK(was_ever_connected_); |
| // TODO(jar): We should transition to marking a socket (or stream) at |
| // construction time as being created for speculative reasons. This current |
| // approach of trying to track use of a socket to convey data can make |
| // mistakes when other sockets (such as ones sitting in the pool for a long |
| // time) are issued. Unused sockets can be left over when a when a set of |
| // connections to a host are made, and one is "unlucky" and takes so long to |
| // complete a connection, that another socket is used, and recycled before a |
| // second connection comes available. Similarly, re-try connections can leave |
| // an original (slow to connect socket) in the pool, and that can be issued |
| // to a speculative requester. In any cases such old sockets will fail when an |
| // attempt is made to used them!... and then it will look like a speculative |
| // socket was discarded without any user!?!?! |
| if (was_used_to_convey_data_) |
| return; |
| subresource_speculation_ = true; |
| } |
| |
| void ClientSocket::UseHistory::set_omnibox_speculation() { |
| DCHECK(was_ever_connected_); |
| if (was_used_to_convey_data_) |
| return; |
| omnibox_speculation_ = true; |
| } |
| |
| bool ClientSocket::UseHistory::was_used_to_convey_data() const { |
| DCHECK(!was_used_to_convey_data_ || was_ever_connected_); |
| return was_used_to_convey_data_; |
| } |
| |
| void ClientSocket::UseHistory::EmitPreconnectionHistograms() const { |
| DCHECK(!subresource_speculation_ || !omnibox_speculation_); |
| // 0 ==> non-speculative, never connected. |
| // 1 ==> non-speculative never used (but connected). |
| // 2 ==> non-speculative and used. |
| // 3 ==> omnibox_speculative never connected. |
| // 4 ==> omnibox_speculative never used (but connected). |
| // 5 ==> omnibox_speculative and used. |
| // 6 ==> subresource_speculative never connected. |
| // 7 ==> subresource_speculative never used (but connected). |
| // 8 ==> subresource_speculative and used. |
| int result; |
| if (was_used_to_convey_data_) |
| result = 2; |
| else if (was_ever_connected_) |
| result = 1; |
| else |
| result = 0; // Never used, and not really connected. |
| |
| if (omnibox_speculation_) |
| result += 3; |
| else if (subresource_speculation_) |
| result += 6; |
| UMA_HISTOGRAM_ENUMERATION("Net.PreconnectUtilization2", result, 9); |
| |
| static const bool connect_backup_jobs_fieldtrial = |
| base::FieldTrialList::Find("ConnnectBackupJobs") && |
| !base::FieldTrialList::Find("ConnnectBackupJobs")->group_name().empty(); |
| if (connect_backup_jobs_fieldtrial) { |
| UMA_HISTOGRAM_ENUMERATION( |
| base::FieldTrial::MakeName("Net.PreconnectUtilization2", |
| "ConnnectBackupJobs"), |
| result, 9); |
| } |
| } |
| |
| void ClientSocket::LogByteTransfer(const BoundNetLog& net_log, |
| NetLog::EventType event_type, |
| int byte_count, |
| char* bytes) const { |
| scoped_refptr<NetLog::EventParameters> params; |
| if (net_log.IsLoggingBytes()) { |
| params = new NetLogBytesTransferredParameter(byte_count, bytes); |
| } else { |
| params = new NetLogBytesTransferredParameter(byte_count, NULL); |
| } |
| net_log.AddEvent(event_type, params); |
| } |
| |
| } // namespace net |