| // Copyright 2019 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/android/java_handler_thread.h" |
| |
| #include "base/synchronization/waitable_event.h" |
| #include "base/task/sequence_manager/sequence_manager_impl.h" |
| #include "base/task/task_observer.h" |
| #include "base/test/android/java_handler_thread_helpers.h" |
| #include "base/test/bind.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace base { |
| namespace { |
| |
| class JavaHandlerThreadForTest : public android::JavaHandlerThread { |
| public: |
| explicit JavaHandlerThreadForTest( |
| const char* name, |
| base::ThreadType thread_type = base::ThreadType::kDefault) |
| : android::JavaHandlerThread(name, thread_type) {} |
| |
| using android::JavaHandlerThread::state; |
| using android::JavaHandlerThread::State; |
| }; |
| |
| class DummyTaskObserver : public TaskObserver { |
| public: |
| explicit DummyTaskObserver(int num_tasks) |
| : num_tasks_started_(0), num_tasks_processed_(0), num_tasks_(num_tasks) {} |
| |
| DummyTaskObserver(int num_tasks, int num_tasks_started) |
| : num_tasks_started_(num_tasks_started), |
| num_tasks_processed_(0), |
| num_tasks_(num_tasks) {} |
| |
| DummyTaskObserver(const DummyTaskObserver&) = delete; |
| DummyTaskObserver& operator=(const DummyTaskObserver&) = delete; |
| |
| ~DummyTaskObserver() override = default; |
| |
| void WillProcessTask(const PendingTask& /* pending_task */, |
| bool /* was_blocked_or_low_priority */) override { |
| num_tasks_started_++; |
| EXPECT_LE(num_tasks_started_, num_tasks_); |
| EXPECT_EQ(num_tasks_started_, num_tasks_processed_ + 1); |
| } |
| |
| void DidProcessTask(const PendingTask& pending_task) override { |
| num_tasks_processed_++; |
| EXPECT_LE(num_tasks_started_, num_tasks_); |
| EXPECT_EQ(num_tasks_started_, num_tasks_processed_); |
| } |
| |
| int num_tasks_started() const { return num_tasks_started_; } |
| int num_tasks_processed() const { return num_tasks_processed_; } |
| |
| private: |
| int num_tasks_started_; |
| int num_tasks_processed_; |
| const int num_tasks_; |
| }; |
| |
| void PostNTasks(int posts_remaining) { |
| if (posts_remaining > 1) { |
| SingleThreadTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, BindOnce(&PostNTasks, posts_remaining - 1)); |
| } |
| } |
| |
| } // namespace |
| |
| class JavaHandlerThreadTest : public ::testing::Test {}; |
| |
| void RunTest_AbortDontRunMoreTasks(bool delayed, bool init_java_first) { |
| WaitableEvent test_done_event; |
| std::unique_ptr<android::JavaHandlerThread> java_thread; |
| if (init_java_first) { |
| java_thread = android::JavaHandlerThreadHelpers::CreateJavaFirst(); |
| } else { |
| java_thread = std::make_unique<android::JavaHandlerThread>( |
| "JavaHandlerThreadForTesting from AbortDontRunMoreTasks"); |
| } |
| java_thread->Start(); |
| java_thread->ListenForUncaughtExceptionsForTesting(); |
| |
| auto target = |
| BindOnce(&android::JavaHandlerThreadHelpers::ThrowExceptionAndAbort, |
| &test_done_event); |
| if (delayed) { |
| java_thread->task_runner()->PostDelayedTask(FROM_HERE, std::move(target), |
| Milliseconds(10)); |
| } else { |
| java_thread->task_runner()->PostTask(FROM_HERE, std::move(target)); |
| java_thread->task_runner()->PostTask(FROM_HERE, |
| MakeExpectedNotRunClosure(FROM_HERE)); |
| } |
| test_done_event.Wait(); |
| java_thread->Stop(); |
| android::ScopedJavaLocalRef<jthrowable> exception = |
| java_thread->GetUncaughtExceptionIfAny(); |
| ASSERT_TRUE( |
| android::JavaHandlerThreadHelpers::IsExceptionTestException(exception)); |
| } |
| |
| TEST_F(JavaHandlerThreadTest, JavaExceptionAbort) { |
| constexpr bool delayed = false; |
| constexpr bool init_java_first = false; |
| RunTest_AbortDontRunMoreTasks(delayed, init_java_first); |
| } |
| |
| TEST_F(JavaHandlerThreadTest, DelayedJavaExceptionAbort) { |
| constexpr bool delayed = true; |
| constexpr bool init_java_first = false; |
| RunTest_AbortDontRunMoreTasks(delayed, init_java_first); |
| } |
| |
| TEST_F(JavaHandlerThreadTest, JavaExceptionAbortInitJavaFirst) { |
| constexpr bool delayed = false; |
| constexpr bool init_java_first = true; |
| RunTest_AbortDontRunMoreTasks(delayed, init_java_first); |
| } |
| |
| TEST_F(JavaHandlerThreadTest, RunTasksWhileShuttingDownJavaThread) { |
| const int kNumPosts = 6; |
| DummyTaskObserver observer(kNumPosts, 1); |
| |
| auto java_thread = std::make_unique<JavaHandlerThreadForTest>("test"); |
| java_thread->Start(); |
| |
| sequence_manager::internal::SequenceManagerImpl* sequence_manager = |
| static_cast<sequence_manager::internal::SequenceManagerImpl*>( |
| java_thread->state()->sequence_manager.get()); |
| |
| java_thread->task_runner()->PostTask( |
| FROM_HERE, BindLambdaForTesting([&]() { |
| sequence_manager->AddTaskObserver(&observer); |
| SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask( |
| FROM_HERE, MakeExpectedNotRunClosure(FROM_HERE), Days(1)); |
| java_thread->StopSequenceManagerForTesting(); |
| PostNTasks(kNumPosts); |
| })); |
| |
| java_thread->JoinForTesting(); |
| java_thread.reset(); |
| |
| EXPECT_EQ(kNumPosts, observer.num_tasks_started()); |
| EXPECT_EQ(kNumPosts, observer.num_tasks_processed()); |
| } |
| |
| } // namespace base |