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

Unified 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: clear to land 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 side-by-side diff with in-line comments
Download patch
« 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 »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: chrome/browser/prerender/prerender_link_manager.cc
diff --git a/chrome/browser/prerender/prerender_link_manager.cc b/chrome/browser/prerender/prerender_link_manager.cc
index ea559977e272d5c0eeace49991130a88ad22457d..ea450129b01d4daeb0a5434c17b2c372562a05fb 100644
--- a/chrome/browser/prerender/prerender_link_manager.cc
+++ b/chrome/browser/prerender/prerender_link_manager.cc
@@ -5,14 +5,16 @@
#include "chrome/browser/prerender/prerender_link_manager.h"
#include <limits>
-#include <queue>
#include <utility>
+#include "base/memory/scoped_ptr.h"
#include "chrome/browser/prerender/prerender_contents.h"
#include "chrome/browser/prerender/prerender_handle.h"
#include "chrome/browser/prerender/prerender_manager.h"
#include "chrome/browser/prerender/prerender_manager_factory.h"
#include "chrome/browser/profiles/profile.h"
+#include "chrome/common/prerender_messages.h"
+#include "content/public/browser/render_process_host.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/session_storage_namespace.h"
#include "content/public/common/referrer.h"
@@ -22,6 +24,20 @@
using content::RenderViewHost;
using content::SessionStorageNamespace;
+namespace {
+
+void Send(int child_id, IPC::Message* raw_message) {
+ using content::RenderProcessHost;
+ scoped_ptr<IPC::Message> own_message(raw_message);
+
+ RenderProcessHost* render_process_host = RenderProcessHost::FromID(child_id);
+ if (!render_process_host)
+ return;
+ render_process_host->Send(own_message.release());
+}
+
+} // namespace
+
namespace prerender {
PrerenderLinkManager::PrerenderLinkManager(PrerenderManager* manager)
@@ -33,7 +49,9 @@ PrerenderLinkManager::~PrerenderLinkManager() {
it != ids_to_handle_map_.end();
++it) {
PrerenderHandle* prerender_handle = it->second;
- prerender_handle->OnCancel();
+ DCHECK(!prerender_handle->IsPrerendering())
+ << "All running prerenders should stop at the same time as the "
+ << "PrerenderManager.";
delete prerender_handle;
}
}
@@ -51,29 +69,25 @@ bool PrerenderLinkManager::OnAddPrerender(int child_id,
<< ", size = (" << size.width() << ", " << size.height() << ")"
<< ", render_view_route_id = " << render_view_route_id;
- const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id);
- DCHECK_EQ(0U, ids_to_handle_map_.count(child_and_prerender_id));
- scoped_ptr<PrerenderHandle> prerender_handle(
+ PrerenderHandle* prerender_handle =
manager_->AddPrerenderFromLinkRelPrerender(
- child_id, render_view_route_id, url, referrer, size));
- if (prerender_handle.get()) {
- std::pair<IdPairToPrerenderHandleMap::iterator, bool> insert_result =
- ids_to_handle_map_.insert(std::make_pair(
- child_and_prerender_id, static_cast<PrerenderHandle*>(NULL)));
- DCHECK(insert_result.second);
- delete insert_result.first->second;
- insert_result.first->second = prerender_handle.release();
- return true;
- }
- return false;
+ child_id, render_view_route_id, url, referrer, size);
+ if (!prerender_handle)
+ return false;
+
+ const ChildAndPrerenderIdPair child_and_prerender_id(child_id, prerender_id);
+ DCHECK_EQ(0u, ids_to_handle_map_.count(child_and_prerender_id));
+ ids_to_handle_map_[child_and_prerender_id] = prerender_handle;
+
+ // If we are given a prerender that is already prerendering, we have missed
+ // the start event.
+ if (prerender_handle->IsPrerendering())
+ OnPrerenderStart(prerender_handle);
+ prerender_handle->SetObserver(this);
+ return true;
}
-// TODO(gavinp): Once an observer interface is provided down to the WebKit
-// layer, we should add DCHECK_NE(0L, ids_to_url_map_.count(...)) to both
-// OnCancelPrerender and OnAbandonPrerender. We can't do this now, since
-// the WebKit layer isn't even aware if we didn't add the prerender to the map
-// in OnAddPrerender above.
void PrerenderLinkManager::OnCancelPrerender(int child_id, int prerender_id) {
DVLOG(2) << "OnCancelPrerender, child_id = " << child_id
<< ", prerender_id = " << prerender_id;
@@ -86,7 +100,12 @@ void PrerenderLinkManager::OnCancelPrerender(int child_id, int prerender_id) {
}
PrerenderHandle* prerender_handle = id_to_handle_iter->second;
prerender_handle->OnCancel();
- RemovePrerender(id_to_handle_iter);
+
+ // Because OnCancel() can remove the prerender from the map, we need to
+ // consider our iterator invalid.
+ id_to_handle_iter = ids_to_handle_map_.find(child_and_prerender_id);
+ if (id_to_handle_iter != ids_to_handle_map_.end())
+ RemovePrerender(id_to_handle_iter);
}
void PrerenderLinkManager::OnAbandonPrerender(int child_id, int prerender_id) {
@@ -99,7 +118,6 @@ void PrerenderLinkManager::OnAbandonPrerender(int child_id, int prerender_id) {
return;
PrerenderHandle* prerender_handle = id_to_handle_iter->second;
prerender_handle->OnNavigateAway();
- RemovePrerender(id_to_handle_iter);
}
void PrerenderLinkManager::OnChannelClosing(int child_id) {
@@ -108,18 +126,21 @@ void PrerenderLinkManager::OnChannelClosing(int child_id) {
child_id, std::numeric_limits<int>::min());
const ChildAndPrerenderIdPair child_and_maximum_prerender_id(
child_id, std::numeric_limits<int>::max());
- std::queue<int> prerender_ids_to_abandon;
- for (IdPairToPrerenderHandleMap::iterator
- i = ids_to_handle_map_.lower_bound(child_and_minimum_prerender_id),
- e = ids_to_handle_map_.upper_bound(child_and_maximum_prerender_id);
- i != e; ++i) {
- prerender_ids_to_abandon.push(i->first.second);
- }
- while (!prerender_ids_to_abandon.empty()) {
- DVLOG(4) << "---> abandon prerender_id = "
- << prerender_ids_to_abandon.front();
- OnAbandonPrerender(child_id, prerender_ids_to_abandon.front());
- prerender_ids_to_abandon.pop();
+
+ IdPairToPrerenderHandleMap::iterator
+ it = ids_to_handle_map_.lower_bound(child_and_minimum_prerender_id);
+ IdPairToPrerenderHandleMap::iterator
+ end = ids_to_handle_map_.upper_bound(child_and_maximum_prerender_id);
+ while (it != end) {
+ IdPairToPrerenderHandleMap::iterator next = it;
+ ++next;
+
+ size_t size_before_abandon = ids_to_handle_map_.size();
+ OnAbandonPrerender(child_id, it->first.second);
+ DCHECK_EQ(size_before_abandon, ids_to_handle_map_.size());
+ RemovePrerender(it);
+
+ it = next;
}
}
@@ -134,4 +155,55 @@ void PrerenderLinkManager::RemovePrerender(
ids_to_handle_map_.erase(id_to_handle_iter);
}
+PrerenderLinkManager::IdPairToPrerenderHandleMap::iterator
+PrerenderLinkManager::FindPrerenderHandle(
+ PrerenderHandle* prerender_handle) {
+ for (IdPairToPrerenderHandleMap::iterator it = ids_to_handle_map_.begin();
+ it != ids_to_handle_map_.end(); ++it) {
+ if (it->second == prerender_handle)
+ return it;
+ }
+ return ids_to_handle_map_.end();
+}
+
+// In practice, this is always called from either
+// PrerenderLinkManager::OnAddPrerender in the regular case, or in the pending
+// prerender case, from PrerenderHandle::AdoptPrerenderDataFrom.
+void PrerenderLinkManager::OnPrerenderStart(
+ PrerenderHandle* prerender_handle) {
+ IdPairToPrerenderHandleMap::iterator it =
+ FindPrerenderHandle(prerender_handle);
+ DCHECK(it != ids_to_handle_map_.end());
+ const int child_id = it->first.first;
+ const int prerender_id = it->first.second;
+
+ Send(child_id, new PrerenderMsg_OnPrerenderStart(prerender_id));
+}
+
+void PrerenderLinkManager::OnPrerenderAddAlias(
+ PrerenderHandle* prerender_handle,
+ const GURL& alias_url) {
+ IdPairToPrerenderHandleMap::iterator it =
+ FindPrerenderHandle(prerender_handle);
+ if (it == ids_to_handle_map_.end())
+ return;
+ const int child_id = it->first.first;
+ const int prerender_id = it->first.second;
+
+ Send(child_id, new PrerenderMsg_OnPrerenderAddAlias(prerender_id, alias_url));
+}
+
+void PrerenderLinkManager::OnPrerenderStop(
+ PrerenderHandle* prerender_handle) {
+ IdPairToPrerenderHandleMap::iterator it =
+ FindPrerenderHandle(prerender_handle);
+ if (it == ids_to_handle_map_.end())
+ return;
+ const int child_id = it->first.first;
+ const int prerender_id = it->first.second;
+
+ Send(child_id, new PrerenderMsg_OnPrerenderStop(prerender_id));
+ RemovePrerender(it);
+}
+
} // namespace prerender
« 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