Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "content/browser/frame_host/render_frame_host_manager.h" | 5 #include "content/browser/frame_host/render_frame_host_manager.h" |
| 6 | 6 |
| 7 #include <utility> | 7 #include <utility> |
| 8 | 8 |
| 9 #include "base/command_line.h" | 9 #include "base/command_line.h" |
| 10 #include "base/debug/trace_event.h" | 10 #include "base/debug/trace_event.h" |
| 11 #include "base/logging.h" | 11 #include "base/logging.h" |
| 12 #include "base/stl_util.h" | 12 #include "base/stl_util.h" |
| 13 #include "content/browser/child_process_security_policy_impl.h" | 13 #include "content/browser/child_process_security_policy_impl.h" |
| 14 #include "content/browser/devtools/render_view_devtools_agent_host.h" | 14 #include "content/browser/devtools/render_view_devtools_agent_host.h" |
| 15 #include "content/browser/frame_host/cross_site_transferring_request.h" | 15 #include "content/browser/frame_host/cross_site_transferring_request.h" |
| 16 #include "content/browser/frame_host/debug_urls.h" | 16 #include "content/browser/frame_host/debug_urls.h" |
| 17 #include "content/browser/frame_host/interstitial_page_impl.h" | 17 #include "content/browser/frame_host/interstitial_page_impl.h" |
| 18 #include "content/browser/frame_host/navigation_before_commit_info.h" | 18 #include "content/browser/frame_host/navigation_before_commit_info.h" |
| 19 #include "content/browser/frame_host/navigation_controller_impl.h" | 19 #include "content/browser/frame_host/navigation_controller_impl.h" |
| 20 #include "content/browser/frame_host/navigation_entry_impl.h" | 20 #include "content/browser/frame_host/navigation_entry_impl.h" |
| 21 #include "content/browser/frame_host/navigation_request.h" | |
| 22 #include "content/browser/frame_host/navigation_request_info.h" | |
| 23 #include "content/browser/frame_host/navigator.h" | 21 #include "content/browser/frame_host/navigator.h" |
| 24 #include "content/browser/frame_host/render_frame_host_factory.h" | 22 #include "content/browser/frame_host/render_frame_host_factory.h" |
| 25 #include "content/browser/frame_host/render_frame_host_impl.h" | 23 #include "content/browser/frame_host/render_frame_host_impl.h" |
| 26 #include "content/browser/frame_host/render_frame_proxy_host.h" | 24 #include "content/browser/frame_host/render_frame_proxy_host.h" |
| 27 #include "content/browser/renderer_host/render_process_host_impl.h" | 25 #include "content/browser/renderer_host/render_process_host_impl.h" |
| 28 #include "content/browser/renderer_host/render_view_host_factory.h" | 26 #include "content/browser/renderer_host/render_view_host_factory.h" |
| 29 #include "content/browser/renderer_host/render_view_host_impl.h" | 27 #include "content/browser/renderer_host/render_view_host_impl.h" |
| 30 #include "content/browser/site_instance_impl.h" | 28 #include "content/browser/site_instance_impl.h" |
| 31 #include "content/browser/webui/web_ui_controller_factory_registry.h" | 29 #include "content/browser/webui/web_ui_controller_factory_registry.h" |
| 32 #include "content/browser/webui/web_ui_impl.h" | 30 #include "content/browser/webui/web_ui_impl.h" |
| 33 #include "content/common/navigation_params.h" | 31 #include "content/common/frame_messages.h" |
| 34 #include "content/common/view_messages.h" | 32 #include "content/common/view_messages.h" |
| 35 #include "content/public/browser/content_browser_client.h" | 33 #include "content/public/browser/content_browser_client.h" |
| 36 #include "content/public/browser/notification_service.h" | 34 #include "content/public/browser/notification_service.h" |
| 37 #include "content/public/browser/notification_types.h" | 35 #include "content/public/browser/notification_types.h" |
| 38 #include "content/public/browser/render_widget_host_iterator.h" | 36 #include "content/public/browser/render_widget_host_iterator.h" |
| 39 #include "content/public/browser/render_widget_host_view.h" | 37 #include "content/public/browser/render_widget_host_view.h" |
| 40 #include "content/public/browser/user_metrics.h" | 38 #include "content/public/browser/user_metrics.h" |
| 41 #include "content/public/browser/web_ui_controller.h" | 39 #include "content/public/browser/web_ui_controller.h" |
| 42 #include "content/public/common/content_switches.h" | 40 #include "content/public/common/content_switches.h" |
| 43 #include "content/public/common/referrer.h" | 41 #include "content/public/common/referrer.h" |
| 44 #include "content/public/common/url_constants.h" | 42 #include "content/public/common/url_constants.h" |
| 45 #include "net/base/load_flags.h" | |
| 46 | 43 |
| 47 namespace content { | 44 namespace content { |
| 48 | 45 |
| 49 // PlzNavigate | |
| 50 // Returns the net load flags to use based on the navigation type. | |
| 51 // TODO(clamy): unify the code with what is happening on the renderer side. | |
| 52 int LoadFlagFromNavigationType(FrameMsg_Navigate_Type::Value navigation_type) { | |
| 53 int load_flags = net::LOAD_NORMAL; | |
| 54 switch (navigation_type) { | |
| 55 case FrameMsg_Navigate_Type::RELOAD: | |
| 56 case FrameMsg_Navigate_Type::RELOAD_ORIGINAL_REQUEST_URL: | |
| 57 load_flags |= net::LOAD_VALIDATE_CACHE; | |
| 58 break; | |
| 59 case FrameMsg_Navigate_Type::RELOAD_IGNORING_CACHE: | |
| 60 load_flags |= net::LOAD_BYPASS_CACHE; | |
| 61 break; | |
| 62 case FrameMsg_Navigate_Type::RESTORE: | |
| 63 load_flags |= net::LOAD_PREFERRING_CACHE; | |
| 64 break; | |
| 65 case FrameMsg_Navigate_Type::RESTORE_WITH_POST: | |
| 66 load_flags |= net::LOAD_ONLY_FROM_CACHE; | |
| 67 break; | |
| 68 case FrameMsg_Navigate_Type::NORMAL: | |
| 69 default: | |
| 70 break; | |
| 71 } | |
| 72 return load_flags; | |
| 73 } | |
| 74 | |
| 75 // PlzNavigate | |
| 76 // Generates a default FrameHostMsg_BeginNavigation_Params to be used when there | |
| 77 // is no live renderer. | |
| 78 FrameHostMsg_BeginNavigation_Params MakeDefaultBeginNavigation( | |
| 79 const RequestNavigationParams& request_params, | |
| 80 FrameMsg_Navigate_Type::Value navigation_type) { | |
| 81 FrameHostMsg_BeginNavigation_Params begin_navigation_params; | |
| 82 begin_navigation_params.method = request_params.is_post ? "POST" : "GET"; | |
| 83 begin_navigation_params.load_flags = | |
| 84 LoadFlagFromNavigationType(navigation_type); | |
| 85 | |
| 86 // TODO(clamy): Post data from the browser should be put in the request body. | |
| 87 // Headers should be filled in as well. | |
| 88 | |
| 89 begin_navigation_params.has_user_gesture = false; | |
| 90 return begin_navigation_params; | |
| 91 } | |
| 92 | |
| 93 bool RenderFrameHostManager::ClearRFHsPendingShutdown(FrameTreeNode* node) { | 46 bool RenderFrameHostManager::ClearRFHsPendingShutdown(FrameTreeNode* node) { |
| 94 node->render_manager()->pending_delete_hosts_.clear(); | 47 node->render_manager()->pending_delete_hosts_.clear(); |
| 95 return true; | 48 return true; |
| 96 } | 49 } |
| 97 | 50 |
| 98 RenderFrameHostManager::RenderFrameHostManager( | 51 RenderFrameHostManager::RenderFrameHostManager( |
| 99 FrameTreeNode* frame_tree_node, | 52 FrameTreeNode* frame_tree_node, |
| 100 RenderFrameHostDelegate* render_frame_delegate, | 53 RenderFrameHostDelegate* render_frame_delegate, |
| 101 RenderViewHostDelegate* render_view_delegate, | 54 RenderViewHostDelegate* render_view_delegate, |
| 102 RenderWidgetHostDelegate* render_widget_delegate, | 55 RenderWidgetHostDelegate* render_widget_delegate, |
| (...skipping 11 matching lines...) Expand all Loading... | |
| 114 | 67 |
| 115 RenderFrameHostManager::~RenderFrameHostManager() { | 68 RenderFrameHostManager::~RenderFrameHostManager() { |
| 116 if (pending_render_frame_host_) | 69 if (pending_render_frame_host_) |
| 117 CancelPending(); | 70 CancelPending(); |
| 118 | 71 |
| 119 // We should always have a current RenderFrameHost except in some tests. | 72 // We should always have a current RenderFrameHost except in some tests. |
| 120 SetRenderFrameHost(scoped_ptr<RenderFrameHostImpl>()); | 73 SetRenderFrameHost(scoped_ptr<RenderFrameHostImpl>()); |
| 121 | 74 |
| 122 // Delete any swapped out RenderFrameHosts. | 75 // Delete any swapped out RenderFrameHosts. |
| 123 STLDeleteValues(&proxy_hosts_); | 76 STLDeleteValues(&proxy_hosts_); |
| 124 | |
| 125 // PlzNavigate | |
| 126 // There is an active navigation request for this RFHM so it needs to be | |
| 127 // canceled. | |
| 128 if (CommandLine::ForCurrentProcess()->HasSwitch( | |
| 129 switches::kEnableBrowserSideNavigation)) { | |
| 130 if (navigation_request_.get()) | |
| 131 navigation_request_->CancelNavigation(); | |
| 132 } | |
| 133 | |
| 134 } | 77 } |
| 135 | 78 |
| 136 void RenderFrameHostManager::Init(BrowserContext* browser_context, | 79 void RenderFrameHostManager::Init(BrowserContext* browser_context, |
| 137 SiteInstance* site_instance, | 80 SiteInstance* site_instance, |
| 138 int view_routing_id, | 81 int view_routing_id, |
| 139 int frame_routing_id) { | 82 int frame_routing_id) { |
| 140 // Create a RenderViewHost and RenderFrameHost, once we have an instance. It | 83 // Create a RenderViewHost and RenderFrameHost, once we have an instance. It |
| 141 // is important to immediately give this SiteInstance to a RenderViewHost so | 84 // is important to immediately give this SiteInstance to a RenderViewHost so |
| 142 // that the SiteInstance is ref counted. | 85 // that the SiteInstance is ref counted. |
| 143 if (!site_instance) | 86 if (!site_instance) |
| (...skipping 302 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 446 static_cast<RenderProcessHostImpl*>(render_frame_host_->GetProcess()); | 389 static_cast<RenderProcessHostImpl*>(render_frame_host_->GetProcess()); |
| 447 process->ResumeResponseDeferredAtStart(*response_started_id_); | 390 process->ResumeResponseDeferredAtStart(*response_started_id_); |
| 448 | 391 |
| 449 render_frame_host_->ClearPendingTransitionRequestData(); | 392 render_frame_host_->ClearPendingTransitionRequestData(); |
| 450 | 393 |
| 451 response_started_id_.reset(); | 394 response_started_id_.reset(); |
| 452 } | 395 } |
| 453 | 396 |
| 454 void RenderFrameHostManager::DidNavigateFrame( | 397 void RenderFrameHostManager::DidNavigateFrame( |
| 455 RenderFrameHostImpl* render_frame_host) { | 398 RenderFrameHostImpl* render_frame_host) { |
| 456 // PlzNavigate | |
| 457 // The navigation request has been committed so the browser process doesn't | |
| 458 // need to care about it anymore. | |
| 459 if (CommandLine::ForCurrentProcess()->HasSwitch( | |
| 460 switches::kEnableBrowserSideNavigation)) { | |
| 461 navigation_request_.reset(); | |
| 462 } | |
| 463 | |
| 464 if (!cross_navigation_pending_) { | 399 if (!cross_navigation_pending_) { |
| 465 DCHECK(!pending_render_frame_host_); | 400 DCHECK(!pending_render_frame_host_); |
| 466 | 401 |
| 467 // We should only hear this from our current renderer. | 402 // We should only hear this from our current renderer. |
| 468 DCHECK_EQ(render_frame_host_, render_frame_host); | 403 DCHECK_EQ(render_frame_host_, render_frame_host); |
| 469 | 404 |
| 470 // Even when there is no pending RVH, there may be a pending Web UI. | 405 // Even when there is no pending RVH, there may be a pending Web UI. |
| 471 if (pending_web_ui()) | 406 if (pending_web_ui()) |
| 472 CommitPending(); | 407 CommitPending(); |
| 473 return; | 408 return; |
| (...skipping 99 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 573 pending_delete_hosts_.find(site_instance_id); | 508 pending_delete_hosts_.find(site_instance_id); |
| 574 if (iter != pending_delete_hosts_.end() && iter->second.get() == rfh) | 509 if (iter != pending_delete_hosts_.end() && iter->second.get() == rfh) |
| 575 pending_delete_hosts_.erase(site_instance_id); | 510 pending_delete_hosts_.erase(site_instance_id); |
| 576 } | 511 } |
| 577 | 512 |
| 578 void RenderFrameHostManager::ResetProxyHosts() { | 513 void RenderFrameHostManager::ResetProxyHosts() { |
| 579 STLDeleteValues(&proxy_hosts_); | 514 STLDeleteValues(&proxy_hosts_); |
| 580 } | 515 } |
| 581 | 516 |
| 582 // PlzNavigate | 517 // PlzNavigate |
| 583 bool RenderFrameHostManager::RequestNavigation( | 518 RenderFrameHostImpl* RenderFrameHostManager::GetFrameHostForNavigation( |
|
nasko
2014/10/01 15:59:50
I just realized that this method acts as though --
clamy
2014/10/01 17:10:43
Well it's pretty close, but UpdateStateForNavigate
nasko
2014/10/01 18:20:19
Either helper functions or flag checks will be fin
clamy
2014/10/01 22:30:47
Done.
| |
| 584 scoped_ptr<NavigationRequest> navigation_request, | 519 const GURL& url, |
| 585 const RequestNavigationParams& request_params) { | 520 ui::PageTransition transition) { |
| 586 CHECK(CommandLine::ForCurrentProcess()->HasSwitch( | 521 CHECK(CommandLine::ForCurrentProcess()->HasSwitch( |
| 587 switches::kEnableBrowserSideNavigation)); | 522 switches::kEnableBrowserSideNavigation)); |
| 588 | 523 |
| 589 // TODO(clamy): Check if navigations are blocked and if so store the | |
| 590 // parameters. | |
| 591 | |
| 592 // If there is an ongoing request it must be canceled. | |
| 593 if (navigation_request_.get()) | |
| 594 navigation_request_->CancelNavigation(); | |
| 595 | |
| 596 navigation_request_ = navigation_request.Pass(); | |
| 597 | |
| 598 if (render_frame_host_->IsRenderFrameLive()) { | |
| 599 // TODO(clamy): send a RequestNavigation IPC. | |
| 600 return true; | |
| 601 } | |
| 602 | |
| 603 // The navigation request is sent directly to the IO thread. | |
| 604 OnBeginNavigation( | |
| 605 MakeDefaultBeginNavigation( | |
| 606 request_params, navigation_request_->common_params().navigation_type), | |
| 607 navigation_request_->common_params()); | |
| 608 return true; | |
| 609 } | |
| 610 | |
| 611 // PlzNavigate | |
| 612 void RenderFrameHostManager::OnBeginNavigation( | |
| 613 const FrameHostMsg_BeginNavigation_Params& params, | |
| 614 const CommonNavigationParams& common_params) { | |
| 615 CHECK(CommandLine::ForCurrentProcess()->HasSwitch( | |
| 616 switches::kEnableBrowserSideNavigation)); | |
| 617 // TODO(clamy): In case of a renderer initiated navigation create a new | |
| 618 // NavigationRequest. | |
| 619 DCHECK(navigation_request_.get()); | |
| 620 // Update the referrer with the one received from the renderer. | |
| 621 navigation_request_->common_params().referrer = common_params.referrer; | |
| 622 | |
| 623 scoped_ptr<NavigationRequestInfo> info(new NavigationRequestInfo(params)); | |
| 624 | |
| 625 info->first_party_for_cookies = | |
| 626 frame_tree_node_->IsMainFrame() | |
| 627 ? navigation_request_->common_params().url | |
| 628 : frame_tree_node_->frame_tree()->root()->current_url(); | |
| 629 info->is_main_frame = frame_tree_node_->IsMainFrame(); | |
| 630 info->parent_is_main_frame = !frame_tree_node_->parent() ? | |
| 631 false : frame_tree_node_->parent()->IsMainFrame(); | |
| 632 | |
| 633 // TODO(clamy): Check if the current RFH should be initialized (in case it has | |
| 634 // crashed) not to display a sad tab while navigating. | |
| 635 // TODO(clamy): Spawn a speculative renderer process if we do not have one to | |
| 636 // use for the navigation. | |
| 637 | |
| 638 navigation_request_->BeginNavigation(info.Pass(), params.request_body); | |
| 639 } | |
| 640 | |
| 641 // PlzNavigate | |
| 642 void RenderFrameHostManager::CommitNavigation( | |
| 643 const NavigationBeforeCommitInfo& info) { | |
| 644 CHECK(CommandLine::ForCurrentProcess()->HasSwitch( | |
| 645 switches::kEnableBrowserSideNavigation)); | |
| 646 DCHECK(navigation_request_.get()); | |
| 647 // Ignores navigation commits if the request ID doesn't match the current | |
| 648 // active request. | |
| 649 if (navigation_request_->navigation_request_id() != | |
| 650 info.navigation_request_id) { | |
| 651 return; | |
| 652 } | |
| 653 | |
| 654 // Pick the right RenderFrameHost to commit the navigation. | 524 // Pick the right RenderFrameHost to commit the navigation. |
| 655 SiteInstance* current_instance = render_frame_host_->GetSiteInstance(); | 525 SiteInstance* current_instance = render_frame_host_->GetSiteInstance(); |
| 656 // TODO(clamy): Replace the default values by the right ones. This may require | 526 // TODO(clamy): Replace the default values by the right ones. |
| 657 // some storing in RequestNavigation. | |
| 658 scoped_refptr<SiteInstance> new_instance = GetSiteInstanceForNavigation( | 527 scoped_refptr<SiteInstance> new_instance = GetSiteInstanceForNavigation( |
| 659 info.navigation_url, | 528 url, |
| 660 NULL, | 529 NULL, |
| 661 navigation_request_->common_params().transition, | 530 transition, |
| 662 false, | 531 false, |
| 663 false); | 532 false); |
| 664 DCHECK(!pending_render_frame_host_.get()); | 533 DCHECK(!pending_render_frame_host_.get()); |
| 665 | 534 |
| 666 // TODO(clamy): Update how pending WebUI objects are handled. | 535 // TODO(clamy): Update how pending WebUI objects are handled. |
| 667 if (current_instance != new_instance.get()) { | 536 if (current_instance != new_instance.get()) { |
| 668 CreateRenderFrameHostForNewSiteInstance( | 537 CreateRenderFrameHostForNewSiteInstance( |
| 669 current_instance, new_instance.get(), frame_tree_node_->IsMainFrame()); | 538 current_instance, new_instance.get(), frame_tree_node_->IsMainFrame()); |
| 670 DCHECK(pending_render_frame_host_.get()); | 539 DCHECK(pending_render_frame_host_.get()); |
| 671 // TODO(clamy): Wait until the navigation has committed before swapping | 540 // TODO(clamy): Wait until the navigation has committed before swapping |
| 672 // renderers. | 541 // renderers. |
| 673 scoped_ptr<RenderFrameHostImpl> old_render_frame_host = | 542 scoped_ptr<RenderFrameHostImpl> old_render_frame_host = |
| 674 SetRenderFrameHost(pending_render_frame_host_.Pass()); | 543 SetRenderFrameHost(pending_render_frame_host_.Pass()); |
| 675 if (frame_tree_node_->IsMainFrame()) | 544 if (frame_tree_node_->IsMainFrame()) |
| 676 render_frame_host_->render_view_host()->AttachToFrameTree(); | 545 render_frame_host_->render_view_host()->AttachToFrameTree(); |
| 677 } | 546 } |
| 678 | 547 |
| 679 // If the renderer that needs to navigate is not live (it was just created or | 548 // If the renderer that needs to navigate is not live (it was just created or |
| 680 // it crashed), initialize it. | 549 // it crashed), initialize it. |
| 681 if (!render_frame_host_->render_view_host()->IsRenderViewLive()) { | 550 if (!render_frame_host_->render_view_host()->IsRenderViewLive()) { |
| 682 // Recreate the opener chain. | 551 // Recreate the opener chain. |
| 683 int opener_route_id = delegate_->CreateOpenerRenderViewsForRenderManager( | 552 int opener_route_id = delegate_->CreateOpenerRenderViewsForRenderManager( |
| 684 render_frame_host_->GetSiteInstance()); | 553 render_frame_host_->GetSiteInstance()); |
| 685 if (!InitRenderView(render_frame_host_->render_view_host(), | 554 if (!InitRenderView(render_frame_host_->render_view_host(), |
| 686 opener_route_id, | 555 opener_route_id, |
| 687 MSG_ROUTING_NONE, | 556 MSG_ROUTING_NONE, |
| 688 frame_tree_node_->IsMainFrame())) { | 557 frame_tree_node_->IsMainFrame())) { |
| 689 return; | 558 return NULL; |
| 690 } | 559 } |
| 691 } | 560 } |
| 692 | 561 return render_frame_host_.get(); |
| 693 frame_tree_node_->navigator()->CommitNavigation( | |
| 694 render_frame_host_.get(), | |
| 695 info.stream_url, | |
| 696 navigation_request_->common_params(), | |
| 697 navigation_request_->commit_params()); | |
| 698 } | 562 } |
| 699 | 563 |
| 700 void RenderFrameHostManager::Observe( | 564 void RenderFrameHostManager::Observe( |
| 701 int type, | 565 int type, |
| 702 const NotificationSource& source, | 566 const NotificationSource& source, |
| 703 const NotificationDetails& details) { | 567 const NotificationDetails& details) { |
| 704 switch (type) { | 568 switch (type) { |
| 705 case NOTIFICATION_RENDERER_PROCESS_CLOSED: | 569 case NOTIFICATION_RENDERER_PROCESS_CLOSED: |
| 706 case NOTIFICATION_RENDERER_PROCESS_CLOSING: | 570 case NOTIFICATION_RENDERER_PROCESS_CLOSING: |
| 707 RendererProcessClosing( | 571 RendererProcessClosing( |
| (...skipping 1036 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1744 void RenderFrameHostManager::DeleteRenderFrameProxyHost( | 1608 void RenderFrameHostManager::DeleteRenderFrameProxyHost( |
| 1745 SiteInstance* instance) { | 1609 SiteInstance* instance) { |
| 1746 RenderFrameProxyHostMap::iterator iter = proxy_hosts_.find(instance->GetId()); | 1610 RenderFrameProxyHostMap::iterator iter = proxy_hosts_.find(instance->GetId()); |
| 1747 if (iter != proxy_hosts_.end()) { | 1611 if (iter != proxy_hosts_.end()) { |
| 1748 delete iter->second; | 1612 delete iter->second; |
| 1749 proxy_hosts_.erase(iter); | 1613 proxy_hosts_.erase(iter); |
| 1750 } | 1614 } |
| 1751 } | 1615 } |
| 1752 | 1616 |
| 1753 } // namespace content | 1617 } // namespace content |
| OLD | NEW |