1*9880d681SAndroid Build Coastguard Worker //===-- Atomic.cpp - Atomic Operations --------------------------*- C++ -*-===// 2*9880d681SAndroid Build Coastguard Worker // 3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure 4*9880d681SAndroid Build Coastguard Worker // 5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source 6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details. 7*9880d681SAndroid Build Coastguard Worker // 8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 9*9880d681SAndroid Build Coastguard Worker // 10*9880d681SAndroid Build Coastguard Worker // This file implements atomic operations. 11*9880d681SAndroid Build Coastguard Worker // 12*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 13*9880d681SAndroid Build Coastguard Worker 14*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Atomic.h" 15*9880d681SAndroid Build Coastguard Worker #include "llvm/Config/llvm-config.h" 16*9880d681SAndroid Build Coastguard Worker 17*9880d681SAndroid Build Coastguard Worker using namespace llvm; 18*9880d681SAndroid Build Coastguard Worker 19*9880d681SAndroid Build Coastguard Worker #if defined(_MSC_VER) 20*9880d681SAndroid Build Coastguard Worker #include <Intrin.h> 21*9880d681SAndroid Build Coastguard Worker #include <windows.h> 22*9880d681SAndroid Build Coastguard Worker #undef MemoryFence 23*9880d681SAndroid Build Coastguard Worker #endif 24*9880d681SAndroid Build Coastguard Worker 25*9880d681SAndroid Build Coastguard Worker #if defined(__GNUC__) || (defined(__IBMCPP__) && __IBMCPP__ >= 1210) 26*9880d681SAndroid Build Coastguard Worker #define GNU_ATOMICS 27*9880d681SAndroid Build Coastguard Worker #endif 28*9880d681SAndroid Build Coastguard Worker MemoryFence()29*9880d681SAndroid Build Coastguard Workervoid sys::MemoryFence() { 30*9880d681SAndroid Build Coastguard Worker #if LLVM_HAS_ATOMICS == 0 31*9880d681SAndroid Build Coastguard Worker return; 32*9880d681SAndroid Build Coastguard Worker #else 33*9880d681SAndroid Build Coastguard Worker # if defined(GNU_ATOMICS) 34*9880d681SAndroid Build Coastguard Worker __sync_synchronize(); 35*9880d681SAndroid Build Coastguard Worker # elif defined(_MSC_VER) 36*9880d681SAndroid Build Coastguard Worker MemoryBarrier(); 37*9880d681SAndroid Build Coastguard Worker # else 38*9880d681SAndroid Build Coastguard Worker # error No memory fence implementation for your platform! 39*9880d681SAndroid Build Coastguard Worker # endif 40*9880d681SAndroid Build Coastguard Worker #endif 41*9880d681SAndroid Build Coastguard Worker } 42*9880d681SAndroid Build Coastguard Worker CompareAndSwap(volatile sys::cas_flag * ptr,sys::cas_flag new_value,sys::cas_flag old_value)43*9880d681SAndroid Build Coastguard Workersys::cas_flag sys::CompareAndSwap(volatile sys::cas_flag* ptr, 44*9880d681SAndroid Build Coastguard Worker sys::cas_flag new_value, 45*9880d681SAndroid Build Coastguard Worker sys::cas_flag old_value) { 46*9880d681SAndroid Build Coastguard Worker #if LLVM_HAS_ATOMICS == 0 47*9880d681SAndroid Build Coastguard Worker sys::cas_flag result = *ptr; 48*9880d681SAndroid Build Coastguard Worker if (result == old_value) 49*9880d681SAndroid Build Coastguard Worker *ptr = new_value; 50*9880d681SAndroid Build Coastguard Worker return result; 51*9880d681SAndroid Build Coastguard Worker #elif defined(GNU_ATOMICS) 52*9880d681SAndroid Build Coastguard Worker return __sync_val_compare_and_swap(ptr, old_value, new_value); 53*9880d681SAndroid Build Coastguard Worker #elif defined(_MSC_VER) 54*9880d681SAndroid Build Coastguard Worker return InterlockedCompareExchange(ptr, new_value, old_value); 55*9880d681SAndroid Build Coastguard Worker #else 56*9880d681SAndroid Build Coastguard Worker # error No compare-and-swap implementation for your platform! 57*9880d681SAndroid Build Coastguard Worker #endif 58*9880d681SAndroid Build Coastguard Worker } 59