| /* |
| * Copyright (C) 2019 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "common_runtime_test.h" |
| |
| #include <thread> |
| |
| #include "android-base/logging.h" |
| #include "base/locks.h" |
| #include "base/mutex.h" |
| #include "runtime.h" |
| #include "thread-current-inl.h" |
| |
| namespace art { |
| |
| class RuntimeTest : public CommonRuntimeTest {}; |
| |
| // Ensure that abort works with ThreadList locks held. |
| |
| TEST_F(RuntimeTest, AbortWithThreadListLockHeld) { |
| // This assumes the test is run single-threaded: do not start the runtime to avoid daemon threads. |
| |
| constexpr const char* kDeathRegex = "Skipping all-threads dump as locks are held"; |
| ASSERT_DEATH({ |
| // The regex only works if we can ensure output goes to stderr. |
| android::base::SetLogger(android::base::StderrLogger); |
| |
| MutexLock mu(Thread::Current(), *Locks::thread_list_lock_); |
| Runtime::Abort("Attempt to abort"); |
| }, kDeathRegex); |
| } |
| |
| |
| TEST_F(RuntimeTest, AbortWithThreadSuspendCountLockHeld) { |
| // This assumes the test is run single-threaded: do not start the runtime to avoid daemon threads. |
| |
| constexpr const char* kDeathRegex = "Skipping all-threads dump as locks are held"; |
| ASSERT_DEATH({ |
| // The regex only works if we can ensure output goes to stderr. |
| android::base::SetLogger(android::base::StderrLogger); |
| |
| MutexLock mu(Thread::Current(), *Locks::thread_suspend_count_lock_); |
| Runtime::Abort("Attempt to abort"); |
| }, kDeathRegex); |
| } |
| |
| TEST_F(RuntimeTest, AbortFromUnattachedThread) { |
| // This assumes the test is run single-threaded: do not start the runtime to avoid daemon threads. |
| |
| constexpr const char* kDeathRegex = "Going down"; |
| ASSERT_EXIT({ |
| // The regex only works if we can ensure output goes to stderr. |
| android::base::SetLogger(android::base::StderrLogger); |
| |
| Thread::Current()->TransitionFromSuspendedToRunnable(); |
| runtime_->Start(); |
| |
| std::thread t([]() { |
| LOG(FATAL) << "Going down"; |
| }); |
| t.join(); |
| }, ::testing::KilledBySignal(SIGABRT), kDeathRegex); |
| } |
| |
| } // namespace art |