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 "chrome/browser/prerender/prerender_contents.h" | 11 #include "chrome/browser/prerender/prerender_contents.h" |
| 12 #include "chrome/browser/prerender/prerender_handle.h" | |
| 12 #include "chrome/browser/prerender/prerender_manager.h" | 13 #include "chrome/browser/prerender/prerender_manager.h" |
| 13 #include "chrome/browser/prerender/prerender_manager_factory.h" | 14 #include "chrome/browser/prerender/prerender_manager_factory.h" |
| 14 #include "chrome/browser/profiles/profile.h" | 15 #include "chrome/browser/profiles/profile.h" |
| 15 #include "content/public/browser/render_view_host.h" | 16 #include "content/public/browser/render_view_host.h" |
| 16 #include "content/public/browser/session_storage_namespace.h" | 17 #include "content/public/browser/session_storage_namespace.h" |
| 17 #include "content/public/common/referrer.h" | 18 #include "content/public/common/referrer.h" |
| 18 #include "googleurl/src/gurl.h" | 19 #include "googleurl/src/gurl.h" |
| 19 #include "googleurl/src/url_canon.h" | 20 #include "googleurl/src/url_canon.h" |
| 20 #include "ui/gfx/size.h" | 21 #include "ui/gfx/size.h" |
| 21 | 22 |
| 22 using content::RenderViewHost; | 23 using content::RenderViewHost; |
| 23 using content::SessionStorageNamespace; | 24 using content::SessionStorageNamespace; |
| 24 | 25 |
| 25 namespace prerender { | 26 namespace prerender { |
| 26 | 27 |
| 27 PrerenderLinkManager::PrerenderLinkManager(PrerenderManager* manager) | 28 PrerenderLinkManager::PrerenderLinkManager(PrerenderManager* manager) |
| 28 : manager_(manager) { | 29 : manager_(manager) { |
| 29 } | 30 } |
| 30 | 31 |
| 31 PrerenderLinkManager::~PrerenderLinkManager() { | 32 PrerenderLinkManager::~PrerenderLinkManager() { |
| 33 for (IdPairToPrerenderHandleMap::iterator it = ids_to_handle_map_.begin(); | |
| 34 it != ids_to_handle_map_.end(); | |
| 35 ++it) { | |
| 36 PrerenderHandle* prerender_handle = it->second.get(); | |
|
dominich
2012/07/03 17:08:39
the more i see this, the more worried i get. I don
mmenke
2012/07/09 18:06:57
I tend to agree, but it's needed by the callback s
| |
| 37 prerender_handle->OnNavigateAway(); | |
|
dominich
2012/07/03 17:08:39
I don't trust refcounting. You have a very clear o
gavinp
2012/07/03 18:45:41
Done.
When we switch to C++11, I'll make it a map
| |
| 38 } | |
| 32 } | 39 } |
| 33 | 40 |
| 34 bool PrerenderLinkManager::OnAddPrerender(int child_id, | 41 bool PrerenderLinkManager::OnAddPrerender(int child_id, |
| 35 int prerender_id, | 42 int prerender_id, |
| 36 const GURL& orig_url, | 43 const GURL& orig_url, |
| 37 const content::Referrer& referrer, | 44 const content::Referrer& referrer, |
| 38 const gfx::Size& size, | 45 const gfx::Size& size, |
| 39 int render_view_route_id) { | 46 int render_view_route_id) { |
| 40 DVLOG(2) << "OnAddPrerender, child_id = " << child_id | 47 DVLOG(2) << "OnAddPrerender, child_id = " << child_id |
| 41 << ", prerender_id = " << prerender_id | 48 << ", prerender_id = " << prerender_id |
| 42 << ", url = " << orig_url.spec(); | 49 << ", url = " << orig_url.spec(); |
| 43 DVLOG(3) << "... referrer url = " << referrer.url.spec() | 50 DVLOG(3) << "... referrer url = " << referrer.url.spec() |
| 44 << ", size = (" << size.width() << ", " << size.height() << ")" | 51 << ", size = (" << size.width() << ", " << size.height() << ")" |
| 45 << ", render_view_route_id = " << render_view_route_id; | 52 << ", render_view_route_id = " << render_view_route_id; |
| 46 | 53 |
| 47 // TODO(gavinp): Add tests to ensure fragments work, then remove this fragment | 54 // TODO(gavinp): Add tests to ensure fragments work, then remove this fragment |
| 48 // clearing code. | 55 // clearing code. |
| 49 url_canon::Replacements<char> replacements; | 56 url_canon::Replacements<char> replacements; |
| 50 replacements.ClearRef(); | 57 replacements.ClearRef(); |
| 51 const GURL url = orig_url.ReplaceComponents(replacements); | 58 const GURL url = orig_url.ReplaceComponents(replacements); |
| 52 | 59 |
| 53 if (!manager_->AddPrerenderFromLinkRelPrerender( | 60 if (PrerenderHandle* prerender_handle = |
| 54 child_id, render_view_route_id, url, referrer, size)) { | 61 manager_->AddPrerenderFromLinkRelPrerender( |
| 55 return false; | 62 child_id, render_view_route_id, url, referrer, size)) { |
| 63 ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | |
| 64 DCHECK_EQ(0U, ids_to_handle_map_.count(child_and_prerender_id)); | |
| 65 ids_to_handle_map_.insert( | |
| 66 std::make_pair(child_and_prerender_id, prerender_handle)); | |
| 67 return true; | |
| 56 } | 68 } |
| 57 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | 69 return false; |
| 58 DCHECK_EQ(0U, ids_to_url_map_.count(child_and_prerender_id)); | |
| 59 ids_to_url_map_.insert(std::make_pair(child_and_prerender_id, url)); | |
| 60 return true; | |
| 61 } | 70 } |
| 62 | 71 |
| 63 // TODO(gavinp): Once an observer interface is provided down to the WebKit | 72 // TODO(gavinp): Once an observer interface is provided down to the WebKit |
| 64 // layer, we should add DCHECK_NE(0L, ids_to_url_map_.count(...)) to both | 73 // layer, we should add DCHECK_NE(0L, ids_to_url_map_.count(...)) to both |
| 65 // OnCancelPrerender and OnAbandonPrerender. We can't do this now, since | 74 // OnCancelPrerender and OnAbandonPrerender. We can't do this now, since |
| 66 // the WebKit layer isn't even aware if we didn't add the prerender to the map | 75 // the WebKit layer isn't even aware if we didn't add the prerender to the map |
| 67 // in OnAddPrerender above. | 76 // in OnAddPrerender above. |
| 68 void PrerenderLinkManager::OnCancelPrerender(int child_id, int prerender_id) { | 77 void PrerenderLinkManager::OnCancelPrerender(int child_id, int prerender_id) { |
| 69 DVLOG(2) << "OnCancelPrerender, child_id = " << child_id | 78 DVLOG(2) << "OnCancelPrerender, child_id = " << child_id |
| 70 << ", prerender_id = " << prerender_id; | 79 << ", prerender_id = " << prerender_id; |
| 71 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | 80 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); |
| 72 IdPairToUrlMap::iterator id_url_iter = | 81 IdPairToPrerenderHandleMap::iterator id_handle_iter = |
| 73 ids_to_url_map_.find(child_and_prerender_id); | 82 ids_to_handle_map_.find(child_and_prerender_id); |
| 74 if (id_url_iter == ids_to_url_map_.end()) { | 83 if (id_handle_iter == ids_to_handle_map_.end()) { |
| 75 DVLOG(5) << "... canceling a prerender that doesn't exist."; | 84 DVLOG(5) << "... canceling a prerender that doesn't exist."; |
| 76 return; | 85 return; |
| 77 } | 86 } |
| 78 const GURL url = id_url_iter->second; | 87 ids_to_handle_map_.erase(id_handle_iter); |
| 79 ids_to_url_map_.erase(id_url_iter); | |
| 80 manager_->MaybeCancelPrerender(url); | |
| 81 } | 88 } |
| 82 | 89 |
| 83 void PrerenderLinkManager::OnAbandonPrerender(int child_id, int prerender_id) { | 90 void PrerenderLinkManager::OnAbandonPrerender(int child_id, int prerender_id) { |
| 84 DVLOG(2) << "OnAbandonPrerender, child_id = " << child_id | 91 DVLOG(2) << "OnAbandonPrerender, child_id = " << child_id |
| 85 << ", prerender_id = " << prerender_id; | 92 << ", prerender_id = " << prerender_id; |
| 86 // TODO(gavinp,cbentzel): Implement reasonable behaviour for | |
| 87 // navigation away from launcher. | |
| 88 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); | 93 const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id); |
| 89 ids_to_url_map_.erase(child_and_prerender_id); | 94 IdPairToPrerenderHandleMap::iterator id_handle_iter = |
| 95 ids_to_handle_map_.find(child_and_prerender_id); | |
| 96 if (id_handle_iter == ids_to_handle_map_.end()) | |
| 97 return; | |
| 98 PrerenderHandle* prerender_handle = id_handle_iter->second.get(); | |
| 99 prerender_handle->OnNavigateAway(); | |
| 100 ids_to_handle_map_.erase(id_handle_iter); | |
| 90 } | 101 } |
| 91 | 102 |
| 92 void PrerenderLinkManager::OnChannelClosing(int child_id) { | 103 void PrerenderLinkManager::OnChannelClosing(int child_id) { |
| 93 DVLOG(2) << "OnChannelClosing, child id = " << child_id; | 104 DVLOG(2) << "OnChannelClosing, child id = " << child_id; |
| 94 const ChildAndPrerenderIdPair child_and_minimum_prerender_id( | 105 const ChildAndPrerenderIdPair child_and_minimum_prerender_id( |
| 95 child_id, std::numeric_limits<int>::min()); | 106 child_id, std::numeric_limits<int>::min()); |
| 96 const ChildAndPrerenderIdPair child_and_maximum_prerender_id( | 107 const ChildAndPrerenderIdPair child_and_maximum_prerender_id( |
| 97 child_id, std::numeric_limits<int>::max()); | 108 child_id, std::numeric_limits<int>::max()); |
| 98 std::queue<int> prerender_ids_to_abandon; | 109 std::queue<int> prerender_ids_to_abandon; |
| 99 for (IdPairToUrlMap::iterator | 110 for (IdPairToPrerenderHandleMap::iterator |
| 100 i = ids_to_url_map_.lower_bound(child_and_minimum_prerender_id), | 111 i = ids_to_handle_map_.lower_bound(child_and_minimum_prerender_id), |
| 101 e = ids_to_url_map_.upper_bound(child_and_maximum_prerender_id); | 112 e = ids_to_handle_map_.upper_bound(child_and_maximum_prerender_id); |
| 102 i != e; ++i) { | 113 i != e; ++i) { |
| 103 prerender_ids_to_abandon.push(i->first.second); | 114 prerender_ids_to_abandon.push(i->first.second); |
| 104 } | 115 } |
| 105 while (!prerender_ids_to_abandon.empty()) { | 116 while (!prerender_ids_to_abandon.empty()) { |
| 106 DVLOG(4) << "---> abandon prerender_id = " | 117 DVLOG(4) << "---> abandon prerender_id = " |
| 107 << prerender_ids_to_abandon.front(); | 118 << prerender_ids_to_abandon.front(); |
| 108 OnAbandonPrerender(child_id, prerender_ids_to_abandon.front()); | 119 OnAbandonPrerender(child_id, prerender_ids_to_abandon.front()); |
| 109 prerender_ids_to_abandon.pop(); | 120 prerender_ids_to_abandon.pop(); |
| 110 } | 121 } |
| 111 } | 122 } |
| 112 | 123 |
| 113 bool PrerenderLinkManager::IsEmpty() const { | 124 bool PrerenderLinkManager::IsEmpty() const { |
| 114 return ids_to_url_map_.empty(); | 125 return ids_to_handle_map_.empty(); |
| 115 } | 126 } |
| 116 | 127 |
| 117 } // namespace prerender | 128 } // namespace prerender |
| 118 | 129 |
| OLD | NEW |