|
|
@ -67,54 +67,53 @@ TEST_F(ThreadLocalTest, UniqueIdTest) { |
|
|
|
port::Mutex mu; |
|
|
|
port::Mutex mu; |
|
|
|
port::CondVar cv(&mu); |
|
|
|
port::CondVar cv(&mu); |
|
|
|
|
|
|
|
|
|
|
|
// perf_context and iostats_context take 2 ids
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 0u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
|
|
|
|
// New ThreadLocal instance bumps id by 1
|
|
|
|
// New ThreadLocal instance bumps id by 1
|
|
|
|
{ |
|
|
|
{ |
|
|
|
// Id used 2
|
|
|
|
// Id used 0
|
|
|
|
Params p1(&mu, &cv, nullptr, 1u); |
|
|
|
Params p1(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 1u); |
|
|
|
// Id used 3
|
|
|
|
// Id used 1
|
|
|
|
Params p2(&mu, &cv, nullptr, 1u); |
|
|
|
Params p2(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 4u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
// Id used 4
|
|
|
|
// Id used 2
|
|
|
|
Params p3(&mu, &cv, nullptr, 1u); |
|
|
|
Params p3(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 5u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
// Id used 5
|
|
|
|
// Id used 3
|
|
|
|
Params p4(&mu, &cv, nullptr, 1u); |
|
|
|
Params p4(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 6u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 4u); |
|
|
|
} |
|
|
|
} |
|
|
|
// id 5, 4, 3, 2 are in the free queue in order
|
|
|
|
// id 3, 2, 1, 0 are in the free queue in order
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 0u); |
|
|
|
|
|
|
|
|
|
|
|
// pick up 2
|
|
|
|
// pick up 0
|
|
|
|
Params p1(&mu, &cv, nullptr, 1u); |
|
|
|
Params p1(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 1u); |
|
|
|
// pick up 3
|
|
|
|
// pick up 1
|
|
|
|
Params* p2 = new Params(&mu, &cv, nullptr, 1u); |
|
|
|
Params* p2 = new Params(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 4u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
// pick up 4
|
|
|
|
// pick up 2
|
|
|
|
Params p3(&mu, &cv, nullptr, 1u); |
|
|
|
Params p3(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 5u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
// return up 3
|
|
|
|
// return up 1
|
|
|
|
delete p2; |
|
|
|
delete p2; |
|
|
|
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 1u); |
|
|
|
|
|
|
|
// Now we have 3, 1 in queue
|
|
|
|
|
|
|
|
// pick up 1
|
|
|
|
|
|
|
|
Params p4(&mu, &cv, nullptr, 1u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
// Now we have 4, 2 in queue
|
|
|
|
|
|
|
|
// pick up 3
|
|
|
|
// pick up 3
|
|
|
|
Params p4(&mu, &cv, nullptr, 1u); |
|
|
|
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 5u); |
|
|
|
|
|
|
|
// pick up 5
|
|
|
|
|
|
|
|
Params p5(&mu, &cv, nullptr, 1u); |
|
|
|
Params p5(&mu, &cv, nullptr, 1u); |
|
|
|
// next new id
|
|
|
|
// next new id
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 6u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 4u); |
|
|
|
// After exit, id sequence in queue:
|
|
|
|
// After exit, id sequence in queue:
|
|
|
|
// 5, 4, 3, 2(, 1, 0)
|
|
|
|
// 3, 1, 2, 0
|
|
|
|
} |
|
|
|
} |
|
|
|
#endif // __clang_analyzer__
|
|
|
|
#endif // __clang_analyzer__
|
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ThreadLocalTest, SequentialReadWriteTest) { |
|
|
|
TEST_F(ThreadLocalTest, SequentialReadWriteTest) { |
|
|
|
// global id list carries over 5, 4, 3, 2
|
|
|
|
// global id list carries over 3, 1, 2, 0
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 0u); |
|
|
|
|
|
|
|
|
|
|
|
port::Mutex mu; |
|
|
|
port::Mutex mu; |
|
|
|
port::CondVar cv(&mu); |
|
|
|
port::CondVar cv(&mu); |
|
|
@ -144,7 +143,7 @@ TEST_F(ThreadLocalTest, SequentialReadWriteTest) { |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
for (int iter = 0; iter < 1024; ++iter) { |
|
|
|
for (int iter = 0; iter < 1024; ++iter) { |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 1u); |
|
|
|
// Another new thread, read/write should not see value from previous thread
|
|
|
|
// Another new thread, read/write should not see value from previous thread
|
|
|
|
env_->StartThread(func, static_cast<void*>(&p)); |
|
|
|
env_->StartThread(func, static_cast<void*>(&p)); |
|
|
|
mu.Lock(); |
|
|
|
mu.Lock(); |
|
|
@ -152,13 +151,13 @@ TEST_F(ThreadLocalTest, SequentialReadWriteTest) { |
|
|
|
cv.Wait(); |
|
|
|
cv.Wait(); |
|
|
|
} |
|
|
|
} |
|
|
|
mu.Unlock(); |
|
|
|
mu.Unlock(); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 1u); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ThreadLocalTest, ConcurrentReadWriteTest) { |
|
|
|
TEST_F(ThreadLocalTest, ConcurrentReadWriteTest) { |
|
|
|
// global id list carries over 5, 4, 3, 2
|
|
|
|
// global id list carries over 3, 1, 2, 0
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 0u); |
|
|
|
|
|
|
|
|
|
|
|
ThreadLocalPtr tls2; |
|
|
|
ThreadLocalPtr tls2; |
|
|
|
port::Mutex mu1; |
|
|
|
port::Mutex mu1; |
|
|
@ -239,11 +238,11 @@ TEST_F(ThreadLocalTest, ConcurrentReadWriteTest) { |
|
|
|
} |
|
|
|
} |
|
|
|
mu2.Unlock(); |
|
|
|
mu2.Unlock(); |
|
|
|
|
|
|
|
|
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 5u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 3u); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
TEST_F(ThreadLocalTest, Unref) { |
|
|
|
TEST_F(ThreadLocalTest, Unref) { |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 2u); |
|
|
|
ASSERT_EQ(IDChecker::PeekId(), 0u); |
|
|
|
|
|
|
|
|
|
|
|
auto unref = [](void* ptr) { |
|
|
|
auto unref = [](void* ptr) { |
|
|
|
auto& p = *static_cast<Params*>(ptr); |
|
|
|
auto& p = *static_cast<Params*>(ptr); |
|
|
|