summaryrefslogtreecommitdiffstats
path: root/libcxx
diff options
context:
space:
mode:
authorDan Albert <danalbert@google.com>2015-01-06 18:39:37 +0000
committerDan Albert <danalbert@google.com>2015-01-06 18:39:37 +0000
commit872bad5ab72ec9e5b5bab1bff98285b1c36bbe58 (patch)
treed1ee51fca9ae2eb19e28f17851791e5c421e7bdf /libcxx
parentd9c605ddae86b6e4b2138d7eec7b301698e3dfcb (diff)
downloadbcm5719-llvm-872bad5ab72ec9e5b5bab1bff98285b1c36bbe58.tar.gz
bcm5719-llvm-872bad5ab72ec9e5b5bab1bff98285b1c36bbe58.zip
Obey [atomics.types.operations.req]/21 for GCC.
Summary: Excerpt from [atomics.types.operations.req]/21: > When only one memory_order argument is supplied, the value of > success is order, and the value of failure is order except that a > value of memory_order_acq_rel shall be replaced by the value > memory_order_acquire and a value of memory_order_release shall be > replaced by the value memory_order_relaxed. Clean up some copy pasta while I'm here (someone added a return statement to a void function). Reviewers: EricWF, jroelofs, mclow.lists Reviewed By: mclow.lists Subscribers: cfe-commits Differential Revision: http://reviews.llvm.org/D6632 llvm-svn: 225280
Diffstat (limited to 'libcxx')
-rw-r--r--libcxx/include/atomic22
-rw-r--r--libcxx/test/atomics/atomics.general/replace_failure_order.pass.cpp41
2 files changed, 57 insertions, 6 deletions
diff --git a/libcxx/include/atomic b/libcxx/include/atomic
index b01a59f5f96..7a6dd24fefc 100644
--- a/libcxx/include/atomic
+++ b/libcxx/include/atomic
@@ -583,6 +583,16 @@ static inline constexpr int __to_gcc_order(memory_order __order) {
__ATOMIC_CONSUME))));
}
+static inline constexpr int __to_gcc_failure_order(memory_order __order) {
+ // Avoid switch statement to make this a constexpr.
+ return __order == memory_order_relaxed ? __ATOMIC_RELAXED:
+ (__order == memory_order_acquire ? __ATOMIC_ACQUIRE:
+ (__order == memory_order_release ? __ATOMIC_RELAXED:
+ (__order == memory_order_seq_cst ? __ATOMIC_SEQ_CST:
+ (__order == memory_order_acq_rel ? __ATOMIC_ACQUIRE:
+ __ATOMIC_CONSUME))));
+}
+
} // namespace __gcc_atomic
template <typename _Tp>
@@ -637,8 +647,8 @@ static inline void __c11_atomic_store(volatile _Atomic(_Tp)* __a, _Tp __val,
template <typename _Tp>
static inline void __c11_atomic_store(_Atomic(_Tp)* __a, _Tp __val,
memory_order __order) {
- return __atomic_store(&__a->__a_value, &__val,
- __gcc_atomic::__to_gcc_order(__order));
+ __atomic_store(&__a->__a_value, &__val,
+ __gcc_atomic::__to_gcc_order(__order));
}
template <typename _Tp>
@@ -683,7 +693,7 @@ static inline bool __c11_atomic_compare_exchange_strong(
return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
false,
__gcc_atomic::__to_gcc_order(__success),
- __gcc_atomic::__to_gcc_order(__failure));
+ __gcc_atomic::__to_gcc_failure_order(__failure));
}
template <typename _Tp>
@@ -693,7 +703,7 @@ static inline bool __c11_atomic_compare_exchange_strong(
return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
false,
__gcc_atomic::__to_gcc_order(__success),
- __gcc_atomic::__to_gcc_order(__failure));
+ __gcc_atomic::__to_gcc_failure_order(__failure));
}
template <typename _Tp>
@@ -703,7 +713,7 @@ static inline bool __c11_atomic_compare_exchange_weak(
return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
true,
__gcc_atomic::__to_gcc_order(__success),
- __gcc_atomic::__to_gcc_order(__failure));
+ __gcc_atomic::__to_gcc_failure_order(__failure));
}
template <typename _Tp>
@@ -713,7 +723,7 @@ static inline bool __c11_atomic_compare_exchange_weak(
return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
true,
__gcc_atomic::__to_gcc_order(__success),
- __gcc_atomic::__to_gcc_order(__failure));
+ __gcc_atomic::__to_gcc_failure_order(__failure));
}
template <typename _Tp>
diff --git a/libcxx/test/atomics/atomics.general/replace_failure_order.pass.cpp b/libcxx/test/atomics/atomics.general/replace_failure_order.pass.cpp
new file mode 100644
index 00000000000..0b37c9db768
--- /dev/null
+++ b/libcxx/test/atomics/atomics.general/replace_failure_order.pass.cpp
@@ -0,0 +1,41 @@
+//===----------------------------------------------------------------------===//
+//
+// The LLVM Compiler Infrastructure
+//
+// This file is dual licensed under the MIT and the University of Illinois Open
+// Source Licenses. See LICENSE.TXT for details.
+//
+//===----------------------------------------------------------------------===//
+
+// This test verifies behavior specified by [atomics.types.operations.req]/21:
+//
+// When only one memory_order argument is supplied, the value of success is
+// order, and the value of failure is order except that a value of
+// memory_order_acq_rel shall be replaced by the value memory_order_acquire
+// and a value of memory_order_release shall be replaced by the value
+// memory_order_relaxed.
+//
+// Clang's atomic intrinsics do this for us, but GCC's do not. We don't actually
+// have visibility to see what these memory orders are lowered to, but we can at
+// least check that they are lowered at all (otherwise there is a compile
+// failure with GCC).
+
+#include <atomic>
+
+int main() {
+ std::atomic<int> i;
+ volatile std::atomic<int> v;
+ int exp;
+
+ i.compare_exchange_weak(exp, 0, std::memory_order_acq_rel);
+ i.compare_exchange_weak(exp, 0, std::memory_order_release);
+ i.compare_exchange_strong(exp, 0, std::memory_order_acq_rel);
+ i.compare_exchange_strong(exp, 0, std::memory_order_release);
+
+ v.compare_exchange_weak(exp, 0, std::memory_order_acq_rel);
+ v.compare_exchange_weak(exp, 0, std::memory_order_release);
+ v.compare_exchange_strong(exp, 0, std::memory_order_acq_rel);
+ v.compare_exchange_strong(exp, 0, std::memory_order_release);
+
+ return 0;
+}
OpenPOWER on IntegriCloud