c++ - 通过 shared_ptr : ThreadSanitzier false positive? 同步

标签 c++ multithreading shared-ptr thread-sanitizer

以下代码通过 shared_ptr 进行同步:

#include <memory>
#include <thread>
#include <future>
#include <chrono>
#include <cassert>
#include <atomic>

using std::shared_ptr;
using std::async;
using std::launch;
using std::this_thread::sleep_for;
using namespace std::literals;

void f1(shared_ptr<int> p)
{
    sleep_for(50ms); // make sure the other has started
    assert(*p == 42);
    p.reset();
    sleep_for(50ms); // make sure the other has deleted *p
}

void f2(shared_ptr<int> p)
{
    while (p.use_count() != 1)
    {
        sleep_for(1ms);
    }
    p.reset();
    sleep_for(50ms);
}

int main()
{
    shared_ptr<int> p(new int(42));
    auto t1 = async(launch::async, f1, p);
    auto t2 = async(launch::async, f2, p);

    p.reset();

    t1.get();
    t2.get();

    return 0;
}

我编译这个

clang++-4.0 -std=c++1z -stdlib=libc++ -Wall -g -O3 -march=native -fsanitize=thread -fno-omit-frame-pointer -pthread sharedPtr.cc -o sharedPtr

运行它时,ThreadSanitizer 给我以下问题:

==================
WARNING: ThreadSanitizer: data race (pid=273)
  Write of size 8 at 0x7b0400000000 by thread T2:
    #0 operator delete(void*) ??:? (sharedPtr+0x4b4af1)
    #1 std::__1::default_delete<int>::operator()(int*) const /usr/include/c++/v1/memory:2516 (discriminator 1) (sharedPtr+0x4b74d8)
    #2 std::__1::__shared_ptr_pointer<int*, std::__1::default_delete<int>, std::__1::allocator<int> >::__on_zero_shared() /usr/include/c++/v1/memory:3759 (discriminator 1) (sharedPtr+0x4b74d8)
[...]

  Previous read of size 4 at 0x7b0400000000 by thread T1:
    #0 f1(std::__1::shared_ptr<int>) /home/dv/src/git/c++-concurrency/test/sharedPtr.cc:22 (sharedPtr+0x4b6fca)
    #1 _ZNSt3__18__invokeIPFvNS_10shared_ptrIiEEEJS2_EEEDTclclsr3std3__1E7forwardIT_Efp_Espclsr3std3__1E7forwardIT0_Efp0_EEEOS5_DpOS6_ /usr/include/c++/v1/__functional_base:415 (sharedPtr+0x4b78cf)
[...]

我假设 C++ 通过 shared_ptr 引用计数保证足够的同步,通过 shared_ptr 读取永远不会与删除器竞争(对于不同的 shared_ptr 对象)。我希望这是很常见的用法,所以令我惊讶的是 ThreadSanitizer 对此有提示。

所以这是我的问题:

  1. 我的使用是否安全(以及我对 shared_ptr 同步的假设是否正确)? (我希望这个问题的答案是肯定的,所以现在请回答我真正的问题:)
  2. libc++ 是否正确实现了同步?
  3. 难道ThreadSanitizer真的看不到通过引用计数的同步吗?

最佳答案

Is my use safe (and my assumptions about shared_ptr synchronization correct)?

我在标准中没有看到任何要求use_count() 的实现中使用内存栅栏。 cppreference 表示大多数实现使用 memory_order_relaxed 进行读取,因此不能保证顺序。

引用:“在多线程环境中,use_count 返回的值是近似值(典型实现使用 memory_order_relaxed 加载)”

因此,严格来说,我认为将 use_count() 用作信号量是不安全的,因为它依赖于对实现的假设。

Does libc++ implement the synchronization correctly?

Does ThreadSanitizer really doesn't see the synchronization through the reference count?

我认为 ThreadSanitizer 正确地召唤了你。

关于c++ - 通过 shared_ptr : ThreadSanitzier false positive? 同步,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42950471/

相关文章:

c++ 要包含哪些文件和库?

c++ - Gtkmm : How to update UI from another thread? 连续

c++ - 将 shared_ptr 转换为 void*

c++ - 分配取消引用的 shared_ptr

java多线程jms队列接收器

C++ reinterpret_cast 对 std::shared_ptr 引用进行优化

c++ - 如何使用具有相互依赖性的模板类?

c++ - 为什么私有(private)构造函数阻止对象创建 C++

c++ - 如何访问 Opencv 中的 RGB 值?

java - 如果我从不使用 ThreadSafeClientConnManager 调用 closeExpiredConnections 会发生什么