| // Copyright (c) 2012 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/renderer/prerender/prerender_dispatcher.h" |
| |
| #include "base/logging.h" |
| #include "chrome/common/prerender_messages.h" |
| #include "chrome/common/prerender_types.h" |
| #include "chrome/renderer/prerender/prerender_extra_data.h" |
| #include "content/public/common/referrer.h" |
| #include "content/public/renderer/render_thread.h" |
| #include "content/public/renderer/render_view.h" |
| #include "third_party/WebKit/public/platform/WebPrerenderingSupport.h" |
| #include "third_party/WebKit/public/platform/WebString.h" |
| #include "third_party/WebKit/public/platform/WebURL.h" |
| #include "url/gurl.h" |
| |
| namespace prerender { |
| |
| using blink::WebPrerender; |
| using blink::WebPrerenderingSupport; |
| |
| PrerenderDispatcher::PrerenderDispatcher() { |
| WebPrerenderingSupport::initialize(this); |
| } |
| |
| PrerenderDispatcher::~PrerenderDispatcher() { |
| WebPrerenderingSupport::shutdown(); |
| } |
| |
| bool PrerenderDispatcher::IsPrerenderURL(const GURL& url) const { |
| return running_prerender_urls_.count(url) >= 1; |
| } |
| |
| void PrerenderDispatcher::OnPrerenderStart(int prerender_id) { |
| std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id); |
| if (it == prerenders_.end()) |
| return; |
| |
| WebPrerender& prerender = it->second; |
| |
| // The prerender should only be null in unit tests. |
| if (prerender.isNull()) |
| return; |
| |
| prerender.didStartPrerender(); |
| } |
| |
| void PrerenderDispatcher::OnPrerenderStopLoading(int prerender_id) { |
| std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id); |
| if (it == prerenders_.end()) |
| return; |
| |
| WebPrerender& prerender = it->second; |
| DCHECK(!prerender.isNull()) |
| << "OnPrerenderStopLoading shouldn't be called from a unit test, the only" |
| << "context in which a WebPrerender in the dispatcher can be null."; |
| |
| prerender.didSendLoadForPrerender(); |
| } |
| |
| void PrerenderDispatcher::OnPrerenderDomContentLoaded(int prerender_id) { |
| std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id); |
| if (it == prerenders_.end()) |
| return; |
| |
| WebPrerender& prerender = it->second; |
| DCHECK(!prerender.isNull()) |
| << "OnPrerenderDomContentLoaded shouldn't be called from a unit test," |
| << " the only context in which a WebPrerender in the dispatcher can be" |
| << " null."; |
| |
| prerender.didSendDOMContentLoadedForPrerender(); |
| } |
| |
| void PrerenderDispatcher::OnPrerenderAddAlias(const GURL& alias) { |
| running_prerender_urls_.insert(alias); |
| } |
| |
| void PrerenderDispatcher::OnPrerenderRemoveAliases( |
| const std::vector<GURL>& aliases) { |
| for (size_t i = 0; i < aliases.size(); ++i) { |
| std::multiset<GURL>::iterator it = running_prerender_urls_.find(aliases[i]); |
| if (it != running_prerender_urls_.end()) { |
| running_prerender_urls_.erase(it); |
| } |
| } |
| } |
| |
| void PrerenderDispatcher::OnPrerenderStop(int prerender_id) { |
| std::map<int, WebPrerender>::iterator it = prerenders_.find(prerender_id); |
| if (it == prerenders_.end()) |
| return; |
| WebPrerender& prerender = it->second; |
| |
| // The prerender should only be null in unit tests. |
| if (!prerender.isNull()) |
| prerender.didStopPrerender(); |
| |
| // TODO(cbentzel): We'd also want to send the map of active prerenders when |
| // creating a new render process, so the Add/Remove go relative to that. |
| // This may not be that big of a deal in practice, since the newly created tab |
| // is unlikely to go to the prerendered page. |
| prerenders_.erase(prerender_id); |
| } |
| |
| bool PrerenderDispatcher::OnControlMessageReceived( |
| const IPC::Message& message) { |
| bool handled = true; |
| IPC_BEGIN_MESSAGE_MAP(PrerenderDispatcher, message) |
| IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStart, OnPrerenderStart) |
| IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStopLoading, |
| OnPrerenderStopLoading) |
| IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderDomContentLoaded, |
| OnPrerenderDomContentLoaded) |
| IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderAddAlias, OnPrerenderAddAlias) |
| IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderRemoveAliases, |
| OnPrerenderRemoveAliases) |
| IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStop, OnPrerenderStop) |
| IPC_MESSAGE_UNHANDLED(handled = false) |
| IPC_END_MESSAGE_MAP() |
| |
| return handled; |
| } |
| |
| void PrerenderDispatcher::add(const WebPrerender& prerender) { |
| const PrerenderExtraData& extra_data = |
| PrerenderExtraData::FromPrerender(prerender); |
| if (prerenders_.count(extra_data.prerender_id()) != 0) { |
| // TODO(gavinp): Determine why these apparently duplicate adds occur. |
| return; |
| } |
| |
| prerenders_[extra_data.prerender_id()] = prerender; |
| |
| PrerenderAttributes attributes; |
| attributes.url = GURL(prerender.url()); |
| attributes.rel_types = prerender.relTypes(); |
| |
| content::RenderThread::Get()->Send(new PrerenderHostMsg_AddLinkRelPrerender( |
| extra_data.prerender_id(), attributes, |
| content::Referrer(GURL(prerender.referrer()), |
| prerender.referrerPolicy()), |
| extra_data.size(), extra_data.render_view_route_id())); |
| } |
| |
| void PrerenderDispatcher::cancel(const WebPrerender& prerender) { |
| const PrerenderExtraData& extra_data = |
| PrerenderExtraData::FromPrerender(prerender); |
| content::RenderThread::Get()->Send( |
| new PrerenderHostMsg_CancelLinkRelPrerender(extra_data.prerender_id())); |
| // The browser will not send an OnPrerenderStop (the prerender may have even |
| // been canceled before it was started), so release it to avoid a |
| // leak. Moreover, if it did, the PrerenderClient in Blink will have been |
| // detached already. |
| prerenders_.erase(extra_data.prerender_id()); |
| } |
| |
| void PrerenderDispatcher::abandon(const WebPrerender& prerender) { |
| const PrerenderExtraData& extra_data = |
| PrerenderExtraData::FromPrerender(prerender); |
| content::RenderThread::Get()->Send( |
| new PrerenderHostMsg_AbandonLinkRelPrerender(extra_data.prerender_id())); |
| // The browser will not send an OnPrerenderStop (the prerender may have even |
| // been canceled before it was started), so release it to avoid a |
| // leak. Moreover, if it did, the PrerenderClient in Blink will have been |
| // detached already. |
| prerenders_.erase(extra_data.prerender_id()); |
| } |
| |
| } // namespace prerender |