// // 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 // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14, c++17 // XFAIL: !has-64-bit-atomics // integral-type operator+=(integral-type) const noexcept; // floating-point-type operator+=(floating-point-type) const noexcept; // T* operator+=(difference_type) const noexcept; #include #include #include #include #include #include "atomic_helpers.h" #include "test_helper.h" #include "test_macros.h" template concept has_operator_plus_equals = requires { std::declval() += std::declval(); }; template struct TestDoesNotHaveOperatorPlusEquals { void operator()() const { static_assert(!has_operator_plus_equals>); } }; template struct TestOperatorPlusEquals { void operator()() const { if constexpr (std::is_arithmetic_v) { T x(T(1)); std::atomic_ref const a(x); std::same_as decltype(auto) y = (a += T(2)); assert(y == T(3)); assert(x == T(3)); ASSERT_NOEXCEPT(a += T(0)); } else if constexpr (std::is_pointer_v) { using U = std::remove_pointer_t; U t[9] = {}; T p{&t[1]}; std::atomic_ref const a(p); std::same_as decltype(auto) y = (a += 2); assert(y == &t[3]); assert(a == &t[3]); ASSERT_NOEXCEPT(a += 0); } else { static_assert(std::is_void_v); } // memory_order::seq_cst { auto plus_equals = [](std::atomic_ref const& x, T old_val, T new_val) { x += (new_val - old_val); }; auto load = [](std::atomic_ref const& x) { return x.load(); }; test_seq_cst(plus_equals, load); } } }; int main(int, char**) { TestEachIntegralType()(); TestOperatorPlusEquals()(); TestOperatorPlusEquals()(); TestEachPointerType()(); TestDoesNotHaveOperatorPlusEquals()(); TestDoesNotHaveOperatorPlusEquals()(); TestDoesNotHaveOperatorPlusEquals()(); return 0; }