Chromium Code Reviews| Index: tests/threads/rwlock_test.c |
| diff --git a/tests/threads/rwlock_test.c b/tests/threads/rwlock_test.c |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..67f3d6657c35bf7f5aa68888a0ae4dda3298b1a4 |
| --- /dev/null |
| +++ b/tests/threads/rwlock_test.c |
| @@ -0,0 +1,215 @@ |
| +/* |
| + * Copyright 2014 The Native Client Authors. All rights reserved. |
| + * Use of this source code is governed by a BSD-style license that can be |
| + * found in the LICENSE file. |
| + */ |
| + |
| +#include <errno.h> |
| +#include <pthread.h> |
| +#include <stdio.h> |
| +#include <time.h> |
| + |
| +#include "native_client/src/include/nacl_assert.h" |
| + |
| +static pthread_rwlock_t rwlock; |
| +volatile int thread_has_lock = 0; |
| +volatile int thread_should_acquire_lock = 0; |
| +volatile int thread_should_release_lock = 0; |
| + |
| +#define READ_LOCK 1 |
|
Mark Seaborn
2014/12/01 22:33:07
Nit: use an enum?
Sam Clegg
2014/12/10 19:17:43
Done.
|
| +#define WRITE_LOCK 2 |
| + |
| +void *locking_thread(void *unused) { |
| + int rc; |
| + for (;;) { |
| + while (!thread_should_acquire_lock) { /* Spin. */ } |
| + |
| + ASSERT_EQ(thread_has_lock, 0); |
| + if (thread_should_acquire_lock == WRITE_LOCK) |
| + rc = pthread_rwlock_wrlock(&rwlock); |
| + else |
| + rc = pthread_rwlock_rdlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + __sync_fetch_and_add(&thread_has_lock, 1); |
| + |
| + while (!thread_should_release_lock) { /* Spin. */ } |
| + |
| + ASSERT_EQ(thread_has_lock, 1); |
| + rc = pthread_rwlock_unlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + __sync_fetch_and_sub(&thread_has_lock, 1); |
| + } |
| + |
| + return NULL; |
| +} |
| + |
| + |
| +void tell_thread_to_acquire_lock(int lock_type) { |
| + fprintf(stderr, "Thread acquiring lock: %s\n", |
| + lock_type == WRITE_LOCK ? "WRITE" : "READ" ); |
| + |
| + ASSERT_EQ(thread_has_lock, 0); |
| + ASSERT_EQ(thread_should_acquire_lock, 0); |
| + __sync_fetch_and_add(&thread_should_acquire_lock, lock_type); |
| + |
| + while (!thread_has_lock) { /* Spin. */ } |
| + |
| + __sync_fetch_and_sub(&thread_should_acquire_lock, lock_type); |
| + ASSERT_EQ(thread_should_acquire_lock, 0); |
| + |
| + fprintf(stderr, "Thread acquired lock.\n"); |
| +} |
| + |
| +void tell_thread_to_release_lock(void) { |
| + fprintf(stderr, "Thread releasing lock.\n"); |
| + |
| + ASSERT_EQ(thread_has_lock, 1); |
| + ASSERT_EQ(thread_should_release_lock, 0); |
| + __sync_fetch_and_add(&thread_should_release_lock, 1); |
| + |
| + while (thread_has_lock) { /* Spin. */ } |
| + |
| + __sync_fetch_and_sub(&thread_should_release_lock, 1); |
| + ASSERT_EQ(thread_should_release_lock, 0); |
| + |
| + fprintf(stderr, "Thread released lock.\n"); |
| +} |
| + |
| +void test_reader_timedwait(void) { |
| + fprintf(stderr, "test_reader_timedwait\n"); |
| + tell_thread_to_acquire_lock(WRITE_LOCK); |
| + |
| + struct timespec t = { 0, 0 }; |
| + int rc = pthread_rwlock_timedrdlock(&rwlock, &t); |
| + ASSERT_EQ(rc, ETIMEDOUT); |
| + |
| + tell_thread_to_release_lock(); |
| +} |
| + |
| +void test_writer_timedwait(void) { |
| + fprintf(stderr, "test_writer_timedwait\n"); |
| + tell_thread_to_acquire_lock(READ_LOCK); |
| + |
| + struct timespec t = { 0, 0 }; |
| + int rc = pthread_rwlock_timedwrlock(&rwlock, &t); |
| + ASSERT_EQ(rc, ETIMEDOUT); |
| + |
| + tell_thread_to_release_lock(); |
| +} |
| + |
| +void test_multiple_writers(void) { |
| + fprintf(stderr, "test_multiple_writers\n"); |
| + tell_thread_to_acquire_lock(WRITE_LOCK); |
| + |
| + /* |
| + * Attempt to acquire second write lock should fail. |
| + */ |
| + int rc = pthread_rwlock_trywrlock(&rwlock); |
| + ASSERT_EQ(rc, EBUSY); |
| + |
| + tell_thread_to_release_lock(); |
| +} |
| + |
| +void test_multiple_readers(void) { |
| + fprintf(stderr, "test_multiple_readers\n"); |
| + tell_thread_to_acquire_lock(READ_LOCK); |
| + |
| + /* |
| + * Now attempt to acquire the lock on the main thread. |
| + * Since they are both readers this should succeed. |
| + * Try with tryrdlock, rdlock and timedrdlock. |
| + */ |
| + int rc = pthread_rwlock_tryrdlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + rc = pthread_rwlock_unlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + |
| + struct timespec t = { 0, 0 }; |
| + rc = pthread_rwlock_timedrdlock(&rwlock, &t); |
| + ASSERT_EQ(rc, 0); |
| + rc = pthread_rwlock_unlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + |
| + rc = pthread_rwlock_rdlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + rc = pthread_rwlock_unlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + |
| + tell_thread_to_release_lock(); |
| +} |
| + |
| +void test_reader_plus_writer(void) { |
| + fprintf(stderr, "test_reader_plus_writer\n"); |
| + tell_thread_to_acquire_lock(READ_LOCK); |
| + |
| + /* |
| + * Now attempt to acquire the write lock on the main thread. |
| + * This should fail. |
| + */ |
| + int rc = pthread_rwlock_trywrlock(&rwlock); |
| + ASSERT_EQ(rc, EBUSY); |
| + |
| + tell_thread_to_release_lock(); |
| +} |
| + |
| +void test_writer_plus_reader(void) { |
| + fprintf(stderr, "test_writer_plus_reader\n"); |
| + |
| + /* |
| + * First get the write lock. |
| + */ |
| + int rc = pthread_rwlock_wrlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| + |
| + /* |
| + * Attempt to acquire read lock should now fail |
| + */ |
| + rc = pthread_rwlock_tryrdlock(&rwlock); |
| + ASSERT_EQ(rc, EBUSY); |
| + |
| + rc = pthread_rwlock_unlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| +} |
| + |
| +void test_unlocked_with_zero_timestamp(void) { |
| + fprintf(stderr, "test_unlocked_with_zero_timestamp\n"); |
| + int rc; |
| + struct timespec abstime = { 0, 0 }; |
| + ASSERT_EQ(thread_has_lock, 0); |
| + fprintf(stderr, "Trying to lock the unlocked rwlock with a valid " |
| + "zero absolute timestamp. " |
| + "Expected to succeed instantly since the lock is free.\n"); |
| + rc = pthread_rwlock_timedrdlock(&rwlock, &abstime); |
| + ASSERT_EQ(rc, 0); |
| + rc = pthread_rwlock_unlock(&rwlock); |
| + ASSERT_EQ(rc, 0); |
| +} |
| + |
| +int main(int argc, char **argv) { |
| + int rc; |
| + fprintf(stderr, "Running...\n"); |
| + |
| + pthread_rwlockattr_t attrs; |
| + rc = pthread_rwlockattr_init(&attrs); |
| + ASSERT_EQ(rc, 0); |
| + rc = pthread_rwlock_init(&rwlock, &attrs); |
| + ASSERT_EQ(rc, 0); |
| + rc = pthread_rwlockattr_destroy(&attrs); |
| + ASSERT_EQ(rc, 0); |
| + |
| + pthread_t thread; |
| + rc = pthread_create(&thread, NULL, locking_thread, NULL); |
| + ASSERT_EQ(rc, 0); |
| + fprintf(stderr, "Thread started.\n"); |
| + |
| + test_unlocked_with_zero_timestamp(); |
| + test_multiple_readers(); |
| + test_multiple_writers(); |
| + test_reader_plus_writer(); |
| + test_writer_plus_reader(); |
| + test_reader_timedwait(); |
| + test_writer_timedwait(); |
| + |
|
Mark Seaborn
2014/12/01 22:33:07
Also test pthread_rwlock_destroy()?
Sam Clegg
2014/12/10 19:17:43
Done.
|
| + fprintf(stderr, "Done.\n"); |
| + return 0; |
| +} |