1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
// Copyright (c) 2010 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/net/preconnect.h"
#include "base/logging.h"
#include "base/metrics/histogram.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/browser_thread.h"
#include "chrome/common/net/url_request_context_getter.h"
#include "net/http/http_network_session.h"
#include "net/http/http_transaction_factory.h"
#include "net/url_request/url_request_context.h"
namespace chrome_browser_net {
// static
void Preconnect::PreconnectOnUIThread(const GURL& url,
UrlInfo::ResolutionMotivation motivation, int count) {
// Prewarm connection to Search URL.
BrowserThread::PostTask(
BrowserThread::IO,
FROM_HERE,
NewRunnableFunction(Preconnect::PreconnectOnIOThread, url, motivation,
count));
return;
}
// static
void Preconnect::PreconnectOnIOThread(const GURL& url,
UrlInfo::ResolutionMotivation motivation, int count) {
Preconnect* preconnect = new Preconnect(motivation);
// TODO(jar): Should I use PostTask for LearnedSubresources to delay the
// preconnection a tad?
preconnect->Connect(url, count);
}
Preconnect::Preconnect(UrlInfo::ResolutionMotivation motivation)
: motivation_(motivation),
ALLOW_THIS_IN_INITIALIZER_LIST(
io_callback_(this, &Preconnect::OnPreconnectComplete)) {}
Preconnect::~Preconnect() {}
void Preconnect::Connect(const GURL& url, int count) {
URLRequestContextGetter* getter = Profile::GetDefaultRequestContext();
if (!getter)
return;
if (!BrowserThread::CurrentlyOn(BrowserThread::IO)) {
LOG(DFATAL) << "This must be run only on the IO thread.";
return;
}
// We are now commited to doing the async preconnection call.
UMA_HISTOGRAM_ENUMERATION("Net.PreconnectMotivation", motivation_,
UrlInfo::MAX_MOTIVATED);
net::URLRequestContext* context = getter->GetURLRequestContext();
net::HttpTransactionFactory* factory = context->http_transaction_factory();
net::HttpNetworkSession* session = factory->GetSession();
request_info_.reset(new net::HttpRequestInfo());
request_info_->url = url;
request_info_->method = "GET";
// It almost doesn't matter whether we use net::LOWEST or net::HIGHEST
// priority here, as we won't make a request, and will surrender the created
// socket to the pool as soon as we can. However, we would like to mark the
// speculative socket as such, and IF we use a net::LOWEST priority, and if
// a navigation asked for a socket (after us) then it would get our socket,
// and we'd get its later-arriving socket, which might make us record that
// the speculation didn't help :-/. By using net::HIGHEST, we ensure that
// a socket is given to us if "we asked first" and this allows us to mark it
// as speculative, and better detect stats (if it gets used).
// TODO(jar): histogram to see how often we accidentally use a previously-
// unused socket, when a previously used socket was available.
request_info_->priority = net::HIGHEST;
// Translate the motivation from UrlRequest motivations to HttpRequest
// motivations.
switch (motivation_) {
case UrlInfo::OMNIBOX_MOTIVATED:
request_info_->motivation = net::HttpRequestInfo::OMNIBOX_MOTIVATED;
break;
case UrlInfo::LEARNED_REFERAL_MOTIVATED:
request_info_->motivation = net::HttpRequestInfo::PRECONNECT_MOTIVATED;
break;
case UrlInfo::EARLY_LOAD_MOTIVATED:
request_info_->motivation = net::HttpRequestInfo::EARLY_LOAD_MOTIVATED;
break;
default:
// Other motivations should never happen here.
NOTREACHED();
break;
}
// Setup the SSL Configuration.
ssl_config_.reset(new net::SSLConfig());
session->ssl_config_service()->GetSSLConfig(ssl_config_.get());
if (session->http_stream_factory()->next_protos())
ssl_config_->next_protos = *session->http_stream_factory()->next_protos();
// All preconnects should perform EV certificate verification.
ssl_config_->verify_ev_cert = true;
proxy_info_.reset(new net::ProxyInfo());
net::StreamFactory* stream_factory = session->http_stream_factory();
int rv = stream_factory->PreconnectStreams(count, request_info_.get(),
ssl_config_.get(),
proxy_info_.get(), session,
net_log_, &io_callback_);
if (rv != net::ERR_IO_PENDING)
delete this;
}
void Preconnect::OnPreconnectComplete(int error_code) {
delete this;
}
} // namespace chrome_browser_net
|