| 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
|
|
|