gpt4 book ai didi

c++11 - 为什么 ThreadSanitizer 会报告这个无锁示例的竞争?

转载 作者:行者123 更新时间:2023-12-03 12:17:39 25 4
gpt4 key购买 nike

我将其归结为一个简单的独立示例。主线程入队 1000 个项目,一个工作线程尝试同时出队。 ThreadSanitizer 提示其中一个元素的读取和写入之间存在竞争,即使有一个获取-释放内存屏障序列保护它们。

#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 输出:
==================
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

命令行:
g++ -std=c++11 -O0 -g -fsanitize=thread issue49.cpp -o issue49 -pthread

g++ 版本:5.3.1

谁能解释一下为什么 tsan 认为这是一场数据竞赛?

更新

这似乎是一个误报。为了安抚 ThreadSanitizer,我添加了注释(参见 here 了解支持的注释和 here 示例)。请注意,通过宏检测是否在 GCC 中启用了 tsan 有 only recently been added ,所以我不得不手动传递 -D__SANITIZE_THREAD__现在到 g++。
#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

现在 ThreadSanitizer 在运行时很开心。

最佳答案

这看起来像 https://gcc.gnu.org/bugzilla/show_bug.cgi?id=78158 .反汇编 GCC 生成的二进制文件表明它没有检测 O0 上的原子操作。
作为一种解决方法,您可以使用带有 -O1/-O2 的 GCC 构建您的代码,或者为自己获取一个新的 Clang 构建并使用它来运行 ThreadSanitizer(这是推荐的方式,因为 TSan 是作为 Clang 的一部分开发的,并且仅向后移植到 GCC)。

上面的评论是无效的:TSan 可以很容易地理解代码中原子之间的发生之前的关系(可以通过在 Clang 中的 TSan 下运行上述复制器来检查)。

我也不推荐使用 AnnotateHappensBefore()/AnnotateHappensAfter() 有两个原因:

  • 在大多数情况下你不应该需要它们;他们表示代码正在做一些非常复杂的事情(在这种情况下,您可能需要仔细检查您是否做对了);
  • 如果您在无锁代码中出错,用注释喷洒它可能会掩盖该错误,这样 TSan 就不会注意到它。
  • 关于c++11 - 为什么 ThreadSanitizer 会报告这个无锁示例的竞争?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37552866/

    25 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com