Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Why does ThreadSanitizer report a race with this lock-free example?

Tags:

I've boiled this down to a simple self-contained example. The main thread enqueues 1000 items, and a worker thread tries to dequeue concurrently. ThreadSanitizer complains that there's a race between the read and the write of one of the elements, even though there is an acquire-release memory barrier sequence protecting them.

#include <atomic> #include <thread> #include <cassert>  struct FakeQueue {     int items[1000];     std::atomic<int> m_enqueueIndex;     int m_dequeueIndex;      FakeQueue() : m_enqueueIndex(0), m_dequeueIndex(0) { }      void enqueue(int x)     {         auto tail = m_enqueueIndex.load(std::memory_order_relaxed);         items[tail] = x;              // <- element written         m_enqueueIndex.store(tail + 1, std::memory_order_release);     }      bool try_dequeue(int& x)     {         auto tail = m_enqueueIndex.load(std::memory_order_acquire);         assert(tail >= m_dequeueIndex);         if (tail == m_dequeueIndex)             return false;         x = items[m_dequeueIndex];    // <- element read -- tsan says race!         ++m_dequeueIndex;         return true;     } };   FakeQueue q;  int main() {     std::thread th([&]() {         int x;         for (int i = 0; i != 1000; ++i)             q.try_dequeue(x);     });      for (int i = 0; i != 1000; ++i)         q.enqueue(i);      th.join(); } 

ThreadSanitizer output:

================== WARNING: ThreadSanitizer: data race (pid=17220)   Read of size 4 at 0x0000006051c0 by thread T1:     #0 FakeQueue::try_dequeue(int&) /home/cameron/projects/concurrentqueue/tests/tsan/issue49.cpp:26 (issue49+0x000000402bcd)     #1 main::{lambda()#1}::operator()() const <null> (issue49+0x000000401132)     #2 _M_invoke<> /usr/include/c++/5.3.1/functional:1531 (issue49+0x0000004025e3)     #3 operator() /usr/include/c++/5.3.1/functional:1520 (issue49+0x0000004024ed)     #4 _M_run /usr/include/c++/5.3.1/thread:115 (issue49+0x00000040244d)     #5 <null> <null> (libstdc++.so.6+0x0000000b8f2f)    Previous write of size 4 at 0x0000006051c0 by main thread:     #0 FakeQueue::enqueue(int) /home/cameron/projects/concurrentqueue/tests/tsan/issue49.cpp:16 (issue49+0x000000402a90)     #1 main /home/cameron/projects/concurrentqueue/tests/tsan/issue49.cpp:44 (issue49+0x000000401187)    Location is global 'q' of size 4008 at 0x0000006051c0 (issue49+0x0000006051c0)    Thread T1 (tid=17222, running) created by main thread at:     #0 pthread_create <null> (libtsan.so.0+0x000000027a67)     #1 std::thread::_M_start_thread(std::shared_ptr<std::thread::_Impl_base>, void (*)()) <null> (libstdc++.so.6+0x0000000b9072)     #2 main /home/cameron/projects/concurrentqueue/tests/tsan/issue49.cpp:41 (issue49+0x000000401168)  SUMMARY: ThreadSanitizer: data race /home/cameron/projects/concurrentqueue/tests/tsan/issue49.cpp:26 FakeQueue::try_dequeue(int&) ================== ThreadSanitizer: reported 1 warnings 

Command line:

g++ -std=c++11 -O0 -g -fsanitize=thread issue49.cpp -o issue49 -pthread 

g++ version: 5.3.1

Can anybody shed some light onto why tsan thinks this is a data race?


UPDATE

It seems like this is a false positive. To appease ThreadSanitizer, I've added annotations (see here for the supported ones and here for an example). Note that detecting whether tsan is enabled in GCC via a macro has only recently been added, so I had to manually pass -D__SANITIZE_THREAD__ to g++ for now.

