Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(69)

Side by Side Diff: net/base/network_quality_estimator.cc

Issue 1316863006: Populate EEP estimate in NQE (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Addressed bengr comments Created 5 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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 160 matching lines...) Expand 10 before | Expand all | Expand 10 after
171 bool allow_smaller_responses_for_tests) 171 bool allow_smaller_responses_for_tests)
172 : allow_localhost_requests_(allow_local_host_requests_for_tests), 172 : allow_localhost_requests_(allow_local_host_requests_for_tests),
173 allow_small_responses_(allow_smaller_responses_for_tests), 173 allow_small_responses_(allow_smaller_responses_for_tests),
174 last_connection_change_(base::TimeTicks::Now()), 174 last_connection_change_(base::TimeTicks::Now()),
175 current_network_id_( 175 current_network_id_(
176 NetworkID(NetworkChangeNotifier::ConnectionType::CONNECTION_UNKNOWN, 176 NetworkID(NetworkChangeNotifier::ConnectionType::CONNECTION_UNKNOWN,
177 std::string())), 177 std::string())),
178 downstream_throughput_kbps_observations_( 178 downstream_throughput_kbps_observations_(
179 GetWeightMultiplierPerSecond(variation_params)), 179 GetWeightMultiplierPerSecond(variation_params)),
180 rtt_msec_observations_(GetWeightMultiplierPerSecond(variation_params)), 180 rtt_msec_observations_(GetWeightMultiplierPerSecond(variation_params)),
181 external_estimates_provider_(external_estimates_provider.Pass()) { 181 external_estimate_provider_(external_estimates_provider.Pass()),
182 external_estimate_request_time_(base::TimeTicks()) {
182 static_assert(kMinRequestDurationMicroseconds > 0, 183 static_assert(kMinRequestDurationMicroseconds > 0,
183 "Minimum request duration must be > 0"); 184 "Minimum request duration must be > 0");
184 static_assert(kDefaultHalfLifeSeconds > 0, 185 static_assert(kDefaultHalfLifeSeconds > 0,
185 "Default half life duration must be > 0"); 186 "Default half life duration must be > 0");
186 static_assert(kMaximumNetworkQualityCacheSize > 0, 187 static_assert(kMaximumNetworkQualityCacheSize > 0,
187 "Size of the network quality cache must be > 0"); 188 "Size of the network quality cache must be > 0");
188 // This limit should not be increased unless the logic for removing the 189 // This limit should not be increased unless the logic for removing the
189 // oldest cache entry is rewritten to use a doubly-linked-list LRU queue. 190 // oldest cache entry is rewritten to use a doubly-linked-list LRU queue.
190 static_assert(kMaximumNetworkQualityCacheSize <= 10, 191 static_assert(kMaximumNetworkQualityCacheSize <= 10,
191 "Size of the network quality cache must <= 10"); 192 "Size of the network quality cache must <= 10");
192 193
193 ObtainOperatingParams(variation_params); 194 ObtainOperatingParams(variation_params);
194 NetworkChangeNotifier::AddConnectionTypeObserver(this); 195 NetworkChangeNotifier::AddConnectionTypeObserver(this);
195 if (external_estimates_provider_) 196 if (external_estimate_provider_) {
196 external_estimates_provider_->SetUpdatedEstimateDelegate(this); 197 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus",
Alexei Svitkine (slow) 2015/09/08 21:33:10 Please make a helper function that logs this histo
tbansal1 2015/09/08 22:29:30 Made the helper function. Had to move the enum def
198 EXTERNAL_ESTIMATE_PROVIDER_STATUS_AVAILABLE,
199 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY);
200 external_estimate_provider_->SetUpdatedEstimateDelegate(this);
201 QueryExternalEstimateProvider();
202 } else {
203 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus",
204 EXTERNAL_ESTIMATE_PROVIDER_STATUS_NOT_AVAILABLE,
205 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY);
206 }
197 current_network_id_ = GetCurrentNetworkID(); 207 current_network_id_ = GetCurrentNetworkID();
198 AddDefaultEstimates(); 208 AddDefaultEstimates();
199 } 209 }
200 210
201 // static 211 // static
202 const base::TimeDelta NetworkQualityEstimator::InvalidRTT() { 212 const base::TimeDelta NetworkQualityEstimator::InvalidRTT() {
203 return base::TimeDelta::Max(); 213 return base::TimeDelta::Max();
204 } 214 }
205 215
206 void NetworkQualityEstimator::ObtainOperatingParams( 216 void NetworkQualityEstimator::ObtainOperatingParams(
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after
257 } 267 }
258 268
259 NetworkQualityEstimator::~NetworkQualityEstimator() { 269 NetworkQualityEstimator::~NetworkQualityEstimator() {
260 DCHECK(thread_checker_.CalledOnValidThread()); 270 DCHECK(thread_checker_.CalledOnValidThread());
261 NetworkChangeNotifier::RemoveConnectionTypeObserver(this); 271 NetworkChangeNotifier::RemoveConnectionTypeObserver(this);
262 } 272 }
263 273
264 void NetworkQualityEstimator::NotifyHeadersReceived(const URLRequest& request) { 274 void NetworkQualityEstimator::NotifyHeadersReceived(const URLRequest& request) {
265 DCHECK(thread_checker_.CalledOnValidThread()); 275 DCHECK(thread_checker_.CalledOnValidThread());
266 276
277 MaybeQueryExternalEstimateProvider();
267 if (!RequestProvidesUsefulObservations(request)) 278 if (!RequestProvidesUsefulObservations(request))
268 return; 279 return;
269 280
270 // Update |estimated_median_network_quality_| if this is a main frame request. 281 // Update |estimated_median_network_quality_| if this is a main frame request.
271 if (request.load_flags() & LOAD_MAIN_FRAME) { 282 if (request.load_flags() & LOAD_MAIN_FRAME) {
272 estimated_median_network_quality_ = NetworkQuality( 283 estimated_median_network_quality_ = NetworkQuality(
273 GetRTTEstimateInternal(base::TimeTicks(), 50), 284 GetRTTEstimateInternal(base::TimeTicks(), 50),
274 GetDownlinkThroughputKbpsEstimateInternal(base::TimeTicks(), 50)); 285 GetDownlinkThroughputKbpsEstimateInternal(base::TimeTicks(), 50));
275 } 286 }
276 287
(...skipping 30 matching lines...) Expand all
307 if (estimated_median_network_quality_.rtt() != InvalidRTT()) { 318 if (estimated_median_network_quality_.rtt() != InvalidRTT()) {
308 RecordRTTUMA(estimated_median_network_quality_.rtt().InMilliseconds(), 319 RecordRTTUMA(estimated_median_network_quality_.rtt().InMilliseconds(),
309 observed_rtt.InMilliseconds()); 320 observed_rtt.InMilliseconds());
310 } 321 }
311 } 322 }
312 323
313 void NetworkQualityEstimator::NotifyRequestCompleted( 324 void NetworkQualityEstimator::NotifyRequestCompleted(
314 const URLRequest& request) { 325 const URLRequest& request) {
315 DCHECK(thread_checker_.CalledOnValidThread()); 326 DCHECK(thread_checker_.CalledOnValidThread());
316 327
328 MaybeQueryExternalEstimateProvider();
317 if (!RequestProvidesUsefulObservations(request)) 329 if (!RequestProvidesUsefulObservations(request))
318 return; 330 return;
319 331
320 base::TimeTicks now = base::TimeTicks::Now(); 332 base::TimeTicks now = base::TimeTicks::Now();
321 LoadTimingInfo load_timing_info; 333 LoadTimingInfo load_timing_info;
322 request.GetLoadTimingInfo(&load_timing_info); 334 request.GetLoadTimingInfo(&load_timing_info);
323 335
324 // If the load timing info is unavailable, it probably means that the request 336 // If the load timing info is unavailable, it probably means that the request
325 // did not go over the network. 337 // did not go over the network.
326 if (load_timing_info.send_start.is_null() || 338 if (load_timing_info.send_start.is_null() ||
(...skipping 204 matching lines...) Expand 10 before | Expand all | Expand 10 after
531 // Write the estimates of the previous network to the cache. 543 // Write the estimates of the previous network to the cache.
532 CacheNetworkQualityEstimate(); 544 CacheNetworkQualityEstimate();
533 545
534 // Clear the local state. 546 // Clear the local state.
535 last_connection_change_ = base::TimeTicks::Now(); 547 last_connection_change_ = base::TimeTicks::Now();
536 peak_network_quality_ = NetworkQuality(); 548 peak_network_quality_ = NetworkQuality();
537 downstream_throughput_kbps_observations_.Clear(); 549 downstream_throughput_kbps_observations_.Clear();
538 rtt_msec_observations_.Clear(); 550 rtt_msec_observations_.Clear();
539 current_network_id_ = GetCurrentNetworkID(); 551 current_network_id_ = GetCurrentNetworkID();
540 552
553 QueryExternalEstimateProvider();
554
541 // Read any cached estimates for the new network. If cached estimates are 555 // Read any cached estimates for the new network. If cached estimates are
542 // unavailable, add the default estimates. 556 // unavailable, add the default estimates.
543 if (!ReadCachedNetworkQualityEstimate()) 557 if (!ReadCachedNetworkQualityEstimate())
544 AddDefaultEstimates(); 558 AddDefaultEstimates();
545 estimated_median_network_quality_ = NetworkQuality(); 559 estimated_median_network_quality_ = NetworkQuality();
546 } 560 }
547 561
548 bool NetworkQualityEstimator::GetRTTEstimate(base::TimeDelta* rtt) const { 562 bool NetworkQualityEstimator::GetRTTEstimate(base::TimeDelta* rtt) {
549 DCHECK(thread_checker_.CalledOnValidThread()); 563 DCHECK(thread_checker_.CalledOnValidThread());
550 DCHECK(rtt); 564 DCHECK(rtt);
565 MaybeQueryExternalEstimateProvider();
551 if (rtt_msec_observations_.Size() == 0) { 566 if (rtt_msec_observations_.Size() == 0) {
552 *rtt = InvalidRTT(); 567 *rtt = InvalidRTT();
553 return false; 568 return false;
554 } 569 }
555 *rtt = GetRTTEstimateInternal(base::TimeTicks(), 50); 570 *rtt = GetRTTEstimateInternal(base::TimeTicks(), 50);
556 return (*rtt != InvalidRTT()); 571 return (*rtt != InvalidRTT());
557 } 572 }
558 573
559 bool NetworkQualityEstimator::GetDownlinkThroughputKbpsEstimate( 574 bool NetworkQualityEstimator::GetDownlinkThroughputKbpsEstimate(int32_t* kbps) {
560 int32_t* kbps) const {
561 DCHECK(thread_checker_.CalledOnValidThread()); 575 DCHECK(thread_checker_.CalledOnValidThread());
562 DCHECK(kbps); 576 DCHECK(kbps);
577 MaybeQueryExternalEstimateProvider();
563 if (downstream_throughput_kbps_observations_.Size() == 0) { 578 if (downstream_throughput_kbps_observations_.Size() == 0) {
564 *kbps = kInvalidThroughput; 579 *kbps = kInvalidThroughput;
565 return false; 580 return false;
566 } 581 }
567 *kbps = GetDownlinkThroughputKbpsEstimateInternal(base::TimeTicks(), 50); 582 *kbps = GetDownlinkThroughputKbpsEstimateInternal(base::TimeTicks(), 50);
568 return (*kbps != kInvalidThroughput); 583 return (*kbps != kInvalidThroughput);
569 } 584 }
570 585
571 bool NetworkQualityEstimator::GetRecentMedianRTT( 586 bool NetworkQualityEstimator::GetRecentMedianRTT(
572 const base::TimeTicks& begin_timestamp, 587 const base::TimeTicks& begin_timestamp,
573 base::TimeDelta* rtt) const { 588 base::TimeDelta* rtt) {
574 DCHECK(thread_checker_.CalledOnValidThread()); 589 DCHECK(thread_checker_.CalledOnValidThread());
575 DCHECK(rtt); 590 DCHECK(rtt);
591
592 MaybeQueryExternalEstimateProvider();
576 *rtt = GetRTTEstimateInternal(begin_timestamp, 50); 593 *rtt = GetRTTEstimateInternal(begin_timestamp, 50);
577 return (*rtt != InvalidRTT()); 594 return (*rtt != InvalidRTT());
578 } 595 }
579 596
580 bool NetworkQualityEstimator::GetRecentMedianDownlinkThroughputKbps( 597 bool NetworkQualityEstimator::GetRecentMedianDownlinkThroughputKbps(
581 const base::TimeTicks& begin_timestamp, 598 const base::TimeTicks& begin_timestamp,
582 int32_t* kbps) const { 599 int32_t* kbps) {
583 DCHECK(thread_checker_.CalledOnValidThread()); 600 DCHECK(thread_checker_.CalledOnValidThread());
584 DCHECK(kbps); 601 DCHECK(kbps);
602
603 MaybeQueryExternalEstimateProvider();
585 *kbps = GetDownlinkThroughputKbpsEstimateInternal(begin_timestamp, 50); 604 *kbps = GetDownlinkThroughputKbpsEstimateInternal(begin_timestamp, 50);
586 return (*kbps != kInvalidThroughput); 605 return (*kbps != kInvalidThroughput);
587 } 606 }
588 607
589 NetworkQualityEstimator::Observation::Observation(int32_t value, 608 NetworkQualityEstimator::Observation::Observation(int32_t value,
590 base::TimeTicks timestamp) 609 base::TimeTicks timestamp)
591 : value(value), timestamp(timestamp) { 610 : value(value), timestamp(timestamp) {
592 DCHECK_GE(value, 0); 611 DCHECK_GE(value, 0);
593 DCHECK(!timestamp.is_null()); 612 DCHECK(!timestamp.is_null());
594 } 613 }
(...skipping 210 matching lines...) Expand 10 before | Expand all | Expand 10 after
805 824
806 downstream_throughput_kbps_observations_.AddObservation(Observation( 825 downstream_throughput_kbps_observations_.AddObservation(Observation(
807 network_quality.downstream_throughput_kbps(), base::TimeTicks::Now())); 826 network_quality.downstream_throughput_kbps(), base::TimeTicks::Now()));
808 rtt_msec_observations_.AddObservation(Observation( 827 rtt_msec_observations_.AddObservation(Observation(
809 network_quality.rtt().InMilliseconds(), base::TimeTicks::Now())); 828 network_quality.rtt().InMilliseconds(), base::TimeTicks::Now()));
810 return true; 829 return true;
811 } 830 }
812 831
813 void NetworkQualityEstimator::OnUpdatedEstimateAvailable() { 832 void NetworkQualityEstimator::OnUpdatedEstimateAvailable() {
814 DCHECK(thread_checker_.CalledOnValidThread()); 833 DCHECK(thread_checker_.CalledOnValidThread());
815 DCHECK(external_estimates_provider_); 834 DCHECK(external_estimate_provider_);
816 // TODO(tbansal): Query provider for the recent value. 835
836 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus",
837 EXTERNAL_ESTIMATE_PROVIDER_STATUS_CALLBACK,
838 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY);
839
840 QueryExternalEstimateProvider();
841 }
842
843 void NetworkQualityEstimator::MaybeQueryExternalEstimateProvider() {
844 if (base::TimeTicks::Now() - external_estimate_request_time_ >=
845 base::TimeDelta::FromMilliseconds(
846 kExternalEstimateProviderQueryIntervalMsec)) {
847 QueryExternalEstimateProvider();
848 }
849 }
850
851 void NetworkQualityEstimator::QueryExternalEstimateProvider() {
852 DCHECK(thread_checker_.CalledOnValidThread());
853
854 if (!external_estimate_provider_)
855 return;
856 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus",
857 EXTERNAL_ESTIMATE_PROVIDER_STATUS_QUERIED,
858 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY);
859
860 external_estimate_request_time_ = base::TimeTicks::Now();
861
862 base::TimeDelta time_since_last_update;
863 if (!external_estimate_provider_->GetTimeSinceLastUpdate(
864 &time_since_last_update) ||
865 time_since_last_update >
866 base::TimeDelta::FromMilliseconds(
867 kExternalEstimateProviderQueryIntervalMsec)) {
868 // Request the external estimate provider for updated estimates. When the
869 // updates estimates are available, OnUpdatedEstimateAvailable() will get
870 // called.
871 external_estimate_provider_->RequestUpdate();
872 return;
873 }
874
875 UMA_HISTOGRAM_ENUMERATION("NQE.ExternalEstimateProviderStatus",
876 EXTERNAL_ESTIMATE_PROVIDER_STATUS_QUERY_SUCCESSFUL,
877 EXTERNAL_ESTIMATE_PROVIDER_STATUS_BOUNDARY);
878
879 base::TimeDelta rtt;
880 if (external_estimate_provider_->GetRTT(&rtt)) {
881 rtt_msec_observations_.AddObservation(Observation(
882 rtt.InMilliseconds(), base::TimeTicks::Now() - time_since_last_update));
883 }
884
885 int32_t downstream_throughput_kbps;
886 if (external_estimate_provider_->GetDownstreamThroughputKbps(
887 &downstream_throughput_kbps)) {
888 downstream_throughput_kbps_observations_.AddObservation(
889 Observation(downstream_throughput_kbps,
890 base::TimeTicks::Now() - time_since_last_update));
891 }
817 } 892 }
818 893
819 void NetworkQualityEstimator::CacheNetworkQualityEstimate() { 894 void NetworkQualityEstimator::CacheNetworkQualityEstimate() {
820 DCHECK(thread_checker_.CalledOnValidThread()); 895 DCHECK(thread_checker_.CalledOnValidThread());
821 DCHECK_LE(cached_network_qualities_.size(), 896 DCHECK_LE(cached_network_qualities_.size(),
822 static_cast<size_t>(kMaximumNetworkQualityCacheSize)); 897 static_cast<size_t>(kMaximumNetworkQualityCacheSize));
823 898
824 // If the network name is unavailable, caching should not be performed. 899 // If the network name is unavailable, caching should not be performed.
825 if (current_network_id_.id.empty()) 900 if (current_network_id_.id.empty())
826 return; 901 return;
(...skipping 82 matching lines...) Expand 10 before | Expand all | Expand 10 after
909 984
910 NetworkQualityEstimator::NetworkQuality& 985 NetworkQualityEstimator::NetworkQuality&
911 NetworkQualityEstimator::NetworkQuality:: 986 NetworkQualityEstimator::NetworkQuality::
912 operator=(const NetworkQuality& other) { 987 operator=(const NetworkQuality& other) {
913 rtt_ = other.rtt_; 988 rtt_ = other.rtt_;
914 downstream_throughput_kbps_ = other.downstream_throughput_kbps_; 989 downstream_throughput_kbps_ = other.downstream_throughput_kbps_;
915 return *this; 990 return *this;
916 } 991 }
917 992
918 } // namespace net 993 } // namespace net
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698