OLD | NEW |
---|---|
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 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 "chrome/browser/prerender/prerender_manager.h" | 5 #include "chrome/browser/prerender/prerender_manager.h" |
6 | 6 |
7 #include "base/logging.h" | 7 #include "base/logging.h" |
8 #include "base/metrics/histogram.h" | 8 #include "base/metrics/histogram.h" |
9 #include "base/time.h" | 9 #include "base/time.h" |
10 #include "base/utf_string_conversions.h" | 10 #include "base/utf_string_conversions.h" |
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
47 struct PrerenderManager::PrerenderContentsData { | 47 struct PrerenderManager::PrerenderContentsData { |
48 PrerenderContents* contents_; | 48 PrerenderContents* contents_; |
49 base::Time start_time_; | 49 base::Time start_time_; |
50 PrerenderContentsData(PrerenderContents* contents, base::Time start_time) | 50 PrerenderContentsData(PrerenderContents* contents, base::Time start_time) |
51 : contents_(contents), | 51 : contents_(contents), |
52 start_time_(start_time) { | 52 start_time_(start_time) { |
53 } | 53 } |
54 }; | 54 }; |
55 | 55 |
56 PrerenderManager::PrerenderManager(Profile* profile) | 56 PrerenderManager::PrerenderManager(Profile* profile) |
57 : profile_(profile), | 57 : rate_limit_enabled_(true), |
58 profile_(profile), | |
58 max_prerender_age_(base::TimeDelta::FromSeconds( | 59 max_prerender_age_(base::TimeDelta::FromSeconds( |
59 kDefaultMaxPrerenderAgeSeconds)), | 60 kDefaultMaxPrerenderAgeSeconds)), |
60 max_elements_(kDefaultMaxPrerenderElements), | 61 max_elements_(kDefaultMaxPrerenderElements), |
61 prerender_contents_factory_(PrerenderContents::CreateFactory()) { | 62 prerender_contents_factory_(PrerenderContents::CreateFactory()), |
63 last_prerender_start_time_(base::TimeTicks::Now()) { | |
cbentzel
2011/03/14 18:18:16
This should be Now - kMinTimeBetweenPrerendersMs s
dominich
2011/03/14 20:03:21
Done.
| |
62 } | 64 } |
63 | 65 |
64 PrerenderManager::~PrerenderManager() { | 66 PrerenderManager::~PrerenderManager() { |
65 while (!prerender_list_.empty()) { | 67 while (!prerender_list_.empty()) { |
66 PrerenderContentsData data = prerender_list_.front(); | 68 PrerenderContentsData data = prerender_list_.front(); |
67 prerender_list_.pop_front(); | 69 prerender_list_.pop_front(); |
68 data.contents_->set_final_status(FINAL_STATUS_MANAGER_SHUTDOWN); | 70 data.contents_->set_final_status(FINAL_STATUS_MANAGER_SHUTDOWN); |
69 delete data.contents_; | 71 delete data.contents_; |
70 } | 72 } |
71 } | 73 } |
(...skipping 13 matching lines...) Expand all Loading... | |
85 // Do not prerender if there are too many render processes, and we would | 87 // Do not prerender if there are too many render processes, and we would |
86 // have to use an existing one. We do not want prerendering to happen in | 88 // have to use an existing one. We do not want prerendering to happen in |
87 // a shared process, so that we can always reliably lower the CPU | 89 // a shared process, so that we can always reliably lower the CPU |
88 // priority for prerendering. | 90 // priority for prerendering. |
89 // TODO(tburkard): Figure out how to cancel prerendering in the opposite | 91 // TODO(tburkard): Figure out how to cancel prerendering in the opposite |
90 // case, when a new tab is added to a process used for prerendering. | 92 // case, when a new tab is added to a process used for prerendering. |
91 if (RenderProcessHost::ShouldTryToUseExistingProcessHost()) { | 93 if (RenderProcessHost::ShouldTryToUseExistingProcessHost()) { |
92 RecordFinalStatus(FINAL_STATUS_TOO_MANY_PROCESSES); | 94 RecordFinalStatus(FINAL_STATUS_TOO_MANY_PROCESSES); |
93 return false; | 95 return false; |
94 } | 96 } |
97 | |
98 // Check if enough time has passed since the last prerender. | |
99 if (!DoesRateLimitAllowPrerender()) { | |
100 // Cancel the prerender. We could add it to the pending prerender list but | |
101 // this doesn't make sense as the next prerender request will be triggered | |
102 // by a navigation and is unlikely to be the same site. | |
103 RecordFinalStatus(FINAL_STATUS_RATE_LIMIT_EXCEEDED); | |
104 return false; | |
105 } | |
106 | |
95 // TODO(cbentzel): Move invalid checks here instead of PrerenderContents? | 107 // TODO(cbentzel): Move invalid checks here instead of PrerenderContents? |
96 PrerenderContentsData data(CreatePrerenderContents(url, alias_urls, referrer), | 108 PrerenderContentsData data(CreatePrerenderContents(url, alias_urls, referrer), |
97 GetCurrentTime()); | 109 GetCurrentTime()); |
98 prerender_list_.push_back(data); | 110 prerender_list_.push_back(data); |
111 last_prerender_start_time_ = base::TimeTicks::Now(); | |
99 data.contents_->StartPrerendering(); | 112 data.contents_->StartPrerendering(); |
100 while (prerender_list_.size() > max_elements_) { | 113 while (prerender_list_.size() > max_elements_) { |
101 data = prerender_list_.front(); | 114 data = prerender_list_.front(); |
102 prerender_list_.pop_front(); | 115 prerender_list_.pop_front(); |
103 data.contents_->set_final_status(FINAL_STATUS_EVICTED); | 116 data.contents_->set_final_status(FINAL_STATUS_EVICTED); |
104 delete data.contents_; | 117 delete data.contents_; |
105 } | 118 } |
106 StartSchedulingPeriodicCleanups(); | 119 StartSchedulingPeriodicCleanups(); |
107 return true; | 120 return true; |
108 } | 121 } |
(...skipping 161 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
270 // static | 283 // static |
271 bool PrerenderManager::ShouldRecordWindowedPPLT() { | 284 bool PrerenderManager::ShouldRecordWindowedPPLT() { |
272 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 285 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
273 if (last_prefetch_seen_time_.is_null()) | 286 if (last_prefetch_seen_time_.is_null()) |
274 return false; | 287 return false; |
275 base::TimeDelta elapsed_time = | 288 base::TimeDelta elapsed_time = |
276 base::TimeTicks::Now() - last_prefetch_seen_time_; | 289 base::TimeTicks::Now() - last_prefetch_seen_time_; |
277 return elapsed_time <= base::TimeDelta::FromSeconds(kWindowedPPLTSeconds); | 290 return elapsed_time <= base::TimeDelta::FromSeconds(kWindowedPPLTSeconds); |
278 } | 291 } |
279 | 292 |
293 // static | |
cbentzel
2011/03/14 18:18:16
Nit: this comment is no longer true.
dominich
2011/03/14 20:03:21
Done.
| |
294 bool PrerenderManager::DoesRateLimitAllowPrerender() { | |
295 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
296 if (!rate_limit_enabled_) | |
297 return true; | |
298 if (last_prerender_start_time_.is_null()) | |
cbentzel
2011/03/14 18:18:16
I don't think you need to test this case given how
dominich
2011/03/14 20:03:21
Done.
| |
299 return true; | |
300 base::TimeDelta elapsed_time = | |
301 base::TimeTicks::Now() - last_prerender_start_time_; | |
302 UMA_HISTOGRAM_TIMES("Prerender.TimeBetweenPrerenderRequests", | |
cbentzel
2011/03/14 18:18:16
Should this be captured at the start of the functi
dominich
2011/03/14 20:03:21
It was here as the last_prerender_start_time_ coul
| |
303 elapsed_time); | |
304 return elapsed_time > | |
305 base::TimeDelta::FromMilliseconds(kMinTimeBetweenPrerendersMs); | |
306 } | |
307 | |
280 void PrerenderManager::StartSchedulingPeriodicCleanups() { | 308 void PrerenderManager::StartSchedulingPeriodicCleanups() { |
281 if (repeating_timer_.IsRunning()) | 309 if (repeating_timer_.IsRunning()) |
282 return; | 310 return; |
283 repeating_timer_.Start( | 311 repeating_timer_.Start( |
284 base::TimeDelta::FromMilliseconds(kPeriodicCleanupIntervalMs), | 312 base::TimeDelta::FromMilliseconds(kPeriodicCleanupIntervalMs), |
285 this, | 313 this, |
286 &PrerenderManager::PeriodicCleanup); | 314 &PrerenderManager::PeriodicCleanup); |
287 } | 315 } |
288 | 316 |
289 void PrerenderManager::StopSchedulingPeriodicCleanups() { | 317 void PrerenderManager::StopSchedulingPeriodicCleanups() { |
(...skipping 24 matching lines...) Expand all Loading... | |
314 | 342 |
315 void PrerenderManager::MarkTabContentsAsNotPrerendered(TabContents* tc) { | 343 void PrerenderManager::MarkTabContentsAsNotPrerendered(TabContents* tc) { |
316 prerendered_tc_set_.erase(tc); | 344 prerendered_tc_set_.erase(tc); |
317 } | 345 } |
318 | 346 |
319 bool PrerenderManager::IsTabContentsPrerendered(TabContents* tc) const { | 347 bool PrerenderManager::IsTabContentsPrerendered(TabContents* tc) const { |
320 return prerendered_tc_set_.count(tc) > 0; | 348 return prerendered_tc_set_.count(tc) > 0; |
321 } | 349 } |
322 | 350 |
323 } // namespace prerender | 351 } // namespace prerender |
OLD | NEW |