| 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 358 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 369 DCHECK(thread_checker_.CalledOnValidThread()); | 369 DCHECK(thread_checker_.CalledOnValidThread()); |
| 370 throughput_observer_list_.AddObserver(throughput_observer); | 370 throughput_observer_list_.AddObserver(throughput_observer); |
| 371 } | 371 } |
| 372 | 372 |
| 373 void NetworkQualityEstimator::RemoveThroughputObserver( | 373 void NetworkQualityEstimator::RemoveThroughputObserver( |
| 374 ThroughputObserver* throughput_observer) { | 374 ThroughputObserver* throughput_observer) { |
| 375 DCHECK(thread_checker_.CalledOnValidThread()); | 375 DCHECK(thread_checker_.CalledOnValidThread()); |
| 376 throughput_observer_list_.RemoveObserver(throughput_observer); | 376 throughput_observer_list_.RemoveObserver(throughput_observer); |
| 377 } | 377 } |
| 378 | 378 |
| 379 void NetworkQualityEstimator::AddPacketCountObserver( |
| 380 PacketCountObserver* packet_count_observer) { |
| 381 DCHECK(thread_checker_.CalledOnValidThread()); |
| 382 packet_count_observer_list_.AddObserver(packet_count_observer); |
| 383 } |
| 384 |
| 385 void NetworkQualityEstimator::RemovePacketCountObserver( |
| 386 PacketCountObserver* packet_count_observer) { |
| 387 DCHECK(thread_checker_.CalledOnValidThread()); |
| 388 packet_count_observer_list_.RemoveObserver(packet_count_observer); |
| 389 } |
| 390 |
| 379 void NetworkQualityEstimator::RecordRTTUMA(int32_t estimated_value_msec, | 391 void NetworkQualityEstimator::RecordRTTUMA(int32_t estimated_value_msec, |
| 380 int32_t actual_value_msec) const { | 392 int32_t actual_value_msec) const { |
| 381 DCHECK(thread_checker_.CalledOnValidThread()); | 393 DCHECK(thread_checker_.CalledOnValidThread()); |
| 382 | 394 |
| 383 // Record the difference between the actual and the estimated value. | 395 // Record the difference between the actual and the estimated value. |
| 384 if (estimated_value_msec >= actual_value_msec) { | 396 if (estimated_value_msec >= actual_value_msec) { |
| 385 base::HistogramBase* difference_rtt = | 397 base::HistogramBase* difference_rtt = |
| 386 GetHistogram("DifferenceRTTEstimatedAndActual.", | 398 GetHistogram("DifferenceRTTEstimatedAndActual.", |
| 387 current_network_id_.type, 10 * 1000); // 10 seconds | 399 current_network_id_.type, 10 * 1000); // 10 seconds |
| 388 difference_rtt->Add(estimated_value_msec - actual_value_msec); | 400 difference_rtt->Add(estimated_value_msec - actual_value_msec); |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 422 !request.was_cached() && | 434 !request.was_cached() && |
| 423 request.creation_time() >= last_connection_change_; | 435 request.creation_time() >= last_connection_change_; |
| 424 } | 436 } |
| 425 | 437 |
| 426 void NetworkQualityEstimator::RecordExternalEstimateProviderMetrics( | 438 void NetworkQualityEstimator::RecordExternalEstimateProviderMetrics( |
| 427 NQEExternalEstimateProviderStatus status) const { | 439 NQEExternalEstimateProviderStatus status) const { |
| 428 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus", status, | 440 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus", status, |
| 429 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY); | 441 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY); |
| 430 } | 442 } |
| 431 | 443 |
| 444 bool NetworkQualityEstimator::GetObservationSourceForProtocol( |
| 445 Protocol protocol, |
| 446 NetworkQualityEstimator::ObservationSource* observation_source) const { |
| 447 switch (protocol) { |
| 448 case PROTOCOL_TCP: |
| 449 *observation_source = TCP; |
| 450 return true; |
| 451 case PROTOCOL_QUIC: |
| 452 *observation_source = QUIC; |
| 453 return true; |
| 454 } |
| 455 NOTREACHED(); |
| 456 return false; |
| 457 } |
| 458 |
| 432 void NetworkQualityEstimator::OnConnectionTypeChanged( | 459 void NetworkQualityEstimator::OnConnectionTypeChanged( |
| 433 NetworkChangeNotifier::ConnectionType type) { | 460 NetworkChangeNotifier::ConnectionType type) { |
| 434 DCHECK(thread_checker_.CalledOnValidThread()); | 461 DCHECK(thread_checker_.CalledOnValidThread()); |
| 435 if (peak_network_quality_.rtt() != InvalidRTT()) { | 462 if (peak_network_quality_.rtt() != InvalidRTT()) { |
| 436 switch (current_network_id_.type) { | 463 switch (current_network_id_.type) { |
| 437 case NetworkChangeNotifier::CONNECTION_UNKNOWN: | 464 case NetworkChangeNotifier::CONNECTION_UNKNOWN: |
| 438 UMA_HISTOGRAM_TIMES("NQE.FastestRTT.Unknown", | 465 UMA_HISTOGRAM_TIMES("NQE.FastestRTT.Unknown", |
| 439 peak_network_quality_.rtt()); | 466 peak_network_quality_.rtt()); |
| 440 break; | 467 break; |
| 441 case NetworkChangeNotifier::CONNECTION_ETHERNET: | 468 case NetworkChangeNotifier::CONNECTION_ETHERNET: |
| (...skipping 451 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 893 | 920 |
| 894 return scoped_ptr<SocketPerformanceWatcher>( | 921 return scoped_ptr<SocketPerformanceWatcher>( |
| 895 new SocketPerformanceWatcher(protocol, this)); | 922 new SocketPerformanceWatcher(protocol, this)); |
| 896 } | 923 } |
| 897 | 924 |
| 898 void NetworkQualityEstimator::OnUpdatedRTTAvailable( | 925 void NetworkQualityEstimator::OnUpdatedRTTAvailable( |
| 899 const Protocol protocol, | 926 const Protocol protocol, |
| 900 const base::TimeDelta& rtt) { | 927 const base::TimeDelta& rtt) { |
| 901 DCHECK(thread_checker_.CalledOnValidThread()); | 928 DCHECK(thread_checker_.CalledOnValidThread()); |
| 902 | 929 |
| 903 switch (protocol) { | 930 ObservationSource observation_source; |
| 904 case PROTOCOL_TCP: | 931 if (!GetObservationSourceForProtocol(protocol, &observation_source)) |
| 905 NotifyObserversOfRTT(RttObservation(rtt, base::TimeTicks::Now(), TCP)); | 932 return; |
| 906 return; | 933 |
| 907 case PROTOCOL_QUIC: | 934 NotifyObserversOfRTT( |
| 908 NotifyObserversOfRTT(RttObservation(rtt, base::TimeTicks::Now(), QUIC)); | 935 RttObservation(rtt, base::TimeTicks::Now(), observation_source)); |
| 909 return; | 936 } |
| 910 default: | 937 |
| 911 NOTREACHED(); | 938 void NetworkQualityEstimator::OnIncrementalPacketCountAvailable( |
| 912 } | 939 Protocol protocol, |
| 940 size_t packets_missing, |
| 941 size_t packets_received_in_order, |
| 942 size_t packets_received_out_of_order) { |
| 943 DCHECK(thread_checker_.CalledOnValidThread()); |
| 944 |
| 945 ObservationSource observation_source; |
| 946 if (!GetObservationSourceForProtocol(protocol, &observation_source)) |
| 947 return; |
| 948 |
| 949 // Consider it as a packet loss only if at least 3 packets are received after |
| 950 // the missing packets. |
| 951 // TODO(tbansal): Maintain a watcher specific buffer that keeps track of |
| 952 // missing, and at most last N-1 in-order packets. If N (=3) packets with |
| 953 // sequence number higher than the sequence number of a missing packet are |
| 954 // received, then the missing packet should be marked as lost, and removed |
| 955 // from the watcher specific buffer. If an out-of-order packet is received, |
| 956 // then the state of the missing packet with lowest sequence number should be |
| 957 // changed to received. This simple approach does not require going back in |
| 958 // time, and changing the previously taken observations. |
| 959 NotifyObserversOfIncrementalPacketCount( |
| 960 packets_missing, packets_received_in_order, packets_received_out_of_order, |
| 961 base::TimeTicks::Now(), observation_source); |
| 913 } | 962 } |
| 914 | 963 |
| 915 void NetworkQualityEstimator::NotifyObserversOfRTT( | 964 void NetworkQualityEstimator::NotifyObserversOfRTT( |
| 916 const RttObservation& observation) { | 965 const RttObservation& observation) { |
| 966 DCHECK(thread_checker_.CalledOnValidThread()); |
| 967 DCHECK_NE(EXTERNAL_ESTIMATE, observation.source); |
| 968 |
| 917 FOR_EACH_OBSERVER( | 969 FOR_EACH_OBSERVER( |
| 918 RTTObserver, rtt_observer_list_, | 970 RTTObserver, rtt_observer_list_, |
| 919 OnRTTObservation(observation.value.InMilliseconds(), | 971 OnRTTObservation(observation.value.InMilliseconds(), |
| 920 observation.timestamp, observation.source)); | 972 observation.timestamp, observation.source)); |
| 921 } | 973 } |
| 922 | 974 |
| 923 void NetworkQualityEstimator::NotifyObserversOfThroughput( | 975 void NetworkQualityEstimator::NotifyObserversOfThroughput( |
| 924 const ThroughputObservation& observation) { | 976 const ThroughputObservation& observation) { |
| 977 DCHECK(thread_checker_.CalledOnValidThread()); |
| 978 DCHECK_NE(EXTERNAL_ESTIMATE, observation.source); |
| 979 |
| 925 FOR_EACH_OBSERVER( | 980 FOR_EACH_OBSERVER( |
| 926 ThroughputObserver, throughput_observer_list_, | 981 ThroughputObserver, throughput_observer_list_, |
| 927 OnThroughputObservation(observation.value, observation.timestamp, | 982 OnThroughputObservation(observation.value, observation.timestamp, |
| 928 observation.source)); | 983 observation.source)); |
| 929 } | 984 } |
| 930 | 985 |
| 986 void NetworkQualityEstimator::NotifyObserversOfIncrementalPacketCount( |
| 987 size_t packets_missing, |
| 988 size_t packets_received_in_order, |
| 989 size_t packets_received_out_of_order, |
| 990 const base::TimeTicks& timestamp, |
| 991 ObservationSource source) { |
| 992 DCHECK(thread_checker_.CalledOnValidThread()); |
| 993 DCHECK_NE(EXTERNAL_ESTIMATE, source); |
| 994 |
| 995 FOR_EACH_OBSERVER(PacketCountObserver, packet_count_observer_list_, |
| 996 OnIncrementalPacketCountObservation( |
| 997 packets_missing, packets_received_in_order, |
| 998 packets_received_out_of_order, timestamp, source)); |
| 999 } |
| 1000 |
| 931 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( | 1001 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( |
| 932 const NetworkQuality& network_quality) | 1002 const NetworkQuality& network_quality) |
| 933 : last_update_time_(base::TimeTicks::Now()), | 1003 : last_update_time_(base::TimeTicks::Now()), |
| 934 network_quality_(network_quality) { | 1004 network_quality_(network_quality) { |
| 935 } | 1005 } |
| 936 | 1006 |
| 937 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( | 1007 NetworkQualityEstimator::CachedNetworkQuality::CachedNetworkQuality( |
| 938 const CachedNetworkQuality& other) | 1008 const CachedNetworkQuality& other) |
| 939 : last_update_time_(other.last_update_time_), | 1009 : last_update_time_(other.last_update_time_), |
| 940 network_quality_(other.network_quality_) { | 1010 network_quality_(other.network_quality_) { |
| (...skipping 27 matching lines...) Expand all Loading... |
| 968 | 1038 |
| 969 NetworkQualityEstimator::NetworkQuality& | 1039 NetworkQualityEstimator::NetworkQuality& |
| 970 NetworkQualityEstimator::NetworkQuality:: | 1040 NetworkQualityEstimator::NetworkQuality:: |
| 971 operator=(const NetworkQuality& other) { | 1041 operator=(const NetworkQuality& other) { |
| 972 rtt_ = other.rtt_; | 1042 rtt_ = other.rtt_; |
| 973 downstream_throughput_kbps_ = other.downstream_throughput_kbps_; | 1043 downstream_throughput_kbps_ = other.downstream_throughput_kbps_; |
| 974 return *this; | 1044 return *this; |
| 975 } | 1045 } |
| 976 | 1046 |
| 977 } // namespace net | 1047 } // namespace net |
| OLD | NEW |