llvm/compiler-rt/test/tsan/signal_in_read.c

// RUN: %clang_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s

#include "test.h"

#include <assert.h>
#include <errno.h>
#include <pthread.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

static int SignalPipeFd[] = {-1, -1};
static int BlockingPipeFd[] = {-1, -1};

static void Handler(int _) { assert(write(SignalPipeFd[1], ".", 1) == 1); }

static void *ThreadFunc(void *_) {
  char C;
  assert(read(BlockingPipeFd[0], &C, sizeof(C)) == 1);
  assert(C == '.');
  return 0;
}

int main() {
  alarm(60); // Kill the test if it hangs.

  assert(pipe(SignalPipeFd) == 0);
  assert(pipe(BlockingPipeFd) == 0);

  struct sigaction act;
  sigemptyset(&act.sa_mask);
  act.sa_flags = SA_RESTART;
  act.sa_handler = Handler;
  assert(sigaction(SIGUSR1, &act, 0) == 0);

  pthread_t Thr;
  assert(pthread_create(&Thr, 0, ThreadFunc, 0) == 0);

  // Give the thread enough time to block in the read call.
  usleep(1000000);

  // Signal the thread, this should run the signal handler and unblock the read
  // below.
  pthread_kill(Thr, SIGUSR1);
  char C;
  assert(read(SignalPipeFd[0], &C, 1) == 1);

  // Unblock the thread and join it.
  assert(write(BlockingPipeFd[1], &C, 1) == 1);
  void *_ = 0;
  assert(pthread_join(Thr, &_) == 0);

  fprintf(stderr, "PASS\n");
  return 0;
}

// CHECK-NOT: WARNING: ThreadSanitizer:
// CHECK: PASS