233 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
//===-- tsan_mop.cpp ------------------------------------------------------===//
 | 
						|
//
 | 
						|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | 
						|
// See https://llvm.org/LICENSE.txt for license information.
 | 
						|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
// This file is a part of ThreadSanitizer (TSan), a race detector.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
#include "tsan_interface.h"
 | 
						|
#include "tsan_test_util.h"
 | 
						|
#include "gtest/gtest.h"
 | 
						|
#include <stddef.h>
 | 
						|
#include <stdint.h>
 | 
						|
 | 
						|
TEST(ThreadSanitizer, SimpleWrite) {
 | 
						|
  ScopedThread t;
 | 
						|
  MemLoc l;
 | 
						|
  t.Write1(l);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, SimpleWriteWrite) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  MemLoc l1, l2;
 | 
						|
  t1.Write1(l1);
 | 
						|
  t2.Write1(l2);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, WriteWriteRace) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  MemLoc l;
 | 
						|
  t1.Write1(l);
 | 
						|
  t2.Write1(l, true);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, ReadWriteRace) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  MemLoc l;
 | 
						|
  t1.Read1(l);
 | 
						|
  t2.Write1(l, true);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, WriteReadRace) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  MemLoc l;
 | 
						|
  t1.Write1(l);
 | 
						|
  t2.Read1(l, true);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, ReadReadNoRace) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  MemLoc l;
 | 
						|
  t1.Read1(l);
 | 
						|
  t2.Read1(l);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, WriteThenRead) {
 | 
						|
  MemLoc l;
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  t1.Write1(l);
 | 
						|
  t1.Read1(l);
 | 
						|
  t2.Read1(l, true);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, WriteThenLockedRead) {
 | 
						|
  Mutex m(Mutex::RW);
 | 
						|
  MainThread t0;
 | 
						|
  t0.Create(m);
 | 
						|
  MemLoc l;
 | 
						|
  {
 | 
						|
    ScopedThread t1, t2;
 | 
						|
 | 
						|
    t1.Write8(l);
 | 
						|
 | 
						|
    t1.Lock(m);
 | 
						|
    t1.Read8(l);
 | 
						|
    t1.Unlock(m);
 | 
						|
 | 
						|
    t2.Read8(l, true);
 | 
						|
  }
 | 
						|
  t0.Destroy(m);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, LockedWriteThenRead) {
 | 
						|
  Mutex m(Mutex::RW);
 | 
						|
  MainThread t0;
 | 
						|
  t0.Create(m);
 | 
						|
  MemLoc l;
 | 
						|
  {
 | 
						|
    ScopedThread t1, t2;
 | 
						|
 | 
						|
    t1.Lock(m);
 | 
						|
    t1.Write8(l);
 | 
						|
    t1.Unlock(m);
 | 
						|
 | 
						|
    t1.Read8(l);
 | 
						|
 | 
						|
    t2.Read8(l, true);
 | 
						|
  }
 | 
						|
  t0.Destroy(m);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
TEST(ThreadSanitizer, RaceWithOffset) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access(l.loc(), true, 8, false);
 | 
						|
    t2.Access((char*)l.loc() + 4, true, 4, true);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access(l.loc(), true, 8, false);
 | 
						|
    t2.Access((char*)l.loc() + 7, true, 1, true);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc() + 4, true, 4, false);
 | 
						|
    t2.Access((char*)l.loc() + 4, true, 2, true);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc() + 4, true, 4, false);
 | 
						|
    t2.Access((char*)l.loc() + 6, true, 2, true);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc() + 3, true, 2, false);
 | 
						|
    t2.Access((char*)l.loc() + 4, true, 1, true);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc() + 1, true, 8, false);
 | 
						|
    t2.Access((char*)l.loc() + 3, true, 1, true);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, RaceWithOffset2) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc(), true, 4, false);
 | 
						|
    t2.Access((char*)l.loc() + 2, true, 1, true);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc() + 2, true, 1, false);
 | 
						|
    t2.Access((char*)l.loc(), true, 4, true);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, NoRaceWithOffset) {
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access(l.loc(), true, 4, false);
 | 
						|
    t2.Access((char*)l.loc() + 4, true, 4, false);
 | 
						|
  }
 | 
						|
  {
 | 
						|
    MemLoc l;
 | 
						|
    t1.Access((char*)l.loc() + 3, true, 2, false);
 | 
						|
    t2.Access((char*)l.loc() + 1, true, 2, false);
 | 
						|
    t2.Access((char*)l.loc() + 5, true, 2, false);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, RaceWithDeadThread) {
 | 
						|
  MemLoc l;
 | 
						|
  ScopedThread t;
 | 
						|
  ScopedThread().Write1(l);
 | 
						|
  t.Write1(l, true);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, BenignRaceOnVptr) {
 | 
						|
  void *vptr_storage;
 | 
						|
  MemLoc vptr(&vptr_storage), val;
 | 
						|
  vptr_storage = val.loc();
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  t1.VptrUpdate(vptr, val);
 | 
						|
  t2.Read8(vptr);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, HarmfulRaceOnVptr) {
 | 
						|
  void *vptr_storage;
 | 
						|
  MemLoc vptr(&vptr_storage), val1, val2;
 | 
						|
  vptr_storage = val1.loc();
 | 
						|
  ScopedThread t1, t2;
 | 
						|
  t1.VptrUpdate(vptr, val2);
 | 
						|
  t2.Read8(vptr, true);
 | 
						|
}
 | 
						|
 | 
						|
static void foo() {
 | 
						|
  volatile int x = 42;
 | 
						|
  int x2 = x;
 | 
						|
  (void)x2;
 | 
						|
}
 | 
						|
 | 
						|
static void bar() {
 | 
						|
  volatile int x = 43;
 | 
						|
  int x2 = x;
 | 
						|
  (void)x2;
 | 
						|
}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, ReportDeadThread) {
 | 
						|
  MemLoc l;
 | 
						|
  ScopedThread t1;
 | 
						|
  {
 | 
						|
    ScopedThread t2;
 | 
						|
    t2.Call(&foo);
 | 
						|
    t2.Call(&bar);
 | 
						|
    t2.Write1(l);
 | 
						|
  }
 | 
						|
  t1.Write1(l, true);
 | 
						|
}
 | 
						|
 | 
						|
struct ClassWithStatic {
 | 
						|
  static int Data[4];
 | 
						|
};
 | 
						|
 | 
						|
int ClassWithStatic::Data[4];
 | 
						|
 | 
						|
static void foobarbaz() {}
 | 
						|
 | 
						|
TEST(ThreadSanitizer, ReportRace) {
 | 
						|
  ScopedThread t1;
 | 
						|
  MainThread().Access(&ClassWithStatic::Data, true, 4, false);
 | 
						|
  t1.Call(&foobarbaz);
 | 
						|
  t1.Access(&ClassWithStatic::Data, true, 2, true);
 | 
						|
  t1.Return();
 | 
						|
}
 |