// Copyright 2016 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/nqe/socket_watcher.h" #include "base/bind.h" #include "base/location.h" #include "base/single_thread_task_runner.h" #include "base/time/tick_clock.h" #include "base/time/time.h" #include "net/base/address_list.h" #include "net/base/ip_address.h" namespace net { namespace nqe { namespace internal { namespace { // Generate a compact representation for the first IP in |address_list|. For // IPv4, all 32 bits are used and for IPv6, the first 64 bits are used as the // remote host identifier. base::Optional CalculateIPHash(const AddressList& address_list) { if (address_list.empty()) return base::nullopt; const IPAddress& ip_addr = address_list.front().address(); IPAddressBytes bytes = ip_addr.bytes(); // For IPv4, the first four bytes are taken. For IPv6, the first 8 bytes are // taken. For IPv4MappedIPv6, the last 4 bytes are taken. int index_min = ip_addr.IsIPv4MappedIPv6() ? 12 : 0; int index_max; if (ip_addr.IsIPv4MappedIPv6()) index_max = 16; else index_max = ip_addr.IsIPv4() ? 4 : 8; DCHECK_LE(index_min, index_max); DCHECK_GE(8, index_max - index_min); uint64_t result = 0ULL; for (int i = index_min; i < index_max; ++i) { result = result << 8; result |= bytes[i]; } return result; } } // namespace SocketWatcher::SocketWatcher( SocketPerformanceWatcherFactory::Protocol protocol, const AddressList& address_list, base::TimeDelta min_notification_interval, bool allow_rtt_private_address, scoped_refptr task_runner, OnUpdatedRTTAvailableCallback updated_rtt_observation_callback, ShouldNotifyRTTCallback should_notify_rtt_callback, base::TickClock* tick_clock) : protocol_(protocol), task_runner_(std::move(task_runner)), updated_rtt_observation_callback_(updated_rtt_observation_callback), should_notify_rtt_callback_(should_notify_rtt_callback), rtt_notifications_minimum_interval_(min_notification_interval), run_rtt_callback_(allow_rtt_private_address || (!address_list.empty() && !address_list.front().address().IsReserved())), tick_clock_(tick_clock), first_quic_rtt_notification_received_(false), host_(CalculateIPHash(address_list)) { DCHECK(tick_clock_); DCHECK(last_rtt_notification_.is_null()); } SocketWatcher::~SocketWatcher() = default; bool SocketWatcher::ShouldNotifyUpdatedRTT() const { DCHECK(thread_checker_.CalledOnValidThread()); if (!run_rtt_callback_) return false; const base::TimeTicks now = tick_clock_->NowTicks(); if (task_runner_->RunsTasksInCurrentSequence()) { // Enables socket watcher to send more frequent RTT observations when very // few sockets are receiving data. if (should_notify_rtt_callback_.Run(now)) return true; } // Do not allow incoming notifications if the last notification was more // recent than |rtt_notifications_minimum_interval_| ago. This helps in // reducing the overhead of obtaining the RTT values. // Enables a socket watcher to send RTT observation, helps in reducing // starvation by allowing every socket watcher to notify at least one RTT // notification every |rtt_notifications_minimum_interval_| duration. return now - last_rtt_notification_ >= rtt_notifications_minimum_interval_; } void SocketWatcher::OnUpdatedRTTAvailable(const base::TimeDelta& rtt) { DCHECK(thread_checker_.CalledOnValidThread()); if (rtt <= base::TimeDelta()) return; if (!first_quic_rtt_notification_received_ && protocol_ == SocketPerformanceWatcherFactory::PROTOCOL_QUIC) { // First RTT sample from QUIC connections may be synthetically generated, // and may not reflect the actual network quality. first_quic_rtt_notification_received_ = true; return; } last_rtt_notification_ = tick_clock_->NowTicks(); task_runner_->PostTask( FROM_HERE, base::Bind(updated_rtt_observation_callback_, protocol_, rtt, host_)); } void SocketWatcher::OnConnectionChanged() { DCHECK(thread_checker_.CalledOnValidThread()); } } // namespace internal } // namespace nqe } // namespace net