ftmemsim-valgrind/drd/tests/atomic_var.c
Bart Van Assche d9e3e0f863 Renamed __sync_add_and_fetch() into sync_add_and_fetch().
git-svn-id: svn://svn.valgrind.org/valgrind/trunk@8630
2008-09-21 11:21:23 +00:00

81 lines
1.8 KiB
C

/** Race condition around use of atomic variable.
* Note: for the i386 and x86_64 memory models, thread 2 must print y = 1.
* On PPC however, both y = 0 and y = 1 are legal results. This is because
* the PPC memory model allows different CPU's to observe stores to variables
* in different cache lines in a different order.
*/
#define _GNU_SOURCE
#include "config.h"
#include <pthread.h>
#include <stdio.h> /* fprintf() */
#include <stdlib.h> /* atoi() */
/** Only gcc 4.1.0 and later have atomic builtins. */
#if defined(HAVE_BUILTIN_ATOMIC)
static __inline__
int sync_add_and_fetch(int* p, int i)
{
return __sync_add_and_fetch(p, i);
}
#else
static __inline__
int sync_add_and_fetch(int* p, int i)
{
if (i == 0)
return *p;
return (*p += i);
}
#endif
#ifdef HAVE_BUILTIN_ATOMIC
static int s_x = 0;
/* s_dummy[] ensures that s_x and s_y are not in the same cache line. */
static char s_dummy[512];
static int s_y = 0;
static void* thread_func_1(void* arg)
{
s_y = 1;
(void) sync_add_and_fetch(&s_x, 1);
return 0;
}
static void* thread_func_2(void* arg)
{
while (sync_add_and_fetch(&s_x, 0) == 0)
;
fprintf(stderr, "y = %d\n", s_y);
return 0;
}
#endif
int main(int argc, char** argv)
{
#ifdef HAVE_BUILTIN_ATOMIC
int i;
const int n_threads = 2;
pthread_t tid[n_threads];
fprintf(stderr, "Start of test.\n");
pthread_create(&tid[0], 0, thread_func_1, 0);
pthread_create(&tid[1], 0, thread_func_2, 0);
for (i = 0; i < n_threads; i++)
pthread_join(tid[i], 0);
fprintf(stderr, "Test finished.\n");
/* Suppress the compiler warning about s_dummy not being used. */
s_dummy[0]++;
#else
fprintf(stderr,
"Sorry, but your compiler does not have built-in support for atomic"
" operations.\n");
#endif
return 0;
}