// Copyright 2014 the V8 project 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 "src/libplatform/default-platform.h" #include "src/base/platform/semaphore.h" #include "src/base/platform/time.h" #include "testing/gmock/include/gmock/gmock.h" using testing::InSequence; using testing::StrictMock; namespace v8 { namespace platform { namespace default_platform_unittest { namespace { struct MockTask : public Task { // See issue v8:8185 ~MockTask() /* override */ { Die(); } MOCK_METHOD(void, Run, (), (override)); MOCK_METHOD(void, Die, ()); }; struct MockIdleTask : public IdleTask { // See issue v8:8185 ~MockIdleTask() /* override */ { Die(); } MOCK_METHOD(void, Run, (double deadline_in_seconds), (override)); MOCK_METHOD(void, Die, ()); }; class DefaultPlatformWithMockTime : public DefaultPlatform { public: explicit DefaultPlatformWithMockTime(int thread_pool_size = 0) : DefaultPlatform(thread_pool_size, IdleTaskSupport::kEnabled, nullptr) { mock_time_ = 0.0; SetTimeFunctionForTesting([]() { return mock_time_; }); } void IncreaseTime(double seconds) { mock_time_ += seconds; } private: static double mock_time_; }; double DefaultPlatformWithMockTime::mock_time_ = 0.0; template class PlatformTest : public ::testing::Test { public: Isolate* isolate() { return reinterpret_cast(dummy_); } Platform* platform() { return &platform_; } std::shared_ptr task_runner() { if (!task_runner_) { task_runner_ = platform_.GetForegroundTaskRunner(isolate()); } DCHECK_NOT_NULL(task_runner_); return task_runner_; } // These methods take ownership of the task. Tests might still reference them, // if the tasks are expected to still exist. void CallOnForegroundThread(Task* task) { task_runner()->PostTask(std::unique_ptr(task)); } void CallNonNestableOnForegroundThread(Task* task) { task_runner()->PostNonNestableTask(std::unique_ptr(task)); } void CallDelayedOnForegroundThread(Task* task, double delay_in_seconds) { task_runner()->PostDelayedTask(std::unique_ptr(task), delay_in_seconds); } void CallIdleOnForegroundThread(IdleTask* task) { task_runner()->PostIdleTask(std::unique_ptr(task)); } bool PumpMessageLoop() { return platform_.PumpMessageLoop(isolate()); } private: Platform platform_; InSequence in_sequence_; std::shared_ptr task_runner_; int dummy_ = 0; }; class DefaultPlatformTest : public PlatformTest {}; class DefaultPlatformTestWithMockTime : public PlatformTest {}; } // namespace TEST_F(DefaultPlatformTest, PumpMessageLoop) { EXPECT_FALSE(platform()->PumpMessageLoop(isolate())); StrictMock* task = new StrictMock; CallOnForegroundThread(task); EXPECT_CALL(*task, Run()); EXPECT_CALL(*task, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_FALSE(PumpMessageLoop()); } TEST_F(DefaultPlatformTest, PumpMessageLoopWithTaskRunner) { std::shared_ptr taskrunner = platform()->GetForegroundTaskRunner(isolate()); EXPECT_FALSE(PumpMessageLoop()); StrictMock* task = new StrictMock; taskrunner->PostTask(std::unique_ptr(task)); EXPECT_CALL(*task, Run()); EXPECT_CALL(*task, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_FALSE(PumpMessageLoop()); } TEST_F(DefaultPlatformTest, PumpMessageLoopNested) { EXPECT_FALSE(PumpMessageLoop()); StrictMock* nestable_task1 = new StrictMock; StrictMock* non_nestable_task2 = new StrictMock; StrictMock* nestable_task3 = new StrictMock; StrictMock* non_nestable_task4 = new StrictMock; CallOnForegroundThread(nestable_task1); CallNonNestableOnForegroundThread(non_nestable_task2); CallOnForegroundThread(nestable_task3); CallNonNestableOnForegroundThread(non_nestable_task4); // Nestable tasks are FIFO; non-nestable tasks are FIFO. A task being // non-nestable may cause it to be executed later, but not earlier. EXPECT_CALL(*nestable_task1, Run).WillOnce([this]() { EXPECT_TRUE(PumpMessageLoop()); }); EXPECT_CALL(*nestable_task3, Run()); EXPECT_CALL(*nestable_task3, Die()); EXPECT_CALL(*nestable_task1, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_CALL(*non_nestable_task2, Run()); EXPECT_CALL(*non_nestable_task2, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_CALL(*non_nestable_task4, Run()); EXPECT_CALL(*non_nestable_task4, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_FALSE(PumpMessageLoop()); } TEST_F(DefaultPlatformTestWithMockTime, PumpMessageLoopDelayed) { EXPECT_FALSE(PumpMessageLoop()); StrictMock* task1 = new StrictMock; StrictMock* task2 = new StrictMock; CallDelayedOnForegroundThread(task2, 100); CallDelayedOnForegroundThread(task1, 10); EXPECT_FALSE(PumpMessageLoop()); platform()->IncreaseTime(11); EXPECT_CALL(*task1, Run()); EXPECT_CALL(*task1, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_FALSE(PumpMessageLoop()); platform()->IncreaseTime(90); EXPECT_CALL(*task2, Run()); EXPECT_CALL(*task2, Die()); EXPECT_TRUE(PumpMessageLoop()); } TEST_F(DefaultPlatformTestWithMockTime, PumpMessageLoopNoStarvation) { EXPECT_FALSE(PumpMessageLoop()); StrictMock* task1 = new StrictMock; StrictMock* task2 = new StrictMock; StrictMock* task3 = new StrictMock; CallOnForegroundThread(task1); CallDelayedOnForegroundThread(task2, 10); platform()->IncreaseTime(11); EXPECT_CALL(*task1, Run()); EXPECT_CALL(*task1, Die()); EXPECT_TRUE(PumpMessageLoop()); CallOnForegroundThread(task3); EXPECT_CALL(*task2, Run()); EXPECT_CALL(*task2, Die()); EXPECT_TRUE(PumpMessageLoop()); EXPECT_CALL(*task3, Run()); EXPECT_CALL(*task3, Die()); EXPECT_TRUE(PumpMessageLoop()); } TEST_F(DefaultPlatformTestWithMockTime, PendingDelayedTasksAreDestroyedOnShutdown) { StrictMock* task = new StrictMock; CallDelayedOnForegroundThread(task, 10); EXPECT_CALL(*task, Die()); } TEST_F(DefaultPlatformTestWithMockTime, RunIdleTasks) { StrictMock* task = new StrictMock; CallIdleOnForegroundThread(task); EXPECT_CALL(*task, Run(42.0 + 23.0)); EXPECT_CALL(*task, Die()); platform()->IncreaseTime(23.0); platform()->RunIdleTasks(isolate(), 42.0); } TEST_F(DefaultPlatformTestWithMockTime, PendingIdleTasksAreDestroyedOnShutdown) { StrictMock* task = new StrictMock; CallIdleOnForegroundThread(task); EXPECT_CALL(*task, Die()); } namespace { class TestBackgroundTask : public Task { public: explicit TestBackgroundTask(base::Semaphore* sem, bool* executed) : sem_(sem), executed_(executed) {} ~TestBackgroundTask() override { Die(); } MOCK_METHOD(void, Die, ()); void Run() override { *executed_ = true; sem_->Signal(); } private: base::Semaphore* sem_; bool* executed_; }; } // namespace TEST(CustomDefaultPlatformTest, RunBackgroundTask) { DefaultPlatform platform(1); base::Semaphore sem(0); bool task_executed = false; StrictMock* task = new StrictMock(&sem, &task_executed); EXPECT_CALL(*task, Die()); platform.CallOnWorkerThread(std::unique_ptr(task)); EXPECT_TRUE(sem.WaitFor(base::TimeDelta::FromSeconds(1))); EXPECT_TRUE(task_executed); } TEST(CustomDefaultPlatformTest, PostForegroundTaskAfterPlatformTermination) { std::shared_ptr foreground_taskrunner; { DefaultPlatformWithMockTime platform(1); int dummy; Isolate* isolate = reinterpret_cast(&dummy); foreground_taskrunner = platform.GetForegroundTaskRunner(isolate); } // It should still be possible to post foreground tasks, even when the // platform does not exist anymore. StrictMock* task1 = new StrictMock; EXPECT_CALL(*task1, Die()); foreground_taskrunner->PostTask(std::unique_ptr(task1)); StrictMock* task2 = new StrictMock; EXPECT_CALL(*task2, Die()); foreground_taskrunner->PostDelayedTask(std::unique_ptr(task2), 10); StrictMock* task3 = new StrictMock; EXPECT_CALL(*task3, Die()); foreground_taskrunner->PostIdleTask(std::unique_ptr(task3)); } } // namespace default_platform_unittest } // namespace platform } // namespace v8