Blame glib/gatomic.c

Packit ae235b
/*
Packit ae235b
 * Copyright © 2011 Ryan Lortie
Packit ae235b
 *
Packit ae235b
 * This library is free software; you can redistribute it and/or
Packit ae235b
 * modify it under the terms of the GNU Lesser General Public
Packit ae235b
 * License as published by the Free Software Foundation; either
Packit ae235b
 * version 2.1 of the License, or (at your option) any later version.
Packit ae235b
 *
Packit ae235b
 * This library is distributed in the hope that it will be useful, but
Packit ae235b
 * WITHOUT ANY WARRANTY; without even the implied warranty of
Packit ae235b
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit ae235b
 * Lesser General Public License for more details.
Packit ae235b
 *
Packit ae235b
 * You should have received a copy of the GNU Lesser General Public
Packit ae235b
 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit ae235b
 *
Packit ae235b
 * Author: Ryan Lortie <desrt@desrt.ca>
Packit ae235b
 */
Packit ae235b
Packit ae235b
#include "config.h"
Packit ae235b
Packit ae235b
#include "gatomic.h"
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * SECTION:atomic_operations
Packit ae235b
 * @title: Atomic Operations
Packit ae235b
 * @short_description: basic atomic integer and pointer operations
Packit ae235b
 * @see_also: #GMutex
Packit ae235b
 *
Packit ae235b
 * The following is a collection of compiler macros to provide atomic
Packit ae235b
 * access to integer and pointer-sized values.
Packit ae235b
 *
Packit ae235b
 * The macros that have 'int' in the name will operate on pointers to
Packit ae235b
 * #gint and #guint.  The macros with 'pointer' in the name will operate
Packit ae235b
 * on pointers to any pointer-sized value, including #gsize.  There is
Packit ae235b
 * no support for 64bit operations on platforms with 32bit pointers
Packit ae235b
 * because it is not generally possible to perform these operations
Packit ae235b
 * atomically.
Packit ae235b
 *
Packit ae235b
 * The get, set and exchange operations for integers and pointers
Packit ae235b
 * nominally operate on #gint and #gpointer, respectively.  Of the
Packit ae235b
 * arithmetic operations, the 'add' operation operates on (and returns)
