| /* |
| * Copyright (C) 2012 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 "mutex-inl.h" |
| |
| #include "common_runtime_test.h" |
| #include "thread-current-inl.h" |
| |
| namespace art { |
| |
| class MutexTest : public CommonRuntimeTest { |
| protected: |
| MutexTest() { |
| use_boot_image_ = true; // Make the Runtime creation cheaper. |
| } |
| }; |
| |
| struct MutexTester { |
| static void AssertDepth(Mutex& mu, uint32_t expected_depth) { |
| ASSERT_EQ(expected_depth, mu.GetDepth()); |
| |
| // This test is single-threaded, so we also know _who_ should hold the lock. |
| if (expected_depth == 0) { |
| mu.AssertNotHeld(Thread::Current()); |
| } else { |
| mu.AssertHeld(Thread::Current()); |
| } |
| } |
| }; |
| |
| TEST_F(MutexTest, LockUnlock) { |
| // TODO: Remove `Mutex` dependency on `Runtime` or at least make sure it works |
| // without a `Runtime` with reasonable defaults (and without dumping stack for timeout). |
| ASSERT_TRUE(Runtime::Current() != nullptr); |
| Mutex mu("test mutex"); |
| MutexTester::AssertDepth(mu, 0U); |
| mu.Lock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 1U); |
| mu.Unlock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 0U); |
| } |
| |
| // GCC has trouble with our mutex tests, so we have to turn off thread safety analysis. |
| static void TryLockUnlockTest() NO_THREAD_SAFETY_ANALYSIS { |
| Mutex mu("test mutex"); |
| MutexTester::AssertDepth(mu, 0U); |
| ASSERT_TRUE(mu.TryLock(Thread::Current())); |
| MutexTester::AssertDepth(mu, 1U); |
| mu.Unlock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 0U); |
| } |
| |
| TEST_F(MutexTest, TryLockUnlock) { |
| TryLockUnlockTest(); |
| } |
| |
| // GCC has trouble with our mutex tests, so we have to turn off thread safety analysis. |
| static void RecursiveLockUnlockTest() NO_THREAD_SAFETY_ANALYSIS { |
| Mutex mu("test mutex", kDefaultMutexLevel, true); |
| MutexTester::AssertDepth(mu, 0U); |
| mu.Lock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 1U); |
| mu.Lock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 2U); |
| mu.Unlock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 1U); |
| mu.Unlock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 0U); |
| } |
| |
| TEST_F(MutexTest, RecursiveLockUnlock) { |
| RecursiveLockUnlockTest(); |
| } |
| |
| // GCC has trouble with our mutex tests, so we have to turn off thread safety analysis. |
| static void RecursiveTryLockUnlockTest() NO_THREAD_SAFETY_ANALYSIS { |
| Mutex mu("test mutex", kDefaultMutexLevel, true); |
| MutexTester::AssertDepth(mu, 0U); |
| ASSERT_TRUE(mu.TryLock(Thread::Current())); |
| MutexTester::AssertDepth(mu, 1U); |
| ASSERT_TRUE(mu.TryLock(Thread::Current())); |
| MutexTester::AssertDepth(mu, 2U); |
| mu.Unlock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 1U); |
| mu.Unlock(Thread::Current()); |
| MutexTester::AssertDepth(mu, 0U); |
| } |
| |
| TEST_F(MutexTest, RecursiveTryLockUnlock) { |
| RecursiveTryLockUnlockTest(); |
| } |
| |
| |
| struct RecursiveLockWait { |
| RecursiveLockWait() |
| : mu("test mutex", kDefaultMutexLevel, true), cv("test condition variable", mu) { |
| } |
| |
| Mutex mu; |
| ConditionVariable cv; |
| }; |
| |
| static void* RecursiveLockWaitCallback(void* arg) { |
| RecursiveLockWait* state = reinterpret_cast<RecursiveLockWait*>(arg); |
| state->mu.Lock(Thread::Current()); |
| state->cv.Signal(Thread::Current()); |
| state->mu.Unlock(Thread::Current()); |
| return nullptr; |
| } |
| |
| // GCC has trouble with our mutex tests, so we have to turn off thread safety analysis. |
| static void RecursiveLockWaitTest() NO_THREAD_SAFETY_ANALYSIS { |
| RecursiveLockWait state; |
| state.mu.Lock(Thread::Current()); |
| state.mu.Lock(Thread::Current()); |
| |
| pthread_t pthread; |
| int pthread_create_result = pthread_create(&pthread, nullptr, RecursiveLockWaitCallback, &state); |
| ASSERT_EQ(0, pthread_create_result); |
| |
| state.cv.Wait(Thread::Current()); |
| |
| state.mu.Unlock(Thread::Current()); |
| state.mu.Unlock(Thread::Current()); |
| EXPECT_EQ(pthread_join(pthread, nullptr), 0); |
| } |
| |
| // This ensures we don't hang when waiting on a recursively locked mutex, |
| // which is not supported with bare pthread_mutex_t. |
| TEST_F(MutexTest, RecursiveLockWait) { |
| RecursiveLockWaitTest(); |
| } |
| |
| TEST_F(MutexTest, SharedLockUnlock) { |
| ReaderWriterMutex mu("test rwmutex"); |
| mu.AssertNotHeld(Thread::Current()); |
| mu.AssertNotExclusiveHeld(Thread::Current()); |
| mu.SharedLock(Thread::Current()); |
| mu.AssertSharedHeld(Thread::Current()); |
| mu.AssertNotExclusiveHeld(Thread::Current()); |
| mu.SharedUnlock(Thread::Current()); |
| mu.AssertNotHeld(Thread::Current()); |
| } |
| |
| TEST_F(MutexTest, ExclusiveLockUnlock) { |
| ReaderWriterMutex mu("test rwmutex"); |
| mu.AssertNotHeld(Thread::Current()); |
| mu.ExclusiveLock(Thread::Current()); |
| mu.AssertSharedHeld(Thread::Current()); |
| mu.AssertExclusiveHeld(Thread::Current()); |
| mu.ExclusiveUnlock(Thread::Current()); |
| mu.AssertNotHeld(Thread::Current()); |
| } |
| |
| // GCC has trouble with our mutex tests, so we have to turn off thread safety analysis. |
| static void SharedTryLockUnlockTest() NO_THREAD_SAFETY_ANALYSIS { |
| ReaderWriterMutex mu("test rwmutex"); |
| mu.AssertNotHeld(Thread::Current()); |
| ASSERT_TRUE(mu.SharedTryLock(Thread::Current())); |
| mu.AssertSharedHeld(Thread::Current()); |
| mu.SharedUnlock(Thread::Current()); |
| mu.AssertNotHeld(Thread::Current()); |
| } |
| |
| TEST_F(MutexTest, SharedTryLockUnlock) { |
| SharedTryLockUnlockTest(); |
| } |
| |
| } // namespace art |