Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 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/renderer/prerender/prerender_dispatcher.h" | 5 #include "chrome/renderer/prerender/prerender_dispatcher.h" |
| 6 | 6 |
| 7 #include <stack> | |
| 8 | |
| 7 #include "base/logging.h" | 9 #include "base/logging.h" |
| 8 #include "chrome/common/prerender_messages.h" | 10 #include "chrome/common/prerender_messages.h" |
| 9 #include "chrome/renderer/prerender/prerender_extra_data.h" | 11 #include "chrome/renderer/prerender/prerender_extra_data.h" |
| 12 #include "content/public/common/referrer.h" | |
| 10 #include "content/public/renderer/render_thread.h" | 13 #include "content/public/renderer/render_thread.h" |
| 14 #include "content/public/renderer/render_view.h" | |
| 11 #include "googleurl/src/gurl.h" | 15 #include "googleurl/src/gurl.h" |
| 12 #include "third_party/WebKit/Source/Platform/chromium/public/WebPrerenderingSupp ort.h" | 16 #include "third_party/WebKit/Source/Platform/chromium/public/WebPrerenderingSupp ort.h" |
| 17 #include "third_party/WebKit/Source/Platform/chromium/public/WebString.h" | |
| 18 #include "third_party/WebKit/Source/Platform/chromium/public/WebURL.h" | |
| 13 | 19 |
| 14 namespace prerender { | 20 namespace prerender { |
| 15 | 21 |
| 22 using WebKit::WebPrerender; | |
| 23 using WebKit::WebPrerenderingSupport; | |
| 24 | |
| 16 PrerenderDispatcher::PrerenderDispatcher() { | 25 PrerenderDispatcher::PrerenderDispatcher() { |
| 17 WebKit::WebPrerenderingSupport::initialize(this); | 26 WebPrerenderingSupport::initialize(this); |
| 18 } | 27 } |
| 19 | 28 |
| 20 PrerenderDispatcher::~PrerenderDispatcher() { | 29 PrerenderDispatcher::~PrerenderDispatcher() { |
| 21 WebKit::WebPrerenderingSupport::shutdown(); | 30 WebPrerenderingSupport::shutdown(); |
| 22 } | 31 } |
| 23 | 32 |
| 24 bool PrerenderDispatcher::IsPrerenderURL(const GURL& url) const { | 33 bool PrerenderDispatcher::IsPrerenderURL(const GURL& url) const { |
| 25 return prerender_urls_.find(url) != prerender_urls_.end(); | 34 return running_prerender_urls_.count(url) >= 1; |
| 26 } | 35 } |
| 27 | 36 |
| 28 void PrerenderDispatcher::OnAddPrerenderURL(const GURL& url) { | 37 void PrerenderDispatcher::OnPrerenderStart(int prerender_id) { |
| 29 PrerenderMap::iterator it = prerender_urls_.find(url); | 38 std::map<int, GURL>::iterator it = prerenders_.find(prerender_id); |
| 30 if (it != prerender_urls_.end()) { | 39 if (it == prerenders_.end()) |
| 31 DCHECK_LT(0, it->second); | 40 return; |
| 32 it->second += 1; | 41 const GURL& url = it->second; |
| 33 } else { | 42 running_prerender_urls_.insert( |
| 34 prerender_urls_[url] = 1; | 43 std::multimap<GURL, int>::value_type(url, prerender_id)); |
| 35 } | |
| 36 } | 44 } |
| 37 | 45 |
| 38 void PrerenderDispatcher::OnRemovePrerenderURL(const GURL& url) { | 46 void PrerenderDispatcher::OnPrerenderAddAlias(int prerender_id, |
| 39 PrerenderMap::iterator it = prerender_urls_.find(url); | 47 const GURL& url) { |
| 40 // This is possible with a spurious remove. | 48 DCHECK_LT(0u, prerenders_.count(prerender_id)); |
| 49 running_prerender_urls_.insert( | |
| 50 std::multimap<GURL, int>::value_type(url, prerender_id)); | |
| 51 } | |
| 52 | |
| 53 void PrerenderDispatcher::OnPrerenderStop(int prerender_id) { | |
| 54 // It's possible we will receive a spurious remove, so we don't crash in that | |
| 55 // situation. | |
|
mmenke
2012/12/12 16:54:12
Is this comment accurate? We only delete prerende
gavinp
2012/12/13 13:38:03
Done.
| |
| 56 if (prerenders_.count(prerender_id) == 0) | |
| 57 return; | |
| 58 | |
| 41 // TODO(cbentzel): We'd also want to send the map of active prerenders when | 59 // TODO(cbentzel): We'd also want to send the map of active prerenders when |
| 42 // creating a new render process, so the Add/Remove go relative to that. | 60 // creating a new render process, so the Add/Remove go relative to that. |
| 43 // This may not be that big of a deal in practice, since the newly created tab | 61 // This may not be that big of a deal in practice, since the newly created tab |
| 44 // is unlikely to go to the prerendered page. | 62 // is unlikely to go to the prerendered page. |
| 45 if (it == prerender_urls_.end()) | 63 prerenders_.erase(prerender_id); |
| 46 return; | 64 |
| 47 DCHECK_LT(0, it->second); | 65 std::stack<std::multimap<GURL, int>::iterator> to_delete; |
| 48 it->second -= 1; | 66 for (std::multimap<GURL, int>::iterator it = running_prerender_urls_.begin(); |
| 49 if (it->second == 0) | 67 it != running_prerender_urls_.end(); ++it) { |
| 50 prerender_urls_.erase(it); | 68 if (it->second == prerender_id) |
| 69 to_delete.push(it); | |
| 70 } | |
| 71 while (!to_delete.empty()) { | |
| 72 running_prerender_urls_.erase(to_delete.top()); | |
| 73 to_delete.pop(); | |
| 74 } | |
| 51 } | 75 } |
| 52 | 76 |
| 53 bool PrerenderDispatcher::OnControlMessageReceived( | 77 bool PrerenderDispatcher::OnControlMessageReceived( |
| 54 const IPC::Message& message) { | 78 const IPC::Message& message) { |
| 55 bool handled = true; | 79 bool handled = true; |
| 56 IPC_BEGIN_MESSAGE_MAP(PrerenderDispatcher, message) | 80 IPC_BEGIN_MESSAGE_MAP(PrerenderDispatcher, message) |
| 57 IPC_MESSAGE_HANDLER(PrerenderMsg_AddPrerenderURL, OnAddPrerenderURL) | 81 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStart, OnPrerenderStart) |
| 58 IPC_MESSAGE_HANDLER(PrerenderMsg_RemovePrerenderURL, OnRemovePrerenderURL) | 82 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderAddAlias, OnPrerenderAddAlias) |
| 83 IPC_MESSAGE_HANDLER(PrerenderMsg_OnPrerenderStop, OnPrerenderStop) | |
| 59 IPC_MESSAGE_UNHANDLED(handled = false) | 84 IPC_MESSAGE_UNHANDLED(handled = false) |
| 60 IPC_END_MESSAGE_MAP() | 85 IPC_END_MESSAGE_MAP() |
| 61 | 86 |
| 62 return handled; | 87 return handled; |
| 63 } | 88 } |
| 64 | 89 |
| 65 void PrerenderDispatcher::add(const WebKit::WebPrerender& prerender) { | 90 void PrerenderDispatcher::add(const WebKit::WebPrerender& prerender) { |
| 66 const PrerenderExtraData& extra_data = | 91 const PrerenderExtraData& extra_data = |
| 67 PrerenderExtraData::FromPrerender(prerender); | 92 PrerenderExtraData::FromPrerender(prerender); |
| 93 prerenders_.insert(std::map<int, GURL>::value_type(extra_data.prerender_id(), | |
| 94 prerender.url())); | |
| 95 | |
| 68 content::RenderThread::Get()->Send(new PrerenderHostMsg_AddLinkRelPrerender( | 96 content::RenderThread::Get()->Send(new PrerenderHostMsg_AddLinkRelPrerender( |
| 69 extra_data.prerender_id(), GURL(prerender.url()), | 97 extra_data.prerender_id(), GURL(prerender.url()), |
| 70 content::Referrer(GURL(prerender.referrer()), prerender.referrerPolicy()), | 98 content::Referrer(GURL(prerender.referrer()), prerender.referrerPolicy()), |
| 71 extra_data.size(), extra_data.render_view_route_id())); | 99 extra_data.size(), extra_data.render_view_route_id())); |
|
mmenke
2012/12/12 16:54:12
For future consideration: Another leak case: If
gavinp
2012/12/13 13:38:03
Good point. I think it'd be good if we tightened u
| |
| 72 } | 100 } |
| 73 | 101 |
| 74 void PrerenderDispatcher::cancel(const WebKit::WebPrerender& prerender) { | 102 void PrerenderDispatcher::cancel(const WebKit::WebPrerender& prerender) { |
| 75 const PrerenderExtraData& extra_data = | 103 const PrerenderExtraData& extra_data = |
| 76 PrerenderExtraData::FromPrerender(prerender); | 104 PrerenderExtraData::FromPrerender(prerender); |
| 77 content::RenderThread::Get()->Send( | 105 content::RenderThread::Get()->Send( |
| 78 new PrerenderHostMsg_CancelLinkRelPrerender(extra_data.prerender_id())); | 106 new PrerenderHostMsg_CancelLinkRelPrerender(extra_data.prerender_id())); |
| 79 } | 107 } |
| 80 | 108 |
| 81 void PrerenderDispatcher::abandon(const WebKit::WebPrerender& prerender) { | 109 void PrerenderDispatcher::abandon(const WebKit::WebPrerender& prerender) { |
| 82 const PrerenderExtraData& extra_data = | 110 const PrerenderExtraData& extra_data = |
| 83 PrerenderExtraData::FromPrerender(prerender); | 111 PrerenderExtraData::FromPrerender(prerender); |
| 84 content::RenderThread::Get()->Send( | 112 content::RenderThread::Get()->Send( |
| 85 new PrerenderHostMsg_AbandonLinkRelPrerender(extra_data.prerender_id())); | 113 new PrerenderHostMsg_AbandonLinkRelPrerender(extra_data.prerender_id())); |
| 86 } | 114 } |
| 87 | 115 |
| 88 } // namespace prerender | 116 } // namespace prerender |
| OLD | NEW |