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/browser/prerender/prerender_link_manager.h" | 5 #include "chrome/browser/prerender/prerender_link_manager.h" |
| 6 | 6 |
| 7 #include <limits> | 7 #include <limits> |
| 8 #include <queue> | 8 #include <queue> |
| 9 #include <utility> | 9 #include <utility> |
| 10 | 10 |
| 11 #include "base/memory/scoped_ptr.h" | |
| 11 #include "chrome/browser/prerender/prerender_contents.h" | 12 #include "chrome/browser/prerender/prerender_contents.h" |
| 12 #include "chrome/browser/prerender/prerender_handle.h" | 13 #include "chrome/browser/prerender/prerender_handle.h" |
| 13 #include "chrome/browser/prerender/prerender_manager.h" | 14 #include "chrome/browser/prerender/prerender_manager.h" |
| 14 #include "chrome/browser/prerender/prerender_manager_factory.h" | 15 #include "chrome/browser/prerender/prerender_manager_factory.h" |
| 15 #include "chrome/browser/profiles/profile.h" | 16 #include "chrome/browser/profiles/profile.h" |
| 17 #include "chrome/common/prerender_messages.h" | |
| 18 #include "content/public/browser/render_process_host.h" | |
| 16 #include "content/public/browser/render_view_host.h" | 19 #include "content/public/browser/render_view_host.h" |
| 17 #include "content/public/browser/session_storage_namespace.h" | 20 #include "content/public/browser/session_storage_namespace.h" |
| 18 #include "content/public/common/referrer.h" | 21 #include "content/public/common/referrer.h" |
| 19 #include "googleurl/src/gurl.h" | 22 #include "googleurl/src/gurl.h" |
| 20 #include "ui/gfx/size.h" | 23 #include "ui/gfx/size.h" |
| 21 | 24 |
| 22 using content::RenderViewHost; | 25 using content::RenderViewHost; |
| 23 using content::SessionStorageNamespace; | 26 using content::SessionStorageNamespace; |
| 24 | 27 |
| 28 namespace { | |
| 29 | |
| 30 void Send(int child_id, IPC::Message* raw_message) { | |
| 31 using content::RenderProcessHost; | |
| 32 scoped_ptr<IPC::Message> own_message(raw_message); | |
| 33 | |
| 34 RenderProcessHost* render_process_host = RenderProcessHost::FromID(child_id); | |
| 35 if (!render_process_host) | |
| 36 return; | |
| 37 render_process_host->Send(own_message.release()); | |
| 38 } | |
| 39 | |
| 40 } // namespace | |
| 41 | |
| 25 namespace prerender { | 42 namespace prerender { |
| 26 | 43 |
| 27 PrerenderLinkManager::PrerenderLinkManager(PrerenderManager* manager) | 44 PrerenderLinkManager::PrerenderLinkManager(PrerenderManager* manager) |
| 28 : manager_(manager) { | 45 : manager_(manager) { |
| 29 } | 46 } |
| 30 | 47 |
| 31 PrerenderLinkManager::~PrerenderLinkManager() { | 48 PrerenderLinkManager::~PrerenderLinkManager() { |
| 32 for (IdPairToPrerenderHandleMap::iterator it = ids_to_handle_map_.begin(); | 49 for (IdPairToPrerenderHandleMap::iterator it = ids_to_handle_map_.begin(); |
| 33 it != ids_to_handle_map_.end(); | 50 it != ids_to_handle_map_.end(); |
| 34 ++it) { | 51 ++it) { |
| 35 PrerenderHandle* prerender_handle = it->second; | 52 PrerenderHandle* prerender_handle = it->second; |
| 36 prerender_handle->OnCancel(); | 53 DCHECK(!prerender_handle->IsPrerendering()) |
| 54 << "All running prerenders should stop at the same time as the " | |
| 55 << "PrerenderManager."; | |
| 37 delete prerender_handle; | 56 delete prerender_handle; |
| 38 } | 57 } |
| 39 } | 58 } |
| 40 | 59 |
| 41 bool PrerenderLinkManager::OnAddPrerender(int child_id, | 60 bool PrerenderLinkManager::OnAddPrerender(int child_id, |
| 42 int prerender_id, | 61 int prerender_id, |
| 43 const GURL& url, | 62 const GURL& url, |
| 44 const content::Referrer& referrer, | 63 const content::Referrer& referrer, |
| 45 const gfx::Size& size, | 64 const gfx::Size& size, |
| 46 int render_view_route_id) { | 65 int render_view_route_id) { |
| 47 DVLOG(2) << "OnAddPrerender, child_id = " << child_id | 66 DVLOG(2) << "OnAddPrerender, child_id = " << child_id |
| 48 << ", prerender_id = " << prerender_id | 67 << ", prerender_id = " << prerender_id |
| 49 << ", url = " << url.spec(); | 68 << ", url = " << url.spec(); |
| 50 DVLOG(3) << "... referrer url = " << referrer.url.spec() | 69 DVLOG(3) << "... referrer url = " << referrer.url.spec() |
| 51 << ", size = (" << size.width() << ", " << size.height() << ")" | 70 << ", size = (" << size.width() << ", " << size.height() << ")" |
| 52 << ", render_view_route_id = " << render_view_route_id; | 71 << ", render_view_route_id = " << render_view_route_id; |
| 53 | 72 |
| 73 | |
| 74 PrerenderHandle* prerender_handle = | |
| 75 manager_->AddPrerenderFromLinkRelPrerender( | |
| 76 child_id, render_view_route_id, url, referrer, size); | |
| 77 if (!prerender_handle) | |
| 78 return false; | |
| 79 | |
| 54 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | 80 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); |
| 55 DCHECK_EQ(0U, ids_to_handle_map_.count(child_and_prerender_id)); | 81 DCHECK_EQ(0u, ids_to_handle_map_.count(child_and_prerender_id)); |
| 82 ids_to_handle_map_[child_and_prerender_id] = prerender_handle; | |
| 56 | 83 |
| 57 scoped_ptr<PrerenderHandle> prerender_handle( | 84 // If we are given a prerender that is already prerendering, we have missed |
| 58 manager_->AddPrerenderFromLinkRelPrerender( | 85 // the start event. |
| 59 child_id, render_view_route_id, url, referrer, size)); | 86 if (prerender_handle->IsPrerendering()) |
| 60 if (prerender_handle.get()) { | 87 OnPrerenderStart(prerender_handle); |
| 61 std::pair<IdPairToPrerenderHandleMap::iterator, bool> insert_result = | 88 prerender_handle->SetObserver(this); |
| 62 ids_to_handle_map_.insert(std::make_pair( | 89 return true; |
| 63 child_and_prerender_id, static_cast<PrerenderHandle*>(NULL))); | |
| 64 DCHECK(insert_result.second); | |
| 65 delete insert_result.first->second; | |
| 66 insert_result.first->second = prerender_handle.release(); | |
| 67 return true; | |
| 68 } | |
| 69 return false; | |
| 70 } | 90 } |
| 71 | 91 |
| 72 // TODO(gavinp): Once an observer interface is provided down to the WebKit | |
| 73 // layer, we should add DCHECK_NE(0L, ids_to_url_map_.count(...)) to both | |
| 74 // OnCancelPrerender and OnAbandonPrerender. We can't do this now, since | |
| 75 // the WebKit layer isn't even aware if we didn't add the prerender to the map | |
| 76 // in OnAddPrerender above. | |
| 77 void PrerenderLinkManager::OnCancelPrerender(int child_id, int prerender_id) { | 92 void PrerenderLinkManager::OnCancelPrerender(int child_id, int prerender_id) { |
| 78 DVLOG(2) << "OnCancelPrerender, child_id = " << child_id | 93 DVLOG(2) << "OnCancelPrerender, child_id = " << child_id |
| 79 << ", prerender_id = " << prerender_id; | 94 << ", prerender_id = " << prerender_id; |
| 80 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | 95 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); |
| 81 IdPairToPrerenderHandleMap::iterator id_to_handle_iter = | 96 IdPairToPrerenderHandleMap::iterator id_to_handle_iter = |
| 82 ids_to_handle_map_.find(child_and_prerender_id); | 97 ids_to_handle_map_.find(child_and_prerender_id); |
| 83 if (id_to_handle_iter == ids_to_handle_map_.end()) { | 98 if (id_to_handle_iter == ids_to_handle_map_.end()) { |
| 84 DVLOG(5) << "... canceling a prerender that doesn't exist."; | 99 DVLOG(5) << "... canceling a prerender that doesn't exist."; |
| 85 return; | 100 return; |
| 86 } | 101 } |
| 87 PrerenderHandle* prerender_handle = id_to_handle_iter->second; | 102 PrerenderHandle* prerender_handle = id_to_handle_iter->second; |
| 88 prerender_handle->OnCancel(); | 103 prerender_handle->OnCancel(); |
| 89 RemovePrerender(id_to_handle_iter); | 104 |
| 105 // Because OnCancel() can remove the prerender from the map, we need to | |
| 106 // consider our iterator invalid. | |
| 107 id_to_handle_iter = ids_to_handle_map_.find(child_and_prerender_id); | |
| 108 if (id_to_handle_iter != ids_to_handle_map_.end()) | |
| 109 RemovePrerender(id_to_handle_iter); | |
| 90 } | 110 } |
| 91 | 111 |
| 92 void PrerenderLinkManager::OnAbandonPrerender(int child_id, int prerender_id) { | 112 void PrerenderLinkManager::OnAbandonPrerender(int child_id, int prerender_id) { |
| 93 DVLOG(2) << "OnAbandonPrerender, child_id = " << child_id | 113 DVLOG(2) << "OnAbandonPrerender, child_id = " << child_id |
| 94 << ", prerender_id = " << prerender_id; | 114 << ", prerender_id = " << prerender_id; |
| 95 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | 115 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); |
| 96 IdPairToPrerenderHandleMap::iterator id_to_handle_iter = | 116 IdPairToPrerenderHandleMap::iterator id_to_handle_iter = |
| 97 ids_to_handle_map_.find(child_and_prerender_id); | 117 ids_to_handle_map_.find(child_and_prerender_id); |
| 98 if (id_to_handle_iter == ids_to_handle_map_.end()) | 118 if (id_to_handle_iter == ids_to_handle_map_.end()) |
| 99 return; | 119 return; |
| 100 PrerenderHandle* prerender_handle = id_to_handle_iter->second; | 120 PrerenderHandle* prerender_handle = id_to_handle_iter->second; |
| 101 prerender_handle->OnNavigateAway(); | 121 prerender_handle->OnNavigateAway(); |
| 102 RemovePrerender(id_to_handle_iter); | |
| 103 } | 122 } |
| 104 | 123 |
| 105 void PrerenderLinkManager::OnChannelClosing(int child_id) { | 124 void PrerenderLinkManager::OnChannelClosing(int child_id) { |
| 106 DVLOG(2) << "OnChannelClosing, child id = " << child_id; | 125 DVLOG(2) << "OnChannelClosing, child id = " << child_id; |
| 107 const ChildAndPrerenderIdPair child_and_minimum_prerender_id( | 126 const ChildAndPrerenderIdPair child_and_minimum_prerender_id( |
| 108 child_id, std::numeric_limits<int>::min()); | 127 child_id, std::numeric_limits<int>::min()); |
| 109 const ChildAndPrerenderIdPair child_and_maximum_prerender_id( | 128 const ChildAndPrerenderIdPair child_and_maximum_prerender_id( |
| 110 child_id, std::numeric_limits<int>::max()); | 129 child_id, std::numeric_limits<int>::max()); |
| 111 std::queue<int> prerender_ids_to_abandon; | 130 |
| 112 for (IdPairToPrerenderHandleMap::iterator | 131 IdPairToPrerenderHandleMap::iterator |
| 113 i = ids_to_handle_map_.lower_bound(child_and_minimum_prerender_id), | 132 it = ids_to_handle_map_.lower_bound(child_and_minimum_prerender_id); |
| 114 e = ids_to_handle_map_.upper_bound(child_and_maximum_prerender_id); | 133 IdPairToPrerenderHandleMap::iterator |
| 115 i != e; ++i) { | 134 end = ids_to_handle_map_.upper_bound(child_and_maximum_prerender_id); |
| 116 prerender_ids_to_abandon.push(i->first.second); | 135 while (it != end) { |
| 117 } | 136 IdPairToPrerenderHandleMap::iterator next = it; |
| 118 while (!prerender_ids_to_abandon.empty()) { | 137 ++next; |
| 119 DVLOG(4) << "---> abandon prerender_id = " | 138 |
| 120 << prerender_ids_to_abandon.front(); | 139 size_t size_before_abandon = ids_to_handle_map_.size(); |
| 121 OnAbandonPrerender(child_id, prerender_ids_to_abandon.front()); | 140 OnAbandonPrerender(child_id, it->first.second); |
| 122 prerender_ids_to_abandon.pop(); | 141 DCHECK_EQ(size_before_abandon, ids_to_handle_map_.size()); |
| 142 ids_to_handle_map_.erase(it); | |
|
mmenke
2012/12/13 21:39:43
Isn't this leaking a handle?
gavinp
2012/12/13 22:01:56
Yes. I've added a test and fixed it.
| |
| 143 | |
| 144 it = next; | |
| 123 } | 145 } |
| 124 } | 146 } |
| 125 | 147 |
| 126 bool PrerenderLinkManager::IsEmpty() const { | 148 bool PrerenderLinkManager::IsEmpty() const { |
| 127 return ids_to_handle_map_.empty(); | 149 return ids_to_handle_map_.empty(); |
| 128 } | 150 } |
| 129 | 151 |
| 130 void PrerenderLinkManager::RemovePrerender( | 152 void PrerenderLinkManager::RemovePrerender( |
| 131 const IdPairToPrerenderHandleMap::iterator& id_to_handle_iter) { | 153 const IdPairToPrerenderHandleMap::iterator& id_to_handle_iter) { |
| 132 PrerenderHandle* prerender_handle = id_to_handle_iter->second; | 154 PrerenderHandle* prerender_handle = id_to_handle_iter->second; |
| 133 delete prerender_handle; | 155 delete prerender_handle; |
| 134 ids_to_handle_map_.erase(id_to_handle_iter); | 156 ids_to_handle_map_.erase(id_to_handle_iter); |
| 135 } | 157 } |
| 136 | 158 |
| 159 PrerenderLinkManager::IdPairToPrerenderHandleMap::iterator | |
| 160 PrerenderLinkManager::FindPrerenderHandle( | |
| 161 PrerenderHandle* prerender_handle) { | |
| 162 for (IdPairToPrerenderHandleMap::iterator it = ids_to_handle_map_.begin(); | |
| 163 it != ids_to_handle_map_.end(); ++it) { | |
| 164 if (it->second == prerender_handle) | |
| 165 return it; | |
| 166 } | |
| 167 return ids_to_handle_map_.end(); | |
| 168 } | |
| 169 | |
| 170 // In practice, this is always called from either | |
| 171 // PrerenderLinkManager::OnAddPrerender in the regular case, or in the pending | |
| 172 // prerender case, from PrerenderHandle::AdoptPrerenderDataFrom. | |
| 173 void PrerenderLinkManager::OnPrerenderStart( | |
| 174 PrerenderHandle* prerender_handle) { | |
| 175 IdPairToPrerenderHandleMap::iterator it = | |
| 176 FindPrerenderHandle(prerender_handle); | |
| 177 DCHECK(it != ids_to_handle_map_.end()); | |
| 178 const int child_id = it->first.first; | |
| 179 const int prerender_id = it->first.second; | |
| 180 | |
| 181 Send(child_id, new PrerenderMsg_OnPrerenderStart(prerender_id)); | |
| 182 } | |
| 183 | |
| 184 void PrerenderLinkManager::OnPrerenderAddAlias( | |
| 185 PrerenderHandle* prerender_handle, | |
| 186 const GURL& alias_url) { | |
| 187 IdPairToPrerenderHandleMap::iterator it = | |
| 188 FindPrerenderHandle(prerender_handle); | |
| 189 if (it == ids_to_handle_map_.end()) | |
| 190 return; | |
| 191 const int child_id = it->first.first; | |
| 192 const int prerender_id = it->first.second; | |
| 193 | |
| 194 Send(child_id, new PrerenderMsg_OnPrerenderAddAlias(prerender_id, alias_url)); | |
| 195 } | |
| 196 | |
| 197 void PrerenderLinkManager::OnPrerenderStop( | |
| 198 PrerenderHandle* prerender_handle) { | |
| 199 IdPairToPrerenderHandleMap::iterator it = | |
| 200 FindPrerenderHandle(prerender_handle); | |
| 201 if (it == ids_to_handle_map_.end()) | |
| 202 return; | |
| 203 const int child_id = it->first.first; | |
| 204 const int prerender_id = it->first.second; | |
| 205 | |
| 206 Send(child_id, new PrerenderMsg_OnPrerenderStop(prerender_id)); | |
| 207 RemovePrerender(it); | |
| 208 } | |
| 209 | |
| 137 } // namespace prerender | 210 } // namespace prerender |
| OLD | NEW |