#if defined(__SANITIZE_THREAD__) #define TSAN_ENABLED #elif defined(__has_feature) #if __has_feature(thread_sanitizer) #define TSAN_ENABLED #endif #endif  #ifdef TSAN_ENABLED #define TSAN_ANNOTATE_HAPPENS_BEFORE(addr) \     AnnotateHappensBefore(__FILE__, __LINE__, (void*)(addr)) #define TSAN_ANNOTATE_HAPPENS_AFTER(addr) \     AnnotateHappensAfter(__FILE__, __LINE__, (void*)(addr)) extern "C" void AnnotateHappensBefore(const char* f, int l, void* addr); extern "C" void AnnotateHappensAfter(const char* f, int l, void* addr); #else #define TSAN_ANNOTATE_HAPPENS_BEFORE(addr) #define TSAN_ANNOTATE_HAPPENS_AFTER(addr) #endif  struct FakeQueue {     int items[1000];     std::atomic<int> m_enqueueIndex;     int m_dequeueIndex;      FakeQueue() : m_enqueueIndex(0), m_dequeueIndex(0) { }      void enqueue(int x)     {         auto tail = m_enqueueIndex.load(std::memory_order_relaxed);         items[tail] = x;         TSAN_ANNOTATE_HAPPENS_BEFORE(&items[tail]);         m_enqueueIndex.store(tail + 1, std::memory_order_release);     }      bool try_dequeue(int& x)     {         auto tail = m_enqueueIndex.load(std::memory_order_acquire);         assert(tail >= m_dequeueIndex);         if (tail == m_dequeueIndex)             return false;         TSAN_ANNOTATE_HAPPENS_AFTER(&items[m_dequeueIndex]);         x = items[m_dequeueIndex];         ++m_dequeueIndex;         return true;     } };  // main() is as before 

Now ThreadSanitizer is happy at runtime.

like image 619
Cameron Avatar asked May 31 '16 18:05

Cameron


People also ask

How does ThreadSanitizer work?

ThreadSanitizer uses a new simple hybrid algorithm which can easily be used in a pure happens-before mode. It supports the dynamic annotations we have suggested for Helgrind. Also, we have tried to make the race reports as informative as possible to make the tool easier to use.

What is data race Swift?

A data race occurs when 2 or more threads trying to access (read/write) the same memory location asynchronously at the same time. In the context of Swift, it usually happens when we try to modify an object's state using a dispatch queue.

What is Thread Sanitizer Swift?

The Thread Sanitizer, also known as TSan, is an LLVM based tool to audit threading issues in your Swift and C language written code. It was first introduced in Xcode 8 and can be a great tool to find less visible bugs in your code, like data races.


2 Answers

This looks like https://gcc.gnu.org/bugzilla/show_bug.cgi?id=78158. Disassembling the binary produced by GCC shows that it doesn't instrument the atomic operations on O0. As a workaround, you can either build your code with GCC with -O1/-O2, or get yourself a fresh Clang build and use it to run ThreadSanitizer (this is the recommended way, as TSan is being developed as part of Clang and only backported to GCC).

The comments above are invalid: TSan can easily comprehend the happens-before relation between the atomics in your code (one can check that by running the above reproducer under TSan in Clang).

I also wouldn't recommend using the AnnotateHappensBefore()/AnnotateHappensAfter() for two reasons:

  • you shouldn't need them in most cases; they denote that the code is doing something really complex (in which case you may want to double-check you're doing it right);

  • if you make an error in your lock-free code, spraying it with annotations may mask that error, so that TSan won't notice it.

like image 177
Glider Avatar answered Oct 17 '22 08:10

Glider


The ThreadSanitizer is not good at counting, it cannot understand that writes to the items always happen before the reads.

The ThreadSanitizer can find that the stores of m_enqueueIndex happen before the loads, but it does not understand that the store to items[m_dequeueIndex] must happen before the load when tail > m_dequeueIndex.

like image 38
user1887915 Avatar answered Oct 17 '22 09:10

user1887915