Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(302)

Side by Side Diff: chrome/browser/prerender/prerender_link_manager.cc

Issue 11316311: Make PrerenderHandle an observer of PrerenderContents. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: fix leak! Created 8 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
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>
gavinp 2012/12/13 22:11:58 Bonus header removal.
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 delete it->second;
143 ids_to_handle_map_.erase(it);
mmenke 2012/12/13 22:05:46 Replace last two with just: RemovePrerender(it);
gavinp 2012/12/13 22:11:58 Done.
144
145 it = next;
123 } 146 }
124 } 147 }
125 148
126 bool PrerenderLinkManager::IsEmpty() const { 149 bool PrerenderLinkManager::IsEmpty() const {
127 return ids_to_handle_map_.empty(); 150 return ids_to_handle_map_.empty();
128 } 151 }
129 152
130 void PrerenderLinkManager::RemovePrerender( 153 void PrerenderLinkManager::RemovePrerender(
131 const IdPairToPrerenderHandleMap::iterator& id_to_handle_iter) { 154 const IdPairToPrerenderHandleMap::iterator& id_to_handle_iter) {
132 PrerenderHandle* prerender_handle = id_to_handle_iter->second; 155 PrerenderHandle* prerender_handle = id_to_handle_iter->second;
133 delete prerender_handle; 156 delete prerender_handle;
134 ids_to_handle_map_.erase(id_to_handle_iter); 157 ids_to_handle_map_.erase(id_to_handle_iter);
135 } 158 }
136 159
160 PrerenderLinkManager::IdPairToPrerenderHandleMap::iterator
161 PrerenderLinkManager::FindPrerenderHandle(
162 PrerenderHandle* prerender_handle) {
163 for (IdPairToPrerenderHandleMap::iterator it = ids_to_handle_map_.begin();
164 it != ids_to_handle_map_.end(); ++it) {
165 if (it->second == prerender_handle)
166 return it;
167 }
168 return ids_to_handle_map_.end();
169 }
170
171 // In practice, this is always called from either
172 // PrerenderLinkManager::OnAddPrerender in the regular case, or in the pending
173 // prerender case, from PrerenderHandle::AdoptPrerenderDataFrom.
174 void PrerenderLinkManager::OnPrerenderStart(
175 PrerenderHandle* prerender_handle) {
176 IdPairToPrerenderHandleMap::iterator it =
177 FindPrerenderHandle(prerender_handle);
178 DCHECK(it != ids_to_handle_map_.end());
179 const int child_id = it->first.first;
180 const int prerender_id = it->first.second;
181
182 Send(child_id, new PrerenderMsg_OnPrerenderStart(prerender_id));
183 }
184
185 void PrerenderLinkManager::OnPrerenderAddAlias(
186 PrerenderHandle* prerender_handle,
187 const GURL& alias_url) {
188 IdPairToPrerenderHandleMap::iterator it =
189 FindPrerenderHandle(prerender_handle);
190 if (it == ids_to_handle_map_.end())
191 return;
192 const int child_id = it->first.first;
193 const int prerender_id = it->first.second;
194
195 Send(child_id, new PrerenderMsg_OnPrerenderAddAlias(prerender_id, alias_url));
196 }
197
198 void PrerenderLinkManager::OnPrerenderStop(
199 PrerenderHandle* prerender_handle) {
200 IdPairToPrerenderHandleMap::iterator it =
201 FindPrerenderHandle(prerender_handle);
202 if (it == ids_to_handle_map_.end())
203 return;
204 const int child_id = it->first.first;
205 const int prerender_id = it->first.second;
206
207 Send(child_id, new PrerenderMsg_OnPrerenderStop(prerender_id));
208 RemovePrerender(it);
209 }
210
137 } // namespace prerender 211 } // namespace prerender
OLDNEW
« no previous file with comments | « chrome/browser/prerender/prerender_link_manager.h ('k') | chrome/browser/prerender/prerender_manager.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698