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

Unified Diff: content/browser/renderer_host/render_view_host.cc

Issue 6319001: Support window.opener after a process swap. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Better message filtering approach. Created 9 years, 8 months 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
Index: content/browser/renderer_host/render_view_host.cc
diff --git a/content/browser/renderer_host/render_view_host.cc b/content/browser/renderer_host/render_view_host.cc
index 98bcbf8561c319317d6a833b99d7d2aa3666b833..91f020eeab13b2997a9c751a053b3406c223cc55 100644
--- a/content/browser/renderer_host/render_view_host.cc
+++ b/content/browser/renderer_host/render_view_host.cc
@@ -92,9 +92,10 @@ RenderViewHost::RenderViewHost(SiteInstance* instance,
delegate_(delegate),
waiting_for_drag_context_response_(false),
enabled_bindings_(0),
- pending_request_id_(0),
+ pending_request_id_(-1),
navigations_suspended_(false),
suspended_nav_message_(NULL),
+ is_swapped_out_(false),
run_modal_reply_msg_(NULL),
is_waiting_for_beforeunload_ack_(false),
is_waiting_for_unload_ack_(false),
@@ -258,11 +259,20 @@ void RenderViewHost::SetNavigationsSuspended(bool suspend) {
navigations_suspended_ = suspend;
if (!suspend && suspended_nav_message_.get()) {
// There's a navigation message waiting to be sent. Now that we're not
- // suspended anymore, resume navigation by sending it.
+ // suspended anymore, resume navigation by sending it. If we were swapped
+ // out, we should also stop filtering out the IPC messages now.
+ is_swapped_out_ = false;
Send(suspended_nav_message_.release());
}
}
+void RenderViewHost::CancelSuspendedNavigations() {
+ // Clear any state if a pending navigation is canceled or pre-empted.
+ if (suspended_nav_message_.get())
+ suspended_nav_message_.reset();
+ navigations_suspended_ = false;
+}
+
void RenderViewHost::FirePageBeforeUnload(bool for_cross_site_transition) {
if (!IsRenderViewLive()) {
// This RenderViewHost doesn't have a live renderer, so just skip running
@@ -295,45 +305,66 @@ void RenderViewHost::FirePageBeforeUnload(bool for_cross_site_transition) {
}
}
-void RenderViewHost::ClosePage(bool for_cross_site_transition,
- int new_render_process_host_id,
- int new_request_id) {
- // This will be set back to false in OnClosePageACK, just before we close the
- // tab or replace it with a pending RVH. There are some cases (such as 204
- // errors) where we'll continue to show this RVH.
+void RenderViewHost::SwapOut(int new_render_process_host_id,
+ int new_request_id) {
+ // Start filtering IPC messages to avoid confusing the delegate. This will
+ // prevent any dialogs from appearing during unload handlers, but we've
+ // already decided to silence them in crbug.com/68780. We will set it back
+ // to false in SetNavigationsSuspended if we swap back in.
+ is_swapped_out_ = true;
+
+ // This will be set back to false in OnSwapOutACK, just before we replace
+ // this RVH with the pending RVH.
is_waiting_for_unload_ack_ = true;
// Start the hang monitor in case the renderer hangs in the unload handler.
StartHangMonitorTimeout(TimeDelta::FromMilliseconds(kUnloadTimeoutMS));
- ViewMsg_ClosePage_Params params;
+ ViewMsg_SwapOut_Params params;
params.closing_process_id = process()->id();
params.closing_route_id = routing_id();
- params.for_cross_site_transition = for_cross_site_transition;
params.new_render_process_host_id = new_render_process_host_id;
params.new_request_id = new_request_id;
if (IsRenderViewLive()) {
- NotificationService::current()->Notify(
- NotificationType::RENDER_VIEW_HOST_WILL_CLOSE_RENDER_VIEW,
- Source<RenderViewHost>(this),
- NotificationService::NoDetails());
-
- Send(new ViewMsg_ClosePage(routing_id(), params));
+ Send(new ViewMsg_SwapOut(routing_id(), params));
} else {
- // This RenderViewHost doesn't have a live renderer, so just skip closing
- // the page. We must notify the ResourceDispatcherHost on the IO thread,
+ // This RenderViewHost doesn't have a live renderer, so just skip the unload
+ // event. We must notify the ResourceDispatcherHost on the IO thread,
// which we will do through the RenderProcessHost's widget helper.
- process()->CrossSiteClosePageACK(params);
+ process()->CrossSiteSwapOutACK(params);
}
}
-void RenderViewHost::OnClosePageACK(bool for_cross_site_transition) {
+void RenderViewHost::OnSwapOutACK() {
+ // Stop the hang monitor now that the unload handler has finished.
StopHangMonitorTimeout();
is_waiting_for_unload_ack_ = false;
+}
+
+void RenderViewHost::WasSwappedOut() {
+ // Don't bother reporting hung state anymore.
+ StopHangMonitorTimeout();
+
+ // Inform the renderer that it can exit if no one else is using it.
+ Send(new ViewMsg_WasSwappedOut(routing_id()));
+}
+
+void RenderViewHost::ClosePage() {
+ // Start the hang monitor in case the renderer hangs in the unload handler.
+ is_waiting_for_unload_ack_ = true;
+ StartHangMonitorTimeout(TimeDelta::FromMilliseconds(kUnloadTimeoutMS));
- // If this ClosePageACK is not for a cross-site transition, then it is for an
- // attempt to close the tab. We have now finished the unload handler and can
- // proceed with closing the tab.
- if (!for_cross_site_transition) {
+ if (IsRenderViewLive()) {
+ // TODO(creis): Should this be moved to Shutdown? It may not be called for
+ // RenderViewHosts that have been swapped out.
+ NotificationService::current()->Notify(
+ NotificationType::RENDER_VIEW_HOST_WILL_CLOSE_RENDER_VIEW,
+ Source<RenderViewHost>(this),
+ NotificationService::NoDetails());
+
+ Send(new ViewMsg_ClosePage(routing_id()));
+ } else {
+ // This RenderViewHost doesn't have a live renderer, so just skip the unload
+ // event and close the page.
ClosePageIgnoringUnloadEvents();
}
}
@@ -687,6 +718,12 @@ bool RenderViewHost::OnMessageReceived(const IPC::Message& msg) {
if (!BrowserMessageFilter::CheckCanDispatchOnUI(msg, this))
return true;
+ // Filter out most IPC messages if this renderer is swapped out.
+ // We still want to certain ACKs to keep our state consistent.
+ if (is_swapped_out_)
+ if (!CanHandleWhileSwappedOut(msg))
+ return true;
+
{
// delegate_->OnMessageReceived can end up deleting |this|, in which case
// the destructor for ObserverListBase::Iterator would access the deleted
@@ -744,6 +781,7 @@ bool RenderViewHost::OnMessageReceived(const IPC::Message& msg) {
IPC_MESSAGE_HANDLER(ViewHostMsg_TakeFocus, OnTakeFocus)
IPC_MESSAGE_HANDLER(ViewHostMsg_AddMessageToConsole, OnAddMessageToConsole)
IPC_MESSAGE_HANDLER(ViewHostMsg_ShouldClose_ACK, OnMsgShouldCloseACK)
+ IPC_MESSAGE_HANDLER(ViewHostMsg_ClosePage_ACK, OnMsgClosePageACK)
IPC_MESSAGE_HANDLER(ViewHostMsg_SelectionChanged, OnMsgSelectionChanged)
IPC_MESSAGE_HANDLER(ViewHostMsg_AccessibilityNotifications,
OnAccessibilityNotifications)
@@ -773,6 +811,50 @@ bool RenderViewHost::OnMessageReceived(const IPC::Message& msg) {
return handled;
}
+bool RenderViewHost::CanHandleWhileSwappedOut(const IPC::Message& msg) {
+ // We drop most messages that arrive from a swapped out renderer. However,
+ // some are important (e.g., ACKs) for keeping the browser and renderer state
+ // consistent in case we later return to the renderer.
+ switch (msg.type()) {
+ // Sends an ACK.
+ case ViewHostMsg_ShowView::ID:
+ // Sends an ACK.
+ case ViewHostMsg_ShowWidget::ID:
+ // Sends an ACK.
+ case ViewHostMsg_ShowFullscreenWidget::ID:
+ // Updates browser state.
+ case ViewHostMsg_RenderViewReady::ID:
+ // Updates browser state.
+ case ViewHostMsg_RenderViewGone::ID:
+ // Updates the previous navigation entry.
+ case ViewHostMsg_UpdateState::ID:
+ // Sends an ACK.
+ case ViewHostMsg_UpdateTargetURL::ID:
+ // Updates browser state.
+ case ViewHostMsg_Snapshot::ID:
+ // We allow closing even if we are in the process of swapping out.
+ case ViewHostMsg_Close::ID:
+ // Sends an ACK.
+ case ViewHostMsg_RequestMove::ID:
+ // Updates browser state, and important for tests.
+ case ViewHostMsg_DomOperationResponse::ID:
+ // Suppresses dialog and sends an ACK.
+ case ViewHostMsg_RunJavaScriptMessage::ID:
+ // Suppresses dialog and sends an ACK.
+ case ViewHostMsg_RunBeforeUnloadConfirm::ID:
+ // Stops hang monitor.
+ case ViewHostMsg_ShouldClose_ACK::ID:
+ // We allow closing even if we are in the process of swapping out.
+ case ViewHostMsg_ClosePage_ACK::ID:
+ // Sends an ACK.
+ case ViewHostMsg_AccessibilityNotifications::ID:
+ return true;
+ default:
+ break;
+ }
+ return false;
+}
+
void RenderViewHost::Shutdown() {
// If we are being run modally (see RunModal), then we need to cleanup.
if (run_modal_reply_msg_) {
@@ -816,7 +898,8 @@ void RenderViewHost::OnMsgShowView(int route_id,
bool user_gesture) {
RenderViewHostDelegate::View* view = delegate_->GetViewDelegate();
if (view) {
- view->ShowCreatedWindow(route_id, disposition, initial_pos, user_gesture);
+ if (!is_swapped_out_)
+ view->ShowCreatedWindow(route_id, disposition, initial_pos, user_gesture);
Send(new ViewMsg_Move_ACK(route_id));
}
}
@@ -825,7 +908,8 @@ void RenderViewHost::OnMsgShowWidget(int route_id,
const gfx::Rect& initial_pos) {
RenderViewHostDelegate::View* view = delegate_->GetViewDelegate();
if (view) {
- view->ShowCreatedWidget(route_id, initial_pos);
+ if (!is_swapped_out_)
+ view->ShowCreatedWidget(route_id, initial_pos);
Send(new ViewMsg_Move_ACK(route_id));
}
}
@@ -833,7 +917,8 @@ void RenderViewHost::OnMsgShowWidget(int route_id,
void RenderViewHost::OnMsgShowFullscreenWidget(int route_id) {
RenderViewHostDelegate::View* view = delegate_->GetViewDelegate();
if (view) {
- view->ShowCreatedFullscreenWidget(route_id);
+ if (!is_swapped_out_)
+ view->ShowCreatedFullscreenWidget(route_id);
Send(new ViewMsg_Move_ACK(route_id));
}
}
@@ -955,7 +1040,8 @@ void RenderViewHost::OnMsgUpdateEncoding(const std::string& encoding_name) {
void RenderViewHost::OnMsgUpdateTargetURL(int32 page_id,
const GURL& url) {
- delegate_->UpdateTargetURL(page_id, url);
+ if (!is_swapped_out_)
+ delegate_->UpdateTargetURL(page_id, url);
// Send a notification back to the renderer that we are ready to
// receive more target urls.
@@ -974,7 +1060,8 @@ void RenderViewHost::OnMsgClose() {
}
void RenderViewHost::OnMsgRequestMove(const gfx::Rect& pos) {
- delegate_->RequestMove(pos);
+ if (!is_swapped_out_)
+ delegate_->RequestMove(pos);
Send(new ViewMsg_Move_ACK(routing_id()));
}
@@ -1095,8 +1182,8 @@ void RenderViewHost::OnMsgRunJavaScriptMessage(
// process input events.
process()->set_ignore_input_events(true);
StopHangMonitorTimeout();
- delegate_->RunJavaScriptMessage(message, default_prompt, frame_url, flags,
- reply_msg,
+ delegate_->RunJavaScriptMessage(this, message, default_prompt, frame_url,
+ flags, reply_msg,
&are_javascript_messages_suppressed_);
}
@@ -1107,7 +1194,7 @@ void RenderViewHost::OnMsgRunBeforeUnloadConfirm(const GURL& frame_url,
// shouldn't process input events.
process()->set_ignore_input_events(true);
StopHangMonitorTimeout();
- delegate_->RunBeforeUnloadConfirm(message, reply_msg);
+ delegate_->RunBeforeUnloadConfirm(this, message, reply_msg);
}
void RenderViewHost::MediaPlayerActionAt(const gfx::Point& location,
@@ -1128,7 +1215,7 @@ void RenderViewHost::OnMsgStartDragging(
const gfx::Point& image_offset) {
RenderViewHostDelegate::View* view = delegate_->GetViewDelegate();
if (view)
- view->StartDragging(drop_data, drag_operations_mask, image, image_offset);
+ view->StartDragging(drop_data, drag_operations_mask, image, image_offset);
}
void RenderViewHost::OnUpdateDragCursor(WebDragOperation current_op) {
@@ -1185,7 +1272,7 @@ void RenderViewHost::OnMsgShouldCloseACK(bool proceed) {
// If this renderer navigated while the beforeunload request was in flight, we
// may have cleared this state in OnMsgNavigate, in which case we can ignore
// this message.
- if (!is_waiting_for_beforeunload_ack_)
+ if (!is_waiting_for_beforeunload_ack_ || is_swapped_out_)
return;
is_waiting_for_beforeunload_ack_ = false;
@@ -1202,6 +1289,10 @@ void RenderViewHost::OnMsgShouldCloseACK(bool proceed) {
delegate_->DidCancelLoading();
}
+void RenderViewHost::OnMsgClosePageACK() {
+ ClosePageIgnoringUnloadEvents();
+}
+
void RenderViewHost::WindowMoveOrResizeStarted() {
Send(new ViewMsg_MoveOrResizeStarted(routing_id()));
}
@@ -1344,7 +1435,7 @@ void RenderViewHost::FilterURL(ChildProcessSecurityPolicy* policy,
void RenderViewHost::OnAccessibilityNotifications(
const std::vector<ViewHostMsg_AccessibilityNotification_Params>& params) {
- if (view())
+ if (view() && !is_swapped_out_)
view()->OnAccessibilityNotifications(params);
if (!params.empty()) {

Powered by Google App Engine
This is Rietveld 408576698