box64/tests/test11.c

132 lines
3.5 KiB
C
Raw Normal View History

2021-02-28 12:19:23 +00:00
#define _MULTI_THREADED
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>
void foo(void); /* Functions that use the TLS data */
void bar(void);
2022-04-10 12:25:16 +00:00
#define checkResults(string, val) { \
if (val) { \
printf("Failed with %d at %s", val, string); \
exit(1); \
} \
2021-02-28 12:19:23 +00:00
}
/*
2022-04-10 12:25:16 +00:00
Use the keyword provided by pthread.h to delcare the following variable
is thread specific, i.e. it is only visible to a specific thread,
not shared/common to all thread.
These variables are stored in thread local storage (TLS) area.
2021-02-28 12:19:23 +00:00
*/
__thread int TLS_data1 = 10;
__thread int TLS_data2 = 20;
__thread char TLS_data3[10];
2022-04-10 12:25:16 +00:00
// Sync, because it's needed apparently...
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
static pthread_mutex_t* mutex_ptr = &mutex;
static pthread_cond_t thread_state_cond = PTHREAD_COND_INITIALIZER;
static pthread_cond_t* thread_state_cond_ptr = &thread_state_cond;
static int status = 0;
2021-02-28 12:19:23 +00:00
#define NUMTHREADS 2
pthread_t thread[NUMTHREADS];
typedef struct {
2022-04-10 12:25:16 +00:00
int data1;
int data2;
2021-02-28 12:19:23 +00:00
} threadparm_t;
void *thread_run(void *parm)
{
2022-04-10 12:25:16 +00:00
int rc;
threadparm_t *gData;
2021-02-28 12:19:23 +00:00
2022-04-10 12:25:16 +00:00
pthread_mutex_lock(mutex_ptr);
if (pthread_self()==thread[0]) {
printf("Thread 1: Entered (%d/%d)\n", TLS_data1, TLS_data2);
fflush(stdout);
} else {
while (!status) pthread_cond_wait(thread_state_cond_ptr, mutex_ptr);
printf("Thread 2: Entered (%d/%d)\n", TLS_data1, TLS_data2);
fflush(stdout);
pthread_mutex_unlock(mutex_ptr);
status = 0;
pthread_cond_broadcast(thread_state_cond_ptr);
}
2021-02-28 12:19:23 +00:00
2022-04-10 12:25:16 +00:00
gData = (threadparm_t *)parm;
2021-02-28 12:19:23 +00:00
2022-04-10 12:25:16 +00:00
/* Assign the value from global variable to thread specific variable*/
TLS_data1 = gData->data1;
TLS_data2 = gData->data2;
strcpy(TLS_data3, "---");
TLS_data3[1] = (pthread_self()==thread[0])?'1':'2';
2021-02-28 12:19:23 +00:00
2022-04-10 12:25:16 +00:00
foo();
return NULL;
2021-02-28 12:19:23 +00:00
}
void foo() {
2022-04-10 12:25:16 +00:00
if (pthread_self()==thread[0]) {
printf("Thread 1: foo(), TLS data=%d %d \"%s\"\n", TLS_data1, TLS_data2, TLS_data3);
fflush(stdout);
status = 1;
pthread_cond_broadcast(thread_state_cond_ptr);
while (status) pthread_cond_wait(thread_state_cond_ptr, mutex_ptr);
pthread_mutex_unlock(mutex_ptr);
} else {
printf("Thread 2: foo(), TLS data=%d %d \"%s\"\n", TLS_data1, TLS_data2, TLS_data3);
fflush(stdout);
}
while(!thread[1])
usleep(300);
if(pthread_self()==thread[0])
pthread_join(thread[1], NULL);
bar();
2021-02-28 12:19:23 +00:00
}
void bar() {
2022-04-10 12:25:16 +00:00
printf("Thread %d: bar(), TLS data=%d %d \"%s\"\n",
(pthread_self()==thread[0])?1:2, TLS_data1, TLS_data2, TLS_data3);
fflush(stdout);
return;
2021-02-28 12:19:23 +00:00
}
int main(int argc, char **argv)
{
2022-04-10 12:25:16 +00:00
int rc=0;
int i;
threadparm_t gData[NUMTHREADS];
2021-02-28 12:19:23 +00:00
2022-04-10 12:25:16 +00:00
printf("Create/start %d threads\n", NUMTHREADS);
fflush(stdout);
for (i=0; i < NUMTHREADS; i++) {
/* Create per-thread TLS data and pass it to the thread */
gData[i].data1 = i;
gData[i].data2 = (i+1)*2;
rc = pthread_create(&thread[i], NULL, thread_run, &gData[i]);
checkResults("pthread_create()\n", rc);
}
//printf("Wait for all threads to complete, and release their resources\n");
for (i=0; i < NUMTHREADS; i++) {
rc = pthread_join(thread[i], NULL);
//checkResults("pthread_join()\n", rc);
}
2021-02-28 12:19:23 +00:00
2022-04-10 12:25:16 +00:00
thread_state_cond_ptr = NULL;
i = pthread_cond_destroy(&thread_state_cond);
checkResults("destroying cond\n", i);
mutex_ptr = NULL;
i = pthread_mutex_destroy(&mutex);
checkResults("destroying mutex\n", i);
printf("Main completed\n");
fflush(stdout);
return 0;
}