1 //===-- sanitizer_atomic_msvc.h ---------------------------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 // This file is a part of ThreadSanitizer/AddressSanitizer runtime.
11 // Not intended for direct inclusion. Include sanitizer_atomic.h.
13 //===----------------------------------------------------------------------===//
15 #ifndef SANITIZER_ATOMIC_MSVC_H
16 #define SANITIZER_ATOMIC_MSVC_H
18 extern "C" void _ReadWriteBarrier();
19 #pragma intrinsic(_ReadWriteBarrier)
20 extern "C" void _mm_mfence();
21 #pragma intrinsic(_mm_mfence)
22 extern "C" void _mm_pause();
23 #pragma intrinsic(_mm_pause)
24 extern "C" long _InterlockedExchangeAdd( // NOLINT
25 long volatile * Addend
, long Value
); // NOLINT
26 #pragma intrinsic(_InterlockedExchangeAdd)
27 extern "C" short _InterlockedCompareExchange16( // NOLINT
28 short volatile *Destination
, // NOLINT
29 short Exchange
, short Comparand
); // NOLINT
30 #pragma intrinsic(_InterlockedCompareExchange16)
32 long long _InterlockedCompareExchange64( // NOLINT
33 long long volatile *Destination
, // NOLINT
34 long long Exchange
, long long Comparand
); // NOLINT
35 #pragma intrinsic(_InterlockedCompareExchange64)
38 extern "C" long long _InterlockedExchangeAdd64( // NOLINT
39 long long volatile * Addend
, long long Value
); // NOLINT
40 #pragma intrinsic(_InterlockedExchangeAdd64)
41 extern "C" void *_InterlockedCompareExchangePointer(
42 void *volatile *Destination
,
43 void *Exchange
, void *Comparand
);
44 #pragma intrinsic(_InterlockedCompareExchangePointer)
46 // There's no _InterlockedCompareExchangePointer intrinsic on x86,
47 // so call _InterlockedCompareExchange instead.
49 long __cdecl
_InterlockedCompareExchange( // NOLINT
50 long volatile *Destination
, // NOLINT
51 long Exchange
, long Comparand
); // NOLINT
52 #pragma intrinsic(_InterlockedCompareExchange)
54 inline static void *_InterlockedCompareExchangePointer(
55 void *volatile *Destination
,
56 void *Exchange
, void *Comparand
) {
57 return reinterpret_cast<void*>(
58 _InterlockedCompareExchange(
59 reinterpret_cast<long volatile*>(Destination
), // NOLINT
60 reinterpret_cast<long>(Exchange
), // NOLINT
61 reinterpret_cast<long>(Comparand
))); // NOLINT
65 namespace __sanitizer
{
67 INLINE
void atomic_signal_fence(memory_order
) {
71 INLINE
void atomic_thread_fence(memory_order
) {
75 INLINE
void proc_yield(int cnt
) {
76 for (int i
= 0; i
< cnt
; i
++)
81 INLINE typename
T::Type
atomic_load(
82 const volatile T
*a
, memory_order mo
) {
83 DCHECK(mo
& (memory_order_relaxed
| memory_order_consume
84 | memory_order_acquire
| memory_order_seq_cst
));
85 DCHECK(!((uptr
)a
% sizeof(*a
)));
87 // FIXME(dvyukov): 64-bit load is not atomic on 32-bits.
88 if (mo
== memory_order_relaxed
) {
91 atomic_signal_fence(memory_order_seq_cst
);
93 atomic_signal_fence(memory_order_seq_cst
);
99 INLINE
void atomic_store(volatile T
*a
, typename
T::Type v
, memory_order mo
) {
100 DCHECK(mo
& (memory_order_relaxed
| memory_order_release
101 | memory_order_seq_cst
));
102 DCHECK(!((uptr
)a
% sizeof(*a
)));
103 // FIXME(dvyukov): 64-bit store is not atomic on 32-bits.
104 if (mo
== memory_order_relaxed
) {
107 atomic_signal_fence(memory_order_seq_cst
);
109 atomic_signal_fence(memory_order_seq_cst
);
111 if (mo
== memory_order_seq_cst
)
112 atomic_thread_fence(memory_order_seq_cst
);
115 INLINE u32
atomic_fetch_add(volatile atomic_uint32_t
*a
,
116 u32 v
, memory_order mo
) {
118 DCHECK(!((uptr
)a
% sizeof(*a
)));
119 return (u32
)_InterlockedExchangeAdd(
120 (volatile long*)&a
->val_dont_use
, (long)v
); // NOLINT
123 INLINE uptr
atomic_fetch_add(volatile atomic_uintptr_t
*a
,
124 uptr v
, memory_order mo
) {
126 DCHECK(!((uptr
)a
% sizeof(*a
)));
128 return (uptr
)_InterlockedExchangeAdd64(
129 (volatile long long*)&a
->val_dont_use
, (long long)v
); // NOLINT
131 return (uptr
)_InterlockedExchangeAdd(
132 (volatile long*)&a
->val_dont_use
, (long)v
); // NOLINT
136 INLINE u32
atomic_fetch_sub(volatile atomic_uint32_t
*a
,
137 u32 v
, memory_order mo
) {
139 DCHECK(!((uptr
)a
% sizeof(*a
)));
140 return (u32
)_InterlockedExchangeAdd(
141 (volatile long*)&a
->val_dont_use
, -(long)v
); // NOLINT
144 INLINE uptr
atomic_fetch_sub(volatile atomic_uintptr_t
*a
,
145 uptr v
, memory_order mo
) {
147 DCHECK(!((uptr
)a
% sizeof(*a
)));
149 return (uptr
)_InterlockedExchangeAdd64(
150 (volatile long long*)&a
->val_dont_use
, -(long long)v
); // NOLINT
152 return (uptr
)_InterlockedExchangeAdd(
153 (volatile long*)&a
->val_dont_use
, -(long)v
); // NOLINT
157 INLINE u8
atomic_exchange(volatile atomic_uint8_t
*a
,
158 u8 v
, memory_order mo
) {
160 DCHECK(!((uptr
)a
% sizeof(*a
)));
164 xchg
[eax
], cl
// NOLINT
170 INLINE u16
atomic_exchange(volatile atomic_uint16_t
*a
,
171 u16 v
, memory_order mo
) {
173 DCHECK(!((uptr
)a
% sizeof(*a
)));
177 xchg
[eax
], cx
// NOLINT
183 INLINE
bool atomic_compare_exchange_strong(volatile atomic_uint8_t
*a
,
188 DCHECK(!((uptr
)a
% sizeof(*a
)));
195 lock cmpxchg
[ecx
], dl
204 INLINE
bool atomic_compare_exchange_strong(volatile atomic_uintptr_t
*a
,
209 uptr prev
= (uptr
)_InterlockedCompareExchangePointer(
210 (void*volatile*)&a
->val_dont_use
, (void*)xchg
, (void*)cmpv
);
217 INLINE
bool atomic_compare_exchange_strong(volatile atomic_uint16_t
*a
,
222 u16 prev
= (u16
)_InterlockedCompareExchange16(
223 (volatile short*)&a
->val_dont_use
, (short)xchg
, (short)cmpv
);
230 INLINE
bool atomic_compare_exchange_strong(volatile atomic_uint32_t
*a
,
235 u32 prev
= (u32
)_InterlockedCompareExchange(
236 (volatile long*)&a
->val_dont_use
, (long)xchg
, (long)cmpv
);
243 INLINE
bool atomic_compare_exchange_strong(volatile atomic_uint64_t
*a
,
248 u64 prev
= (u64
)_InterlockedCompareExchange64(
249 (volatile long long*)&a
->val_dont_use
, (long long)xchg
, (long long)cmpv
);
257 INLINE
bool atomic_compare_exchange_weak(volatile T
*a
,
258 typename
T::Type
*cmp
,
259 typename
T::Type xchg
,
261 return atomic_compare_exchange_strong(a
, cmp
, xchg
, mo
);
264 } // namespace __sanitizer
266 #endif // SANITIZER_ATOMIC_CLANG_H