#ifndef BOOST_ATOMIC_DETAIL_GCC_ALPHA_HPP #define BOOST_ATOMIC_DETAIL_GCC_ALPHA_HPP // Copyright (c) 2009 Helge Bahmann // Copyright (c) 2013 Tim Blechmann // // Distributed under the Boost Software License, Version 1.0. // See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #include #include #include #ifdef BOOST_HAS_PRAGMA_ONCE #pragma once #endif /* Refer to http://h71000.www7.hp.com/doc/82final/5601/5601pro_004.html (HP OpenVMS systems documentation) and the alpha reference manual. */ /* NB: The most natural thing would be to write the increment/decrement operators along the following lines: __asm__ __volatile__( "1: ldl_l %0,%1 \n" "addl %0,1,%0 \n" "stl_c %0,%1 \n" "beq %0,1b\n" : "=&b" (tmp) : "m" (value) : "cc" ); However according to the comments on the HP website and matching comments in the Linux kernel sources this defies branch prediction, as the cpu assumes that backward branches are always taken; so instead copy the trick from the Linux kernel, introduce a forward branch and back again. I have, however, had a hard time measuring the difference between the two versions in microbenchmarks -- I am leaving it in nevertheless as it apparently does not hurt either. */ namespace boost { namespace atomics { namespace detail { inline void fence_before(memory_order order) { switch(order) { case memory_order_consume: case memory_order_release: case memory_order_acq_rel: case memory_order_seq_cst: __asm__ __volatile__ ("mb" ::: "memory"); default:; } } inline void fence_after(memory_order order) { switch(order) { case memory_order_acquire: case memory_order_acq_rel: case memory_order_seq_cst: __asm__ __volatile__ ("mb" ::: "memory"); default:; } } template<> inline void platform_atomic_thread_fence(memory_order order) { switch(order) { case memory_order_acquire: case memory_order_consume: case memory_order_release: case memory_order_acq_rel: case memory_order_seq_cst: __asm__ __volatile__ ("mb" ::: "memory"); default:; } } template class atomic_alpha_32 { public: typedef T integral_type; BOOST_CONSTEXPR atomic_alpha_32(T v) BOOST_NOEXCEPT: i(v) {} atomic_alpha_32() {} T load(memory_order order=memory_order_seq_cst) const volatile BOOST_NOEXCEPT { T v=*reinterpret_cast(&i); fence_after(order); return v; } void store(T v, memory_order order=memory_order_seq_cst) volatile BOOST_NOEXCEPT { fence_before(order); *reinterpret_cast(&i)=(int)v; } bool compare_exchange_weak( T &expected, T desired, memory_order success_order, memory_order failure_order) volatile BOOST_NOEXCEPT { fence_before(success_order); int current, success; __asm__ __volatile__( "1: ldl_l %2, %4\n" "cmpeq %2, %0, %3\n" "mov %2, %0\n" "beq %3, 3f\n" "stl_c %1, %4\n" "2:\n" ".subsection 2\n" "3: mov %3, %1\n" "br 2b\n" ".previous\n" : "+&r" (expected), "+&r" (desired), "=&r"(current), "=&r"(success) : "m" (i) : ); if (desired) fence_after(success_order); else fence_after(failure_order); return desired; } bool is_lock_free(void) const volatile BOOST_NOEXCEPT {return true;} protected: inline T fetch_add_var(T c, memory_order order) volatile BOOST_NOEXCEPT { fence_before(order); T original, modified; __asm__ __volatile__( "1: ldl_l %0, %2\n" "addl %0, %3, %1\n" "stl_c %1, %2\n" "beq %1, 2f\n" ".subsection 2\n" "2: br 1b\n" ".previous\n" : "=&r" (original), "=&r" (modified) : "m" (i), "r" (c) : ); fence_after(order); return original; } inline T fetch_inc(memory_order order) volatile BOOST_NOEXCEPT { fence_before(order); int original, modified; __asm__ __volatile__( "1: ldl_l %0, %2\n" "addl %0, 1, %1\n" "stl_c %1, %2\n" "beq %1, 2f\n" ".subsection 2\n" "2: br 1b\n" ".previous\n" : "=&r" (original), "=&r" (modified) : "m" (i) : ); fence_after(order); return original; } inline T fetch_dec(memory_order order) volatile BOOST_NOEXCEPT { fence_before(order); int original, modified; __asm__ __volatile__( "1: ldl_l %0, %2\n" "subl %0, 1, %1\n" "stl_c %1, %2\n" "beq %1, 2f\n" ".subsection 2\n" "2: br 1b\n" ".previous\n" : "=&r" (original), "=&r" (modified) : "m" (i) : ); fence_after(order); return original; } private: T i; }; template class atomic_alpha_64 { public: typedef T integral_type; BOOST_CONSTEXPR atomic_alpha_64(T v) BOOST_NOEXCEPT: i(v) {} atomic_alpha_64() {} T load(memory_order order=memory_order_seq_cst) const volatile { T v=*reinterpret_cast(&i); fence_after(order); return v; } void store(T v, memory_order order=memory_order_seq_cst) volatile BOOST_NOEXCEPT { fence_before(order); *reinterpret_cast(&i)=v; } bool compare_exchange_weak( T &expected, T desired, memory_order success_order, memory_order failure_order) volatile BOOST_NOEXCEPT { fence_before(success_order); int current, success; __asm__ __volatile__( "1: ldq_l %2, %4\n" "cmpeq %2, %0, %3\n" "mov %2, %0\n" "beq %3, 3f\n" "stq_c %1, %4\n" "2:\n" ".subsection 2\n" "3: mov %3, %1\n" "br 2b\n" ".previous\n" : "+&r" (expected), "+&r" (desired), "=&r"(current), "=&r"(success) : "m" (i) : ); if (desired) fence_after(success_order); else fence_after(failure_order); return desired; } bool is_lock_free(void) const volatile BOOST_NOEXCEPT {return true;} protected: inline T fetch_add_var(T c, memory_order order) volatile BOOST_NOEXCEPT { fence_before(order); T original, modified; __asm__ __volatile__( "1: ldq_l %0, %2\n" "addq %0, %3, %1\n" "stq_c %1, %2\n" "beq %1, 2f\n" ".subsection 2\n" "2: br 1b\n" ".previous\n" : "=&r" (original), "=&r" (modified) : "m" (i), "r" (c) : ); fence_after(order); return original; } inline T fetch_inc(memory_order order) volatile BOOST_NOEXCEPT { fence_before(order); T original, modified; __asm__ __volatile__( "1: ldq_l %0, %2\n" "addq %0, 1, %1\n" "stq_c %1, %2\n" "beq %1, 2f\n" ".subsection 2\n" "2: br 1b\n" ".previous\n" : "=&r" (original), "=&r" (modified) : "m" (i) : ); fence_after(order); return original; } inline T fetch_dec(memory_order order) volatile BOOST_NOEXCEPT { fence_before(order); T original, modified; __asm__ __volatile__( "1: ldq_l %0, %2\n" "subq %0, 1, %1\n" "stq_c %1, %2\n" "beq %1, 2f\n" ".subsection 2\n" "2: br 1b\n" ".previous\n" : "=&r" (original), "=&r" (modified) : "m" (i) : ); fence_after(order); return original; } private: T i; }; template class platform_atomic_integral : public build_atomic_from_typical > > { public: typedef build_atomic_from_typical > > super; BOOST_CONSTEXPR platform_atomic_integral(T v) BOOST_NOEXCEPT: super(v) {} platform_atomic_integral(void) {} }; template class platform_atomic_integral : public build_atomic_from_typical > > { public: typedef build_atomic_from_typical > > super; BOOST_CONSTEXPR platform_atomic_integral(T v) BOOST_NOEXCEPT: super(v) {} platform_atomic_integral(void) {} }; template class platform_atomic_integral : public build_atomic_from_larger_type, T> { public: typedef build_atomic_from_larger_type, T> super; BOOST_CONSTEXPR platform_atomic_integral(T v) BOOST_NOEXCEPT: super(v) {} platform_atomic_integral(void) {} }; template class platform_atomic_integral : public build_atomic_from_larger_type, T> { public: typedef build_atomic_from_larger_type, T> super; BOOST_CONSTEXPR platform_atomic_integral(T v) BOOST_NOEXCEPT: super(v) {} platform_atomic_integral(void) {} }; } } } #endif