blob: fca8c901b938b7cc4c17e7f3c0050f73e82b1c94 [file] [log] [blame]
/*
* 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 HIDDEN {
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