// 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/command_line.h"
#include "base/memory/discardable_memory.h"
#include "base/memory/scoped_vector.h"
#include "content/public/browser/content_browser_client.h"
#include "content/public/common/content_client.h"
#include "content/public/common/content_switches.h"
#include "content/public/renderer/content_renderer_client.h"
#include "content/renderer/render_process_impl.h"
#include "content/renderer/render_thread_impl.h"
#include "content/test/mock_render_process.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace content {
namespace {
class RenderThreadImplBrowserTest : public testing::Test {
public:
virtual ~RenderThreadImplBrowserTest() {}
};
class DummyListener : public IPC::Listener {
public:
virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE {
return true;
}
};
void CheckRenderThreadInputHandlerManager(RenderThreadImpl* thread) {
ASSERT_TRUE(thread->input_handler_manager());
}
// Check that InputHandlerManager outlives compositor thread because it uses
// raw pointers to post tasks.
// Disabled under LeakSanitizer due to memory leaks. http://crbug.com/348994
#if defined(LEAK_SANITIZER)
#define MAYBE_InputHandlerManagerDestroyedAfterCompositorThread \
DISABLED_InputHandlerManagerDestroyedAfterCompositorThread
#else
#define MAYBE_InputHandlerManagerDestroyedAfterCompositorThread \
InputHandlerManagerDestroyedAfterCompositorThread
#endif
TEST_F(RenderThreadImplBrowserTest,
MAYBE_InputHandlerManagerDestroyedAfterCompositorThread) {
ContentClient content_client;
ContentBrowserClient content_browser_client;
ContentRendererClient content_renderer_client;
SetContentClient(&content_client);
SetBrowserClientForTesting(&content_browser_client);
SetRendererClientForTesting(&content_renderer_client);
base::MessageLoopForIO message_loop_;
std::string channel_id = IPC::Channel::GenerateVerifiedChannelID(
std::string());
DummyListener dummy_listener;
scoped_ptr<IPC::Channel> channel(
IPC::Channel::CreateServer(channel_id, &dummy_listener));
ASSERT_TRUE(channel->Connect());
scoped_ptr<MockRenderProcess> mock_process(new MockRenderProcess);
// Owned by mock_process.
RenderThreadImpl* thread = new RenderThreadImpl(channel_id);
thread->EnsureWebKitInitialized();
ASSERT_TRUE(thread->input_handler_manager());
thread->compositor_message_loop_proxy()->PostTask(
FROM_HERE,
base::Bind(&CheckRenderThreadInputHandlerManager, thread));
}
// Checks that emulated discardable memory is discarded when the last widget
// is hidden.
// Disabled under LeakSanitizer due to memory leaks.
#if defined(LEAK_SANITIZER)
#define MAYBE_EmulatedDiscardableMemoryDiscardedWhenWidgetsHidden \
DISABLED_EmulatedDiscardableMemoryDiscardedWhenWidgetsHidden
#else
#define MAYBE_EmulatedDiscardableMemoryDiscardedWhenWidgetsHidden \
EmulatedDiscardableMemoryDiscardedWhenWidgetsHidden
#endif
TEST_F(RenderThreadImplBrowserTest,
MAYBE_EmulatedDiscardableMemoryDiscardedWhenWidgetsHidden) {
ContentClient content_client;
ContentBrowserClient content_browser_client;
ContentRendererClient content_renderer_client;
SetContentClient(&content_client);
SetBrowserClientForTesting(&content_browser_client);
SetRendererClientForTesting(&content_renderer_client);
base::MessageLoopForIO message_loop_;
std::string channel_id =
IPC::Channel::GenerateVerifiedChannelID(std::string());
DummyListener dummy_listener;
scoped_ptr<IPC::Channel> channel(
IPC::Channel::CreateServer(channel_id, &dummy_listener));
ASSERT_TRUE(channel->Connect());
scoped_ptr<MockRenderProcess> mock_process(new MockRenderProcess);
// Owned by mock_process.
RenderThreadImpl* thread = new RenderThreadImpl(channel_id);
thread->EnsureWebKitInitialized();
thread->WidgetCreated();
// Allocate 128MB of discardable memory.
ScopedVector<base::DiscardableMemory> discardable_memory;
for (int i = 0; i < 32; ++i) {
discardable_memory.push_back(
base::DiscardableMemory::CreateLockedMemoryWithType(
base::DISCARDABLE_MEMORY_TYPE_EMULATED, 4 * 1024 * 1024).release());
ASSERT_TRUE(discardable_memory.back());
discardable_memory.back()->Unlock();
}
// Hide all widgets.
thread->WidgetHidden();
// Count how much memory is left, should be at most one block.
int blocks_left = 0;
for (auto iter = discardable_memory.begin(); iter != discardable_memory.end();
++iter) {
if ((*iter)->Lock() == base::DISCARDABLE_MEMORY_LOCK_STATUS_SUCCESS)
++blocks_left;
}
EXPECT_LE(blocks_left, 1);
thread->WidgetDestroyed();
}
} // namespace
} // namespace content