0
0
mirror of https://github.com/zeromq/libzmq.git synced 2025-01-22 07:29:31 +08:00
libzmq/src/atomic_counter.hpp

223 lines
7.3 KiB
C++
Raw Normal View History

2009-07-29 12:07:54 +02:00
/*
Copyright (c) 2007-2015 Contributors as noted in the AUTHORS file
2009-07-29 12:07:54 +02:00
This file is part of libzmq, the ZeroMQ core engine in C++.
2009-07-29 12:07:54 +02:00
libzmq is free software; you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License (LGPL) as published
by the Free Software Foundation; either version 3 of the License, or
2009-07-29 12:07:54 +02:00
(at your option) any later version.
As a special exception, the Contributors give you permission to link
this library with independent modules to produce an executable,
regardless of the license terms of these independent modules, and to
copy and distribute the resulting executable under terms of your choice,
provided that you also meet, for each linked independent module, the
terms and conditions of the license of that module. An independent
module is a module which is not derived from or based on this library.
If you modify this library, you must extend this exception to your
version of the library.
libzmq is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
License for more details.
2009-07-29 12:07:54 +02:00
You should have received a copy of the GNU Lesser General Public License
2009-07-29 12:07:54 +02:00
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
2009-08-03 11:30:13 +02:00
#ifndef __ZMQ_ATOMIC_COUNTER_HPP_INCLUDED__
#define __ZMQ_ATOMIC_COUNTER_HPP_INCLUDED__
2009-07-29 12:07:54 +02:00
#include "stdint.hpp"
#include "platform.hpp"
2009-08-03 11:30:13 +02:00
#if defined ZMQ_FORCE_MUTEXES
#define ZMQ_ATOMIC_COUNTER_MUTEX
#elif defined ZMQ_HAVE_ATOMIC_INTRINSICS
#define ZMQ_ATOMIC_INTRINSIC
2009-07-29 12:07:54 +02:00
#elif (defined __i386__ || defined __x86_64__) && defined __GNUC__
2009-08-03 11:30:13 +02:00
#define ZMQ_ATOMIC_COUNTER_X86
#elif defined __ARM_ARCH_7A__ && defined __GNUC__
#define ZMQ_ATOMIC_COUNTER_ARM
2009-08-03 11:30:13 +02:00
#elif defined ZMQ_HAVE_WINDOWS
#define ZMQ_ATOMIC_COUNTER_WINDOWS
#elif (defined ZMQ_HAVE_SOLARIS || defined ZMQ_HAVE_NETBSD)
#define ZMQ_ATOMIC_COUNTER_ATOMIC_H
#elif defined __tile__
#define ZMQ_ATOMIC_COUNTER_TILE
2009-07-29 12:07:54 +02:00
#else
2009-08-03 11:30:13 +02:00
#define ZMQ_ATOMIC_COUNTER_MUTEX
2009-07-29 12:07:54 +02:00
#endif
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_MUTEX
2009-07-29 12:07:54 +02:00
#include "mutex.hpp"
2009-08-03 11:30:13 +02:00
#elif defined ZMQ_ATOMIC_COUNTER_WINDOWS
2009-07-29 12:07:54 +02:00
#include "windows.hpp"
#elif defined ZMQ_ATOMIC_COUNTER_ATOMIC_H
2009-07-29 12:07:54 +02:00
#include <atomic.h>
#elif defined ZMQ_ATOMIC_COUNTER_TILE
#include <arch/atomic.h>
2009-07-29 12:07:54 +02:00
#endif
2009-08-03 11:30:13 +02:00
namespace zmq
2009-07-29 12:07:54 +02:00
{
// This class represents an integer that can be incremented/decremented
// in atomic fashion.
class atomic_counter_t
{
public:
typedef uint32_t integer_t;
inline atomic_counter_t (integer_t value_ = 0) :
value (value_)
{
}
inline ~atomic_counter_t ()
{
}
// Set counter value (not thread-safe).
inline void set (integer_t value_)
{
value = value_;
}
// Atomic addition. Returns the old value.
inline integer_t add (integer_t increment_)
{
integer_t old_value;
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_WINDOWS
2009-07-29 12:07:54 +02:00
old_value = InterlockedExchangeAdd ((LONG*) &value, increment_);
#elif defined ZMQ_ATOMIC_INTRINSIC
old_value = __atomic_fetch_add(&value, increment_, __ATOMIC_ACQ_REL);
#elif defined ZMQ_ATOMIC_COUNTER_ATOMIC_H
2009-07-29 12:07:54 +02:00
integer_t new_value = atomic_add_32_nv (&value, increment_);
old_value = new_value - increment_;
#elif defined ZMQ_ATOMIC_COUNTER_TILE
old_value = arch_atomic_add (&value, increment_);
2009-08-03 11:30:13 +02:00
#elif defined ZMQ_ATOMIC_COUNTER_X86
2009-07-29 12:07:54 +02:00
__asm__ volatile (
"lock; xadd %0, %1 \n\t"
2009-07-29 12:07:54 +02:00
: "=r" (old_value), "=m" (value)
: "0" (increment_), "m" (value)
: "cc", "memory");
#elif defined ZMQ_ATOMIC_COUNTER_ARM
integer_t flag, tmp;
__asm__ volatile (
" dmb sy\n\t"
"1: ldrex %0, [%5]\n\t"
" add %2, %0, %4\n\t"
" strex %1, %2, [%5]\n\t"
" teq %1, #0\n\t"
" bne 1b\n\t"
" dmb sy\n\t"
: "=&r"(old_value), "=&r"(flag), "=&r"(tmp), "+Qo"(value)
: "Ir"(increment_), "r"(&value)
: "cc");
2009-08-03 11:30:13 +02:00
#elif defined ZMQ_ATOMIC_COUNTER_MUTEX
2009-07-29 12:07:54 +02:00
sync.lock ();
old_value = value;
value += increment_;
sync.unlock ();
#else
#error atomic_counter is not implemented for this platform
2009-07-29 12:07:54 +02:00
#endif
return old_value;
}
// Atomic subtraction. Returns false if the counter drops to zero.
inline bool sub (integer_t decrement)
{
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_WINDOWS
2009-07-29 12:07:54 +02:00
LONG delta = - ((LONG) decrement);
integer_t old = InterlockedExchangeAdd ((LONG*) &value, delta);
return old - decrement != 0;
#elif defined ZMQ_ATOMIC_INTRINSIC
integer_t nv = __atomic_sub_fetch(&value, decrement, __ATOMIC_ACQ_REL);
return nv != 0;
#elif defined ZMQ_ATOMIC_COUNTER_ATOMIC_H
2009-07-29 12:07:54 +02:00
int32_t delta = - ((int32_t) decrement);
integer_t nv = atomic_add_32_nv (&value, delta);
return nv != 0;
#elif defined ZMQ_ATOMIC_COUNTER_TILE
int32_t delta = - ((int32_t) decrement);
integer_t nv = arch_atomic_add (&value, delta);
return nv != 0;
2009-08-03 11:30:13 +02:00
#elif defined ZMQ_ATOMIC_COUNTER_X86
2009-07-29 12:07:54 +02:00
integer_t oldval = -decrement;
volatile integer_t *val = &value;
__asm__ volatile ("lock; xaddl %0,%1"
: "=r" (oldval), "=m" (*val)
: "0" (oldval), "m" (*val)
: "cc", "memory");
2009-07-29 12:07:54 +02:00
return oldval != decrement;
#elif defined ZMQ_ATOMIC_COUNTER_ARM
integer_t old_value, flag, tmp;
__asm__ volatile (
" dmb sy\n\t"
"1: ldrex %0, [%5]\n\t"
" sub %2, %0, %4\n\t"
" strex %1, %2, [%5]\n\t"
" teq %1, #0\n\t"
" bne 1b\n\t"
" dmb sy\n\t"
: "=&r"(old_value), "=&r"(flag), "=&r"(tmp), "+Qo"(value)
: "Ir"(decrement), "r"(&value)
: "cc");
return old_value - decrement != 0;
2009-08-03 11:30:13 +02:00
#elif defined ZMQ_ATOMIC_COUNTER_MUTEX
2009-07-29 12:07:54 +02:00
sync.lock ();
value -= decrement;
bool result = value ? true : false;
sync.unlock ();
return result;
#else
#error atomic_counter is not implemented for this platform
2009-07-29 12:07:54 +02:00
#endif
}
inline integer_t get () const
2009-07-29 12:07:54 +02:00
{
return value;
}
private:
volatile integer_t value;
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_MUTEX
2009-07-29 12:07:54 +02:00
mutex_t sync;
#endif
atomic_counter_t (const atomic_counter_t&);
const atomic_counter_t& operator = (const atomic_counter_t&);
2009-07-29 12:07:54 +02:00
};
}
// Remove macros local to this file.
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_WINDOWS
#undef ZMQ_ATOMIC_COUNTER_WINDOWS
2009-07-29 12:07:54 +02:00
#endif
#if defined ZMQ_ATOMIC_COUNTER_ATOMIC_H
#undef ZMQ_ATOMIC_COUNTER_ATOMIC_H
2009-07-29 12:07:54 +02:00
#endif
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_X86
#undef ZMQ_ATOMIC_COUNTER_X86
2009-07-29 12:07:54 +02:00
#endif
#if defined ZMQ_ATOMIC_COUNTER_ARM
#undef ZMQ_ATOMIC_COUNTER_ARM
#endif
2009-08-03 11:30:13 +02:00
#if defined ZMQ_ATOMIC_COUNTER_MUTEX
#undef ZMQ_ATOMIC_COUNTER_MUTEX
2009-07-29 12:07:54 +02:00
#endif
#endif