Packit ae235b
 * signed integer values (#gint and #gssize) and the 'and', 'or', and
Packit ae235b
 * 'xor' operations operate on (and return) unsigned integer values
Packit ae235b
 * (#guint and #gsize).
Packit ae235b
 *
Packit ae235b
 * All of the operations act as a full compiler and (where appropriate)
Packit ae235b
 * hardware memory barrier.  Acquire and release or producer and
Packit ae235b
 * consumer barrier semantics are not available through this API.
Packit ae235b
 *
Packit ae235b
 * It is very important that all accesses to a particular integer or
Packit ae235b
 * pointer be performed using only this API and that different sizes of
Packit ae235b
 * operation are not mixed or used on overlapping memory regions.  Never
Packit ae235b
 * read or assign directly from or to a value -- always use this API.
Packit ae235b
 *
Packit ae235b
 * For simple reference counting purposes you should use
Packit ae235b
 * g_atomic_int_inc() and g_atomic_int_dec_and_test().  Other uses that
Packit ae235b
 * fall outside of simple reference counting patterns are prone to
Packit ae235b
 * subtle bugs and occasionally undefined behaviour.  It is also worth
Packit ae235b
 * noting that since all of these operations require global
Packit ae235b
 * synchronisation of the entire machine, they can be quite slow.  In
Packit ae235b
 * the case of performing multiple atomic operations it can often be
Packit ae235b
 * faster to simply acquire a mutex lock around the critical area,
Packit ae235b
 * perform the operations normally and then release the lock.
Packit ae235b
 **/
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * G_ATOMIC_LOCK_FREE:
Packit ae235b
 *
Packit ae235b
 * This macro is defined if the atomic operations of GLib are
Packit ae235b
 * implemented using real hardware atomic operations.  This means that
Packit ae235b
 * the GLib atomic API can be used between processes and safely mixed
Packit ae235b
 * with other (hardware) atomic APIs.
Packit ae235b
 *
Packit ae235b
 * If this macro is not defined, the atomic operations may be
Packit ae235b
 * emulated using a mutex.  In that case, the GLib atomic operations are
Packit ae235b
 * only atomic relative to themselves and within a single process.
Packit ae235b
 **/
Packit ae235b
Packit ae235b
/* NOTE CAREFULLY:
Packit ae235b
 *
Packit ae235b
 * This file is the lowest-level part of GLib.
Packit ae235b
 *
Packit ae235b
 * Other lowlevel parts of GLib (threads, slice allocator, g_malloc,
Packit ae235b
 * messages, etc) call into these functions and macros to get work done.
Packit ae235b
 *
Packit ae235b
 * As such, these functions can not call back into any part of GLib
Packit ae235b
 * without risking recursion.
Packit ae235b
 */
Packit ae235b
Packit ae235b
#ifdef G_ATOMIC_LOCK_FREE
Packit ae235b
Packit ae235b
/* if G_ATOMIC_LOCK_FREE was defined by ./configure then we MUST
Packit ae235b
 * implement the atomic operations in a lock-free manner.
Packit ae235b
 */
Packit ae235b
Packit ae235b
#if defined (__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4)
Packit ae235b
Packit ae235b
#if defined(__ATOMIC_SEQ_CST) && !defined(__clang__)
Packit ae235b
/* The implementation used in this code path in gatomic.h assumes
Packit ae235b
 * 4-byte int */
Packit ae235b
G_STATIC_ASSERT (sizeof (gint) == 4);
Packit ae235b
Packit ae235b
/* The implementations in gatomic.h assume 4- or 8-byte pointers */
Packit ae235b
G_STATIC_ASSERT (sizeof (void *) == 4 || sizeof (void *) == 8);
Packit ae235b
#endif
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_get:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 *
Packit ae235b
 * Gets the current value of @atomic.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware
Packit ae235b
 * memory barrier (before the get).
Packit ae235b
 *
Packit ae235b
 * Returns: the value of the integer
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
gint
Packit ae235b
(g_atomic_int_get) (const volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_get (atomic);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_set:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 * @newval: a new value to store
Packit ae235b
 *
Packit ae235b
 * Sets the value of @atomic to @newval.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware
Packit ae235b
 * memory barrier (after the set).
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 */
Packit ae235b
void
Packit ae235b
(g_atomic_int_set) (volatile gint *atomic,
Packit ae235b
                    gint           newval)
Packit ae235b
{
Packit ae235b
  g_atomic_int_set (atomic, newval);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_inc:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 *
Packit ae235b
 * Increments the value of @atomic by 1.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of `{ *atomic += 1; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
void
Packit ae235b
(g_atomic_int_inc) (volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  g_atomic_int_inc (atomic);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_dec_and_test:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 *
Packit ae235b
 * Decrements the value of @atomic by 1.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ *atomic -= 1; return (*atomic == 0); }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: %TRUE if the resultant value is zero
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
gboolean
Packit ae235b
(g_atomic_int_dec_and_test) (volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_dec_and_test (atomic);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_compare_and_exchange:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 * @oldval: the value to compare with
Packit ae235b
 * @newval: the value to conditionally replace with
Packit ae235b
 *
Packit ae235b
 * Compares @atomic to @oldval and, if equal, sets it to @newval.
Packit ae235b
 * If @atomic was not equal to @oldval then no change occurs.
Packit ae235b
 *
Packit ae235b
 * This compare and exchange is done atomically.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ if (*atomic == oldval) { *atomic = newval; return TRUE; } else return FALSE; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: %TRUE if the exchange took place
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
gboolean
Packit ae235b
(g_atomic_int_compare_and_exchange) (volatile gint *atomic,
Packit ae235b
                                     gint           oldval,
Packit ae235b
                                     gint           newval)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_compare_and_exchange (atomic, oldval, newval);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_add:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 * @val: the value to add
Packit ae235b
 *
Packit ae235b
 * Atomically adds @val to the value of @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic += val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Before version 2.30, this function did not return a value
Packit ae235b
 * (but g_atomic_int_exchange_and_add() did, and had the same meaning).
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the add, signed
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
gint
Packit ae235b
(g_atomic_int_add) (volatile gint *atomic,
Packit ae235b
                    gint           val)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_add (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_and:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 * @val: the value to 'and'
Packit ae235b
 *
Packit ae235b
 * Performs an atomic bitwise 'and' of the value of @atomic and @val,
Packit ae235b
 * storing the result back in @atomic.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic &= val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the operation, unsigned
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
guint
Packit ae235b
(g_atomic_int_and) (volatile guint *atomic,
Packit ae235b
                    guint           val)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_and (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_or:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 * @val: the value to 'or'
Packit ae235b
 *
Packit ae235b
 * Performs an atomic bitwise 'or' of the value of @atomic and @val,
Packit ae235b
 * storing the result back in @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic |= val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the operation, unsigned
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
guint
Packit ae235b
(g_atomic_int_or) (volatile guint *atomic,
Packit ae235b
                   guint           val)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_or (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_xor:
Packit ae235b
 * @atomic: a pointer to a #gint or #guint
Packit ae235b
 * @val: the value to 'xor'
Packit ae235b
 *
Packit ae235b
 * Performs an atomic bitwise 'xor' of the value of @atomic and @val,
Packit ae235b
 * storing the result back in @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic ^= val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the operation, unsigned
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
guint
Packit ae235b
(g_atomic_int_xor) (volatile guint *atomic,
Packit ae235b
                    guint           val)
Packit ae235b
{
Packit ae235b
  return g_atomic_int_xor (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_get:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 *
Packit ae235b
 * Gets the current value of @atomic.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware
Packit ae235b
 * memory barrier (before the get).
Packit ae235b
 *
Packit ae235b
 * Returns: the value of the pointer
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
gpointer
Packit ae235b
(g_atomic_pointer_get) (const volatile void *atomic)
Packit ae235b
{
Packit ae235b
  return g_atomic_pointer_get ((const volatile gpointer *) atomic);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_set:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 * @newval: a new value to store
Packit ae235b
 *
Packit ae235b
 * Sets the value of @atomic to @newval.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware
Packit ae235b
 * memory barrier (after the set).
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
void
Packit ae235b
(g_atomic_pointer_set) (volatile void *atomic,
Packit ae235b
                        gpointer       newval)
Packit ae235b
{
Packit ae235b
  g_atomic_pointer_set ((volatile gpointer *) atomic, newval);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_compare_and_exchange:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 * @oldval: the value to compare with
Packit ae235b
 * @newval: the value to conditionally replace with
Packit ae235b
 *
Packit ae235b
 * Compares @atomic to @oldval and, if equal, sets it to @newval.
Packit ae235b
 * If @atomic was not equal to @oldval then no change occurs.
Packit ae235b
 *
Packit ae235b
 * This compare and exchange is done atomically.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ if (*atomic == oldval) { *atomic = newval; return TRUE; } else return FALSE; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: %TRUE if the exchange took place
Packit ae235b
 *
Packit ae235b
 * Since: 2.4
Packit ae235b
 **/
Packit ae235b
gboolean
Packit ae235b
(g_atomic_pointer_compare_and_exchange) (volatile void *atomic,
Packit ae235b
                                         gpointer       oldval,
Packit ae235b
                                         gpointer       newval)
Packit ae235b
{
Packit ae235b
  return g_atomic_pointer_compare_and_exchange ((volatile gpointer *) atomic,
Packit ae235b
                                                oldval, newval);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_add:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 * @val: the value to add
Packit ae235b
 *
Packit ae235b
 * Atomically adds @val to the value of @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic += val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the add, signed
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
gssize
Packit ae235b
(g_atomic_pointer_add) (volatile void *atomic,
Packit ae235b
                        gssize         val)
Packit ae235b
{
Packit ae235b
  return g_atomic_pointer_add ((volatile gpointer *) atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_and:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 * @val: the value to 'and'
Packit ae235b
 *
Packit ae235b
 * Performs an atomic bitwise 'and' of the value of @atomic and @val,
Packit ae235b
 * storing the result back in @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic &= val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the operation, unsigned
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_and) (volatile void *atomic,
Packit ae235b
                        gsize          val)
Packit ae235b
{
Packit ae235b
  return g_atomic_pointer_and ((volatile gpointer *) atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_or:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 * @val: the value to 'or'
Packit ae235b
 *
Packit ae235b
 * Performs an atomic bitwise 'or' of the value of @atomic and @val,
Packit ae235b
 * storing the result back in @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic |= val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the operation, unsigned
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_or) (volatile void *atomic,
Packit ae235b
                       gsize          val)
Packit ae235b
{
Packit ae235b
  return g_atomic_pointer_or ((volatile gpointer *) atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_pointer_xor:
Packit ae235b
 * @atomic: (not nullable): a pointer to a #gpointer-sized value
Packit ae235b
 * @val: the value to 'xor'
Packit ae235b
 *
Packit ae235b
 * Performs an atomic bitwise 'xor' of the value of @atomic and @val,
Packit ae235b
 * storing the result back in @atomic.
Packit ae235b
 *
Packit ae235b
 * Think of this operation as an atomic version of
Packit ae235b
 * `{ tmp = *atomic; *atomic ^= val; return tmp; }`.
Packit ae235b
 *
Packit ae235b
 * This call acts as a full compiler and hardware memory barrier.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the operation, unsigned
Packit ae235b
 *
Packit ae235b
 * Since: 2.30
Packit ae235b
 **/
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_xor) (volatile void *atomic,
Packit ae235b
                        gsize          val)
Packit ae235b
{
Packit ae235b
  return g_atomic_pointer_xor ((volatile gpointer *) atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
#elif defined (G_PLATFORM_WIN32)
Packit ae235b
Packit ae235b
#include <windows.h>
Packit ae235b
#if !defined(_M_AMD64) && !defined (_M_IA64) && !defined(_M_X64) && !(defined _MSC_VER && _MSC_VER <= 1200)
Packit ae235b
#define InterlockedAnd _InterlockedAnd
Packit ae235b
#define InterlockedOr _InterlockedOr
Packit ae235b
#define InterlockedXor _InterlockedXor
Packit ae235b
#endif
Packit ae235b
Packit ae235b
#if !defined (_MSC_VER) || _MSC_VER <= 1200
Packit ae235b
#include "gmessages.h"
Packit ae235b
/* Inlined versions for older compiler */
Packit ae235b
static LONG
Packit ae235b
_gInterlockedAnd (volatile guint *atomic,
Packit ae235b
                  guint           val)
Packit ae235b
{
Packit ae235b
  LONG i, j;
Packit ae235b
Packit ae235b
  j = *atomic;
Packit ae235b
  do {
Packit ae235b
    i = j;
Packit ae235b
    j = InterlockedCompareExchange(atomic, i & val, i);
Packit ae235b
  } while (i != j);
Packit ae235b
Packit ae235b
  return j;
Packit ae235b
}
Packit ae235b
#define InterlockedAnd(a,b) _gInterlockedAnd(a,b)
Packit ae235b
static LONG
Packit ae235b
_gInterlockedOr (volatile guint *atomic,
Packit ae235b
                 guint           val)
Packit ae235b
{
Packit ae235b
  LONG i, j;
Packit ae235b
Packit ae235b
  j = *atomic;
Packit ae235b
  do {
Packit ae235b
    i = j;
Packit ae235b
    j = InterlockedCompareExchange(atomic, i | val, i);
Packit ae235b
  } while (i != j);
Packit ae235b
Packit ae235b
  return j;
Packit ae235b
}
Packit ae235b
#define InterlockedOr(a,b) _gInterlockedOr(a,b)
Packit ae235b
static LONG
Packit ae235b
_gInterlockedXor (volatile guint *atomic,
Packit ae235b
                  guint           val)
Packit ae235b
{
Packit ae235b
  LONG i, j;
Packit ae235b
Packit ae235b
  j = *atomic;
Packit ae235b
  do {
Packit ae235b
    i = j;
Packit ae235b
    j = InterlockedCompareExchange(atomic, i ^ val, i);
Packit ae235b
  } while (i != j);
Packit ae235b
Packit ae235b
  return j;
Packit ae235b
}
Packit ae235b
#define InterlockedXor(a,b) _gInterlockedXor(a,b)
Packit ae235b
#endif
Packit ae235b
Packit ae235b
/*
Packit ae235b
 * http://msdn.microsoft.com/en-us/library/ms684122(v=vs.85).aspx
Packit ae235b
 */
Packit ae235b
gint
Packit ae235b
(g_atomic_int_get) (const volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  MemoryBarrier ();
Packit ae235b
  return *atomic;
Packit ae235b
}
Packit ae235b
Packit ae235b
void
Packit ae235b
(g_atomic_int_set) (volatile gint *atomic,
Packit ae235b
                    gint           newval)
Packit ae235b
{
Packit ae235b
  *atomic = newval;
Packit ae235b
  MemoryBarrier ();
Packit ae235b
}
Packit ae235b
Packit ae235b
void
Packit ae235b
(g_atomic_int_inc) (volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  InterlockedIncrement (atomic);
Packit ae235b
}
Packit ae235b
Packit ae235b
gboolean
Packit ae235b
(g_atomic_int_dec_and_test) (volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  return InterlockedDecrement (atomic) == 0;
Packit ae235b
}
Packit ae235b
Packit ae235b
gboolean
Packit ae235b
(g_atomic_int_compare_and_exchange) (volatile gint *atomic,
Packit ae235b
                                     gint           oldval,
Packit ae235b
                                     gint           newval)
Packit ae235b
{
Packit ae235b
  return InterlockedCompareExchange (atomic, newval, oldval) == oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
gint
Packit ae235b
(g_atomic_int_add) (volatile gint *atomic,
Packit ae235b
                    gint           val)
Packit ae235b
{
Packit ae235b
  return InterlockedExchangeAdd (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
guint
Packit ae235b
(g_atomic_int_and) (volatile guint *atomic,
Packit ae235b
                    guint           val)
Packit ae235b
{
Packit ae235b
  return InterlockedAnd (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
guint
Packit ae235b
(g_atomic_int_or) (volatile guint *atomic,
Packit ae235b
                   guint           val)
Packit ae235b
{
Packit ae235b
  return InterlockedOr (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
guint
Packit ae235b
(g_atomic_int_xor) (volatile guint *atomic,
Packit ae235b
                    guint           val)
Packit ae235b
{
Packit ae235b
  return InterlockedXor (atomic, val);
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
gpointer
Packit ae235b
(g_atomic_pointer_get) (const volatile void *atomic)
Packit ae235b
{
Packit ae235b
  const volatile gpointer *ptr = atomic;
Packit ae235b
Packit ae235b
  MemoryBarrier ();
Packit ae235b
  return *ptr;
Packit ae235b
}
Packit ae235b
Packit ae235b
void
Packit ae235b
(g_atomic_pointer_set) (volatile void *atomic,
Packit ae235b
                        gpointer       newval)
Packit ae235b
{
Packit ae235b
  volatile gpointer *ptr = atomic;
Packit ae235b
Packit ae235b
  *ptr = newval;
Packit ae235b
  MemoryBarrier ();
Packit ae235b
}
Packit ae235b
Packit ae235b
gboolean
Packit ae235b
(g_atomic_pointer_compare_and_exchange) (volatile void *atomic,
Packit ae235b
                                         gpointer       oldval,
Packit ae235b
                                         gpointer       newval)
Packit ae235b
{
Packit ae235b
  return InterlockedCompareExchangePointer (atomic, newval, oldval) == oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
gssize
Packit ae235b
(g_atomic_pointer_add) (volatile void *atomic,
Packit ae235b
                        gssize         val)
Packit ae235b
{
Packit ae235b
#if GLIB_SIZEOF_VOID_P == 8
Packit ae235b
  return InterlockedExchangeAdd64 (atomic, val);
Packit ae235b
#else
Packit ae235b
  return InterlockedExchangeAdd (atomic, val);
Packit ae235b
#endif
Packit ae235b
}
Packit ae235b
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_and) (volatile void *atomic,
Packit ae235b
                        gsize          val)
Packit ae235b
{
Packit ae235b
#if GLIB_SIZEOF_VOID_P == 8
Packit ae235b
  return InterlockedAnd64 (atomic, val);
Packit ae235b
#else
Packit ae235b
  return InterlockedAnd (atomic, val);
Packit ae235b
#endif
Packit ae235b
}
Packit ae235b
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_or) (volatile void *atomic,
Packit ae235b
                       gsize          val)
Packit ae235b
{
Packit ae235b
#if GLIB_SIZEOF_VOID_P == 8
Packit ae235b
  return InterlockedOr64 (atomic, val);
Packit ae235b
#else
Packit ae235b
  return InterlockedOr (atomic, val);
Packit ae235b
#endif
Packit ae235b
}
Packit ae235b
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_xor) (volatile void *atomic,
Packit ae235b
                        gsize          val)
Packit ae235b
{
Packit ae235b
#if GLIB_SIZEOF_VOID_P == 8
Packit ae235b
  return InterlockedXor64 (atomic, val);
Packit ae235b
#else
Packit ae235b
  return InterlockedXor (atomic, val);
Packit ae235b
#endif
Packit ae235b
}
Packit ae235b
#else
Packit ae235b
Packit ae235b
/* This error occurs when ./configure decided that we should be capable
Packit ae235b
 * of lock-free atomics but we find at compile-time that we are not.
Packit ae235b
 */
Packit ae235b
#error G_ATOMIC_LOCK_FREE defined, but incapable of lock-free atomics.
Packit ae235b
Packit ae235b
#endif /* defined (__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4) */
Packit ae235b
Packit ae235b
#else /* G_ATOMIC_LOCK_FREE */
Packit ae235b
Packit ae235b
/* We are not permitted to call into any GLib functions from here, so we
Packit ae235b
 * can not use GMutex.
Packit ae235b
 *
Packit ae235b
 * Fortunately, we already take care of the Windows case above, and all
Packit ae235b
 * non-Windows platforms on which glib runs have pthreads.  Use those.
Packit ae235b
 */
Packit ae235b
#include <pthread.h>
Packit ae235b
Packit ae235b
static pthread_mutex_t g_atomic_lock = PTHREAD_MUTEX_INITIALIZER;
Packit ae235b
Packit ae235b
gint
Packit ae235b
(g_atomic_int_get) (const volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  gint value;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  value = *atomic;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return value;
Packit ae235b
}
Packit ae235b
Packit ae235b
void
Packit ae235b
(g_atomic_int_set) (volatile gint *atomic,
Packit ae235b
                    gint           value)
Packit ae235b
{
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  *atomic = value;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
}
Packit ae235b
Packit ae235b
void
Packit ae235b
(g_atomic_int_inc) (volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  (*atomic)++;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
}
Packit ae235b
Packit ae235b
gboolean
Packit ae235b
(g_atomic_int_dec_and_test) (volatile gint *atomic)
Packit ae235b
{
Packit ae235b
  gboolean is_zero;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  is_zero = --(*atomic) == 0;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return is_zero;
Packit ae235b
}
Packit ae235b
Packit ae235b
gboolean
Packit ae235b
(g_atomic_int_compare_and_exchange) (volatile gint *atomic,
Packit ae235b
                                     gint           oldval,
Packit ae235b
                                     gint           newval)
Packit ae235b
{
Packit ae235b
  gboolean success;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  if ((success = (*atomic == oldval)))
Packit ae235b
    *atomic = newval;
Packit ae235b
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return success;
Packit ae235b
}
Packit ae235b
Packit ae235b
gint
Packit ae235b
(g_atomic_int_add) (volatile gint *atomic,
Packit ae235b
                    gint           val)
Packit ae235b
{
Packit ae235b
  gint oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *atomic;
Packit ae235b
  *atomic = oldval + val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
guint
Packit ae235b
(g_atomic_int_and) (volatile guint *atomic,
Packit ae235b
                    guint           val)
Packit ae235b
{
Packit ae235b
  guint oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *atomic;
Packit ae235b
  *atomic = oldval & val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
guint
Packit ae235b
(g_atomic_int_or) (volatile guint *atomic,
Packit ae235b
                   guint           val)
Packit ae235b
{
Packit ae235b
  guint oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *atomic;
Packit ae235b
  *atomic = oldval | val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
guint
Packit ae235b
(g_atomic_int_xor) (volatile guint *atomic,
Packit ae235b
                    guint           val)
Packit ae235b
{
Packit ae235b
  guint oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *atomic;
Packit ae235b
  *atomic = oldval ^ val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
gpointer
Packit ae235b
(g_atomic_pointer_get) (const volatile void *atomic)
Packit ae235b
{
Packit ae235b
  const volatile gpointer *ptr = atomic;
Packit ae235b
  gpointer value;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  value = *ptr;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return value;
Packit ae235b
}
Packit ae235b
Packit ae235b
void
Packit ae235b
(g_atomic_pointer_set) (volatile void *atomic,
Packit ae235b
                        gpointer       newval)
Packit ae235b
{
Packit ae235b
  volatile gpointer *ptr = atomic;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  *ptr = newval;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
}
Packit ae235b
Packit ae235b
gboolean
Packit ae235b
(g_atomic_pointer_compare_and_exchange) (volatile void *atomic,
Packit ae235b
                                         gpointer       oldval,
Packit ae235b
                                         gpointer       newval)
Packit ae235b
{
Packit ae235b
  volatile gpointer *ptr = atomic;
Packit ae235b
  gboolean success;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  if ((success = (*ptr == oldval)))
Packit ae235b
    *ptr = newval;
Packit ae235b
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return success;
Packit ae235b
}
Packit ae235b
Packit ae235b
gssize
Packit ae235b
(g_atomic_pointer_add) (volatile void *atomic,
Packit ae235b
                        gssize         val)
Packit ae235b
{
Packit ae235b
  volatile gssize *ptr = atomic;
Packit ae235b
  gssize oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *ptr;
Packit ae235b
  *ptr = oldval + val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_and) (volatile void *atomic,
Packit ae235b
                        gsize          val)
Packit ae235b
{
Packit ae235b
  volatile gsize *ptr = atomic;
Packit ae235b
  gsize oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *ptr;
Packit ae235b
  *ptr = oldval & val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_or) (volatile void *atomic,
Packit ae235b
                       gsize          val)
Packit ae235b
{
Packit ae235b
  volatile gsize *ptr = atomic;
Packit ae235b
  gsize oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *ptr;
Packit ae235b
  *ptr = oldval | val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
gsize
Packit ae235b
(g_atomic_pointer_xor) (volatile void *atomic,
Packit ae235b
                        gsize          val)
Packit ae235b
{
Packit ae235b
  volatile gsize *ptr = atomic;
Packit ae235b
  gsize oldval;
Packit ae235b
Packit ae235b
  pthread_mutex_lock (&g_atomic_lock);
Packit ae235b
  oldval = *ptr;
Packit ae235b
  *ptr = oldval ^ val;
Packit ae235b
  pthread_mutex_unlock (&g_atomic_lock);
Packit ae235b
Packit ae235b
  return oldval;
Packit ae235b
}
Packit ae235b
Packit ae235b
#endif
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_atomic_int_exchange_and_add:
Packit ae235b
 * @atomic: a pointer to a #gint
Packit ae235b
 * @val: the value to add
Packit ae235b
 *
Packit ae235b
 * This function existed before g_atomic_int_add() returned the prior
Packit ae235b
 * value of the integer (which it now does).  It is retained only for
Packit ae235b
 * compatibility reasons.  Don't use this function in new code.
Packit ae235b
 *
Packit ae235b
 * Returns: the value of @atomic before the add, signed
Packit ae235b
 * Since: 2.4
Packit ae235b
 * Deprecated: 2.30: Use g_atomic_int_add() instead.
Packit ae235b
 **/
Packit ae235b
gint
Packit ae235b
g_atomic_int_exchange_and_add (volatile gint *atomic,
Packit ae235b
                               gint           val)
Packit ae235b
{
Packit ae235b
  return (g_atomic_int_add) (atomic, val);
Packit ae235b
}