yunque9/xcode/Libraries/libil2cpp/include/os/c-api/Posix/AtomicImpl-c-api.h

143 lines
3.7 KiB
C

#pragma once
#include <assert.h>
#ifdef __EMSCRIPTEN__
#include <emscripten/threading.h>
#endif
#ifdef IL2CPP_TARGET_ARM64
#define IL2CPP_BARRIER_PATCH(INTRIN) \
({ \
__typeof__(INTRIN) atomic_ret__ = (INTRIN); \
__sync_synchronize(); \
atomic_ret__; \
})
#else
#define IL2CPP_BARRIER_PATCH(INTRIN) INTRIN;
#endif
inline int32_t UnityPalAdd(volatile int32_t* location1, int32_t value)
{
ASSERT_ALIGNMENT(location1, 4);
return IL2CPP_BARRIER_PATCH(__sync_add_and_fetch(location1, value));
}
#if IL2CPP_ENABLE_INTERLOCKED_64_REQUIRED_ALIGNMENT
inline int64_t UnityPalAdd64(volatile int64_t* location1, int64_t value)
{
ASSERT_ALIGNMENT(location1, 8);
return IL2CPP_BARRIER_PATCH(__sync_add_and_fetch(location1, value));
}
#endif
inline int32_t UnityPalIncrement(volatile int32_t* value)
{
ASSERT_ALIGNMENT(value, 4);
return IL2CPP_BARRIER_PATCH(__sync_add_and_fetch(value, 1));
}
#if IL2CPP_ENABLE_INTERLOCKED_64_REQUIRED_ALIGNMENT
inline int64_t UnityPalIncrement64(volatile int64_t* value)
{
ASSERT_ALIGNMENT(value, 8);
return IL2CPP_BARRIER_PATCH(__sync_add_and_fetch(value, 1));
}
#endif
inline int32_t UnityPalDecrement(volatile int32_t* value)
{
ASSERT_ALIGNMENT(value, 4);
return IL2CPP_BARRIER_PATCH(__sync_add_and_fetch(value, -1));
}
#if IL2CPP_ENABLE_INTERLOCKED_64_REQUIRED_ALIGNMENT
inline int64_t UnityPalDecrement64(volatile int64_t* value)
{
ASSERT_ALIGNMENT(value, 8);
return IL2CPP_BARRIER_PATCH(__sync_add_and_fetch(value, -1));
}
#endif
inline int32_t UnityPalCompareExchange(volatile int32_t* dest, int32_t exchange, int32_t comparand)
{
ASSERT_ALIGNMENT(dest, 4);
return IL2CPP_BARRIER_PATCH(__sync_val_compare_and_swap(dest, comparand, exchange));
}
inline int64_t UnityPalCompareExchange64(volatile int64_t* dest, int64_t exchange, int64_t comparand)
{
ASSERT_ALIGNMENT(dest, 8);
#ifdef __EMSCRIPTEN__
return emscripten_atomic_cas_u64((void*)dest, comparand, exchange) == comparand ? comparand : *dest;
#else
return IL2CPP_BARRIER_PATCH(__sync_val_compare_and_swap(dest, comparand, exchange));
#endif
}
inline void* UnityPalCompareExchangePointer(void* volatile* dest, void* exchange, void* comparand)
{
ASSERT_ALIGNMENT(dest, IL2CPP_SIZEOF_VOID_P);
return IL2CPP_BARRIER_PATCH(__sync_val_compare_and_swap(dest, comparand, exchange));
}
inline int32_t UnityPalExchange(volatile int32_t* dest, int32_t exchange)
{
ASSERT_ALIGNMENT(dest, 4);
#ifdef __EMSCRIPTEN__
return emscripten_atomic_exchange_u32((void*)dest, exchange);
#else
int32_t prev;
do
{
prev = *dest;
}
while (!IL2CPP_BARRIER_PATCH(__sync_bool_compare_and_swap(dest, prev, exchange)));
return prev;
#endif
}
#if IL2CPP_ENABLE_INTERLOCKED_64_REQUIRED_ALIGNMENT
inline int64_t UnityPalExchange64(volatile int64_t* dest, int64_t exchange)
{
ASSERT_ALIGNMENT(dest, 8);
#ifdef __EMSCRIPTEN__
return emscripten_atomic_exchange_u64((void*)dest, exchange);
#else
int64_t prev;
do
{
prev = *dest;
}
while (!IL2CPP_BARRIER_PATCH(__sync_bool_compare_and_swap(dest, prev, exchange)));
return prev;
#endif
}
#endif
inline void* UnityPalExchangePointer(void* volatile* dest, void* exchange)
{
ASSERT_ALIGNMENT(dest, IL2CPP_SIZEOF_VOID_P);
#ifdef __EMSCRIPTEN__
return (void*)emscripten_atomic_exchange_u32((void*)dest, (uint32_t)exchange);
#else
void* prev;
do
{
prev = *dest;
}
while (!IL2CPP_BARRIER_PATCH(__sync_bool_compare_and_swap(dest, prev, exchange)));
return prev;
#endif
}
inline int64_t UnityPalRead64(volatile int64_t* addr)
{
ASSERT_ALIGNMENT(addr, 8);
return IL2CPP_BARRIER_PATCH(__sync_fetch_and_add(addr, 0));
}