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

Unified Diff: content/renderer/render_view.cc

Issue 6319001: Support window.opener after a process swap. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Update comments. 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
« no previous file with comments | « content/renderer/render_view.h ('k') | no next file » | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: content/renderer/render_view.cc
diff --git a/content/renderer/render_view.cc b/content/renderer/render_view.cc
index 4eea883eba10f8cfb8c037525c14b427422a00e0..ccc4b70f1d5d5cc6148b19c05ad8ec8c0dd88fb0 100644
--- a/content/renderer/render_view.cc
+++ b/content/renderer/render_view.cc
@@ -481,6 +481,7 @@ RenderView::RenderView(RenderThreadBase* render_thread,
opened_by_user_gesture_(true),
opener_suppressed_(false),
is_prerendering_(false),
+ is_swapped_out_(false),
page_id_(-1),
last_page_id_sent_to_browser_(-1),
last_indexed_page_id_(-1),
@@ -663,6 +664,15 @@ void RenderView::RemoveObserver(RenderViewObserver* observer) {
observers_.RemoveObserver(observer);
}
+void RenderView::SwapIn() {
+ DCHECK(is_swapped_out_);
+ is_swapped_out_ = false;
+
+ // Increment the ChildProcess's ref count now that this RenderView is active
+ // again. This prevents the process from exiting.
+ RenderProcess::current()->AddRefProcess();
+}
+
bool RenderView::RendererAccessibilityNotification::ShouldIncludeChildren() {
typedef ViewHostMsg_AccessibilityNotification_Params params;
if (type == WebKit::WebAccessibilityNotificationChildrenChanged ||
@@ -945,6 +955,7 @@ bool RenderView::OnMessageReceived(const IPC::Message& message) {
OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)
IPC_MESSAGE_HANDLER(ExtensionMsg_GetApplicationInfo, OnGetApplicationInfo)
IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnShouldClose)
+ IPC_MESSAGE_HANDLER(ViewMsg_SwapOut, OnSwapOut)
IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)
IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)
IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,
@@ -1233,6 +1244,11 @@ void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {
if (!webview())
return;
+ // Swap this renderer back in if necessary.
+ if (is_swapped_out_) {
+ SwapIn();
+ }
+
history_list_offset_ = params.current_history_list_offset;
history_list_length_ = params.current_history_list_length;
@@ -1487,6 +1503,14 @@ void RenderView::UpdateURL(WebFrame* frame) {
const WebURLRequest& original_request = ds->originalRequest();
const WebURLResponse& response = ds->response();
+ // Don't send an update if this renderer is swapped out.
+ // TODO(creis): This won't be necessary once OnSwapOut is fixed to clear
+ // the frame without a full navigation.
+ if (is_swapped_out_) {
+ DCHECK(request.url() == GURL(chrome::kAboutBlankURL));
+ return;
+ }
+
NavigationState* navigation_state = NavigationState::FromDataSource(ds);
DCHECK(navigation_state);
@@ -2176,6 +2200,12 @@ bool RenderView::runModalPromptDialog(
bool RenderView::runModalBeforeUnloadDialog(
WebFrame* frame, const WebString& message) {
+ // If we are swapping out, we have already run the beforeunload handler.
+ // TODO(creis): Fix OnSwapOut to clear the frame without running beforeunload
+ // at all, to avoid running it twice.
+ if (is_swapped_out_)
+ return true;
+
bool success = false;
// This is an ignored return value, but is included so we can accept the same
// response as RunJavaScriptMessage.
@@ -4427,7 +4457,39 @@ void RenderView::OnShouldClose() {
Send(new ViewHostMsg_ShouldClose_ACK(routing_id_, should_close));
}
-void RenderView::OnClosePage(const ViewMsg_ClosePage_Params& params) {
+void RenderView::OnSwapOut(const ViewMsg_SwapOut_Params& params) {
+ // Swap this RenderView out so the tab can navigate to a page rendered by a
+ // different process. This involves running the unload handler, clearing
+ // the page, and allowing the process to exit if there are no other active
+ // RenderViews in it.
+ DCHECK(!is_swapped_out_);
+ is_swapped_out_ = true;
+
+ // TODO(davemoore) This code should be removed once willClose() gets
+ // called when a page is destroyed. page_load_histograms_.Dump() is safe
+ // to call multiple times for the same frame, but it will simplify things.
+ page_load_histograms_.Dump(webview()->mainFrame());
+ page_load_histograms_.ResetCrossFramePropertyAccess();
+
+ // Send an UpdateState message before we get swapped out.
+ SyncNavigationState();
+
+ // Replace the page with a blank dummy URL, triggering the unload handler.
+ // TODO(creis): Need to add a better way to do this that avoids running the
+ // beforeunload handler and that can't match an existing URL.
+ WebURLRequest request(GURL(chrome::kAboutBlankURL));
+ webview()->mainFrame()->loadRequest(request);
Charlie Reis 2011/04/13 22:57:21 This is the main sticking point right now. We nee
+
+ // Just echo back the params in the ACK.
+ Send(new ViewHostMsg_SwapOut_ACK(routing_id_, params));
+
+ // Remove this RenderView from the ChildProcess's ref count, allowing the
+ // process to exit if all of the RenderViews are swapped out. We will
+ // increment the count if this RenderView is swapped back in.
+ RenderProcess::current()->ReleaseProcess();
+}
+
+void RenderView::OnClosePage() {
// TODO(creis): We'd rather use webview()->Close() here, but that currently
// sets the WebView's delegate_ to NULL, preventing any JavaScript dialogs
// in the onunload handler from appearing. For now, we're bypassing that and
@@ -4442,8 +4504,7 @@ void RenderView::OnClosePage(const ViewMsg_ClosePage_Params& params) {
page_load_histograms_.ResetCrossFramePropertyAccess();
webview()->dispatchUnloadEvent();
- // Just echo back the params in the ACK.
- Send(new ViewHostMsg_ClosePage_ACK(routing_id_, params));
+ Send(new ViewHostMsg_ClosePage_ACK(routing_id_));
}
void RenderView::OnThemeChanged() {
« no previous file with comments | « content/renderer/render_view.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698