summaryrefslogtreecommitdiff
path: root/libvtv/testsuite/other-tests/dlopen_mt.cc
diff options
context:
space:
mode:
Diffstat (limited to 'libvtv/testsuite/other-tests/dlopen_mt.cc')
-rw-r--r--libvtv/testsuite/other-tests/dlopen_mt.cc112
1 files changed, 112 insertions, 0 deletions
diff --git a/libvtv/testsuite/other-tests/dlopen_mt.cc b/libvtv/testsuite/other-tests/dlopen_mt.cc
new file mode 100644
index 00000000000..772e8a733ed
--- /dev/null
+++ b/libvtv/testsuite/other-tests/dlopen_mt.cc
@@ -0,0 +1,112 @@
+#include <stdlib.h>
+#include <dlfcn.h>
+#include <stdio.h>
+
+#include "vtv_utils.h"
+#include "vtv_rts.h"
+#include "pthread.h"
+
+#define NUM_REPEATS 10
+#define NUM_THREADS 10
+#define NUM_SOS 100
+#define NUM_SOS_PER_THREAD (NUM_SOS/NUM_THREADS)
+
+typedef void (*voidfn)(void);
+
+int failures = 0;
+
+void
+__vtv_verify_fail (void **data_set_ptr, const void *vtbl_pointer)
+{
+ failures++;
+ return;
+}
+
+
+void do_dlopen(int so_num)
+{
+ char so_name [sizeof("soxxx.so")];
+ sprintf(so_name, "so%d.so", so_num);
+ // printf("dl-opening %s\n", so_name);
+ void * dlhandle = dlopen(so_name, RTLD_NOW);
+ if (!dlhandle)
+ {
+ fprintf(stderr, "dlopen so:%s error: %s\n", so_name, dlerror());
+ exit(1);
+ }
+ char so_entry [sizeof("so_entry_xxx")];
+ sprintf(so_entry, "so_entry_%d", so_num);
+ voidfn so_entry_fn = (voidfn)dlsym(dlhandle, so_entry);
+ if (!so_entry_fn)
+ {
+ fprintf(stderr, "so:%s dlsym error: %s\n", so_name, dlerror());
+ exit(2);
+ }
+
+ so_entry_fn();
+
+ dlclose(dlhandle);
+}
+
+volatile int threads_completed_it = 0;
+volatile int current_wave = -1;
+
+void * do_dlopens(void * ptid)
+{
+ for (int k = 0; k < NUM_REPEATS; k++)
+ {
+
+ for (int i = 0; i < NUM_SOS_PER_THREAD; i++)
+ {
+ while (current_wave < (k*NUM_SOS_PER_THREAD + i)) /* from 0 to 99 */
+ ;
+
+ do_dlopen((NUM_SOS_PER_THREAD * *(int *)ptid) + i);
+
+ int old_value;
+ do {
+ old_value = threads_completed_it;
+ } while (!__sync_bool_compare_and_swap(&threads_completed_it, old_value, old_value + 1));
+
+ if (old_value == (NUM_THREADS - 1)) // Only one thread will do this.
+ {
+ threads_completed_it = 0;
+ printf("%c%d", 13, current_wave + 1);
+ fflush(stdout);
+ current_wave++;
+ }
+ }
+ }
+
+ return NULL;
+}
+
+
+int main()
+{
+ pthread_t thread_ids[NUM_THREADS];
+ int thread_nids[NUM_THREADS];
+
+ for (int t = 0; t < NUM_THREADS; t++ )
+ {
+ thread_nids[t] = t;
+ if (pthread_create(&thread_ids[t], NULL, do_dlopens, &thread_nids[t]) != 0)
+ {
+ printf("failed pthread_create\n");
+ exit(1);
+ }
+ }
+
+ current_wave = 0; // start the work on the other threads
+
+ for (int t = 0; t < NUM_THREADS; t++)
+ if (pthread_join(thread_ids[t], NULL) != 0)
+ {
+ printf("failed pthread_join\n");
+ exit(2);
+ }
+
+ printf("\n");
+
+ return 0;
+}