/** * \file * Mapping of the arch memory model. * * Author: * Rodrigo Kumpera (kumpera@gmail.com) * * (C) 2011 Xamarin, Inc */ #ifndef _MONO_UTILS_MONO_MEMMODEL_H_ #define _MONO_UTILS_MONO_MEMMODEL_H_ #include #include /* In order to allow for fast concurrent code, we must use fencing to properly order memory access - specially on arch with weaker memory models such as ARM or PPC. On the other hand, we can't use arm's weak model on targets such as x86 that have a stronger model that requires much much less fencing. The idea of exposing each arch memory model is to avoid fencing whenever possible but at the same time make all required ordering explicit. There are four kinds of barriers, LoadLoad, LoadStore, StoreLoad and StoreStore. Each arch must define which ones needs fencing. We assume 3 kinds of barriers are available: load, store and memory (load+store). TODO: Add support for weaker forms of CAS such as present on ARM. TODO: replace all explicit uses of memory barriers with macros from this section. This will make a nicer read of lazy init code. TODO: if we find places where a data depencency could replace barriers, add macros here to help with it TODO: some arch with strong consistency, such as x86, support weaker access. We might need to expose more kinds of barriers once we exploit this. */ /* * Keep in sync with the enum in mini/mini-llvm-cpp.h. */ enum { MONO_MEMORY_BARRIER_NONE = 0, MONO_MEMORY_BARRIER_ACQ = 1, MONO_MEMORY_BARRIER_REL = 2, MONO_MEMORY_BARRIER_SEQ = 3, }; #define MEMORY_BARRIER mono_memory_barrier () #define LOAD_BARRIER mono_memory_read_barrier () #define STORE_BARRIER mono_memory_write_barrier () #if defined(__i386__) || defined(__x86_64__) /* Both x86 and amd64 follow the SPO memory model: -Loads are not reordered with other loads -Stores are not reordered with others stores -Stores are not reordered with earlier loads */ /*Neither sfence or mfence provide the required semantics here*/ #define STORE_LOAD_FENCE MEMORY_BARRIER #define LOAD_RELEASE_FENCE MEMORY_BARRIER #define STORE_ACQUIRE_FENCE MEMORY_BARRIER #elif defined(__arm__) /* ARM memory model is as weak as it can get. the only guarantee are data dependent accesses. LoadStore fences are much better handled using a data depencency such as: load x; if (x = x) store y; This trick can be applied to other fences such as LoadLoad, but require some assembly: LDR R0, [R1] AND R0, R0, #0 LDR R3, [R4, R0] */ #define STORE_STORE_FENCE STORE_BARRIER #define LOAD_LOAD_FENCE LOAD_BARRIER #define STORE_LOAD_FENCE MEMORY_BARRIER #define STORE_ACQUIRE_FENCE MEMORY_BARRIER #define STORE_RELEASE_FENCE MEMORY_BARRIER #define LOAD_ACQUIRE_FENCE MEMORY_BARRIER #define LOAD_RELEASE_FENCE MEMORY_BARRIER #elif defined(__s390x__) #define STORE_STORE_FENCE do {} while (0) #define LOAD_LOAD_FENCE do {} while (0) #define STORE_LOAD_FENCE do {} while (0) #define LOAD_STORE_FENCE do {} while (0) #define STORE_RELEASE_FENCE do {} while (0) #else /*default implementation with the weakest possible memory model */ #define STORE_STORE_FENCE STORE_BARRIER #define LOAD_LOAD_FENCE LOAD_BARRIER #define STORE_LOAD_FENCE MEMORY_BARRIER #define LOAD_STORE_FENCE MEMORY_BARRIER #define STORE_ACQUIRE_FENCE MEMORY_BARRIER #define STORE_RELEASE_FENCE MEMORY_BARRIER #define LOAD_ACQUIRE_FENCE MEMORY_BARRIER #define LOAD_RELEASE_FENCE MEMORY_BARRIER #endif #ifndef STORE_STORE_FENCE #define STORE_STORE_FENCE #endif #ifndef LOAD_LOAD_FENCE #define LOAD_LOAD_FENCE #endif #ifndef STORE_LOAD_FENCE #define STORE_LOAD_FENCE #endif #ifndef LOAD_STORE_FENCE #define LOAD_STORE_FENCE #endif #ifndef STORE_RELEASE_FENCE #define STORE_RELEASE_FENCE #endif #ifndef LOAD_RELEASE_FENCE #define LOAD_RELEASE_FENCE #endif #ifndef STORE_ACQUIRE_FENCE #define STORE_ACQUIRE_FENCE #endif #ifndef LOAD_ACQUIRE_FENCE #define LOAD_ACQUIRE_FENCE #endif /*Makes sure all previous stores as visible before */ #define mono_atomic_store_seq(target,value) do { \ STORE_STORE_FENCE; \ *(target) = (value); \ } while (0) /* Acquire/release semantics macros. */ #define mono_atomic_store_release(target,value) do { \ STORE_RELEASE_FENCE; \ *(target) = (value); \ } while (0) #define mono_atomic_load_release(_type,target) ({ \ _type __tmp; \ LOAD_RELEASE_FENCE; \ __tmp = *(target); \ __tmp; }) #define mono_atomic_load_acquire(var,_type,target) do { \ _type __tmp = *(target); \ LOAD_ACQUIRE_FENCE; \ (var) = __tmp; \ } while (0) #define mono_atomic_store_acquire(target,value) { \ *(target) = (value); \ STORE_ACQUIRE_FENCE; \ } #endif /* _MONO_UTILS_MONO_MEMMODEL_H_ */