Is std::atomic_compare_exchange_weak thread-unsafe by design?

后端 未结 5 1766
北海茫月
北海茫月 2020-12-28 17:47

It was brought up on cppreference atomic_compare_exchange Talk page that the existing implementations of std::atomic_compare_exchange_weak compute the boolea

5条回答
  •  [愿得一人]
    2020-12-28 18:35

    TL;DR: atomic_compare_exchange_weak is safe by design, but actual implementations are buggy.

    Here's the code that Clang actually generates for this little snippet:

    struct node {
      int data;
      node* next;
    };
    
    std::atomic head;
    
    void push(int data) {
      node* new_node = new node{data};
      new_node->next = head.load(std::memory_order_relaxed);
      while (!head.compare_exchange_weak(new_node->next, new_node,
          std::memory_order_release, std::memory_order_relaxed)) {}
    }
    

    Result:

      movl  %edi, %ebx
      # Allocate memory
      movl  $16, %edi
      callq _Znwm
      movq  %rax, %rcx
      # Initialize with data and 0
      movl  %ebx, (%rcx)
      movq  $0, 8(%rcx) ; dead store, should have been optimized away
      # Overwrite next with head.load
      movq  head(%rip), %rdx
      movq  %rdx, 8(%rcx)
      .align  16, 0x90
    .LBB0_1:                                # %while.cond
                                            # =>This Inner Loop Header: Depth=1
      # put value of head into comparand/result position
      movq  %rdx, %rax
      # atomic operation here, compares second argument to %rax, stores first argument
      # in second if same, and second in %rax otherwise
      lock
      cmpxchgq  %rcx, head(%rip)
      # unconditionally write old value back to next - wait, what?
      movq  %rax, 8(%rcx)
      # check if cmpxchg modified the result position
      cmpq  %rdx, %rax
      movq  %rax, %rdx
      jne .LBB0_1
    

    The comparison is perfectly safe: it's just comparing registers. However, the whole operation is not safe.

    The critical point is this: the description of compare_exchange_(weak|strong) says:

    Atomically [...] if true, replace the contents of the memory point to by this with that in desired, and if false, updates the contents of the memory in expected with the contents of the memory pointed to by this

    Or in pseudo-code:

    if (*this == expected)
      *this = desired;
    else
      expected = *this;
    

    Note that expected is only written to if the comparison is false, and *this is only written to if comparison is true. The abstract model of C++ does not allow an execution where both are written to. This is important for the correctness of push above, because if the write to head happens, suddenly new_node points to a location that is visible to other threads, which means other threads can start reading next (by accessing head->next), and if the write to expected (which aliases new_node->next) also happens, that's a race.

    And Clang writes to new_node->next unconditionally. In the case where the comparison is true, that's an invented write.

    This is a bug in Clang. I don't know whether GCC does the same thing.

    In addition, the wording of the standard is suboptimal. It claims that the entire operation must happen atomically, but this is impossible, because expected is not an atomic object; writes to there cannot happen atomically. What the standard should say is that the comparison and the write to *this happen atomically, but the write to expected does not. But this isn't that bad, because no one really expects that write to be atomic anyway.

    So there should be a bug report for Clang (and possibly GCC), and a defect report for the standard.

提交回复
热议问题