xref: /llvm-project/compiler-rt/lib/tsan/tests/rtl/tsan_posix.cpp (revision c0fa632236308100e1031fc86edfde13ddc4bcef)
15de29a4bSNico Weber //===-- tsan_posix.cpp ----------------------------------------------------===//
25de29a4bSNico Weber //
35de29a4bSNico Weber // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
45de29a4bSNico Weber // See https://llvm.org/LICENSE.txt for license information.
55de29a4bSNico Weber // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65de29a4bSNico Weber //
75de29a4bSNico Weber //===----------------------------------------------------------------------===//
85de29a4bSNico Weber //
95de29a4bSNico Weber // This file is a part of ThreadSanitizer (TSan), a race detector.
105de29a4bSNico Weber //
115de29a4bSNico Weber //===----------------------------------------------------------------------===//
125de29a4bSNico Weber #include "tsan_interface.h"
135de29a4bSNico Weber #include "tsan_posix_util.h"
145de29a4bSNico Weber #include "tsan_test_util.h"
155de29a4bSNico Weber #include "gtest/gtest.h"
165de29a4bSNico Weber #include <pthread.h>
175de29a4bSNico Weber 
185de29a4bSNico Weber struct thread_key {
195de29a4bSNico Weber   pthread_key_t key;
205de29a4bSNico Weber   pthread_mutex_t *mtx;
215de29a4bSNico Weber   int val;
225de29a4bSNico Weber   int *cnt;
thread_keythread_key235de29a4bSNico Weber   thread_key(pthread_key_t key, pthread_mutex_t *mtx, int val, int *cnt)
245de29a4bSNico Weber     : key(key)
255de29a4bSNico Weber     , mtx(mtx)
265de29a4bSNico Weber     , val(val)
275de29a4bSNico Weber     , cnt(cnt) {
285de29a4bSNico Weber   }
295de29a4bSNico Weber };
305de29a4bSNico Weber 
thread_secific_dtor(void * v)315de29a4bSNico Weber static void thread_secific_dtor(void *v) {
325de29a4bSNico Weber   thread_key *k = (thread_key *)v;
335de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_lock(k->mtx), 0);
345de29a4bSNico Weber   (*k->cnt)++;
355de29a4bSNico Weber   __tsan_write4(&k->cnt);
365de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_unlock(k->mtx), 0);
375de29a4bSNico Weber   if (k->val == 42) {
385de29a4bSNico Weber     // Okay.
395de29a4bSNico Weber   } else if (k->val == 43 || k->val == 44) {
405de29a4bSNico Weber     k->val--;
415de29a4bSNico Weber     EXPECT_EQ(pthread_setspecific(k->key, k), 0);
425de29a4bSNico Weber   } else {
435de29a4bSNico Weber     ASSERT_TRUE(false);
445de29a4bSNico Weber   }
455de29a4bSNico Weber }
465de29a4bSNico Weber 
dtors_thread(void * p)475de29a4bSNico Weber static void *dtors_thread(void *p) {
485de29a4bSNico Weber   thread_key *k = (thread_key *)p;
495de29a4bSNico Weber   EXPECT_EQ(pthread_setspecific(k->key, k), 0);
505de29a4bSNico Weber   return 0;
515de29a4bSNico Weber }
525de29a4bSNico Weber 
TEST(Posix,ThreadSpecificDtors)535de29a4bSNico Weber TEST(Posix, ThreadSpecificDtors) {
545de29a4bSNico Weber   int cnt = 0;
555de29a4bSNico Weber   pthread_key_t key;
565de29a4bSNico Weber   EXPECT_EQ(pthread_key_create(&key, thread_secific_dtor), 0);
575de29a4bSNico Weber   pthread_mutex_t mtx;
585de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_init(&mtx, 0), 0);
595de29a4bSNico Weber   pthread_t th[3];
605de29a4bSNico Weber   thread_key k1 = thread_key(key, &mtx, 42, &cnt);
615de29a4bSNico Weber   thread_key k2 = thread_key(key, &mtx, 43, &cnt);
625de29a4bSNico Weber   thread_key k3 = thread_key(key, &mtx, 44, &cnt);
635de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_create(&th[0], 0, dtors_thread, &k1), 0);
645de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_create(&th[1], 0, dtors_thread, &k2), 0);
655de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_join(th[0], 0), 0);
665de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_create(&th[2], 0, dtors_thread, &k3), 0);
675de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_join(th[1], 0), 0);
685de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_join(th[2], 0), 0);
695de29a4bSNico Weber   EXPECT_EQ(pthread_key_delete(key), 0);
705de29a4bSNico Weber   EXPECT_EQ(6, cnt);
715de29a4bSNico Weber }
725de29a4bSNico Weber 
735de29a4bSNico Weber #if !defined(__aarch64__) && !defined(__APPLE__)
745de29a4bSNico Weber static __thread int local_var;
755de29a4bSNico Weber 
local_thread(void * p)765de29a4bSNico Weber static void *local_thread(void *p) {
775de29a4bSNico Weber   __tsan_write1(&local_var);
785de29a4bSNico Weber   __tsan_write1(&p);
795de29a4bSNico Weber   if (p == 0)
805de29a4bSNico Weber     return 0;
815de29a4bSNico Weber   const int kThreads = 4;
825de29a4bSNico Weber   pthread_t th[kThreads];
835de29a4bSNico Weber   for (int i = 0; i < kThreads; i++)
845de29a4bSNico Weber     EXPECT_EQ(__interceptor_pthread_create(&th[i], 0, local_thread,
85*c0fa6322SVitaly Buka                                            (void *)((long)p - 1)),
86*c0fa6322SVitaly Buka               0);
875de29a4bSNico Weber   for (int i = 0; i < kThreads; i++)
885de29a4bSNico Weber     EXPECT_EQ(__interceptor_pthread_join(th[i], 0), 0);
895de29a4bSNico Weber   return 0;
905de29a4bSNico Weber }
915de29a4bSNico Weber #endif
925de29a4bSNico Weber 
TEST(Posix,ThreadLocalAccesses)935de29a4bSNico Weber TEST(Posix, ThreadLocalAccesses) {
945de29a4bSNico Weber // The test is failing with high thread count for aarch64.
955de29a4bSNico Weber // FIXME: track down the issue and re-enable the test.
965de29a4bSNico Weber // On Darwin, we're running unit tests without interceptors and __thread is
975de29a4bSNico Weber // using malloc and free, which causes false data race reports.  On rare
985de29a4bSNico Weber // occasions on powerpc64le this test also fails.
995de29a4bSNico Weber #if !defined(__aarch64__) && !defined(__APPLE__) && !defined(powerpc64le)
1005de29a4bSNico Weber   local_thread((void*)2);
1015de29a4bSNico Weber #endif
1025de29a4bSNico Weber }
1035de29a4bSNico Weber 
1045de29a4bSNico Weber struct CondContext {
1055de29a4bSNico Weber   pthread_mutex_t m;
1065de29a4bSNico Weber   pthread_cond_t c;
1075de29a4bSNico Weber   int data;
1085de29a4bSNico Weber };
1095de29a4bSNico Weber 
cond_thread(void * p)1105de29a4bSNico Weber static void *cond_thread(void *p) {
1115de29a4bSNico Weber   CondContext &ctx = *static_cast<CondContext*>(p);
1125de29a4bSNico Weber 
1135de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_lock(&ctx.m), 0);
1145de29a4bSNico Weber   EXPECT_EQ(ctx.data, 0);
1155de29a4bSNico Weber   ctx.data = 1;
1165de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_cond_signal(&ctx.c), 0);
1175de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_unlock(&ctx.m), 0);
1185de29a4bSNico Weber 
1195de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_lock(&ctx.m), 0);
1205de29a4bSNico Weber   while (ctx.data != 2)
1215de29a4bSNico Weber     EXPECT_EQ(__interceptor_pthread_cond_wait(&ctx.c, &ctx.m), 0);
1225de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_unlock(&ctx.m), 0);
1235de29a4bSNico Weber 
1245de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_lock(&ctx.m), 0);
1255de29a4bSNico Weber   ctx.data = 3;
1265de29a4bSNico Weber   EXPECT_EQ(pthread_cond_broadcast(&ctx.c), 0);
1275de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_unlock(&ctx.m), 0);
1285de29a4bSNico Weber 
1295de29a4bSNico Weber   return 0;
1305de29a4bSNico Weber }
1315de29a4bSNico Weber 
TEST(Posix,CondBasic)1325de29a4bSNico Weber TEST(Posix, CondBasic) {
1335de29a4bSNico Weber   CondContext ctx;
1345de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_init(&ctx.m, 0), 0);
1355de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_cond_init(&ctx.c, 0), 0);
1365de29a4bSNico Weber   ctx.data = 0;
1375de29a4bSNico Weber   pthread_t th;
1385de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_create(&th, 0, cond_thread, &ctx), 0);
1395de29a4bSNico Weber 
1405de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_lock(&ctx.m), 0);
1415de29a4bSNico Weber   while (ctx.data != 1)
1425de29a4bSNico Weber     EXPECT_EQ(__interceptor_pthread_cond_wait(&ctx.c, &ctx.m), 0);
1435de29a4bSNico Weber   ctx.data = 2;
1445de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_unlock(&ctx.m), 0);
1455de29a4bSNico Weber   EXPECT_EQ(pthread_cond_broadcast(&ctx.c), 0);
1465de29a4bSNico Weber 
1475de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_lock(&ctx.m), 0);
1485de29a4bSNico Weber   while (ctx.data != 3)
1495de29a4bSNico Weber     EXPECT_EQ(__interceptor_pthread_cond_wait(&ctx.c, &ctx.m), 0);
1505de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_unlock(&ctx.m), 0);
1515de29a4bSNico Weber 
1525de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_join(th, 0), 0);
1535de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_cond_destroy(&ctx.c), 0);
1545de29a4bSNico Weber   EXPECT_EQ(__interceptor_pthread_mutex_destroy(&ctx.m), 0);
1555de29a4bSNico Weber }
156