// 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::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::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::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& aliases) { for (size_t i = 0; i < aliases.size(); ++i) { std::multiset::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::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