You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
5.3 KiB
175 lines
5.3 KiB
4 months ago
|
/*
|
||
|
* 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
|