binutils-gdb/gdb/testsuite/gdb.threads/tls.c

231 lines
4.9 KiB
C

/* BeginSourceFile tls.c
This file creates and deletes threads. It uses thread local storage
variables too. */
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <assert.h>
#include <pthread.h>
#include <semaphore.h>
#include <errno.h>
#define N_THREADS 3
/* Uncomment to turn on debugging output */
/*#define START_DEBUG*/
/* Thread-local storage. */
__thread int a_thread_local;
__thread int another_thread_local;
/* psymtabs->symtabs resolving check. */
extern __thread int file2_thread_local;
/* Global variable just for info addr in gdb. */
int a_global;
/* Print the results of thread-local storage. */
int thread_local_val[ N_THREADS ];
int another_thread_local_val[ N_THREADS ];
/* Semaphores to make sure the threads are alive when we print the TLS
variables from gdb. */
sem_t tell_main, tell_thread;
void print_error ()
{
switch (errno)
{
case EAGAIN:
fprintf (stderr, "EAGAIN\n");
break;
case EINTR:
fprintf (stderr, "EINTR\n");
break;
case EINVAL:
fprintf (stderr, "EINVAL\n");
break;
case ENOSYS:
fprintf (stderr, "ENOSYS\n");
break;
case ENOENT:
fprintf (stderr, "ENOENT\n");
break;
case EDEADLK:
fprintf (stderr, "EDEADLK\n");
break;
default:
fprintf (stderr, "Unknown error\n");
break;
}
}
/* Routine for each thread to run, does nothing. */
void *spin( vp )
void * vp;
{
int me = (long) vp;
int i;
/* Use a_global. */
a_global++;
a_thread_local = 0;
another_thread_local = me;
for( i = 0; i <= me; i++ ) {
a_thread_local += i;
}
another_thread_local_val[me] = another_thread_local;
thread_local_val[ me ] = a_thread_local; /* here we know tls value */
if (sem_post (&tell_main) == -1)
{
fprintf (stderr, "th %d post on sem tell_main failed\n", me);
print_error ();
return;
}
#ifdef START_DEBUG
fprintf (stderr, "th %d post on tell main\n", me);
#endif
while (1)
{
#ifdef START_DEBUG
fprintf (stderr, "th %d start wait on tell_thread\n", me);
#endif
if (sem_wait (&tell_thread) == 0)
break;
if (errno == EINTR)
{
#ifdef START_DEBUG
fprintf (stderr, "th %d wait tell_thread got EINTR, rewaiting\n", me);
#endif
continue;
}
else
{
fprintf (stderr, "th %d wait on sem tell_thread failed\n", me);
print_error ();
return;
}
}
#ifdef START_DEBUG
fprintf (stderr, "th %d Wait on tell_thread\n", me);
#endif
}
void
function_referencing_file2_thread_local (void)
{
file2_thread_local = file2_thread_local;
}
void
do_pass()
{
int i;
pthread_t t[ N_THREADS ];
int err;
for( i = 0; i < N_THREADS; i++)
{
thread_local_val[i] = 0;
another_thread_local_val[i] = 0;
}
if (sem_init (&tell_main, 0, 0) == -1)
{
fprintf (stderr, "tell_main semaphore init failed\n");
return;
}
if (sem_init (&tell_thread, 0, 0) == -1)
{
fprintf (stderr, "tell_thread semaphore init failed\n");
return;
}
/* Start N_THREADS threads, then join them so that they are terminated. */
for( i = 0; i < N_THREADS; i++ )
{
err = pthread_create( &t[i], NULL, spin, (void *) (long) i );
if( err != 0 ) {
fprintf(stderr, "Error in thread %d create\n", i );
}
}
for( i = 0; i < N_THREADS; i++ )
{
while (1)
{
#ifdef START_DEBUG
fprintf (stderr, "main %d start wait on tell_main\n", i);
#endif
if (sem_wait (&tell_main) == 0)
break;
if (errno == EINTR)
{
#ifdef START_DEBUG
fprintf (stderr, "main %d wait tell_main got EINTR, rewaiting\n", i);
#endif
continue;
}
else
{
fprintf (stderr, "main %d wait on sem tell_main failed\n", i);
print_error ();
return;
}
}
}
#ifdef START_DEBUG
fprintf (stderr, "main done waiting on tell_main\n");
#endif
i = 10; /* Here all threads should be still alive. */
for( i = 0; i < N_THREADS; i++ )
{
if (sem_post (&tell_thread) == -1)
{
fprintf (stderr, "main %d post on sem tell_thread failed\n", i);
print_error ();
return;
}
#ifdef START_DEBUG
fprintf (stderr, "main %d post on tell_thread\n", i);
#endif
}
for( i = 0; i < N_THREADS; i++ )
{
err = pthread_join(t[i], NULL );
if( err != 0 )
{
fprintf (stderr, "error in thread %d join\n", i );
}
}
i = 10; /* Null line for setting bpts on. */
}
int
main()
{
do_pass ();
return 0; /* Set breakpoint here before exit. */
}
/* EndSourceFile */