blob: 74238a2beb47ca0d3f08af81cce418fe6e76ac13 [file] [log] [blame]
// 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 "media/base/user_input_monitor.h"
#include <memory>
#include <utility>
#include "base/run_loop.h"
#include "base/test/task_environment.h"
#include "base/threading/thread_task_runner_handle.h"
#include "build/build_config.h"
#include "testing/gtest/include/gtest/gtest.h"
#if defined(OS_LINUX) || defined(OS_CHROMEOS)
#include "base/files/file_descriptor_watcher_posix.h"
#endif
#if defined(USE_OZONE)
#include "ui/base/ui_base_features.h" // nogncheck
#include "ui/ozone/public/ozone_platform.h" // nogncheck
#endif
#if defined(OS_WIN)
#include "ui/events/test/keyboard_hook_monitor_utils.h"
#endif
namespace media {
namespace {
class UserInputMonitorTest : public testing::Test {
protected:
// testing::Test.
void SetUp() override {
#if defined(USE_OZONE)
if (features::IsUsingOzonePlatform()) {
if (ui::OzonePlatform::GetPlatformNameForTest() == "drm") {
// OzonePlatformDrm::InitializeUI hangs in tests on the DRM platform.
GTEST_SKIP();
}
// Initialise Ozone in single process mode, as all tests do.
ui::OzonePlatform::InitParams params;
params.single_process = true;
ui::OzonePlatform::InitializeForUI(params);
}
#endif
}
};
} // namespace
TEST_F(UserInputMonitorTest, CreatePlatformSpecific) {
#if defined(OS_LINUX) || defined(OS_CHROMEOS)
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::IO);
#else
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::UI);
#endif // defined(OS_LINUX) || defined(OS_CHROMEOS)
std::unique_ptr<UserInputMonitor> monitor = UserInputMonitor::Create(
base::ThreadTaskRunnerHandle::Get(), base::ThreadTaskRunnerHandle::Get());
if (!monitor)
return;
monitor->EnableKeyPressMonitoring();
monitor->DisableKeyPressMonitoring();
monitor.reset();
base::RunLoop().RunUntilIdle();
}
TEST_F(UserInputMonitorTest, CreatePlatformSpecificWithMapping) {
#if defined(OS_LINUX) || defined(OS_CHROMEOS)
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::IO);
#else
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::UI);
#endif // defined(OS_LINUX) || defined(OS_CHROMEOS)
std::unique_ptr<UserInputMonitor> monitor = UserInputMonitor::Create(
base::ThreadTaskRunnerHandle::Get(), base::ThreadTaskRunnerHandle::Get());
if (!monitor)
return;
base::ReadOnlySharedMemoryMapping readonly_mapping =
static_cast<UserInputMonitorBase*>(monitor.get())
->EnableKeyPressMonitoringWithMapping()
.Map();
EXPECT_EQ(0u, ReadKeyPressMonitorCount(readonly_mapping));
monitor->DisableKeyPressMonitoring();
monitor.reset();
base::RunLoop().RunUntilIdle();
// Check that read only region remains valid after disable.
EXPECT_EQ(0u, ReadKeyPressMonitorCount(readonly_mapping));
}
TEST_F(UserInputMonitorTest, ReadWriteKeyPressMonitorCount) {
std::unique_ptr<base::MappedReadOnlyRegion> shmem =
std::make_unique<base::MappedReadOnlyRegion>(
base::ReadOnlySharedMemoryRegion::Create(sizeof(uint32_t)));
ASSERT_TRUE(shmem->IsValid());
constexpr uint32_t count = 10;
WriteKeyPressMonitorCount(shmem->mapping, count);
base::ReadOnlySharedMemoryMapping readonly_mapping = shmem->region.Map();
EXPECT_EQ(count, ReadKeyPressMonitorCount(readonly_mapping));
}
#if defined(OS_WIN)
//
// Windows specific scenarios which require simulating keyboard hook events.
//
TEST_F(UserInputMonitorTest, BlockMonitoringAfterMonitoringEnabled) {
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::UI);
std::unique_ptr<UserInputMonitor> monitor = UserInputMonitor::Create(
base::ThreadTaskRunnerHandle::Get(), base::ThreadTaskRunnerHandle::Get());
if (!monitor)
return;
monitor->EnableKeyPressMonitoring();
ui::SimulateKeyboardHookRegistered();
ui::SimulateKeyboardHookUnregistered();
monitor->DisableKeyPressMonitoring();
monitor.reset();
base::RunLoop().RunUntilIdle();
}
TEST_F(UserInputMonitorTest, BlockMonitoringBeforeMonitoringEnabled) {
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::UI);
std::unique_ptr<UserInputMonitor> monitor = UserInputMonitor::Create(
base::ThreadTaskRunnerHandle::Get(), base::ThreadTaskRunnerHandle::Get());
if (!monitor)
return;
ui::SimulateKeyboardHookRegistered();
monitor->EnableKeyPressMonitoring();
ui::SimulateKeyboardHookUnregistered();
monitor->DisableKeyPressMonitoring();
monitor.reset();
base::RunLoop().RunUntilIdle();
}
TEST_F(UserInputMonitorTest, UnblockMonitoringAfterMonitoringDisabled) {
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::UI);
std::unique_ptr<UserInputMonitor> monitor = UserInputMonitor::Create(
base::ThreadTaskRunnerHandle::Get(), base::ThreadTaskRunnerHandle::Get());
if (!monitor)
return;
monitor->EnableKeyPressMonitoring();
ui::SimulateKeyboardHookRegistered();
monitor->DisableKeyPressMonitoring();
ui::SimulateKeyboardHookUnregistered();
monitor.reset();
base::RunLoop().RunUntilIdle();
}
TEST_F(UserInputMonitorTest, BlockKeypressMonitoringWithSharedMemoryBuffer) {
base::test::TaskEnvironment task_environment(
base::test::TaskEnvironment::MainThreadType::UI);
std::unique_ptr<UserInputMonitor> monitor = UserInputMonitor::Create(
base::ThreadTaskRunnerHandle::Get(), base::ThreadTaskRunnerHandle::Get());
if (!monitor)
return;
base::ReadOnlySharedMemoryMapping readonly_mapping =
static_cast<UserInputMonitorBase*>(monitor.get())
->EnableKeyPressMonitoringWithMapping()
.Map();
EXPECT_EQ(0u, ReadKeyPressMonitorCount(readonly_mapping));
ui::SimulateKeyboardHookRegistered();
EXPECT_EQ(0u, ReadKeyPressMonitorCount(readonly_mapping));
ui::SimulateKeyboardHookUnregistered();
EXPECT_EQ(0u, ReadKeyPressMonitorCount(readonly_mapping));
monitor->DisableKeyPressMonitoring();
monitor.reset();
base::RunLoop().RunUntilIdle();
// Check that read only region remains valid after disable.
EXPECT_EQ(0u, ReadKeyPressMonitorCount(readonly_mapping));
}
#endif // defined(OS_WIN)
} // namespace media