blob: b21275b9c860b14128490c8b2951025eca8eacec [file] [log] [blame]
Dmitry Vyukove8cee122012-06-29 18:00:38 +00001//===-- sanitizer_atomic_msvc.h ---------------------------------*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file is a part of ThreadSanitizer/AddressSanitizer runtime.
11// Not intended for direct inclusion. Include sanitizer_atomic.h.
12//
13//===----------------------------------------------------------------------===//
14
15#ifndef SANITIZER_ATOMIC_MSVC_H
16#define SANITIZER_ATOMIC_MSVC_H
17
Dmitry Vyukov3c5c9e72012-06-29 18:37:45 +000018extern "C" void _ReadWriteBarrier();
Dmitry Vyukovb379fe52012-06-29 18:28:02 +000019#pragma intrinsic(_ReadWriteBarrier)
Dmitry Vyukov3c5c9e72012-06-29 18:37:45 +000020extern "C" void _mm_mfence();
Dmitry Vyukovb379fe52012-06-29 18:28:02 +000021#pragma intrinsic(_mm_mfence)
Dmitry Vyukov3c5c9e72012-06-29 18:37:45 +000022extern "C" void _mm_pause();
Dmitry Vyukovb379fe52012-06-29 18:28:02 +000023#pragma intrinsic(_mm_pause)
Dmitry Vyukov3c5c9e72012-06-29 18:37:45 +000024extern "C" long _InterlockedExchangeAdd( // NOLINT
25 long volatile * Addend, long Value); // NOLINT
Dmitry Vyukovb379fe52012-06-29 18:28:02 +000026#pragma intrinsic(_InterlockedExchangeAdd)
Dmitry Vyukovfa90fa32012-08-31 14:01:33 +000027extern "C" void *_InterlockedCompareExchangePointer(
28 void *volatile *Destination,
29 void *Exchange, void *Comparand);
30#pragma intrinsic(_InterlockedCompareExchangePointer)
Dmitry Vyukove8cee122012-06-29 18:00:38 +000031
32namespace __sanitizer {
33
34INLINE void atomic_signal_fence(memory_order) {
35 _ReadWriteBarrier();
36}
37
38INLINE void atomic_thread_fence(memory_order) {
39 _mm_mfence();
40}
41
42INLINE void proc_yield(int cnt) {
43 for (int i = 0; i < cnt; i++)
44 _mm_pause();
45}
46
47template<typename T>
48INLINE typename T::Type atomic_load(
49 const volatile T *a, memory_order mo) {
50 DCHECK(mo & (memory_order_relaxed | memory_order_consume
51 | memory_order_acquire | memory_order_seq_cst));
52 DCHECK(!((uptr)a % sizeof(*a)));
53 typename T::Type v;
54 if (mo == memory_order_relaxed) {
55 v = a->val_dont_use;
56 } else {
57 atomic_signal_fence(memory_order_seq_cst);
58 v = a->val_dont_use;
59 atomic_signal_fence(memory_order_seq_cst);
60 }
61 return v;
62}
63
64template<typename T>
65INLINE void atomic_store(volatile T *a, typename T::Type v, memory_order mo) {
66 DCHECK(mo & (memory_order_relaxed | memory_order_release
67 | memory_order_seq_cst));
68 DCHECK(!((uptr)a % sizeof(*a)));
69 if (mo == memory_order_relaxed) {
70 a->val_dont_use = v;
71 } else {
72 atomic_signal_fence(memory_order_seq_cst);
73 a->val_dont_use = v;
74 atomic_signal_fence(memory_order_seq_cst);
75 }
76 if (mo == memory_order_seq_cst)
77 atomic_thread_fence(memory_order_seq_cst);
78}
79
80INLINE u32 atomic_fetch_add(volatile atomic_uint32_t *a,
81 u32 v, memory_order mo) {
82 (void)mo;
83 DCHECK(!((uptr)a % sizeof(*a)));
84 return (u32)_InterlockedExchangeAdd(
85 (volatile long*)&a->val_dont_use, (long)v); // NOLINT
86}
87
88INLINE u8 atomic_exchange(volatile atomic_uint8_t *a,
89 u8 v, memory_order mo) {
90 (void)mo;
91 DCHECK(!((uptr)a % sizeof(*a)));
92 __asm {
93 mov eax, a
Dmitry Vyukovb379fe52012-06-29 18:28:02 +000094 mov cl, v
95 xchg [eax], cl // NOLINT
96 mov v, cl
Dmitry Vyukove8cee122012-06-29 18:00:38 +000097 }
98 return v;
99}
100
101INLINE u16 atomic_exchange(volatile atomic_uint16_t *a,
102 u16 v, memory_order mo) {
103 (void)mo;
104 DCHECK(!((uptr)a % sizeof(*a)));
105 __asm {
106 mov eax, a
Dmitry Vyukovb379fe52012-06-29 18:28:02 +0000107 mov cx, v
108 xchg [eax], cx // NOLINT
109 mov v, cx
Dmitry Vyukove8cee122012-06-29 18:00:38 +0000110 }
111 return v;
112}
113
Dmitry Vyukovfa90fa32012-08-31 14:01:33 +0000114INLINE bool atomic_compare_exchange_strong(volatile uptr *a,
115 uptr *cmp,
116 uptr xchg,
117 memory_order mo) {
118 uptr cmpv = *cmp;
119 uptr prev = (uptr)_InterlockedCompareExchangePointer(
120 (void*volatile*)&a->val_dont_use, (void*)xchg, (void*)cmpv);
121 if (prev == cmpv)
122 return true;
123 *cmp = prev;
124 return false;
125}
126
127template<typename T>
128INLINE bool atomic_compare_exchange_weak(volatile T *a,
129 typename T::Type *cmp,
130 typename T::Type xchg,
131 memory_order mo) {
132 return atomic_compare_exchange_strong(a, cmp, xchg, mo);
133}
134
Dmitry Vyukove8cee122012-06-29 18:00:38 +0000135} // namespace __sanitizer
136
137#endif // SANITIZER_ATOMIC_CLANG_H