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; |
+} |