113 lines
2.2 KiB
C++
113 lines
2.2 KiB
C++
|
#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;
|
||
|
}
|