OLD | NEW |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "net/base/network_quality_estimator.h" | 5 #include "net/base/network_quality_estimator.h" |
6 | 6 |
7 #include <float.h> | 7 #include <float.h> |
8 #include <algorithm> | 8 #include <algorithm> |
9 #include <cmath> | 9 #include <cmath> |
10 #include <limits> | 10 #include <limits> |
(...skipping 235 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
246 variations_value >= kMinimumThroughputVariationParameterKbps) { | 246 variations_value >= kMinimumThroughputVariationParameterKbps) { |
247 default_observations_[i] = | 247 default_observations_[i] = |
248 NetworkQuality(default_observations_[i].rtt(), variations_value); | 248 NetworkQuality(default_observations_[i].rtt(), variations_value); |
249 } | 249 } |
250 } | 250 } |
251 } | 251 } |
252 | 252 |
253 void NetworkQualityEstimator::AddDefaultEstimates() { | 253 void NetworkQualityEstimator::AddDefaultEstimates() { |
254 DCHECK(thread_checker_.CalledOnValidThread()); | 254 DCHECK(thread_checker_.CalledOnValidThread()); |
255 if (default_observations_[current_network_id_.type].rtt() != InvalidRTT()) { | 255 if (default_observations_[current_network_id_.type].rtt() != InvalidRTT()) { |
256 rtt_msec_observations_.AddObservation(Observation( | 256 Observation rtt_observation( |
257 default_observations_[current_network_id_.type].rtt().InMilliseconds(), | 257 default_observations_[current_network_id_.type].rtt().InMilliseconds(), |
258 base::TimeTicks::Now())); | 258 base::TimeTicks::Now(), DEFAULT_FROM_PLATFORM); |
| 259 rtt_msec_observations_.AddObservation(rtt_observation); |
| 260 NotifyObserversOfRTT(rtt_observation); |
259 } | 261 } |
260 if (default_observations_[current_network_id_.type] | 262 if (default_observations_[current_network_id_.type] |
261 .downstream_throughput_kbps() != kInvalidThroughput) { | 263 .downstream_throughput_kbps() != kInvalidThroughput) { |
| 264 Observation throughput_observation( |
| 265 default_observations_[current_network_id_.type] |
| 266 .downstream_throughput_kbps(), |
| 267 base::TimeTicks::Now(), DEFAULT_FROM_PLATFORM); |
262 downstream_throughput_kbps_observations_.AddObservation( | 268 downstream_throughput_kbps_observations_.AddObservation( |
263 Observation(default_observations_[current_network_id_.type] | 269 throughput_observation); |
264 .downstream_throughput_kbps(), | 270 NotifyObserversOfThroughput(throughput_observation); |
265 base::TimeTicks::Now())); | |
266 } | 271 } |
267 } | 272 } |
268 | 273 |
269 NetworkQualityEstimator::~NetworkQualityEstimator() { | 274 NetworkQualityEstimator::~NetworkQualityEstimator() { |
270 DCHECK(thread_checker_.CalledOnValidThread()); | 275 DCHECK(thread_checker_.CalledOnValidThread()); |
271 NetworkChangeNotifier::RemoveConnectionTypeObserver(this); | 276 NetworkChangeNotifier::RemoveConnectionTypeObserver(this); |
272 } | 277 } |
273 | 278 |
274 void NetworkQualityEstimator::NotifyHeadersReceived(const URLRequest& request) { | 279 void NetworkQualityEstimator::NotifyHeadersReceived(const URLRequest& request) { |
275 DCHECK(thread_checker_.CalledOnValidThread()); | 280 DCHECK(thread_checker_.CalledOnValidThread()); |
(...skipping 27 matching lines...) Expand all Loading... |
303 | 308 |
304 // Duration between when the resource was requested and when response | 309 // Duration between when the resource was requested and when response |
305 // headers were received. | 310 // headers were received. |
306 base::TimeDelta observed_rtt = headers_received_time - request_start_time; | 311 base::TimeDelta observed_rtt = headers_received_time - request_start_time; |
307 DCHECK_GE(observed_rtt, base::TimeDelta()); | 312 DCHECK_GE(observed_rtt, base::TimeDelta()); |
308 if (observed_rtt < peak_network_quality_.rtt()) { | 313 if (observed_rtt < peak_network_quality_.rtt()) { |
309 peak_network_quality_ = NetworkQuality( | 314 peak_network_quality_ = NetworkQuality( |
310 observed_rtt, peak_network_quality_.downstream_throughput_kbps()); | 315 observed_rtt, peak_network_quality_.downstream_throughput_kbps()); |
311 } | 316 } |
312 | 317 |
313 rtt_msec_observations_.AddObservation( | 318 Observation rtt_observation(observed_rtt.InMilliseconds(), now, |
314 Observation(observed_rtt.InMilliseconds(), now)); | 319 URL_REQUEST); |
| 320 rtt_msec_observations_.AddObservation(rtt_observation); |
| 321 NotifyObserversOfRTT(rtt_observation); |
315 | 322 |
316 // Compare the RTT observation with the estimated value and record it. | 323 // Compare the RTT observation with the estimated value and record it. |
317 if (estimated_median_network_quality_.rtt() != InvalidRTT()) { | 324 if (estimated_median_network_quality_.rtt() != InvalidRTT()) { |
318 RecordRTTUMA(estimated_median_network_quality_.rtt().InMilliseconds(), | 325 RecordRTTUMA(estimated_median_network_quality_.rtt().InMilliseconds(), |
319 observed_rtt.InMilliseconds()); | 326 observed_rtt.InMilliseconds()); |
320 } | 327 } |
321 } | 328 } |
322 | 329 |
323 void NetworkQualityEstimator::NotifyRequestCompleted( | 330 void NetworkQualityEstimator::NotifyRequestCompleted( |
324 const URLRequest& request) { | 331 const URLRequest& request) { |
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
371 // connection. | 378 // connection. |
372 if (downstream_kbps - downstream_kbps_as_integer > 0) | 379 if (downstream_kbps - downstream_kbps_as_integer > 0) |
373 downstream_kbps_as_integer++; | 380 downstream_kbps_as_integer++; |
374 | 381 |
375 DCHECK_GT(downstream_kbps_as_integer, 0.0); | 382 DCHECK_GT(downstream_kbps_as_integer, 0.0); |
376 if (downstream_kbps_as_integer > | 383 if (downstream_kbps_as_integer > |
377 peak_network_quality_.downstream_throughput_kbps()) | 384 peak_network_quality_.downstream_throughput_kbps()) |
378 peak_network_quality_ = | 385 peak_network_quality_ = |
379 NetworkQuality(peak_network_quality_.rtt(), downstream_kbps_as_integer); | 386 NetworkQuality(peak_network_quality_.rtt(), downstream_kbps_as_integer); |
380 | 387 |
| 388 Observation throughput_observation(downstream_kbps_as_integer, now, |
| 389 URL_REQUEST); |
381 downstream_throughput_kbps_observations_.AddObservation( | 390 downstream_throughput_kbps_observations_.AddObservation( |
382 Observation(downstream_kbps_as_integer, now)); | 391 throughput_observation); |
| 392 NotifyObserversOfThroughput(throughput_observation); |
| 393 } |
| 394 |
| 395 void NetworkQualityEstimator::AddRTTObserver(RTTObserver* rtt_observer) { |
| 396 DCHECK(thread_checker_.CalledOnValidThread()); |
| 397 rtt_observer_list_.AddObserver(rtt_observer); |
| 398 } |
| 399 |
| 400 void NetworkQualityEstimator::RemoveRTTObserver(RTTObserver* rtt_observer) { |
| 401 DCHECK(thread_checker_.CalledOnValidThread()); |
| 402 rtt_observer_list_.RemoveObserver(rtt_observer); |
| 403 } |
| 404 |
| 405 void NetworkQualityEstimator::AddThroughputObserver( |
| 406 ThroughputObserver* throughput_observer) { |
| 407 DCHECK(thread_checker_.CalledOnValidThread()); |
| 408 throughput_observer_list_.AddObserver(throughput_observer); |
| 409 } |
| 410 |
| 411 void NetworkQualityEstimator::RemoveThroughputObserver( |
| 412 ThroughputObserver* throughput_observer) { |
| 413 DCHECK(thread_checker_.CalledOnValidThread()); |
| 414 throughput_observer_list_.RemoveObserver(throughput_observer); |
| 415 } |
| 416 |
| 417 void NetworkQualityEstimator::ConfigureForTests(bool allow_local_host_requests, |
| 418 bool allow_smaller_responses) { |
| 419 DCHECK(thread_checker_.CalledOnValidThread()); |
| 420 allow_localhost_requests_ = allow_local_host_requests, |
| 421 allow_small_responses_ = allow_smaller_responses; |
383 } | 422 } |
384 | 423 |
385 void NetworkQualityEstimator::RecordRTTUMA(int32_t estimated_value_msec, | 424 void NetworkQualityEstimator::RecordRTTUMA(int32_t estimated_value_msec, |
386 int32_t actual_value_msec) const { | 425 int32_t actual_value_msec) const { |
387 DCHECK(thread_checker_.CalledOnValidThread()); | 426 DCHECK(thread_checker_.CalledOnValidThread()); |
388 | 427 |
389 // Record the difference between the actual and the estimated value. | 428 // Record the difference between the actual and the estimated value. |
390 if (estimated_value_msec >= actual_value_msec) { | 429 if (estimated_value_msec >= actual_value_msec) { |
391 base::HistogramBase* difference_rtt = | 430 base::HistogramBase* difference_rtt = |
392 GetHistogram("DifferenceRTTEstimatedAndActual.", | 431 GetHistogram("DifferenceRTTEstimatedAndActual.", |
(...skipping 205 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
598 bool NetworkQualityEstimator::GetRecentMedianDownlinkThroughputKbps( | 637 bool NetworkQualityEstimator::GetRecentMedianDownlinkThroughputKbps( |
599 const base::TimeTicks& begin_timestamp, | 638 const base::TimeTicks& begin_timestamp, |
600 int32_t* kbps) const { | 639 int32_t* kbps) const { |
601 DCHECK(thread_checker_.CalledOnValidThread()); | 640 DCHECK(thread_checker_.CalledOnValidThread()); |
602 DCHECK(kbps); | 641 DCHECK(kbps); |
603 *kbps = GetDownlinkThroughputKbpsEstimateInternal(begin_timestamp, 50); | 642 *kbps = GetDownlinkThroughputKbpsEstimateInternal(begin_timestamp, 50); |
604 return (*kbps != kInvalidThroughput); | 643 return (*kbps != kInvalidThroughput); |
605 } | 644 } |
606 | 645 |
607 NetworkQualityEstimator::Observation::Observation(int32_t value, | 646 NetworkQualityEstimator::Observation::Observation(int32_t value, |
608 base::TimeTicks timestamp) | 647 base::TimeTicks timestamp, |
609 : value(value), timestamp(timestamp) { | 648 ObservationSource source) |
| 649 : value(value), timestamp(timestamp), source(source) { |
610 DCHECK_GE(value, 0); | 650 DCHECK_GE(value, 0); |
611 DCHECK(!timestamp.is_null()); | 651 DCHECK(!timestamp.is_null()); |
612 } | 652 } |
613 | 653 |
614 NetworkQualityEstimator::Observation::~Observation() { | 654 NetworkQualityEstimator::Observation::~Observation() { |
615 } | 655 } |
616 | 656 |
617 NetworkQualityEstimator::ObservationBuffer::ObservationBuffer( | 657 NetworkQualityEstimator::ObservationBuffer::ObservationBuffer( |
618 double weight_multiplier_per_second) | 658 double weight_multiplier_per_second) |
619 : weight_multiplier_per_second_(weight_multiplier_per_second) { | 659 : weight_multiplier_per_second_(weight_multiplier_per_second) { |
(...skipping 194 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
814 cached_network_qualities_.find(current_network_id_); | 854 cached_network_qualities_.find(current_network_id_); |
815 | 855 |
816 if (it == cached_network_qualities_.end()) | 856 if (it == cached_network_qualities_.end()) |
817 return false; | 857 return false; |
818 | 858 |
819 NetworkQuality network_quality(it->second.network_quality()); | 859 NetworkQuality network_quality(it->second.network_quality()); |
820 | 860 |
821 DCHECK_NE(InvalidRTT(), network_quality.rtt()); | 861 DCHECK_NE(InvalidRTT(), network_quality.rtt()); |
822 DCHECK_NE(kInvalidThroughput, network_quality.downstream_throughput_kbps()); | 862 DCHECK_NE(kInvalidThroughput, network_quality.downstream_throughput_kbps()); |
823 | 863 |
824 downstream_throughput_kbps_observations_.AddObservation(Observation( | 864 Observation througphput_observation( |
825 network_quality.downstream_throughput_kbps(), base::TimeTicks::Now())); | 865 network_quality.downstream_throughput_kbps(), base::TimeTicks::Now(), |
826 rtt_msec_observations_.AddObservation(Observation( | 866 CACHED_ESTIMATE); |
827 network_quality.rtt().InMilliseconds(), base::TimeTicks::Now())); | 867 downstream_throughput_kbps_observations_.AddObservation( |
| 868 througphput_observation); |
| 869 NotifyObserversOfThroughput(througphput_observation); |
| 870 |
| 871 Observation rtt_observation(network_quality.rtt().InMilliseconds(), |
| 872 base::TimeTicks::Now(), CACHED_ESTIMATE); |
| 873 rtt_msec_observations_.AddObservation(rtt_observation); |
| 874 NotifyObserversOfRTT(rtt_observation); |
| 875 |
828 return true; | 876 return true; |
829 } | 877 } |
830 | 878 |
831 void NetworkQualityEstimator::OnUpdatedEstimateAvailable() { | 879 void NetworkQualityEstimator::OnUpdatedEstimateAvailable() { |
832 DCHECK(thread_checker_.CalledOnValidThread()); | 880 DCHECK(thread_checker_.CalledOnValidThread()); |
833 DCHECK(external_estimate_provider_); | 881 DCHECK(external_estimate_provider_); |
834 | 882 |
835 RecordExternalEstimateProviderMetrics( | 883 RecordExternalEstimateProviderMetrics( |
836 EXTERNAL_ESTIMATE_PROVIDER_STATUS_CALLBACK); | 884 EXTERNAL_ESTIMATE_PROVIDER_STATUS_CALLBACK); |
837 QueryExternalEstimateProvider(); | 885 QueryExternalEstimateProvider(); |
(...skipping 20 matching lines...) Expand all Loading... |
858 // updates estimates are available, OnUpdatedEstimateAvailable() will be | 906 // updates estimates are available, OnUpdatedEstimateAvailable() will be |
859 // called. | 907 // called. |
860 external_estimate_provider_->Update(); | 908 external_estimate_provider_->Update(); |
861 return; | 909 return; |
862 } | 910 } |
863 | 911 |
864 RecordExternalEstimateProviderMetrics( | 912 RecordExternalEstimateProviderMetrics( |
865 EXTERNAL_ESTIMATE_PROVIDER_STATUS_QUERY_SUCCESSFUL); | 913 EXTERNAL_ESTIMATE_PROVIDER_STATUS_QUERY_SUCCESSFUL); |
866 base::TimeDelta rtt; | 914 base::TimeDelta rtt; |
867 if (external_estimate_provider_->GetRTT(&rtt)) { | 915 if (external_estimate_provider_->GetRTT(&rtt)) { |
868 rtt_msec_observations_.AddObservation( | 916 rtt_msec_observations_.AddObservation(Observation( |
869 Observation(rtt.InMilliseconds(), base::TimeTicks::Now())); | 917 rtt.InMilliseconds(), base::TimeTicks::Now(), EXTERNAL_ESTIMATE)); |
870 } | 918 } |
871 | 919 |
872 int32_t downstream_throughput_kbps; | 920 int32_t downstream_throughput_kbps; |
873 if (external_estimate_provider_->GetDownstreamThroughputKbps( | 921 if (external_estimate_provider_->GetDownstreamThroughputKbps( |
874 &downstream_throughput_kbps)) { | 922 &downstream_throughput_kbps)) { |
875 downstream_throughput_kbps_observations_.AddObservation( | 923 downstream_throughput_kbps_observations_.AddObservation(Observation( |
876 Observation(downstream_throughput_kbps, base::TimeTicks::Now())); | 924 downstream_throughput_kbps, base::TimeTicks::Now(), EXTERNAL_ESTIMATE)); |
877 } | 925 } |
878 } | 926 } |
879 | 927 |
880 void NetworkQualityEstimator::CacheNetworkQualityEstimate() { | 928 void NetworkQualityEstimator::CacheNetworkQualityEstimate() { |
881 DCHECK(thread_checker_.CalledOnValidThread()); | 929 DCHECK(thread_checker_.CalledOnValidThread()); |
882 DCHECK_LE(cached_network_qualities_.size(), | 930 DCHECK_LE(cached_network_qualities_.size(), |
883 static_cast<size_t>(kMaximumNetworkQualityCacheSize)); | 931 static_cast<size_t>(kMaximumNetworkQualityCacheSize)); |
884 | 932 |
885 // If the network name is unavailable, caching should not be performed. | 933 // If the network name is unavailable, caching should not be performed. |
886 if (current_network_id_.id.empty()) | 934 if (current_network_id_.id.empty()) |
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
923 new SocketPerformanceWatcherTCP()); | 971 new SocketPerformanceWatcherTCP()); |
924 } | 972 } |
925 | 973 |
926 scoped_ptr<SocketPerformanceWatcher> | 974 scoped_ptr<SocketPerformanceWatcher> |
927 NetworkQualityEstimator::CreateUDPSocketPerformanceWatcher() const { | 975 NetworkQualityEstimator::CreateUDPSocketPerformanceWatcher() const { |
928 DCHECK(thread_checker_.CalledOnValidThread()); | 976 DCHECK(thread_checker_.CalledOnValidThread()); |
929 return scoped_ptr<SocketPerformanceWatcher>( | 977 return scoped_ptr<SocketPerformanceWatcher>( |
930 new SocketPerformanceWatcherUDP()); | 978 new SocketPerformanceWatcherUDP()); |
931 } | 979 } |
932 | 980 |
| 981 void NetworkQualityEstimator::NotifyObserversOfRTT( |
| 982 const Observation& observation) { |
| 983 FOR_EACH_OBSERVER(RTTObserver, rtt_observer_list_, |
| 984 OnRTTObservation(observation.value, observation.timestamp, |
| 985 observation.source)); |
| 986 } |
| 987 |
| 988 void NetworkQualityEstimator::NotifyObserversOfThroughput( |
| 989 const Observation& observation) { |
| 990 FOR_EACH_OBSERVER( |
| 991 ThroughputObserver, throughput_observer_list_, |
| 992 OnThroughputObservation(observation.value, observation.timestamp, |
| 993 observation.source)); |
| 994 } |
| 995 |
933 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( | 996 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( |
934 const NetworkQuality& network_quality) | 997 const NetworkQuality& network_quality) |
935 : last_update_time_(base::TimeTicks::Now()), | 998 : last_update_time_(base::TimeTicks::Now()), |
936 network_quality_(network_quality) { | 999 network_quality_(network_quality) { |
937 } | 1000 } |
938 | 1001 |
939 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( | 1002 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( |
940 const CachedNetworkQuality& other) | 1003 const CachedNetworkQuality& other) |
941 : last_update_time_(other.last_update_time_), | 1004 : last_update_time_(other.last_update_time_), |
942 network_quality_(other.network_quality_) { | 1005 network_quality_(other.network_quality_) { |
(...skipping 27 matching lines...) Expand all Loading... |
970 | 1033 |
971 NetworkQualityEstimator::NetworkQuality& | 1034 NetworkQualityEstimator::NetworkQuality& |
972 NetworkQualityEstimator::NetworkQuality:: | 1035 NetworkQualityEstimator::NetworkQuality:: |
973 operator=(const NetworkQuality& other) { | 1036 operator=(const NetworkQuality& other) { |
974 rtt_ = other.rtt_; | 1037 rtt_ = other.rtt_; |
975 downstream_throughput_kbps_ = other.downstream_throughput_kbps_; | 1038 downstream_throughput_kbps_ = other.downstream_throughput_kbps_; |
976 return *this; | 1039 return *this; |
977 } | 1040 } |
978 | 1041 |
979 } // namespace net | 1042 } // namespace net |
OLD | NEW |