mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 22:36:09 +03:00
135 lines
4.3 KiB
C++
135 lines
4.3 KiB
C++
|
// 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<IPHash> 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<base::SingleThreadTaskRunner> task_runner,
|
||
|
OnUpdatedRTTAvailableCallback updated_rtt_observation_callback,
|
||
|
ShouldNotifyRTTCallback should_notify_rtt_callback,
|
||
|
const 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().IsPubliclyRoutable())),
|
||
|
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
|