blob: a7922a42e42cedb06219c37bd73e851ab5f671bd [file] [log] [blame]
Alexey Samsonov3b2f9f42012-06-04 13:55:19 +00001//===-- tsan_interface_ann.cc ---------------------------------------------===//
Kostya Serebryany4ad375f2012-05-10 13:48:04 +00002//
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 (TSan), a race detector.
11//
12//===----------------------------------------------------------------------===//
Alexey Samsonov90f96302012-06-04 13:27:49 +000013#include "sanitizer_common/sanitizer_libc.h"
Dmitry Vyukov01ea6532012-12-04 14:01:21 +000014#include "sanitizer_common/sanitizer_internal_defs.h"
Alexey Samsonov8bd90982012-06-07 09:50:16 +000015#include "sanitizer_common/sanitizer_placement_new.h"
Dmitry Vyukov5cf581a2013-06-17 19:57:03 +000016#include "sanitizer_common/sanitizer_stacktrace.h"
Kamil Rytarowski64fc9cf2017-12-04 12:30:09 +000017#include "sanitizer_common/sanitizer_vector.h"
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000018#include "tsan_interface_ann.h"
19#include "tsan_mutex.h"
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000020#include "tsan_report.h"
21#include "tsan_rtl.h"
22#include "tsan_mman.h"
23#include "tsan_flags.h"
Dmitry Vyukovea03fc22012-06-14 21:40:35 +000024#include "tsan_platform.h"
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000025
26#define CALLERPC ((uptr)__builtin_return_address(0))
27
28using namespace __tsan; // NOLINT
29
30namespace __tsan {
31
32class ScopedAnnotation {
33 public:
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +000034 ScopedAnnotation(ThreadState *thr, const char *aname, uptr pc)
Dmitry Vyukovce372102013-12-24 12:55:56 +000035 : thr_(thr) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000036 FuncEntry(thr_, pc);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +000037 DPrintf("#%d: annotation %s()\n", thr_->tid, aname);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000038 }
39
40 ~ScopedAnnotation() {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000041 FuncExit(thr_);
Dmitry Vyukovbde4c9c2014-05-29 13:50:54 +000042 CheckNoLocks(thr_);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000043 }
44 private:
45 ThreadState *const thr_;
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000046};
47
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +000048#define SCOPED_ANNOTATION_RET(typ, ret) \
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000049 if (!flags()->enable_annotations) \
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +000050 return ret; \
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000051 ThreadState *thr = cur_thread(); \
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +000052 const uptr caller_pc = (uptr)__builtin_return_address(0); \
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000053 StatInc(thr, StatAnnotation); \
54 StatInc(thr, Stat##typ); \
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +000055 ScopedAnnotation sa(thr, __func__, caller_pc); \
Alexey Samsonov40733a82014-11-03 22:23:44 +000056 const uptr pc = StackTrace::GetCurrentPc(); \
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000057 (void)pc; \
58/**/
59
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +000060#define SCOPED_ANNOTATION(typ) SCOPED_ANNOTATION_RET(typ, )
61
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000062static const int kMaxDescLen = 128;
63
64struct ExpectRace {
65 ExpectRace *next;
66 ExpectRace *prev;
Dmitry Vyukov3464dac2015-09-03 11:20:46 +000067 atomic_uintptr_t hitcount;
68 atomic_uintptr_t addcount;
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000069 uptr addr;
70 uptr size;
71 char *file;
72 int line;
73 char desc[kMaxDescLen];
74};
75
76struct DynamicAnnContext {
77 Mutex mtx;
78 ExpectRace expect;
79 ExpectRace benign;
80
81 DynamicAnnContext()
82 : mtx(MutexTypeAnnotations, StatMtxAnnotations) {
83 }
84};
85
86static DynamicAnnContext *dyn_ann_ctx;
Alexey Samsonovef2e2cf2012-06-05 13:50:57 +000087static char dyn_ann_ctx_placeholder[sizeof(DynamicAnnContext)] ALIGNED(64);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000088
89static void AddExpectRace(ExpectRace *list,
90 char *f, int l, uptr addr, uptr size, char *desc) {
91 ExpectRace *race = list->next;
92 for (; race != list; race = race->next) {
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +000093 if (race->addr == addr && race->size == size) {
Dmitry Vyukov3464dac2015-09-03 11:20:46 +000094 atomic_store_relaxed(&race->addcount,
95 atomic_load_relaxed(&race->addcount) + 1);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000096 return;
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +000097 }
Kostya Serebryany4ad375f2012-05-10 13:48:04 +000098 }
99 race = (ExpectRace*)internal_alloc(MBlockExpectRace, sizeof(ExpectRace));
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000100 race->addr = addr;
101 race->size = size;
102 race->file = f;
103 race->line = l;
104 race->desc[0] = 0;
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000105 atomic_store_relaxed(&race->hitcount, 0);
106 atomic_store_relaxed(&race->addcount, 1);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000107 if (desc) {
108 int i = 0;
109 for (; i < kMaxDescLen - 1 && desc[i]; i++)
110 race->desc[i] = desc[i];
111 race->desc[i] = 0;
112 }
113 race->prev = list;
114 race->next = list->next;
115 race->next->prev = race;
116 list->next = race;
117}
118
119static ExpectRace *FindRace(ExpectRace *list, uptr addr, uptr size) {
120 for (ExpectRace *race = list->next; race != list; race = race->next) {
121 uptr maxbegin = max(race->addr, addr);
122 uptr minend = min(race->addr + race->size, addr + size);
123 if (maxbegin < minend)
124 return race;
125 }
126 return 0;
127}
128
129static bool CheckContains(ExpectRace *list, uptr addr, uptr size) {
130 ExpectRace *race = FindRace(list, addr, size);
131 if (race == 0)
132 return false;
Alexey Samsonov51ae9832012-06-06 13:11:29 +0000133 DPrintf("Hit expected/benign race: %s addr=%zx:%d %s:%d\n",
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000134 race->desc, race->addr, (int)race->size, race->file, race->line);
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000135 atomic_fetch_add(&race->hitcount, 1, memory_order_relaxed);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000136 return true;
137}
138
139static void InitList(ExpectRace *list) {
140 list->next = list;
141 list->prev = list;
142}
143
144void InitializeDynamicAnnotations() {
145 dyn_ann_ctx = new(dyn_ann_ctx_placeholder) DynamicAnnContext;
146 InitList(&dyn_ann_ctx->expect);
147 InitList(&dyn_ann_ctx->benign);
148}
149
150bool IsExpectedReport(uptr addr, uptr size) {
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000151 ReadLock lock(&dyn_ann_ctx->mtx);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000152 if (CheckContains(&dyn_ann_ctx->expect, addr, size))
153 return true;
154 if (CheckContains(&dyn_ann_ctx->benign, addr, size))
155 return true;
156 return false;
157}
158
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000159static void CollectMatchedBenignRaces(Vector<ExpectRace> *matched,
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000160 int *unique_count, int *hit_count, atomic_uintptr_t ExpectRace::*counter) {
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000161 ExpectRace *list = &dyn_ann_ctx->benign;
162 for (ExpectRace *race = list->next; race != list; race = race->next) {
163 (*unique_count)++;
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000164 const uptr cnt = atomic_load_relaxed(&(race->*counter));
165 if (cnt == 0)
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000166 continue;
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000167 *hit_count += cnt;
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000168 uptr i = 0;
169 for (; i < matched->Size(); i++) {
170 ExpectRace *race0 = &(*matched)[i];
171 if (race->line == race0->line
172 && internal_strcmp(race->file, race0->file) == 0
173 && internal_strcmp(race->desc, race0->desc) == 0) {
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000174 atomic_fetch_add(&(race0->*counter), cnt, memory_order_relaxed);
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000175 break;
176 }
177 }
178 if (i == matched->Size())
179 matched->PushBack(*race);
180 }
181}
182
183void PrintMatchedBenignRaces() {
184 Lock lock(&dyn_ann_ctx->mtx);
185 int unique_count = 0;
186 int hit_count = 0;
187 int add_count = 0;
Kamil Rytarowski64fc9cf2017-12-04 12:30:09 +0000188 Vector<ExpectRace> hit_matched;
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000189 CollectMatchedBenignRaces(&hit_matched, &unique_count, &hit_count,
190 &ExpectRace::hitcount);
Kamil Rytarowski64fc9cf2017-12-04 12:30:09 +0000191 Vector<ExpectRace> add_matched;
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000192 CollectMatchedBenignRaces(&add_matched, &unique_count, &add_count,
193 &ExpectRace::addcount);
194 if (hit_matched.Size()) {
195 Printf("ThreadSanitizer: Matched %d \"benign\" races (pid=%d):\n",
Peter Collingbourneffaf2ea2013-05-17 16:56:53 +0000196 hit_count, (int)internal_getpid());
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000197 for (uptr i = 0; i < hit_matched.Size(); i++) {
198 Printf("%d %s:%d %s\n",
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000199 atomic_load_relaxed(&hit_matched[i].hitcount),
200 hit_matched[i].file, hit_matched[i].line, hit_matched[i].desc);
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000201 }
202 }
203 if (hit_matched.Size()) {
204 Printf("ThreadSanitizer: Annotated %d \"benign\" races, %d unique"
205 " (pid=%d):\n",
Peter Collingbourneffaf2ea2013-05-17 16:56:53 +0000206 add_count, unique_count, (int)internal_getpid());
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000207 for (uptr i = 0; i < add_matched.Size(); i++) {
208 Printf("%d %s:%d %s\n",
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000209 atomic_load_relaxed(&add_matched[i].addcount),
210 add_matched[i].file, add_matched[i].line, add_matched[i].desc);
Dmitry Vyukovf2cbda42013-03-28 16:21:19 +0000211 }
212 }
213}
214
215static void ReportMissedExpectedRace(ExpectRace *race) {
216 Printf("==================\n");
217 Printf("WARNING: ThreadSanitizer: missed expected data race\n");
218 Printf(" %s addr=%zx %s:%d\n",
219 race->desc, race->addr, race->file, race->line);
220 Printf("==================\n");
221}
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000222} // namespace __tsan
223
224using namespace __tsan; // NOLINT
225
226extern "C" {
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000227void INTERFACE_ATTRIBUTE AnnotateHappensBefore(char *f, int l, uptr addr) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000228 SCOPED_ANNOTATION(AnnotateHappensBefore);
Dmitry Vyukovc2437ff2013-09-19 04:42:25 +0000229 Release(thr, pc, addr);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000230}
231
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000232void INTERFACE_ATTRIBUTE AnnotateHappensAfter(char *f, int l, uptr addr) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000233 SCOPED_ANNOTATION(AnnotateHappensAfter);
Dmitry Vyukovc2437ff2013-09-19 04:42:25 +0000234 Acquire(thr, pc, addr);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000235}
236
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000237void INTERFACE_ATTRIBUTE AnnotateCondVarSignal(char *f, int l, uptr cv) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000238 SCOPED_ANNOTATION(AnnotateCondVarSignal);
239}
240
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000241void INTERFACE_ATTRIBUTE AnnotateCondVarSignalAll(char *f, int l, uptr cv) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000242 SCOPED_ANNOTATION(AnnotateCondVarSignalAll);
243}
244
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000245void INTERFACE_ATTRIBUTE AnnotateMutexIsNotPHB(char *f, int l, uptr mu) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000246 SCOPED_ANNOTATION(AnnotateMutexIsNotPHB);
247}
248
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000249void INTERFACE_ATTRIBUTE AnnotateCondVarWait(char *f, int l, uptr cv,
250 uptr lock) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000251 SCOPED_ANNOTATION(AnnotateCondVarWait);
252}
253
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000254void INTERFACE_ATTRIBUTE AnnotateRWLockCreate(char *f, int l, uptr m) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000255 SCOPED_ANNOTATION(AnnotateRWLockCreate);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000256 MutexCreate(thr, pc, m, MutexFlagWriteReentrant);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000257}
258
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000259void INTERFACE_ATTRIBUTE AnnotateRWLockCreateStatic(char *f, int l, uptr m) {
Dmitry Vyukov4723e6b2012-08-16 13:29:41 +0000260 SCOPED_ANNOTATION(AnnotateRWLockCreateStatic);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000261 MutexCreate(thr, pc, m, MutexFlagWriteReentrant | MutexFlagLinkerInit);
Dmitry Vyukov4723e6b2012-08-16 13:29:41 +0000262}
263
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000264void INTERFACE_ATTRIBUTE AnnotateRWLockDestroy(char *f, int l, uptr m) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000265 SCOPED_ANNOTATION(AnnotateRWLockDestroy);
Dmitry Vyukov4723e6b2012-08-16 13:29:41 +0000266 MutexDestroy(thr, pc, m);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000267}
268
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000269void INTERFACE_ATTRIBUTE AnnotateRWLockAcquired(char *f, int l, uptr m,
270 uptr is_w) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000271 SCOPED_ANNOTATION(AnnotateRWLockAcquired);
Dmitry Vyukov4723e6b2012-08-16 13:29:41 +0000272 if (is_w)
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000273 MutexPostLock(thr, pc, m, MutexFlagDoPreLockOnPostLock);
Dmitry Vyukov4723e6b2012-08-16 13:29:41 +0000274 else
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000275 MutexPostReadLock(thr, pc, m, MutexFlagDoPreLockOnPostLock);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000276}
277
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000278void INTERFACE_ATTRIBUTE AnnotateRWLockReleased(char *f, int l, uptr m,
279 uptr is_w) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000280 SCOPED_ANNOTATION(AnnotateRWLockReleased);
Dmitry Vyukov4723e6b2012-08-16 13:29:41 +0000281 if (is_w)
282 MutexUnlock(thr, pc, m);
283 else
284 MutexReadUnlock(thr, pc, m);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000285}
286
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000287void INTERFACE_ATTRIBUTE AnnotateTraceMemory(char *f, int l, uptr mem) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000288 SCOPED_ANNOTATION(AnnotateTraceMemory);
289}
290
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000291void INTERFACE_ATTRIBUTE AnnotateFlushState(char *f, int l) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000292 SCOPED_ANNOTATION(AnnotateFlushState);
293}
294
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000295void INTERFACE_ATTRIBUTE AnnotateNewMemory(char *f, int l, uptr mem,
296 uptr size) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000297 SCOPED_ANNOTATION(AnnotateNewMemory);
298}
299
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000300void INTERFACE_ATTRIBUTE AnnotateNoOp(char *f, int l, uptr mem) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000301 SCOPED_ANNOTATION(AnnotateNoOp);
302}
303
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000304void INTERFACE_ATTRIBUTE AnnotateFlushExpectedRaces(char *f, int l) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000305 SCOPED_ANNOTATION(AnnotateFlushExpectedRaces);
306 Lock lock(&dyn_ann_ctx->mtx);
307 while (dyn_ann_ctx->expect.next != &dyn_ann_ctx->expect) {
308 ExpectRace *race = dyn_ann_ctx->expect.next;
Dmitry Vyukov3464dac2015-09-03 11:20:46 +0000309 if (atomic_load_relaxed(&race->hitcount) == 0) {
Dmitry Vyukovc9e12aa2014-03-20 10:36:20 +0000310 ctx->nmissed_expected++;
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000311 ReportMissedExpectedRace(race);
312 }
313 race->prev->next = race->next;
314 race->next->prev = race->prev;
315 internal_free(race);
316 }
317}
318
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000319void INTERFACE_ATTRIBUTE AnnotateEnableRaceDetection(
320 char *f, int l, int enable) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000321 SCOPED_ANNOTATION(AnnotateEnableRaceDetection);
322 // FIXME: Reconsider this functionality later. It may be irrelevant.
323}
324
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000325void INTERFACE_ATTRIBUTE AnnotateMutexIsUsedAsCondVar(
326 char *f, int l, uptr mu) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000327 SCOPED_ANNOTATION(AnnotateMutexIsUsedAsCondVar);
328}
329
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000330void INTERFACE_ATTRIBUTE AnnotatePCQGet(
331 char *f, int l, uptr pcq) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000332 SCOPED_ANNOTATION(AnnotatePCQGet);
333}
334
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000335void INTERFACE_ATTRIBUTE AnnotatePCQPut(
336 char *f, int l, uptr pcq) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000337 SCOPED_ANNOTATION(AnnotatePCQPut);
338}
339
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000340void INTERFACE_ATTRIBUTE AnnotatePCQDestroy(
341 char *f, int l, uptr pcq) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000342 SCOPED_ANNOTATION(AnnotatePCQDestroy);
343}
344
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000345void INTERFACE_ATTRIBUTE AnnotatePCQCreate(
346 char *f, int l, uptr pcq) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000347 SCOPED_ANNOTATION(AnnotatePCQCreate);
348}
349
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000350void INTERFACE_ATTRIBUTE AnnotateExpectRace(
351 char *f, int l, uptr mem, char *desc) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000352 SCOPED_ANNOTATION(AnnotateExpectRace);
353 Lock lock(&dyn_ann_ctx->mtx);
354 AddExpectRace(&dyn_ann_ctx->expect,
355 f, l, mem, 1, desc);
Alexey Samsonov51ae9832012-06-06 13:11:29 +0000356 DPrintf("Add expected race: %s addr=%zx %s:%d\n", desc, mem, f, l);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000357}
358
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000359static void BenignRaceImpl(
360 char *f, int l, uptr mem, uptr size, char *desc) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000361 Lock lock(&dyn_ann_ctx->mtx);
362 AddExpectRace(&dyn_ann_ctx->benign,
363 f, l, mem, size, desc);
Alexey Samsonov51ae9832012-06-06 13:11:29 +0000364 DPrintf("Add benign race: %s addr=%zx %s:%d\n", desc, mem, f, l);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000365}
366
367// FIXME: Turn it off later. WTF is benign race?1?? Go talk to Hans Boehm.
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000368void INTERFACE_ATTRIBUTE AnnotateBenignRaceSized(
369 char *f, int l, uptr mem, uptr size, char *desc) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000370 SCOPED_ANNOTATION(AnnotateBenignRaceSized);
371 BenignRaceImpl(f, l, mem, size, desc);
372}
373
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000374void INTERFACE_ATTRIBUTE AnnotateBenignRace(
375 char *f, int l, uptr mem, char *desc) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000376 SCOPED_ANNOTATION(AnnotateBenignRace);
377 BenignRaceImpl(f, l, mem, 1, desc);
378}
379
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000380void INTERFACE_ATTRIBUTE AnnotateIgnoreReadsBegin(char *f, int l) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000381 SCOPED_ANNOTATION(AnnotateIgnoreReadsBegin);
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +0000382 ThreadIgnoreBegin(thr, pc);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000383}
384
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000385void INTERFACE_ATTRIBUTE AnnotateIgnoreReadsEnd(char *f, int l) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000386 SCOPED_ANNOTATION(AnnotateIgnoreReadsEnd);
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +0000387 ThreadIgnoreEnd(thr, pc);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000388}
389
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000390void INTERFACE_ATTRIBUTE AnnotateIgnoreWritesBegin(char *f, int l) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000391 SCOPED_ANNOTATION(AnnotateIgnoreWritesBegin);
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +0000392 ThreadIgnoreBegin(thr, pc);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000393}
394
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000395void INTERFACE_ATTRIBUTE AnnotateIgnoreWritesEnd(char *f, int l) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000396 SCOPED_ANNOTATION(AnnotateIgnoreWritesEnd);
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +0000397 ThreadIgnoreEnd(thr, pc);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000398}
399
Dmitry Vyukovfbb194f2013-10-10 15:58:12 +0000400void INTERFACE_ATTRIBUTE AnnotateIgnoreSyncBegin(char *f, int l) {
401 SCOPED_ANNOTATION(AnnotateIgnoreSyncBegin);
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +0000402 ThreadIgnoreSyncBegin(thr, pc);
Dmitry Vyukovfbb194f2013-10-10 15:58:12 +0000403}
404
405void INTERFACE_ATTRIBUTE AnnotateIgnoreSyncEnd(char *f, int l) {
406 SCOPED_ANNOTATION(AnnotateIgnoreSyncEnd);
Dmitry Vyukov3238e1c2013-11-27 11:30:28 +0000407 ThreadIgnoreSyncEnd(thr, pc);
Dmitry Vyukovfbb194f2013-10-10 15:58:12 +0000408}
409
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000410void INTERFACE_ATTRIBUTE AnnotatePublishMemoryRange(
411 char *f, int l, uptr addr, uptr size) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000412 SCOPED_ANNOTATION(AnnotatePublishMemoryRange);
413}
414
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000415void INTERFACE_ATTRIBUTE AnnotateUnpublishMemoryRange(
416 char *f, int l, uptr addr, uptr size) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000417 SCOPED_ANNOTATION(AnnotateUnpublishMemoryRange);
418}
419
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000420void INTERFACE_ATTRIBUTE AnnotateThreadName(
421 char *f, int l, char *name) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000422 SCOPED_ANNOTATION(AnnotateThreadName);
Dmitry Vyukov1b469932012-12-04 15:46:05 +0000423 ThreadSetName(thr, name);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000424}
425
Alexander Potapenkobbfc7222013-04-02 11:21:53 +0000426// We deliberately omit the implementation of WTFAnnotateHappensBefore() and
427// WTFAnnotateHappensAfter(). Those are being used by Webkit to annotate
428// atomic operations, which should be handled by ThreadSanitizer correctly.
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000429void INTERFACE_ATTRIBUTE WTFAnnotateHappensBefore(char *f, int l, uptr addr) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000430 SCOPED_ANNOTATION(AnnotateHappensBefore);
431}
432
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000433void INTERFACE_ATTRIBUTE WTFAnnotateHappensAfter(char *f, int l, uptr addr) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000434 SCOPED_ANNOTATION(AnnotateHappensAfter);
435}
436
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000437void INTERFACE_ATTRIBUTE WTFAnnotateBenignRaceSized(
438 char *f, int l, uptr mem, uptr sz, char *desc) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000439 SCOPED_ANNOTATION(AnnotateBenignRaceSized);
Dmitry Vyukov2918b672013-11-19 16:03:15 +0000440 BenignRaceImpl(f, l, mem, sz, desc);
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000441}
442
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000443int INTERFACE_ATTRIBUTE RunningOnValgrind() {
Dmitry Vyukov65c21a52012-05-24 09:24:45 +0000444 return flags()->running_on_valgrind;
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000445}
446
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000447double __attribute__((weak)) INTERFACE_ATTRIBUTE ValgrindSlowdown(void) {
Dmitry Vyukov166b8e52012-05-17 08:04:41 +0000448 return 10.0;
449}
450
Dmitry Vyukov01ea6532012-12-04 14:01:21 +0000451const char INTERFACE_ATTRIBUTE* ThreadSanitizerQuery(const char *query) {
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000452 if (internal_strcmp(query, "pure_happens_before") == 0)
453 return "1";
454 else
455 return "0";
456}
Evgeniy Stepanov91375fd2013-09-18 11:20:31 +0000457
458void INTERFACE_ATTRIBUTE
459AnnotateMemoryIsInitialized(char *f, int l, uptr mem, uptr sz) {}
Kostya Serebryany2bd76312014-10-22 05:02:20 +0000460void INTERFACE_ATTRIBUTE
461AnnotateMemoryIsUninitialized(char *f, int l, uptr mem, uptr sz) {}
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000462
463// Note: the parameter is called flagz, because flags is already taken
464// by the global function that returns flags.
465INTERFACE_ATTRIBUTE
466void __tsan_mutex_create(void *m, unsigned flagz) {
467 SCOPED_ANNOTATION(__tsan_mutex_create);
468 MutexCreate(thr, pc, (uptr)m, flagz & MutexCreationFlagMask);
469}
470
471INTERFACE_ATTRIBUTE
472void __tsan_mutex_destroy(void *m, unsigned flagz) {
473 SCOPED_ANNOTATION(__tsan_mutex_destroy);
Dmitry Vyukov5fa91752017-05-01 10:01:13 +0000474 MutexDestroy(thr, pc, (uptr)m, flagz);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000475}
476
477INTERFACE_ATTRIBUTE
478void __tsan_mutex_pre_lock(void *m, unsigned flagz) {
479 SCOPED_ANNOTATION(__tsan_mutex_pre_lock);
480 if (!(flagz & MutexFlagTryLock)) {
481 if (flagz & MutexFlagReadLock)
482 MutexPreReadLock(thr, pc, (uptr)m);
483 else
484 MutexPreLock(thr, pc, (uptr)m);
485 }
Kuba Mracek7cd7c1a2017-07-10 15:37:13 +0000486 ThreadIgnoreBegin(thr, pc, /*save_stack=*/false);
487 ThreadIgnoreSyncBegin(thr, pc, /*save_stack=*/false);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000488}
489
490INTERFACE_ATTRIBUTE
491void __tsan_mutex_post_lock(void *m, unsigned flagz, int rec) {
492 SCOPED_ANNOTATION(__tsan_mutex_post_lock);
493 ThreadIgnoreSyncEnd(thr, pc);
494 ThreadIgnoreEnd(thr, pc);
495 if (!(flagz & MutexFlagTryLockFailed)) {
496 if (flagz & MutexFlagReadLock)
497 MutexPostReadLock(thr, pc, (uptr)m, flagz);
498 else
499 MutexPostLock(thr, pc, (uptr)m, flagz, rec);
500 }
501}
502
503INTERFACE_ATTRIBUTE
504int __tsan_mutex_pre_unlock(void *m, unsigned flagz) {
505 SCOPED_ANNOTATION_RET(__tsan_mutex_pre_unlock, 0);
506 int ret = 0;
507 if (flagz & MutexFlagReadLock) {
508 CHECK(!(flagz & MutexFlagRecursiveUnlock));
509 MutexReadUnlock(thr, pc, (uptr)m);
510 } else {
511 ret = MutexUnlock(thr, pc, (uptr)m, flagz);
512 }
Kuba Mracek7cd7c1a2017-07-10 15:37:13 +0000513 ThreadIgnoreBegin(thr, pc, /*save_stack=*/false);
514 ThreadIgnoreSyncBegin(thr, pc, /*save_stack=*/false);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000515 return ret;
516}
517
518INTERFACE_ATTRIBUTE
519void __tsan_mutex_post_unlock(void *m, unsigned flagz) {
520 SCOPED_ANNOTATION(__tsan_mutex_post_unlock);
521 ThreadIgnoreSyncEnd(thr, pc);
522 ThreadIgnoreEnd(thr, pc);
523}
524
525INTERFACE_ATTRIBUTE
526void __tsan_mutex_pre_signal(void *addr, unsigned flagz) {
527 SCOPED_ANNOTATION(__tsan_mutex_pre_signal);
Kuba Mracek7cd7c1a2017-07-10 15:37:13 +0000528 ThreadIgnoreBegin(thr, pc, /*save_stack=*/false);
529 ThreadIgnoreSyncBegin(thr, pc, /*save_stack=*/false);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000530}
531
532INTERFACE_ATTRIBUTE
533void __tsan_mutex_post_signal(void *addr, unsigned flagz) {
534 SCOPED_ANNOTATION(__tsan_mutex_post_signal);
535 ThreadIgnoreSyncEnd(thr, pc);
536 ThreadIgnoreEnd(thr, pc);
537}
538
539INTERFACE_ATTRIBUTE
540void __tsan_mutex_pre_divert(void *addr, unsigned flagz) {
541 SCOPED_ANNOTATION(__tsan_mutex_pre_divert);
542 // Exit from ignore region started in __tsan_mutex_pre_lock/unlock/signal.
543 ThreadIgnoreSyncEnd(thr, pc);
544 ThreadIgnoreEnd(thr, pc);
545}
546
547INTERFACE_ATTRIBUTE
548void __tsan_mutex_post_divert(void *addr, unsigned flagz) {
549 SCOPED_ANNOTATION(__tsan_mutex_post_divert);
Kuba Mracek7cd7c1a2017-07-10 15:37:13 +0000550 ThreadIgnoreBegin(thr, pc, /*save_stack=*/false);
551 ThreadIgnoreSyncBegin(thr, pc, /*save_stack=*/false);
Dmitry Vyukov8096a8c2017-03-26 15:27:04 +0000552}
Kostya Serebryany4ad375f2012-05-10 13:48:04 +0000553} // extern "C"