175 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			175 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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 {};
 | |
| 
 | |
| 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) {
 | |
|   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
 |