|  | // Copyright 2013 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/command_line.h" | 
|  | #include "base/run_loop.h" | 
|  | #include "base/synchronization/waitable_event.h" | 
|  | #include "base/test/scoped_feature_list.h" | 
|  | #include "base/test/test_timeouts.h" | 
|  | #include "build/build_config.h" | 
|  | #include "content/browser/frame_host/render_frame_host_impl.h" | 
|  | #include "content/browser/renderer_host/render_process_host_impl.h" | 
|  | #include "content/public/browser/browser_thread.h" | 
|  | #include "content/public/browser/child_process_launcher_utils.h" | 
|  | #include "content/public/browser/render_frame_host.h" | 
|  | #include "content/public/browser/render_process_host.h" | 
|  | #include "content/public/browser/render_process_host_observer.h" | 
|  | #include "content/public/browser/web_contents.h" | 
|  | #include "content/public/browser/web_contents_observer.h" | 
|  | #include "content/public/common/content_features.h" | 
|  | #include "content/public/common/content_switches.h" | 
|  | #include "content/public/common/url_constants.h" | 
|  | #include "content/public/test/browser_test_utils.h" | 
|  | #include "content/public/test/content_browser_test.h" | 
|  | #include "content/public/test/content_browser_test_utils.h" | 
|  | #include "content/public/test/test_service.mojom.h" | 
|  | #include "content/public/test/test_utils.h" | 
|  | #include "content/shell/browser/shell.h" | 
|  | #include "content/shell/browser/shell_browser_context.h" | 
|  | #include "content/shell/browser/shell_browser_main_parts.h" | 
|  | #include "content/shell/browser/shell_content_browser_client.h" | 
|  | #include "content/test/test_content_browser_client.h" | 
|  | #include "media/base/bind_to_current_loop.h" | 
|  | #include "media/base/media_switches.h" | 
|  | #include "media/base/test_data_util.h" | 
|  | #include "media/mojo/buildflags.h" | 
|  | #include "net/test/embedded_test_server/embedded_test_server.h" | 
|  | #include "net/test/embedded_test_server/http_request.h" | 
|  | #include "net/test/embedded_test_server/http_response.h" | 
|  |  | 
|  | #if defined(OS_WIN) | 
|  | #include "base/win/windows_version.h" | 
|  | #endif | 
|  |  | 
|  | namespace content { | 
|  | namespace { | 
|  |  | 
|  | std::unique_ptr<net::test_server::HttpResponse> HandleBeacon( | 
|  | const net::test_server::HttpRequest& request) { | 
|  | if (request.relative_url != "/beacon") | 
|  | return nullptr; | 
|  | return std::make_unique<net::test_server::BasicHttpResponse>(); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<net::test_server::HttpResponse> HandleHungBeacon( | 
|  | const net::test_server::HttpRequest& request) { | 
|  | if (request.relative_url != "/beacon") | 
|  | return nullptr; | 
|  | return std::make_unique<net::test_server::HungResponse>(); | 
|  | } | 
|  |  | 
|  | class RenderProcessHostTest : public ContentBrowserTest, | 
|  | public RenderProcessHostObserver { | 
|  | public: | 
|  | RenderProcessHostTest() : process_exits_(0), host_destructions_(0) {} | 
|  |  | 
|  | void SetUpCommandLine(base::CommandLine* command_line) override { | 
|  | command_line->AppendSwitchASCII( | 
|  | switches::kAutoplayPolicy, | 
|  | switches::autoplay::kNoUserGestureRequiredPolicy); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | void set_process_exit_callback(const base::Closure& callback) { | 
|  | process_exit_callback_ = callback; | 
|  | } | 
|  |  | 
|  | // RenderProcessHostObserver: | 
|  | void RenderProcessExited(RenderProcessHost* host, | 
|  | const ChildProcessTerminationInfo& info) override { | 
|  | ++process_exits_; | 
|  | if (!process_exit_callback_.is_null()) | 
|  | process_exit_callback_.Run(); | 
|  | } | 
|  | void RenderProcessHostDestroyed(RenderProcessHost* host) override { | 
|  | ++host_destructions_; | 
|  | } | 
|  | void WaitUntilProcessExits(int target) { | 
|  | while (process_exits_ < target) | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | } | 
|  |  | 
|  | int process_exits_; | 
|  | int host_destructions_; | 
|  | base::Closure process_exit_callback_; | 
|  | }; | 
|  |  | 
|  | // A mock ContentBrowserClient that only considers a spare renderer to be a | 
|  | // suitable host. | 
|  | class SpareRendererContentBrowserClient : public TestContentBrowserClient { | 
|  | public: | 
|  | bool IsSuitableHost(RenderProcessHost* process_host, | 
|  | const GURL& site_url) override { | 
|  | if (RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()) { | 
|  | return process_host == | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | } | 
|  | return true; | 
|  | } | 
|  | }; | 
|  |  | 
|  | // A mock ContentBrowserClient that only considers a non-spare renderer to be a | 
|  | // suitable host, but otherwise tries to reuse processes. | 
|  | class NonSpareRendererContentBrowserClient : public TestContentBrowserClient { | 
|  | public: | 
|  | NonSpareRendererContentBrowserClient() = default; | 
|  |  | 
|  | bool IsSuitableHost(RenderProcessHost* process_host, | 
|  | const GURL& site_url) override { | 
|  | return RenderProcessHostImpl::GetSpareRenderProcessHostForTesting() != | 
|  | process_host; | 
|  | } | 
|  |  | 
|  | bool ShouldTryToUseExistingProcessHost(BrowserContext* context, | 
|  | const GURL& url) override { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ShouldUseSpareRenderProcessHost(BrowserContext* browser_context, | 
|  | const GURL& site_url) override { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(NonSpareRendererContentBrowserClient); | 
|  | }; | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | GuestsAreNotSuitableHosts) { | 
|  | // Set max renderers to 1 to force running out of processes. | 
|  | RenderProcessHost::SetMaxRendererProcessCount(1); | 
|  |  | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | NavigateToURL(shell(), test_url); | 
|  | RenderProcessHost* rph = | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess(); | 
|  | // Make it believe it's a guest. | 
|  | reinterpret_cast<RenderProcessHostImpl*>(rph)-> | 
|  | set_is_for_guests_only_for_testing(true); | 
|  | EXPECT_EQ(1, RenderProcessHost::GetCurrentRenderProcessCountForTesting()); | 
|  |  | 
|  | // Navigate to a different page. | 
|  | GURL::Replacements replace_host; | 
|  | replace_host.SetHostStr("localhost"); | 
|  | GURL another_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | another_url = another_url.ReplaceComponents(replace_host); | 
|  | NavigateToURL(CreateBrowser(), another_url); | 
|  |  | 
|  | // Expect that we got another process (the guest renderer was not reused). | 
|  | EXPECT_EQ(2, RenderProcessHost::GetCurrentRenderProcessCountForTesting()); | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareRenderProcessHostTaken) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | EXPECT_NE(nullptr, spare_renderer); | 
|  |  | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | Shell* window = CreateBrowser(); | 
|  | NavigateToURL(window, test_url); | 
|  |  | 
|  | EXPECT_EQ(spare_renderer, | 
|  | window->web_contents()->GetMainFrame()->GetProcess()); | 
|  |  | 
|  | // The old spare render process host should no longer be available. | 
|  | EXPECT_NE(spare_renderer, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  |  | 
|  | // Check if a fresh spare is available (depending on the operating mode). | 
|  | if (RenderProcessHostImpl::IsSpareProcessKeptAtAllTimes()) { | 
|  | EXPECT_NE(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } else { | 
|  | EXPECT_EQ(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareRenderProcessHostNotTaken) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->off_the_record_browser_context()); | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | Shell* window = CreateBrowser(); | 
|  | NavigateToURL(window, test_url); | 
|  |  | 
|  | // There should have been another process created for the navigation. | 
|  | EXPECT_NE(spare_renderer, | 
|  | window->web_contents()->GetMainFrame()->GetProcess()); | 
|  |  | 
|  | // Check if a fresh spare is available (depending on the operating mode). | 
|  | // Note this behavior is identical to what would have happened if the | 
|  | // RenderProcessHost were taken. | 
|  | if (RenderProcessHostImpl::IsSpareProcessKeptAtAllTimes()) { | 
|  | EXPECT_NE(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } else { | 
|  | EXPECT_EQ(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareRenderProcessHostKilled) { | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  |  | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | mojom::TestServicePtr service; | 
|  | ASSERT_NE(nullptr, spare_renderer); | 
|  | BindInterface(spare_renderer, &service); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | set_process_exit_callback(run_loop.QuitClosure()); | 
|  | spare_renderer->AddObserver(this);  // For process_exit_callback. | 
|  |  | 
|  | // Should reply with a bad message and cause process death. | 
|  | service->DoSomething(base::DoNothing()); | 
|  | run_loop.Run(); | 
|  |  | 
|  | // The spare RenderProcessHost should disappear when its process dies. | 
|  | EXPECT_EQ(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } | 
|  |  | 
|  | // Test that the spare renderer works correctly when the limit on the maximum | 
|  | // number of processes is small. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | SpareRendererSurpressedMaxProcesses) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | SpareRendererContentBrowserClient browser_client; | 
|  | ContentBrowserClient* old_client = | 
|  | SetBrowserClientForTesting(&browser_client); | 
|  |  | 
|  | RenderProcessHost::SetMaxRendererProcessCount(1); | 
|  |  | 
|  | // A process is created with shell startup, so with a maximum of one renderer | 
|  | // process the spare RPH should not be created. | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  | EXPECT_EQ(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  |  | 
|  | // A spare RPH should be created with a max of 2 renderer processes. | 
|  | RenderProcessHost::SetMaxRendererProcessCount(2); | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | EXPECT_NE(nullptr, spare_renderer); | 
|  |  | 
|  | // Thanks to the injected SpareRendererContentBrowserClient and the limit on | 
|  | // processes, the spare RPH will always be used via GetExistingProcessHost() | 
|  | // rather than picked up via MaybeTakeSpareRenderProcessHost(). | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | Shell* new_window = CreateBrowser(); | 
|  | NavigateToURL(new_window, test_url); | 
|  | // Outside of RenderProcessHostImpl::IsSpareProcessKeptAtAllTimes mode, the | 
|  | // spare RPH should have been dropped during CreateBrowser() and given to the | 
|  | // new window.  OTOH, even in the IsSpareProcessKeptAtAllTimes mode, the spare | 
|  | // shouldn't be created because of the low process limit. | 
|  | EXPECT_EQ(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | EXPECT_EQ(spare_renderer, | 
|  | new_window->web_contents()->GetMainFrame()->GetProcess()); | 
|  |  | 
|  | // Revert to the default process limit and original ContentBrowserClient. | 
|  | RenderProcessHost::SetMaxRendererProcessCount(0); | 
|  | SetBrowserClientForTesting(old_client); | 
|  | } | 
|  |  | 
|  | // Check that the spare renderer is dropped if an existing process is reused. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareRendererOnProcessReuse) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | NonSpareRendererContentBrowserClient browser_client; | 
|  | ContentBrowserClient* old_client = | 
|  | SetBrowserClientForTesting(&browser_client); | 
|  |  | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | EXPECT_NE(nullptr, spare_renderer); | 
|  |  | 
|  | // This should reuse the existing process. | 
|  | Shell* new_browser = CreateBrowser(); | 
|  | EXPECT_EQ(shell()->web_contents()->GetMainFrame()->GetProcess(), | 
|  | new_browser->web_contents()->GetMainFrame()->GetProcess()); | 
|  | EXPECT_NE(spare_renderer, | 
|  | new_browser->web_contents()->GetMainFrame()->GetProcess()); | 
|  | if (RenderProcessHostImpl::IsSpareProcessKeptAtAllTimes()) { | 
|  | EXPECT_NE(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } else { | 
|  | EXPECT_EQ(nullptr, | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting()); | 
|  | } | 
|  |  | 
|  | // The launcher thread reads state from browser_client, need to wait for it to | 
|  | // be done before resetting the browser client. crbug.com/742533. | 
|  | base::WaitableEvent launcher_thread_done( | 
|  | base::WaitableEvent::ResetPolicy::MANUAL, | 
|  | base::WaitableEvent::InitialState::NOT_SIGNALED); | 
|  | GetProcessLauncherTaskRunner()->PostTask( | 
|  | FROM_HERE, | 
|  | base::BindOnce([](base::WaitableEvent* done) { done->Signal(); }, | 
|  | base::Unretained(&launcher_thread_done))); | 
|  | ASSERT_TRUE(launcher_thread_done.TimedWait(TestTimeouts::action_timeout())); | 
|  |  | 
|  | SetBrowserClientForTesting(old_client); | 
|  | } | 
|  |  | 
|  | // Verifies that the spare renderer maintained by SpareRenderProcessHostManager | 
|  | // is correctly destroyed during browser shutdown.  This test is an analogue | 
|  | // to the //chrome-layer FastShutdown.SpareRenderProcessHost test. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | SpareRenderProcessHostDuringShutdown) { | 
|  | content::RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | shell()->web_contents()->GetBrowserContext()); | 
|  |  | 
|  | // The verification is that there are no DCHECKs anywhere during test tear | 
|  | // down. | 
|  | } | 
|  |  | 
|  | // Verifies that the spare renderer maintained by SpareRenderProcessHostManager | 
|  | // is correctly destroyed when closing the last content shell. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareRendererDuringClosing) { | 
|  | content::RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | shell()->web_contents()->GetBrowserContext()); | 
|  | shell()->web_contents()->Close(); | 
|  |  | 
|  | // The verification is that there are no DCHECKs or UaF anywhere during test | 
|  | // tear down. | 
|  | } | 
|  |  | 
|  | // Class that simulates the fact that some //content embedders (e.g. by | 
|  | // overriding ChromeContentBrowserClient::GetStoragePartitionConfigForSite) can | 
|  | // cause BrowserContext::GetDefaultStoragePartition(browser_context) to differ | 
|  | // from BrowserContext::GetStoragePartition(browser_context, site_instance) even | 
|  | // if |site_instance| is not for guests. | 
|  | class CustomStoragePartitionForSomeSites : public TestContentBrowserClient { | 
|  | public: | 
|  | explicit CustomStoragePartitionForSomeSites(const GURL& site_to_isolate) | 
|  | : site_to_isolate_(site_to_isolate) {} | 
|  |  | 
|  | void GetStoragePartitionConfigForSite(BrowserContext* browser_context, | 
|  | const GURL& site, | 
|  | bool can_be_default, | 
|  | std::string* partition_domain, | 
|  | std::string* partition_name, | 
|  | bool* in_memory) override { | 
|  | // Default to the browser-wide storage partition and override based on | 
|  | // |site| below. | 
|  | partition_domain->clear(); | 
|  | partition_name->clear(); | 
|  | *in_memory = false; | 
|  |  | 
|  | // Override for |site_to_isolate_|. | 
|  | if (site == site_to_isolate_) { | 
|  | *partition_domain = "blah_isolated_storage"; | 
|  | *partition_name = "blah_isolated_storage"; | 
|  | *in_memory = false; | 
|  | } | 
|  | } | 
|  |  | 
|  | std::string GetStoragePartitionIdForSite(BrowserContext* browser_context, | 
|  | const GURL& site) override { | 
|  | if (site == site_to_isolate_) | 
|  | return "custom"; | 
|  | return ""; | 
|  | } | 
|  |  | 
|  | bool IsValidStoragePartitionId(BrowserContext* browser_context, | 
|  | const std::string& partition_id) override { | 
|  | return partition_id == "" || partition_id == "custom"; | 
|  | } | 
|  |  | 
|  | private: | 
|  | GURL site_to_isolate_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(CustomStoragePartitionForSomeSites); | 
|  | }; | 
|  |  | 
|  | // This test verifies that SpareRenderProcessHostManager correctly accounts | 
|  | // for StoragePartition differences when handing out the spare process. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | SpareProcessVsCustomStoragePartition) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | // Provide custom storage partition for test sites. | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | BrowserContext* browser_context = | 
|  | ShellContentBrowserClient::Get()->browser_context(); | 
|  | GURL test_site = SiteInstance::GetSiteForURL(browser_context, test_url); | 
|  | CustomStoragePartitionForSomeSites modified_client(test_site); | 
|  | ContentBrowserClient* old_client = | 
|  | SetBrowserClientForTesting(&modified_client); | 
|  | StoragePartition* default_storage = | 
|  | BrowserContext::GetDefaultStoragePartition(browser_context); | 
|  | StoragePartition* custom_storage = | 
|  | BrowserContext::GetStoragePartitionForSite(browser_context, test_site); | 
|  | EXPECT_NE(default_storage, custom_storage); | 
|  |  | 
|  | // Open a test window - it should be associated with the default storage | 
|  | // partition. | 
|  | Shell* window = CreateBrowser(); | 
|  | RenderProcessHost* old_process = | 
|  | window->web_contents()->GetMainFrame()->GetProcess(); | 
|  | EXPECT_EQ(default_storage, old_process->GetStoragePartition()); | 
|  |  | 
|  | // Warm up the spare process - it should be associated with the default | 
|  | // storage partition. | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost(browser_context); | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | ASSERT_TRUE(spare_renderer); | 
|  | EXPECT_EQ(default_storage, spare_renderer->GetStoragePartition()); | 
|  |  | 
|  | // Navigate to a URL that requires a custom storage partition. | 
|  | EXPECT_TRUE(NavigateToURL(window, test_url)); | 
|  | RenderProcessHost* new_process = | 
|  | window->web_contents()->GetMainFrame()->GetProcess(); | 
|  | // Requirement to use a custom storage partition should force a process swap. | 
|  | EXPECT_NE(new_process, old_process); | 
|  | // The new process should be associated with the custom storage partition. | 
|  | EXPECT_EQ(custom_storage, new_process->GetStoragePartition()); | 
|  | // And consequently, the spare shouldn't have been used. | 
|  | EXPECT_NE(spare_renderer, new_process); | 
|  |  | 
|  | // Restore the original ContentBrowserClient. | 
|  | SetBrowserClientForTesting(old_client); | 
|  | } | 
|  |  | 
|  | class RenderProcessHostObserverCounter : public RenderProcessHostObserver { | 
|  | public: | 
|  | explicit RenderProcessHostObserverCounter(RenderProcessHost* host) { | 
|  | host->AddObserver(this); | 
|  | observing_ = true; | 
|  | observed_host_ = host; | 
|  | } | 
|  |  | 
|  | ~RenderProcessHostObserverCounter() override { | 
|  | if (observing_) | 
|  | observed_host_->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | void RenderProcessExited(RenderProcessHost* host, | 
|  | const ChildProcessTerminationInfo& info) override { | 
|  | DCHECK(observing_); | 
|  | DCHECK_EQ(host, observed_host_); | 
|  | exited_count_++; | 
|  | } | 
|  |  | 
|  | void RenderProcessHostDestroyed(RenderProcessHost* host) override { | 
|  | DCHECK(observing_); | 
|  | DCHECK_EQ(host, observed_host_); | 
|  | destroyed_count_++; | 
|  |  | 
|  | host->RemoveObserver(this); | 
|  | observing_ = false; | 
|  | observed_host_ = nullptr; | 
|  | } | 
|  |  | 
|  | int exited_count() const { return exited_count_; } | 
|  | int destroyed_count() const { return destroyed_count_; } | 
|  |  | 
|  | private: | 
|  | int exited_count_ = 0; | 
|  | int destroyed_count_ = 0; | 
|  | bool observing_ = false; | 
|  | RenderProcessHost* observed_host_ = nullptr; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(RenderProcessHostObserverCounter); | 
|  | }; | 
|  |  | 
|  | // Check that the spare renderer is properly destroyed via | 
|  | // DisableKeepAliveRefCount. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareVsDisableKeepAliveRefCount) { | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | RenderProcessHostObserverCounter counter(spare_renderer); | 
|  |  | 
|  | RenderProcessHostWatcher process_watcher( | 
|  | spare_renderer, RenderProcessHostWatcher::WATCH_FOR_HOST_DESTRUCTION); | 
|  |  | 
|  | spare_renderer->DisableKeepAliveRefCount(); | 
|  |  | 
|  | process_watcher.Wait(); | 
|  | EXPECT_TRUE(process_watcher.did_exit_normally()); | 
|  |  | 
|  | // An important part of test verification is that UaF doesn't happen in the | 
|  | // next revolution of the message pump - without extra care in the | 
|  | // SpareRenderProcessHostManager RenderProcessHost::Cleanup could be called | 
|  | // twice leading to a crash caused by double-free flavour of UaF in | 
|  | // base::DeleteHelper<...>::DoDelete. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | DCHECK_EQ(1, counter.exited_count()); | 
|  | DCHECK_EQ(1, counter.destroyed_count()); | 
|  | } | 
|  |  | 
|  | // Check that the spare renderer is properly destroyed via | 
|  | // DisableKeepAliveRefCount. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, SpareVsFastShutdown) { | 
|  | RenderProcessHost::WarmupSpareRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context()); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | RenderProcessHost* spare_renderer = | 
|  | RenderProcessHostImpl::GetSpareRenderProcessHostForTesting(); | 
|  | RenderProcessHostObserverCounter counter(spare_renderer); | 
|  |  | 
|  | RenderProcessHostWatcher process_watcher( | 
|  | spare_renderer, RenderProcessHostWatcher::WATCH_FOR_HOST_DESTRUCTION); | 
|  |  | 
|  | spare_renderer->FastShutdownIfPossible(); | 
|  |  | 
|  | process_watcher.Wait(); | 
|  | EXPECT_FALSE(process_watcher.did_exit_normally()); | 
|  |  | 
|  | // An important part of test verification is that UaF doesn't happen in the | 
|  | // next revolution of the message pump - without extra care in the | 
|  | // SpareRenderProcessHostManager RenderProcessHost::Cleanup could be called | 
|  | // twice leading to a crash caused by double-free flavour of UaF in | 
|  | // base::DeleteHelper<...>::DoDelete. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | DCHECK_EQ(1, counter.exited_count()); | 
|  | DCHECK_EQ(1, counter.destroyed_count()); | 
|  | } | 
|  |  | 
|  | class ShellCloser : public RenderProcessHostObserver { | 
|  | public: | 
|  | ShellCloser(Shell* shell, std::string* logging_string) | 
|  | : shell_(shell), logging_string_(logging_string) {} | 
|  |  | 
|  | protected: | 
|  | // RenderProcessHostObserver: | 
|  | void RenderProcessExited(RenderProcessHost* host, | 
|  | const ChildProcessTerminationInfo& info) override { | 
|  | logging_string_->append("ShellCloser::RenderProcessExited "); | 
|  | shell_->Close(); | 
|  | } | 
|  |  | 
|  | void RenderProcessHostDestroyed(RenderProcessHost* host) override { | 
|  | logging_string_->append("ShellCloser::RenderProcessHostDestroyed "); | 
|  | } | 
|  |  | 
|  | Shell* shell_; | 
|  | std::string* logging_string_; | 
|  | }; | 
|  |  | 
|  | class ObserverLogger : public RenderProcessHostObserver { | 
|  | public: | 
|  | explicit ObserverLogger(std::string* logging_string) | 
|  | : logging_string_(logging_string), host_destroyed_(false) {} | 
|  |  | 
|  | bool host_destroyed() { return host_destroyed_; } | 
|  |  | 
|  | protected: | 
|  | // RenderProcessHostObserver: | 
|  | void RenderProcessExited(RenderProcessHost* host, | 
|  | const ChildProcessTerminationInfo& info) override { | 
|  | logging_string_->append("ObserverLogger::RenderProcessExited "); | 
|  | } | 
|  |  | 
|  | void RenderProcessHostDestroyed(RenderProcessHost* host) override { | 
|  | logging_string_->append("ObserverLogger::RenderProcessHostDestroyed "); | 
|  | host_destroyed_ = true; | 
|  | } | 
|  |  | 
|  | std::string* logging_string_; | 
|  | bool host_destroyed_; | 
|  | }; | 
|  |  | 
|  | // Flaky on Android. http://crbug.com/759514. | 
|  | #if defined(OS_ANDROID) | 
|  | #define MAYBE_AllProcessExitedCallsBeforeAnyHostDestroyedCalls \ | 
|  | DISABLED_AllProcessExitedCallsBeforeAnyHostDestroyedCalls | 
|  | #else | 
|  | #define MAYBE_AllProcessExitedCallsBeforeAnyHostDestroyedCalls \ | 
|  | AllProcessExitedCallsBeforeAnyHostDestroyedCalls | 
|  | #endif | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | MAYBE_AllProcessExitedCallsBeforeAnyHostDestroyedCalls) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | NavigateToURL(shell(), test_url); | 
|  |  | 
|  | std::string logging_string; | 
|  | ShellCloser shell_closer(shell(), &logging_string); | 
|  | ObserverLogger observer_logger(&logging_string); | 
|  | RenderProcessHost* rph = | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess(); | 
|  |  | 
|  | // Ensure that the ShellCloser observer is first, so that it will have first | 
|  | // dibs on the ProcessExited callback. | 
|  | rph->AddObserver(&shell_closer); | 
|  | rph->AddObserver(&observer_logger); | 
|  |  | 
|  | // This will crash the render process, and start all the callbacks. | 
|  | // We can't use NavigateToURL here since it accesses the shell() after | 
|  | // navigating, which the shell_closer deletes. | 
|  | NavigateToURLBlockUntilNavigationsComplete( | 
|  | shell(), GURL(kChromeUICrashURL), 1); | 
|  |  | 
|  | // The key here is that all the RenderProcessExited callbacks precede all the | 
|  | // RenderProcessHostDestroyed callbacks. | 
|  | EXPECT_EQ("ShellCloser::RenderProcessExited " | 
|  | "ObserverLogger::RenderProcessExited " | 
|  | "ShellCloser::RenderProcessHostDestroyed " | 
|  | "ObserverLogger::RenderProcessHostDestroyed ", logging_string); | 
|  |  | 
|  | // If the test fails, and somehow the RPH is still alive somehow, at least | 
|  | // deregister the observers so that the test fails and doesn't also crash. | 
|  | if (!observer_logger.host_destroyed()) { | 
|  | rph->RemoveObserver(&shell_closer); | 
|  | rph->RemoveObserver(&observer_logger); | 
|  | } | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, KillProcessOnBadMojoMessage) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | GURL test_url = embedded_test_server()->GetURL("/simple_page.html"); | 
|  | NavigateToURL(shell(), test_url); | 
|  | RenderProcessHost* rph = | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess(); | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  |  | 
|  | mojom::TestServicePtr service; | 
|  | BindInterface(rph, &service); | 
|  |  | 
|  | base::RunLoop run_loop; | 
|  | set_process_exit_callback(run_loop.QuitClosure()); | 
|  |  | 
|  | // Should reply with a bad message and cause process death. | 
|  | service->DoSomething(base::DoNothing()); | 
|  |  | 
|  | run_loop.Run(); | 
|  |  | 
|  | EXPECT_EQ(1, process_exits_); | 
|  | EXPECT_EQ(0, host_destructions_); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | class AudioStartObserver : public WebContentsObserver { | 
|  | public: | 
|  | AudioStartObserver(WebContents* web_contents, | 
|  | base::OnceClosure audible_closure) | 
|  | : WebContentsObserver(web_contents), | 
|  | audible_closure_(std::move(audible_closure)) {} | 
|  | ~AudioStartObserver() override {} | 
|  |  | 
|  | // WebContentsObserver: | 
|  | void OnAudioStateChanged(bool audible) override { | 
|  | if (audible && audible_closure_) | 
|  | std::move(audible_closure_).Run(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | base::OnceClosure audible_closure_; | 
|  | }; | 
|  |  | 
|  | // Tests that audio stream counts (used for process priority calculations) are | 
|  | // properly set and cleared during media playback and renderer terminations. | 
|  | // | 
|  | // Note: This test can't run when the Mojo Renderer is used since it does not | 
|  | // create audio streams through the normal audio pathways; at present this is | 
|  | // only used by Chromecast. | 
|  | // | 
|  | // crbug.com/864476: flaky on Android for unclear reasons. | 
|  | #if BUILDFLAG(ENABLE_MOJO_RENDERER) || defined(OS_ANDROID) | 
|  | #define KillProcessZerosAudioStreams DISABLED_KillProcessZerosAudioStreams | 
|  | #endif | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, KillProcessZerosAudioStreams) { | 
|  | // TODO(maxmorin): This test only uses an output stream. There should be a | 
|  | // similar test for input streams. | 
|  | embedded_test_server()->ServeFilesFromSourceDirectory( | 
|  | media::GetTestDataPath()); | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  | ASSERT_TRUE(NavigateToURL( | 
|  | shell(), embedded_test_server()->GetURL("/webaudio_oscillator.html"))); | 
|  | RenderProcessHostImpl* rph = static_cast<RenderProcessHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess()); | 
|  |  | 
|  | { | 
|  | // Start audio and wait for it to become audible. | 
|  | base::RunLoop run_loop; | 
|  | AudioStartObserver observer(shell()->web_contents(), | 
|  | run_loop.QuitClosure()); | 
|  |  | 
|  | std::string result; | 
|  | EXPECT_TRUE( | 
|  | ExecuteScriptAndExtractString(shell(), "StartOscillator();", &result)) | 
|  | << "Failed to execute javascript."; | 
|  | run_loop.Run(); | 
|  |  | 
|  | // No point in running the rest of the test if this is wrong. | 
|  | ASSERT_EQ(1, rph->get_media_stream_count_for_testing()); | 
|  | } | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  |  | 
|  | mojom::TestServicePtr service; | 
|  | BindInterface(rph, &service); | 
|  |  | 
|  | { | 
|  | // Force a bad message event to occur which will terminate the renderer. | 
|  | // Note: We post task the QuitClosure since RenderProcessExited() is called | 
|  | // before destroying BrowserMessageFilters; and the next portion of the test | 
|  | // must run after these notifications have been delivered. | 
|  | base::RunLoop run_loop; | 
|  | set_process_exit_callback(media::BindToCurrentLoop(run_loop.QuitClosure())); | 
|  | service->DoSomething(base::DoNothing()); | 
|  | run_loop.Run(); | 
|  | } | 
|  |  | 
|  | { | 
|  | // Cycle UI and IO loop once to ensure OnChannelClosing() has been delivered | 
|  | // to audio stream owners and they get a chance to notify of stream closure. | 
|  | base::RunLoop run_loop; | 
|  | BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, | 
|  | media::BindToCurrentLoop(run_loop.QuitClosure())); | 
|  | run_loop.Run(); | 
|  | } | 
|  |  | 
|  | // Verify shutdown went as expected. | 
|  | EXPECT_EQ(0, rph->get_media_stream_count_for_testing()); | 
|  | EXPECT_EQ(1, process_exits_); | 
|  | EXPECT_EQ(0, host_destructions_); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | // Test class instance to run specific setup steps for capture streams. | 
|  | class CaptureStreamRenderProcessHostTest : public RenderProcessHostTest { | 
|  | public: | 
|  | void SetUp() override { | 
|  | // Pixel output is needed when digging pixel values out of video tags for | 
|  | // verification in VideoCaptureStream tests. | 
|  | EnablePixelOutput(); | 
|  | RenderProcessHostTest::SetUp(); | 
|  | } | 
|  |  | 
|  | void SetUpCommandLine(base::CommandLine* command_line) override { | 
|  | // These flags are necessary to emulate camera input for getUserMedia() | 
|  | // tests. | 
|  | command_line->AppendSwitch(switches::kUseFakeDeviceForMediaStream); | 
|  | command_line->AppendSwitch(switches::kUseFakeUIForMediaStream); | 
|  | RenderProcessHostTest::SetUpCommandLine(command_line); | 
|  | } | 
|  | }; | 
|  |  | 
|  | // Tests that video capture stream count increments when getUserMedia() is | 
|  | // called. | 
|  | IN_PROC_BROWSER_TEST_F(CaptureStreamRenderProcessHostTest, | 
|  | GetUserMediaIncrementsVideoCaptureStreams) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  | NavigateToURL(shell(), | 
|  | embedded_test_server()->GetURL("/media/getusermedia.html")); | 
|  | RenderProcessHostImpl* rph = static_cast<RenderProcessHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess()); | 
|  | std::string result; | 
|  | EXPECT_TRUE(ExecuteScriptAndExtractString( | 
|  | shell(), "getUserMediaAndExpectSuccess({video: true});", &result)) | 
|  | << "Failed to execute javascript."; | 
|  | EXPECT_EQ(1, rph->get_media_stream_count_for_testing()); | 
|  | } | 
|  |  | 
|  | // Tests that video capture stream count resets when getUserMedia() is called | 
|  | // and stopped. | 
|  | IN_PROC_BROWSER_TEST_F(CaptureStreamRenderProcessHostTest, | 
|  | StopResetsVideoCaptureStreams) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  | NavigateToURL(shell(), | 
|  | embedded_test_server()->GetURL("/media/getusermedia.html")); | 
|  | RenderProcessHostImpl* rph = static_cast<RenderProcessHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess()); | 
|  | std::string result; | 
|  | EXPECT_TRUE(ExecuteScriptAndExtractString( | 
|  | shell(), "getUserMediaAndStop({video: true});", &result)) | 
|  | << "Failed to execute javascript."; | 
|  | EXPECT_EQ(0, rph->get_media_stream_count_for_testing()); | 
|  | } | 
|  |  | 
|  | // Tests that video capture stream counts (used for process priority | 
|  | // calculations) are properly set and cleared during media playback and renderer | 
|  | // terminations. | 
|  | IN_PROC_BROWSER_TEST_F(CaptureStreamRenderProcessHostTest, | 
|  | KillProcessZerosVideoCaptureStreams) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  | NavigateToURL(shell(), | 
|  | embedded_test_server()->GetURL("/media/getusermedia.html")); | 
|  | RenderProcessHostImpl* rph = static_cast<RenderProcessHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess()); | 
|  | std::string result; | 
|  | EXPECT_TRUE(ExecuteScriptAndExtractString( | 
|  | shell(), "getUserMediaAndExpectSuccess({video: true});", &result)) | 
|  | << "Failed to execute javascript."; | 
|  | EXPECT_EQ(1, rph->get_media_stream_count_for_testing()); | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  |  | 
|  | mojom::TestServicePtr service; | 
|  | BindInterface(rph, &service); | 
|  |  | 
|  | { | 
|  | // Force a bad message event to occur which will terminate the renderer. | 
|  | base::RunLoop run_loop; | 
|  | set_process_exit_callback(media::BindToCurrentLoop(run_loop.QuitClosure())); | 
|  | service->DoSomething(base::DoNothing()); | 
|  | run_loop.Run(); | 
|  | } | 
|  |  | 
|  | { | 
|  | // Cycle UI and IO loop once to ensure OnChannelClosing() has been delivered | 
|  | // to audio stream owners and they get a chance to notify of stream closure. | 
|  | base::RunLoop run_loop; | 
|  | BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, | 
|  | media::BindToCurrentLoop(run_loop.QuitClosure())); | 
|  | run_loop.Run(); | 
|  | } | 
|  |  | 
|  | EXPECT_EQ(0, rph->get_media_stream_count_for_testing()); | 
|  | EXPECT_EQ(1, process_exits_); | 
|  | EXPECT_EQ(0, host_destructions_); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | // Tests that media stream count increments when getUserMedia() is | 
|  | // called with audio only. | 
|  | IN_PROC_BROWSER_TEST_F(CaptureStreamRenderProcessHostTest, | 
|  | GetUserMediaAudioOnlyIncrementsMediaStreams) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  | NavigateToURL(shell(), | 
|  | embedded_test_server()->GetURL("/media/getusermedia.html")); | 
|  | RenderProcessHostImpl* rph = static_cast<RenderProcessHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess()); | 
|  | std::string result; | 
|  | EXPECT_TRUE(ExecuteScriptAndExtractString( | 
|  | shell(), "getUserMediaAndExpectSuccess({video: false, audio: true});", | 
|  | &result)) | 
|  | << "Failed to execute javascript."; | 
|  | EXPECT_EQ(1, rph->get_media_stream_count_for_testing()); | 
|  | } | 
|  |  | 
|  | // Tests that media stream counts (used for process priority | 
|  | // calculations) are properly set and cleared during media playback and renderer | 
|  | // terminations for audio only streams. | 
|  | IN_PROC_BROWSER_TEST_F(CaptureStreamRenderProcessHostTest, | 
|  | KillProcessZerosAudioCaptureStreams) { | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  | NavigateToURL(shell(), | 
|  | embedded_test_server()->GetURL("/media/getusermedia.html")); | 
|  | RenderProcessHostImpl* rph = static_cast<RenderProcessHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()->GetProcess()); | 
|  | std::string result; | 
|  | EXPECT_TRUE(ExecuteScriptAndExtractString( | 
|  | shell(), "getUserMediaAndExpectSuccess({video: false, audio: true});", | 
|  | &result)) | 
|  | << "Failed to execute javascript."; | 
|  | EXPECT_EQ(1, rph->get_media_stream_count_for_testing()); | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  |  | 
|  | mojom::TestServicePtr service; | 
|  | BindInterface(rph, &service); | 
|  |  | 
|  | { | 
|  | // Force a bad message event to occur which will terminate the renderer. | 
|  | base::RunLoop run_loop; | 
|  | set_process_exit_callback(media::BindToCurrentLoop(run_loop.QuitClosure())); | 
|  | service->DoSomething(base::DoNothing()); | 
|  | run_loop.Run(); | 
|  | } | 
|  |  | 
|  | { | 
|  | // Cycle UI and IO loop once to ensure OnChannelClosing() has been delivered | 
|  | // to audio stream owners and they get a chance to notify of stream closure. | 
|  | base::RunLoop run_loop; | 
|  | BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, | 
|  | media::BindToCurrentLoop(run_loop.QuitClosure())); | 
|  | run_loop.Run(); | 
|  | } | 
|  |  | 
|  | EXPECT_EQ(0, rph->get_media_stream_count_for_testing()); | 
|  | EXPECT_EQ(1, process_exits_); | 
|  | EXPECT_EQ(0, host_destructions_); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, KeepAliveRendererProcess) { | 
|  | embedded_test_server()->RegisterRequestHandler( | 
|  | base::BindRepeating(HandleBeacon)); | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | NavigateToURL(shell(), embedded_test_server()->GetURL("/send-beacon.html")); | 
|  |  | 
|  | RenderFrameHostImpl* rfh = static_cast<RenderFrameHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()); | 
|  | RenderProcessHostImpl* rph = | 
|  | static_cast<RenderProcessHostImpl*>(rfh->GetProcess()); | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  | rfh->SetKeepAliveTimeoutForTesting(base::TimeDelta::FromSeconds(30)); | 
|  |  | 
|  | base::TimeTicks start = base::TimeTicks::Now(); | 
|  | NavigateToURL(shell(), GURL("data:text/html,<p>hello</p>")); | 
|  |  | 
|  | WaitUntilProcessExits(1); | 
|  |  | 
|  | EXPECT_LT(base::TimeTicks::Now() - start, base::TimeDelta::FromSeconds(30)); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | // Test is flaky on Android builders: https://crbug.com/875179 | 
|  | #if defined(OS_ANDROID) | 
|  | #define MAYBE_KeepAliveRendererProcess_Hung \ | 
|  | DISABLED_KeepAliveRendererProcess_Hung | 
|  | #else | 
|  | #define MAYBE_KeepAliveRendererProcess_Hung KeepAliveRendererProcess_Hung | 
|  | #endif | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | MAYBE_KeepAliveRendererProcess_Hung) { | 
|  | embedded_test_server()->RegisterRequestHandler( | 
|  | base::BindRepeating(HandleHungBeacon)); | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | NavigateToURL(shell(), embedded_test_server()->GetURL("/send-beacon.html")); | 
|  |  | 
|  | RenderFrameHostImpl* rfh = static_cast<RenderFrameHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()); | 
|  | RenderProcessHostImpl* rph = | 
|  | static_cast<RenderProcessHostImpl*>(rfh->GetProcess()); | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  | rfh->SetKeepAliveTimeoutForTesting(base::TimeDelta::FromSeconds(1)); | 
|  |  | 
|  | base::TimeTicks start = base::TimeTicks::Now(); | 
|  | NavigateToURL(shell(), GURL("data:text/html,<p>hello</p>")); | 
|  |  | 
|  | WaitUntilProcessExits(1); | 
|  |  | 
|  | EXPECT_GE(base::TimeTicks::Now() - start, base::TimeDelta::FromSeconds(1)); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | // Test is flaky on Android builders: https://crbug.com/875179 | 
|  | #if defined(OS_ANDROID) | 
|  | #define MAYBE_FetchKeepAliveRendererProcess_Hung \ | 
|  | DISABLED_FetchKeepAliveRendererProcess_Hung | 
|  | #else | 
|  | #define MAYBE_FetchKeepAliveRendererProcess_Hung \ | 
|  | FetchKeepAliveRendererProcess_Hung | 
|  | #endif | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, | 
|  | MAYBE_FetchKeepAliveRendererProcess_Hung) { | 
|  | embedded_test_server()->RegisterRequestHandler( | 
|  | base::BindRepeating(HandleHungBeacon)); | 
|  | ASSERT_TRUE(embedded_test_server()->Start()); | 
|  |  | 
|  | NavigateToURL(shell(), | 
|  | embedded_test_server()->GetURL("/fetch-keepalive.html")); | 
|  |  | 
|  | RenderFrameHostImpl* rfh = static_cast<RenderFrameHostImpl*>( | 
|  | shell()->web_contents()->GetMainFrame()); | 
|  | RenderProcessHostImpl* rph = | 
|  | static_cast<RenderProcessHostImpl*>(rfh->GetProcess()); | 
|  |  | 
|  | host_destructions_ = 0; | 
|  | process_exits_ = 0; | 
|  | rph->AddObserver(this); | 
|  | rfh->SetKeepAliveTimeoutForTesting(base::TimeDelta::FromSeconds(1)); | 
|  |  | 
|  | base::TimeTicks start = base::TimeTicks::Now(); | 
|  | NavigateToURL(shell(), GURL("data:text/html,<p>hello</p>")); | 
|  |  | 
|  | WaitUntilProcessExits(1); | 
|  |  | 
|  | EXPECT_GE(base::TimeTicks::Now() - start, base::TimeDelta::FromSeconds(1)); | 
|  | if (!host_destructions_) | 
|  | rph->RemoveObserver(this); | 
|  | } | 
|  |  | 
|  | // This test verifies properties of RenderProcessHostImpl *before* Init method | 
|  | // is called. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, ConstructedButNotInitializedYet) { | 
|  | RenderProcessHost* process = RenderProcessHostImpl::CreateRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context(), nullptr, nullptr, | 
|  | false /* is_for_guests_only */); | 
|  |  | 
|  | // Just verifying that the arguments of CreateRenderProcessHost got processed | 
|  | // correctly. | 
|  | EXPECT_EQ(ShellContentBrowserClient::Get()->browser_context(), | 
|  | process->GetBrowserContext()); | 
|  | EXPECT_FALSE(process->IsForGuestsOnly()); | 
|  |  | 
|  | // There should be no OS process before Init() method is called. | 
|  | EXPECT_FALSE(process->IsInitializedAndNotDead()); | 
|  | EXPECT_FALSE(process->IsReady()); | 
|  | EXPECT_FALSE(process->GetProcess().IsValid()); | 
|  | EXPECT_EQ(base::kNullProcessHandle, process->GetProcess().Handle()); | 
|  |  | 
|  | // TODO(lukasza): https://crbug.com/813045: RenderProcessHost shouldn't have | 
|  | // an associated IPC channel (and shouldn't accumulate IPC messages) unless | 
|  | // the Init() method was called and the RPH either has connection to an actual | 
|  | // OS process or is currently attempting to spawn the OS process.  After this | 
|  | // bug is fixed the 1st test assertion below should be reversed (unsure about | 
|  | // the 2nd one). | 
|  | EXPECT_TRUE(process->GetChannel()); | 
|  | EXPECT_TRUE(process->GetRendererInterface()); | 
|  |  | 
|  | // Cleanup the resources acquired by the test. | 
|  | process->Cleanup(); | 
|  | } | 
|  |  | 
|  | // This test verifies that a fast shutdown is possible for a starting process. | 
|  | IN_PROC_BROWSER_TEST_F(RenderProcessHostTest, FastShutdownForStartingProcess) { | 
|  | RenderProcessHost* process = RenderProcessHostImpl::CreateRenderProcessHost( | 
|  | ShellContentBrowserClient::Get()->browser_context(), nullptr, nullptr, | 
|  | false /* is_for_guests_only */); | 
|  | process->Init(); | 
|  | EXPECT_TRUE(process->FastShutdownIfPossible()); | 
|  | process->Cleanup(); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace content |