8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-23 08:03:04 +01:00

Improvement CORE-2619: make atomic counters 64-bit for 64-bit builds on AIX, linux and with libatomic

This commit is contained in:
alexpeshkoff 2009-09-22 09:08:57 +00:00
parent e7416d030c
commit 0396da5147

View File

@ -28,6 +28,13 @@
#ifndef CLASSES_FB_ATOMIC_H #ifndef CLASSES_FB_ATOMIC_H
#define CLASSES_FB_ATOMIC_H #define CLASSES_FB_ATOMIC_H
// Here we define an atomic type, which is to be used in interface for all OS
#if SIZEOF_VOID_P == 8
typedef SINT64 AtomicType;
#else
typedef SLONG AtomicType;
#endif
#if defined(WIN_NT) #if defined(WIN_NT)
#include <windows.h> #include <windows.h>
@ -108,6 +115,10 @@ private:
#elif defined(AIX) #elif defined(AIX)
#if SIZEOF_VOID_P != 8
#error: Only 64-bit mode supported on AIX
#endif
#include <sys/atomic_op.h> #include <sys/atomic_op.h>
namespace Firebird { namespace Firebird {
@ -115,23 +126,24 @@ namespace Firebird {
// AIX version - uses AIX atomic API // AIX version - uses AIX atomic API
class AtomicCounter class AtomicCounter
{ {
public: private:
typedef int counter_type; typedef long counter_type;
explicit AtomicCounter(counter_type value = 0) : counter(value) {} public:
explicit AtomicCounter(AtomicType value = 0) : counter(value) {}
~AtomicCounter() {} ~AtomicCounter() {}
counter_type exchangeAdd(counter_type value) counter_type exchangeAdd(AtomicType value)
{ {
return fetch_and_add(&counter, value); return fetch_and_addlp(&counter, static_cast<unsigned long>(value));
} }
counter_type operator +=(counter_type value) counter_type operator +=(AtomicType value)
{ {
return exchangeAdd(value) + value; return exchangeAdd(value) + value;
} }
counter_type operator -=(counter_type value) counter_type operator -=(AtomicType value)
{ {
return exchangeAdd(-value) - value; return exchangeAdd(-value) - value;
} }
@ -148,13 +160,13 @@ public:
counter_type value() const { return counter; } counter_type value() const { return counter; }
counter_type setValue(counter_type val) counter_type setValue(AtomicType val)
{ {
counter_type old; counter_type old;
do do
{ {
old = counter; old = counter;
} while (!compare_and_swap(&counter, &old, val)); } while (!compare_and_swaplp(&counter, &old, val));
return old; return old;
} }
@ -355,48 +367,56 @@ namespace Firebird {
class AtomicCounter class AtomicCounter
{ {
public: public:
typedef int counter_type; typedef AtomicType counter_type;
explicit AtomicCounter(counter_type value = 0) : counter(value) {} explicit AtomicCounter(AtomicType value = 0) : counter(value) {}
~AtomicCounter() {} ~AtomicCounter() {}
counter_type exchangeAdd(counter_type value) AtomicType exchangeAdd(AtomicType value)
{ {
register counter_type result; register counter_type result;
__asm __volatile ( __asm __volatile (
#if SIZEOF_VOID_P == 8
"lock; xaddq %0, %1"
#else
"lock; xaddl %0, %1" "lock; xaddl %0, %1"
#endif
: "=r" (result), "=m" (counter) : "=r" (result), "=m" (counter)
: "0" (value), "m" (counter)); : "0" (value), "m" (counter));
return result; return result;
} }
counter_type operator +=(counter_type value) AtomicType operator +=(AtomicType value)
{ {
return exchangeAdd(value) + value; return exchangeAdd(value) + value;
} }
counter_type operator -=(counter_type value) AtomicType operator -=(AtomicType value)
{ {
return exchangeAdd(-value) - value; return exchangeAdd(-value) - value;
} }
counter_type operator ++() AtomicType operator ++()
{ {
return exchangeAdd(1) + 1; return exchangeAdd(1) + 1;
} }
counter_type operator --() AtomicType operator --()
{ {
return exchangeAdd(-1) - 1; return exchangeAdd(-1) - 1;
} }
counter_type value() const { return counter; } AtomicType value() const { return counter; }
counter_type setValue(counter_type val) AtomicType setValue(AtomicType val)
{ {
register counter_type result; register counter_type result;
__asm __volatile ( __asm __volatile (
"lock; xchg %0, %1" #if SIZEOF_VOID_P == 8
"lock; xchgq %0, %1"
#else
"lock; xchgl %0, %1"
#endif
: "=r" (result), "=m" (counter) : "=r" (result), "=m" (counter)
: "0" (val), "m" (counter)); : "0" (val), "m" (counter));
return result; return result;
@ -421,7 +441,7 @@ namespace Firebird {
class AtomicCounter class AtomicCounter
{ {
public: public:
typedef AO_t counter_type; typedef AO_t counter_type; // AO_t should match maximum native in type
explicit AtomicCounter(counter_type value = 0) : counter(value) {} explicit AtomicCounter(counter_type value = 0) : counter(value) {}
~AtomicCounter() {} ~AtomicCounter() {}
@ -430,46 +450,56 @@ public:
{ {
return AO_fetch_and_add_full(&counter, value); return AO_fetch_and_add_full(&counter, value);
} */ } */
counter_type exchangeAdd(counter_type value) AtomicType exchangeAdd(AtomicType value)
{ {
#ifdef DEV_BUILD
// check that AO_t size is as we expect (can't fb_assert here)
if (sizeof(AtomicType) != sizeof(AO_t)) abort();
#endif
counter_type old; counter_type old;
do do
{ {
old = counter; old = counter;
} while (!AO_compare_and_swap_full(&counter, old, old + value)); } while (!AO_compare_and_swap_full(&counter, old, old + counter_type(value)));
return old; return AtomicType(old);
} }
counter_type operator +=(counter_type value) AtomicType operator +=(AtomicType value)
{ {
return exchangeAdd(value) + value; return exchangeAdd(value) + value;
} }
counter_type operator -=(counter_type value) AtomicType operator -=(AtomicType value)
{ {
return exchangeAdd(-value) - value; return exchangeAdd(-value) - value;
} }
counter_type operator ++() AtomicType operator ++()
{ {
return exchangeAdd(1) + 1; return exchangeAdd(1) + 1;
} }
counter_type operator --() AtomicType operator --()
{ {
return exchangeAdd(-1) - 1; return exchangeAdd(-1) - 1;
} }
counter_type value() const { return counter; } AtomicType value() const { return counter; }
counter_type setValue(counter_type val) AtomicType setValue(AtomicType val)
{ {
#ifdef DEV_BUILD
// check that AO_t size is as we expect (can't fb_assert here)
if (sizeof(AtomicType) != sizeof(AO_t)) abort();
#endif
counter_type old; counter_type old;
do do
{ {
old = counter; old = counter;
} while (!AO_compare_and_swap_full(&counter, old, val)); } while (!AO_compare_and_swap_full(&counter, old, counter_type(val)));
return old; return AtomicType(old);
} }
private: private: