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

Side by Side Diff: chrome/browser/net/url_info.cc

Issue 3032014: Support both preconnection, and pre-resolution for subresources... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: '' Created 10 years, 4 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 | Annotate | Revision Log
« no previous file with comments | « chrome/browser/net/url_info.h ('k') | chrome/browser/renderer_host/resource_dispatcher_host.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2006-2010 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2006-2010 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/net/url_info.h" 5 #include "chrome/browser/net/url_info.h"
6 6
7 #include <math.h> 7 #include <math.h>
8 8
9 #include <algorithm> 9 #include <algorithm>
10 #include <string> 10 #include <string>
(...skipping 113 matching lines...) Expand 10 before | Expand all | Expand 10 after
124 kMaxNonNetworkDnsLookupDuration, TimeDelta::FromMinutes(15), 100); 124 kMaxNonNetworkDnsLookupDuration, TimeDelta::FromMinutes(15), 100);
125 125
126 static bool use_ipv6_histogram(FieldTrialList::Find("IPv6_Probe") && 126 static bool use_ipv6_histogram(FieldTrialList::Find("IPv6_Probe") &&
127 !FieldTrialList::Find("IPv6_Probe")->group_name().empty()); 127 !FieldTrialList::Find("IPv6_Probe")->group_name().empty());
128 if (use_ipv6_histogram) { 128 if (use_ipv6_histogram) {
129 UMA_HISTOGRAM_CUSTOM_TIMES( 129 UMA_HISTOGRAM_CUSTOM_TIMES(
130 FieldTrial::MakeName("DNS.PrefetchResolution", "IPv6_Probe"), 130 FieldTrial::MakeName("DNS.PrefetchResolution", "IPv6_Probe"),
131 resolve_duration_, kMaxNonNetworkDnsLookupDuration, 131 resolve_duration_, kMaxNonNetworkDnsLookupDuration,
132 TimeDelta::FromMinutes(15), 100); 132 TimeDelta::FromMinutes(15), 100);
133 } 133 }
134
135 // Record potential beneficial time, and maybe we'll get a cache hit.
136 // We keep the maximum, as the warming we did earlier may still be
137 // helping with a cache upstream in DNS resolution.
138 benefits_remaining_ = std::max(resolve_duration_, benefits_remaining_);
139 } 134 }
140 sequence_number_ = sequence_counter++; 135 sequence_number_ = sequence_counter++;
141 DLogResultsStats("DNS PrefetchFound"); 136 DLogResultsStats("DNS PrefetchFound");
142 } 137 }
143 138
144 void UrlInfo::SetNoSuchNameState() { 139 void UrlInfo::SetNoSuchNameState() {
145 DCHECK(ASSIGNED == state_); 140 DCHECK(ASSIGNED == state_);
146 state_ = NO_SUCH_NAME; 141 state_ = NO_SUCH_NAME;
147 resolve_duration_ = GetDuration(); 142 resolve_duration_ = GetDuration();
148 if (kMaxNonNetworkDnsLookupDuration <= resolve_duration_) { 143 if (kMaxNonNetworkDnsLookupDuration <= resolve_duration_) {
149 DHISTOGRAM_TIMES("DNS.PrefetchNotFoundName", resolve_duration_); 144 DHISTOGRAM_TIMES("DNS.PrefetchNotFoundName", resolve_duration_);
150 // Record potential beneficial time, and maybe we'll get a cache hit.
151 benefits_remaining_ = std::max(resolve_duration_, benefits_remaining_);
152 } 145 }
153 sequence_number_ = sequence_counter++; 146 sequence_number_ = sequence_counter++;
154 DLogResultsStats("DNS PrefetchNotFound"); 147 DLogResultsStats("DNS PrefetchNotFound");
155 } 148 }
156 149
157 void UrlInfo::SetStartedState() {
158 DCHECK(PENDING == state_);
159 state_ = STARTED;
160 queue_duration_ = resolve_duration_ = TimeDelta(); // 0ms.
161 SetMotivation(NO_PREFETCH_MOTIVATION);
162 GetDuration(); // Set time.
163 }
164
165 void UrlInfo::SetFinishedState(bool was_resolved) {
166 DCHECK(STARTED == state_);
167 state_ = was_resolved ? FINISHED : FINISHED_UNRESOLVED;
168 resolve_duration_ = GetDuration();
169 // TODO(jar): Sequence number should be incremented in prefetched HostInfo.
170 DLogResultsStats("DNS HTTP Finished");
171 }
172
173 void UrlInfo::SetUrl(const GURL& url) { 150 void UrlInfo::SetUrl(const GURL& url) {
174 if (url_.is_empty()) // Not yet initialized. 151 if (url_.is_empty()) // Not yet initialized.
175 url_ = url; 152 url_ = url;
176 else 153 else
177 DCHECK_EQ(url_, url); 154 DCHECK_EQ(url_, url);
178 } 155 }
179 156
180 // IsStillCached() guesses if the DNS cache still has IP data, 157 // IsStillCached() guesses if the DNS cache still has IP data,
181 // or at least remembers results about "not finding host." 158 // or at least remembers results about "not finding host."
182 bool UrlInfo::IsStillCached() const { 159 bool UrlInfo::IsStillCached() const {
183 DCHECK(FOUND == state_ || NO_SUCH_NAME == state_); 160 DCHECK(FOUND == state_ || NO_SUCH_NAME == state_);
184 161
185 // Default MS OS does not cache failures. Hence we could return false almost 162 // Default MS OS does not cache failures. Hence we could return false almost
186 // all the time for that case. However, we'd never try again to prefetch 163 // all the time for that case. However, we'd never try again to prefetch
187 // the value if we returned false that way. Hence we'll just let the lookup 164 // the value if we returned false that way. Hence we'll just let the lookup
188 // time out the same way as FOUND case. 165 // time out the same way as FOUND case.
189 166
190 if (sequence_counter - sequence_number_ > kMaxGuaranteedDnsCacheSize) 167 if (sequence_counter - sequence_number_ > kMaxGuaranteedDnsCacheSize)
191 return false; 168 return false;
192 169
193 TimeDelta time_since_resolution = TimeTicks::Now() - time_; 170 TimeDelta time_since_resolution = TimeTicks::Now() - time_;
194 171
195 return time_since_resolution < kCacheExpirationDuration; 172 return time_since_resolution < kCacheExpirationDuration;
196 } 173 }
197 174
198 // Compare the actual navigation DNS latency found in navigation_info, to the
199 // previously prefetched info.
200 DnsBenefit UrlInfo::AccruePrefetchBenefits(UrlInfo* navigation_info) {
201 DCHECK(FINISHED == navigation_info->state_ ||
202 FINISHED_UNRESOLVED == navigation_info->state_);
203 DCHECK(navigation_info->url() == url_);
204
205 if ((0 == benefits_remaining_.InMilliseconds()) ||
206 (FOUND != state_ && NO_SUCH_NAME != state_)) {
207 if (FINISHED == navigation_info->state_)
208 UMA_HISTOGRAM_LONG_TIMES("DNS.IndependentNavigation",
209 navigation_info->resolve_duration_);
210 else
211 UMA_HISTOGRAM_LONG_TIMES("DNS.IndependentFailedNavigation",
212 navigation_info->resolve_duration_);
213 return PREFETCH_NO_BENEFIT;
214 }
215
216 TimeDelta benefit = benefits_remaining_ - navigation_info->resolve_duration_;
217 navigation_info->benefits_remaining_ = benefits_remaining_;
218 benefits_remaining_ = TimeDelta(); // We used up all our benefits here.
219
220 navigation_info->motivation_ = motivation_;
221 if (LEARNED_REFERAL_MOTIVATED == motivation_ ||
222 STATIC_REFERAL_MOTIVATED == motivation_)
223 navigation_info->referring_url_ = referring_url_;
224
225 if (navigation_info->resolve_duration_ > kMaxNonNetworkDnsLookupDuration) {
226 // Our precache effort didn't help since HTTP stack hit the network.
227 UMA_HISTOGRAM_LONG_TIMES("DNS.PrefetchCacheEvictionL", resolve_duration_);
228 DLogResultsStats("DNS PrefetchCacheEviction");
229 return PREFETCH_CACHE_EVICTION;
230 }
231
232 if (NO_SUCH_NAME == state_) {
233 UMA_HISTOGRAM_LONG_TIMES("DNS.PrefetchNegativeHitL", benefit);
234 DLogResultsStats("DNS PrefetchNegativeHit");
235 return PREFETCH_NAME_NONEXISTANT;
236 }
237
238 DCHECK_EQ(FOUND, state_);
239 if (LEARNED_REFERAL_MOTIVATED == motivation_ ||
240 STATIC_REFERAL_MOTIVATED == motivation_) {
241 UMA_HISTOGRAM_TIMES("DNS.PrefetchReferredPositiveHit", benefit);
242 DLogResultsStats("DNS PrefetchReferredPositiveHit");
243 } else {
244 UMA_HISTOGRAM_LONG_TIMES("DNS.PrefetchPositiveHitL", benefit);
245 DLogResultsStats("DNS PrefetchPositiveHit");
246 }
247 return PREFETCH_NAME_FOUND;
248 }
249
250 void UrlInfo::DLogResultsStats(const char* message) const { 175 void UrlInfo::DLogResultsStats(const char* message) const {
251 if (!detailed_logging_enabled) 176 if (!detailed_logging_enabled)
252 return; 177 return;
253 DLOG(INFO) << "\t" << message << "\tq=" 178 DLOG(INFO) << "\t" << message << "\tq="
254 << queue_duration().InMilliseconds() << "ms,\tr=" 179 << queue_duration().InMilliseconds() << "ms,\tr="
255 << resolve_duration().InMilliseconds() << "ms\tp=" 180 << resolve_duration().InMilliseconds() << "ms\tp="
256 << benefits_remaining_.InMilliseconds() << "ms\tseq="
257 << sequence_number_ 181 << sequence_number_
258 << "\t" << url_.spec(); 182 << "\t" << url_.spec();
259 } 183 }
260 184
261 //------------------------------------------------------------------------------ 185 //------------------------------------------------------------------------------
262 // This last section supports HTML output, such as seen in about:dns. 186 // This last section supports HTML output, such as seen in about:dns.
263 //------------------------------------------------------------------------------ 187 //------------------------------------------------------------------------------
264 188
265 // Preclude any possibility of Java Script or markup in the text, by only 189 // Preclude any possibility of Java Script or markup in the text, by only
266 // allowing alphanumerics, '.', '-', ':', and whitespace. 190 // allowing alphanumerics, '.', '-', ':', and whitespace.
(...skipping 55 matching lines...) Expand 10 before | Expand all | Expand 10 after
322 int print_hours = minutes/60; 246 int print_hours = minutes/60;
323 if (print_hours) 247 if (print_hours)
324 StringAppendF(&result, "%.2d:", print_hours); 248 StringAppendF(&result, "%.2d:", print_hours);
325 if (print_hours || print_minutes) 249 if (print_hours || print_minutes)
326 StringAppendF(&result, "%2.2d:", print_minutes); 250 StringAppendF(&result, "%2.2d:", print_minutes);
327 StringAppendF(&result, "%2.2d", print_seconds); 251 StringAppendF(&result, "%2.2d", print_seconds);
328 return result; 252 return result;
329 } 253 }
330 254
331 // static 255 // static
332 void UrlInfo::GetHtmlTable(const DnsInfoTable host_infos, 256 void UrlInfo::GetHtmlTable(const UrlInfoTable host_infos,
333 const char* description, 257 const char* description,
334 const bool brief, 258 const bool brief,
335 std::string* output) { 259 std::string* output) {
336 if (0 == host_infos.size()) 260 if (0 == host_infos.size())
337 return; 261 return;
338 output->append(description); 262 output->append(description);
339 StringAppendF(output, "%" PRIuS " %s", host_infos.size(), 263 StringAppendF(output, "%" PRIuS " %s", host_infos.size(),
340 (1 == host_infos.size()) ? "hostname" : "hostnames"); 264 (1 == host_infos.size()) ? "hostname" : "hostnames");
341 265
342 if (brief) { 266 if (brief) {
343 output->append("<br><br>"); 267 output->append("<br><br>");
344 return; 268 return;
345 } 269 }
346 270
347 const char* row_format = "<tr align=right><td>%s</td>" 271 output->append("<br><table border=1>"
348 "<td>%d</td><td>%d</td><td>%s</td><td>%s</td></tr>"; 272 "<tr><th>Host name</th>"
273 "<th>How long ago<br>(HH:MM:SS)</th>"
274 "<th>Motivation</th>"
275 "</tr>");
349 276
350 output->append("<br><table border=1>"); 277 const char* row_format = "<tr align=right><td>%s</td>" // Host name.
351 StringAppendF(output, 278 "<td>%s</td>" // How long ago.
352 "<tr><th>%s</th><th>%s</th><th>%s</th><th>%s</th><th>%s</th></tr>", 279 "<td>%s</td>" // Motivation.
353 "Host name", "Applicable Prefetch<br>Time (ms)", 280 "</tr>";
354 "Recent Resolution<br>Time(ms)", "How long ago<br>(HH:MM:SS)",
355 "Motivation");
356 281
357 // Print bulk of table, and gather stats at same time. 282 // Print bulk of table, and gather stats at same time.
358 MinMaxAverage queue, resolve, preresolve, when; 283 MinMaxAverage queue, when;
359 TimeTicks current_time = TimeTicks::Now(); 284 TimeTicks current_time = TimeTicks::Now();
360 for (DnsInfoTable::const_iterator it(host_infos.begin()); 285 for (UrlInfoTable::const_iterator it(host_infos.begin());
361 it != host_infos.end(); it++) { 286 it != host_infos.end(); it++) {
362 queue.sample((it->queue_duration_.InMilliseconds())); 287 queue.sample((it->queue_duration_.InMilliseconds()));
363 StringAppendF(output, row_format, 288 StringAppendF(output, row_format,
364 RemoveJs(it->url_.spec()).c_str(), 289 RemoveJs(it->url_.spec()).c_str(),
365 preresolve.sample((it->benefits_remaining_.InMilliseconds())),
366 resolve.sample((it->resolve_duration_.InMilliseconds())),
367 HoursMinutesSeconds(when.sample( 290 HoursMinutesSeconds(when.sample(
368 (current_time - it->time_).InSeconds())).c_str(), 291 (current_time - it->time_).InSeconds())).c_str(),
369 it->GetAsciiMotivation().c_str()); 292 it->GetAsciiMotivation().c_str());
370 } 293 }
371 // Write min, max, and average summary lines.
372 if (host_infos.size() > 2) {
373 output->append("<B>");
374 StringAppendF(output, row_format,
375 "<b>---minimum---</b>",
376 preresolve.minimum(), resolve.minimum(),
377 HoursMinutesSeconds(when.minimum()).c_str(), "");
378 StringAppendF(output, row_format,
379 "<b>---average---</b>",
380 preresolve.average(), resolve.average(),
381 HoursMinutesSeconds(when.average()).c_str(), "");
382 StringAppendF(output, row_format,
383 "<b>standard deviation</b>",
384 preresolve.standard_deviation(),
385 resolve.standard_deviation(), "n/a", "");
386 StringAppendF(output, row_format,
387 "<b>---maximum---</b>",
388 preresolve.maximum(), resolve.maximum(),
389 HoursMinutesSeconds(when.maximum()).c_str(), "");
390 StringAppendF(output, row_format,
391 "<b>-----SUM-----</b>",
392 preresolve.sum(), resolve.sum(), "n/a", "");
393 }
394 output->append("</table>"); 294 output->append("</table>");
395 295
396 #ifdef DEBUG 296 #ifdef DEBUG
397 StringAppendF(output, 297 StringAppendF(output,
398 "Prefetch Queue Durations: min=%d, avg=%d, max=%d<br><br>", 298 "Prefetch Queue Durations: min=%d, avg=%d, max=%d<br><br>",
399 queue.minimum(), queue.average(), queue.maximum()); 299 queue.minimum(), queue.average(), queue.maximum());
400 #endif 300 #endif
401 301
402 output->append("<br>"); 302 output->append("<br>");
403 } 303 }
(...skipping 26 matching lines...) Expand all
430 330
431 case LEARNED_REFERAL_MOTIVATED: 331 case LEARNED_REFERAL_MOTIVATED:
432 return RemoveJs(referring_url_.spec()); 332 return RemoveJs(referring_url_.spec());
433 333
434 default: 334 default:
435 return ""; 335 return "";
436 } 336 }
437 } 337 }
438 338
439 } // namespace chrome_browser_net 339 } // namespace chrome_browser_net
OLDNEW
« no previous file with comments | « chrome/browser/net/url_info.h ('k') | chrome/browser/renderer_host/resource_dispatcher_host.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698