 Chromium Code Reviews
 Chromium Code Reviews Issue 6319001:
  Support window.opener after a process swap.   (Closed) 
  Base URL: svn://svn.chromium.org/chrome/trunk/src
    
  
    Issue 6319001:
  Support window.opener after a process swap.   (Closed) 
  Base URL: svn://svn.chromium.org/chrome/trunk/src| 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() { |