1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome/renderer/prerender/prerender_dispatcher.h"
7 #include "base/logging.h"
8 #include "chrome/common/prerender_messages.h"
9 #include "chrome/renderer/prerender/prerender_extra_data.h"
10 #include "content/public/common/referrer.h"
11 #include "content/public/renderer/render_thread.h"
12 #include "content/public/renderer/render_view.h"
13 #include "third_party/WebKit/public/platform/WebPrerenderingSupport.h"
14 #include "third_party/WebKit/public/platform/WebString.h"
15 #include "third_party/WebKit/public/platform/WebURL.h"
20 using WebKit::WebPrerender;
21 using WebKit::WebPrerenderingSupport;
23 PrerenderDispatcher::PrerenderDispatcher() {
24 WebPrerenderingSupport::initialize(this);
27 PrerenderDispatcher::~PrerenderDispatcher() {
28 WebPrerenderingSupport::shutdown();
31 bool PrerenderDispatcher::IsPrerenderURL(const GURL& url) const {
32 return running_prerender_urls_.count(url) >= 1;
35 void PrerenderDispatcher::OnPrerenderStart(int prerender_id) {
36 std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id);
37 if (it == prerenders_.end())
40 WebPrerender& prerender = it->second;
42 // The prerender should only be null in unit tests.
43 if (prerender.isNull())
46 prerender.didStartPrerender();
49 void PrerenderDispatcher::OnPrerenderStopLoading(int prerender_id) {
50 std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id);
51 if (it == prerenders_.end())
54 WebPrerender& prerender = it->second;
55 DCHECK(!prerender.isNull())
56 << "OnPrerenderStopLoading shouldn't be called from a unit test, the only"
57 << "context in which a WebPrerender in the dispatcher can be null.";
59 prerender.didSendLoadForPrerender();
62 void PrerenderDispatcher::OnPrerenderAddAlias(const GURL& alias) {
63 running_prerender_urls_.insert(alias);
66 void PrerenderDispatcher::OnPrerenderRemoveAliases(
67 const std::vector<GURL>& aliases) {
68 for (size_t i = 0; i < aliases.size(); ++i) {
69 std::multiset<GURL>::iterator it = running_prerender_urls_.find(aliases[i]);
70 if (it != running_prerender_urls_.end()) {
71 running_prerender_urls_.erase(it);
76 void PrerenderDispatcher::OnPrerenderStop(int prerender_id) {
77 std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id);
78 if (it == prerenders_.end())
80 WebPrerender& prerender = it->second;
82 // The prerender should only be null in unit tests.
83 if (!prerender.isNull())
84 prerender.didStopPrerender();
86 // TODO(cbentzel): We'd also want to send the map of active prerenders when
87 // creating a new render process, so the Add/Remove go relative to that.
88 // This may not be that big of a deal in practice, since the newly created tab
89 // is unlikely to go to the prerendered page.
90 prerenders_.erase(prerender_id);
93 bool PrerenderDispatcher::OnControlMessageReceived(
94 const IPC::Message& message) {
96 IPC_BEGIN_MESSAGE_MAP(PrerenderDispatcher, message)
97 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStart, OnPrerenderStart)
98 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStopLoading,
99 OnPrerenderStopLoading)
100 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderAddAlias, OnPrerenderAddAlias)
101 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderRemoveAliases,
102 OnPrerenderRemoveAliases)
103 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStop, OnPrerenderStop)
104 IPC_MESSAGE_UNHANDLED(handled = false)
105 IPC_END_MESSAGE_MAP()
110 void PrerenderDispatcher::add(const WebPrerender& prerender) {
111 const PrerenderExtraData& extra_data =
112 PrerenderExtraData::FromPrerender(prerender);
113 if (prerenders_.count(extra_data.prerender_id()) != 0) {
114 // TODO(gavinp): Determine why these apparently duplicate adds occur.
118 prerenders_[extra_data.prerender_id()] = prerender;
120 content::RenderThread::Get()->Send(new PrerenderHostMsg_AddLinkRelPrerender(
121 extra_data.prerender_id(), GURL(prerender.url()),
122 content::Referrer(GURL(prerender.referrer()),
123 prerender.referrerPolicy()),
124 extra_data.size(), extra_data.render_view_route_id()));
127 void PrerenderDispatcher::cancel(const WebPrerender& prerender) {
128 const PrerenderExtraData& extra_data =
129 PrerenderExtraData::FromPrerender(prerender);
130 content::RenderThread::Get()->Send(
131 new PrerenderHostMsg_CancelLinkRelPrerender(extra_data.prerender_id()));
132 // The browser will not send an OnPrerenderStop (the prerender may have even
133 // been canceled before it was started), so release it to avoid a
134 // leak. Moreover, if it did, the PrerenderClient in Blink will have been
136 prerenders_.erase(extra_data.prerender_id());
139 void PrerenderDispatcher::abandon(const WebPrerender& prerender) {
140 const PrerenderExtraData& extra_data =
141 PrerenderExtraData::FromPrerender(prerender);
142 content::RenderThread::Get()->Send(
143 new PrerenderHostMsg_AbandonLinkRelPrerender(extra_data.prerender_id()));
144 // The browser will not send an OnPrerenderStop (the prerender may have even
145 // been canceled before it was started), so release it to avoid a
146 // leak. Moreover, if it did, the PrerenderClient in Blink will have been
148 prerenders_.erase(extra_data.prerender_id());
151 } // namespace prerender