summaryrefslogtreecommitdiff
path: root/3rdparty/plibsys/tests/prwlock_test.cpp
diff options
context:
space:
mode:
Diffstat (limited to '3rdparty/plibsys/tests/prwlock_test.cpp')
-rw-r--r--3rdparty/plibsys/tests/prwlock_test.cpp224
1 files changed, 224 insertions, 0 deletions
diff --git a/3rdparty/plibsys/tests/prwlock_test.cpp b/3rdparty/plibsys/tests/prwlock_test.cpp
new file mode 100644
index 0000000..c8bce38
--- /dev/null
+++ b/3rdparty/plibsys/tests/prwlock_test.cpp
@@ -0,0 +1,224 @@
+/*
+ * The MIT License
+ *
+ * Copyright (C) 2016-2019 Alexander Saprykin <saprykin.spb@gmail.com>
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining
+ * a copy of this software and associated documentation files (the
+ * 'Software'), to deal in the Software without restriction, including
+ * without limitation the rights to use, copy, modify, merge, publish,
+ * distribute, sublicense, and/or sell copies of the Software, and to
+ * permit persons to whom the Software is furnished to do so, subject to
+ * the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be
+ * included in all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
+ * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+ * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
+ * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
+ * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
+ * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
+ * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+ */
+
+#include "plibsys.h"
+#include "ptestmacros.h"
+
+#include <string.h>
+
+P_TEST_MODULE_INIT ();
+
+#define PRWLOCK_TEST_STRING_1 "This is a test string."
+#define PRWLOCK_TEST_STRING_2 "Ouh, yet another string to check!"
+
+static PRWLock * test_rwlock = NULL;
+static volatile pboolean is_threads_working = FALSE;
+static volatile pint writers_counter = 0;
+static pchar string_buf[50];
+
+extern "C" ppointer pmem_alloc (psize nbytes)
+{
+ P_UNUSED (nbytes);
+ return (ppointer) NULL;
+}
+
+extern "C" ppointer pmem_realloc (ppointer block, psize nbytes)
+{
+ P_UNUSED (block);
+ P_UNUSED (nbytes);
+ return (ppointer) NULL;
+}
+
+extern "C" void pmem_free (ppointer block)
+{
+ P_UNUSED (block);
+}
+
+static void * reader_thread_func (void *data)
+{
+ P_UNUSED (data);
+
+ pint counter = 0;
+
+ while (p_atomic_int_get (&writers_counter) == 0)
+ p_uthread_sleep (10);
+
+ while (is_threads_working == TRUE) {
+ p_uthread_sleep (10);
+
+ if (p_rwlock_reader_trylock (test_rwlock) == FALSE) {
+ if (p_rwlock_reader_lock (test_rwlock) == FALSE)
+ p_uthread_exit (-1);
+ }
+
+ if (strcmp (string_buf, PRWLOCK_TEST_STRING_1) != 0 &&
+ strcmp (string_buf, PRWLOCK_TEST_STRING_2) != 0) {
+ p_rwlock_reader_unlock (test_rwlock);
+ p_uthread_exit (-1);
+ }
+
+ if (p_rwlock_reader_unlock (test_rwlock) == FALSE)
+ p_uthread_exit (-1);
+
+ ++counter;
+ }
+
+ p_uthread_exit (counter);
+
+ return NULL;
+}
+
+static void * writer_thread_func (void *data)
+{
+ pint string_num = PPOINTER_TO_INT (data);
+ pint counter = 0;
+
+ while (is_threads_working == TRUE) {
+ p_uthread_sleep (10);
+
+ if (p_rwlock_writer_trylock (test_rwlock) == FALSE) {
+ if (p_rwlock_writer_lock (test_rwlock) == FALSE)
+ p_uthread_exit (-1);
+ }
+
+ memset (string_buf, 0, sizeof (string_buf));
+
+ if (string_num == 1)
+ strcpy (string_buf, PRWLOCK_TEST_STRING_1);
+ else
+ strcpy (string_buf, PRWLOCK_TEST_STRING_1);
+
+ if (p_rwlock_writer_unlock (test_rwlock) == FALSE)
+ p_uthread_exit (-1);
+
+ ++counter;
+
+ p_atomic_int_inc ((&writers_counter));
+ }
+
+ p_uthread_exit (counter);
+
+ return NULL;
+}
+
+P_TEST_CASE_BEGIN (prwlock_nomem_test)
+{
+ p_libsys_init ();
+
+ PMemVTable vtable;
+
+ vtable.free = pmem_free;
+ vtable.malloc = pmem_alloc;
+ vtable.realloc = pmem_realloc;
+
+ P_TEST_CHECK (p_mem_set_vtable (&vtable) == TRUE);
+
+ P_TEST_CHECK (p_rwlock_new () == NULL);
+
+ p_mem_restore_vtable ();
+
+ p_libsys_shutdown ();
+}
+P_TEST_CASE_END ()
+
+P_TEST_CASE_BEGIN (prwlock_bad_input_test)
+{
+ p_libsys_init ();
+
+ P_TEST_CHECK (p_rwlock_reader_lock (NULL) == FALSE);
+ P_TEST_CHECK (p_rwlock_reader_trylock (NULL) == FALSE);
+ P_TEST_CHECK (p_rwlock_reader_unlock (NULL) == FALSE);
+ P_TEST_CHECK (p_rwlock_writer_lock (NULL) == FALSE);
+ P_TEST_CHECK (p_rwlock_writer_trylock (NULL) == FALSE);
+ P_TEST_CHECK (p_rwlock_writer_unlock (NULL) == FALSE);
+ p_rwlock_free (NULL);
+
+ p_libsys_shutdown ();
+}
+P_TEST_CASE_END ()
+
+P_TEST_CASE_BEGIN (prwlock_general_test)
+{
+ p_libsys_init ();
+
+ test_rwlock = p_rwlock_new ();
+
+ P_TEST_REQUIRE (test_rwlock != NULL);
+
+ is_threads_working = TRUE;
+ writers_counter = 0;
+
+ PUThread *reader_thr1 = p_uthread_create ((PUThreadFunc) reader_thread_func,
+ NULL,
+ TRUE,
+ NULL);
+
+ PUThread *reader_thr2 = p_uthread_create ((PUThreadFunc) reader_thread_func,
+ NULL,
+ TRUE,
+ NULL);
+
+ PUThread *writer_thr1 = p_uthread_create ((PUThreadFunc) writer_thread_func,
+ NULL,
+ TRUE,
+ NULL);
+
+ PUThread *writer_thr2 = p_uthread_create ((PUThreadFunc) writer_thread_func,
+ NULL,
+ TRUE,
+ NULL);
+
+ P_TEST_REQUIRE (reader_thr1 != NULL);
+ P_TEST_REQUIRE (reader_thr2 != NULL);
+ P_TEST_REQUIRE (writer_thr1 != NULL);
+ P_TEST_REQUIRE (writer_thr2 != NULL);
+
+ p_uthread_sleep (10000);
+
+ is_threads_working = FALSE;
+
+ P_TEST_CHECK (p_uthread_join (reader_thr1) > 0);
+ P_TEST_CHECK (p_uthread_join (reader_thr2) > 0);
+ P_TEST_CHECK (p_uthread_join (writer_thr1) > 0);
+ P_TEST_CHECK (p_uthread_join (writer_thr2) > 0);
+
+ p_uthread_unref (reader_thr1);
+ p_uthread_unref (reader_thr2);
+ p_uthread_unref (writer_thr1);
+ p_uthread_unref (writer_thr2);
+
+ p_rwlock_free (test_rwlock);
+
+ p_libsys_shutdown ();
+}
+P_TEST_CASE_END ()
+
+P_TEST_SUITE_BEGIN()
+{
+ P_TEST_SUITE_RUN_CASE (prwlock_nomem_test);
+ P_TEST_SUITE_RUN_CASE (prwlock_bad_input_test);
+ P_TEST_SUITE_RUN_CASE (prwlock_general_test);
+}
+P_TEST_SUITE_END()