gc-alloc 0.1.1

A Rust library for garbage-collected memory allocation using libgc (bdwgc).
Documentation
/*
 * THIS MATERIAL IS PROVIDED AS IS, WITH ABSOLUTELY NO WARRANTY EXPRESSED
 * OR IMPLIED.  ANY USE IS AT YOUR OWN RISK.
 *
 * Permission is hereby granted to use or copy this program
 * for any purpose, provided the above notices are retained on all copies.
 * Permission to modify the code and to distribute modified code is granted,
 * provided the above notices are retained, and a notice that the code was
 * modified is included with the above copyright notice.
 */

#if !defined(AO_ATOMIC_OPS_H) || defined(AO_ATOMIC_OPS_INCLUDED)
# error This file should not be included directly.
#endif

#if defined(__clang__) || AO_GNUC_PREREQ(14, 0) \
    || defined(AO_PREFER_BUILTIN_ATOMICS)
  /* The operations are lock-free even for the types smaller than word. */
# if !(AO_GNUC_PREREQ(13, 0) || AO_CLANG_PREREQ(16, 0))
    /* __GCC_HAVE_SYNC_COMPARE_AND_SWAP_n macros were missing.  */
#   define AO_GCC_FORCE_HAVE_CAS
# endif
#else
  /* For older versions of gcc (e.g. gcc-7.5), CAS and arithmetic       */
  /* atomic operations for char and short are supported by the compiler */
  /* but require -latomic flag.                                         */
# if !defined(__GCC_HAVE_SYNC_COMPARE_AND_SWAP_1)
#   define AO_NO_char_ARITHM
# endif
# if !defined(__GCC_HAVE_SYNC_COMPARE_AND_SWAP_2)
#   define AO_NO_short_ARITHM
# endif
#endif

#if defined(__riscv_zacas) && !defined(AO_NO_DOUBLE_CAS) \
    && __SIZEOF_SIZE_T__ == 8
  /* TODO: Support also rv32 and rv64ilp32 (i.e. use amocas.d). */

# define AO_SKIPATOMIC_double_load
# define AO_SKIPATOMIC_double_load_acquire
# define AO_SKIPATOMIC_double_store
# define AO_SKIPATOMIC_double_store_release

# include "../standard_ao_double_t.h"

  AO_INLINE int
  AO_double_compare_and_swap(volatile AO_double_t *addr,
                             AO_double_t old_val, AO_double_t new_val)
  {
    AO_double_t expected = old_val;

    __asm__ __volatile__("amocas.q %0, %z2, %1"
                         : "+rJ" (expected.AO_whole), "+A" (*addr)
                         : "rJ" (new_val.AO_whole));
    return expected.AO_whole == old_val.AO_whole;
  }
# define AO_HAVE_double_compare_and_swap

  AO_INLINE int
  AO_double_compare_and_swap_acquire(volatile AO_double_t *addr,
                                     AO_double_t old_val, AO_double_t new_val)
  {
    AO_double_t expected = old_val;

    __asm__ __volatile__("amocas.q.aq %0, %z2, %1"
                         : "+rJ" (expected.AO_whole), "+A" (*addr)
                         : "rJ" (new_val.AO_whole));
    return expected.AO_whole == old_val.AO_whole;
  }
# define AO_HAVE_double_compare_and_swap_acquire

  AO_INLINE int
  AO_double_compare_and_swap_release(volatile AO_double_t *addr,
                                     AO_double_t old_val, AO_double_t new_val)
  {
    AO_double_t expected = old_val;

    __asm__ __volatile__("amocas.q.rl %0, %z2, %1"
                         : "+rJ" (expected.AO_whole), "+A" (*addr)
                         : "rJ" (new_val.AO_whole));
    return expected.AO_whole == old_val.AO_whole;
  }
# define AO_HAVE_double_compare_and_swap_release

  AO_INLINE int
  AO_double_compare_and_swap_full(volatile AO_double_t *addr,
                                  AO_double_t old_val, AO_double_t new_val)
  {
    AO_double_t expected = old_val;

    __asm__ __volatile__("amocas.q.aqrl %0, %z2, %1"
                         : "+rJ" (expected.AO_whole), "+A" (*addr)
                         : "rJ" (new_val.AO_whole));
    return expected.AO_whole == old_val.AO_whole;
  }
# define AO_HAVE_double_compare_and_swap_full

#endif /* __riscv_zacas */

#include "generic.h"

#undef AO_GCC_FORCE_HAVE_CAS
#undef AO_NO_char_ARITHM
#undef AO_NO_short_ARITHM
#undef AO_SKIPATOMIC_double_load
#undef AO_SKIPATOMIC_double_load_acquire
#undef AO_SKIPATOMIC_double_store
#undef AO_SKIPATOMIC_double_store_release