forked from OSchip/llvm-project
446 lines
14 KiB
C++
446 lines
14 KiB
C++
//===-- tsan_rtl_mutex.cc -------------------------------------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file is a part of ThreadSanitizer (TSan), a race detector.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include <sanitizer_common/sanitizer_deadlock_detector_interface.h>
|
|
#include <sanitizer_common/sanitizer_stackdepot.h>
|
|
|
|
#include "tsan_rtl.h"
|
|
#include "tsan_flags.h"
|
|
#include "tsan_sync.h"
|
|
#include "tsan_report.h"
|
|
#include "tsan_symbolize.h"
|
|
#include "tsan_platform.h"
|
|
|
|
namespace __tsan {
|
|
|
|
void ReportDeadlock(ThreadState *thr, uptr pc, DDReport *r);
|
|
|
|
struct Callback : DDCallback {
|
|
ThreadState *thr;
|
|
uptr pc;
|
|
|
|
Callback(ThreadState *thr, uptr pc)
|
|
: thr(thr)
|
|
, pc(pc) {
|
|
DDCallback::pt = thr->dd_pt;
|
|
DDCallback::lt = thr->dd_lt;
|
|
}
|
|
|
|
virtual u32 Unwind() {
|
|
#ifdef TSAN_GO
|
|
return 0;
|
|
#else
|
|
return CurrentStackId(thr, pc);
|
|
#endif
|
|
}
|
|
};
|
|
|
|
void DDMutexInit(ThreadState *thr, uptr pc, SyncVar *s) {
|
|
Callback cb(thr, pc);
|
|
CTX()->dd->MutexInit(&cb, &s->dd);
|
|
s->dd.ctx = s->GetId();
|
|
}
|
|
|
|
void MutexCreate(ThreadState *thr, uptr pc, uptr addr,
|
|
bool rw, bool recursive, bool linker_init) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexCreate %zx\n", thr->tid, addr);
|
|
StatInc(thr, StatMutexCreate);
|
|
if (!linker_init && IsAppMem(addr)) {
|
|
CHECK(!thr->is_freeing);
|
|
thr->is_freeing = true;
|
|
MemoryWrite(thr, pc, addr, kSizeLog1);
|
|
thr->is_freeing = false;
|
|
}
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
s->is_rw = rw;
|
|
s->is_recursive = recursive;
|
|
s->is_linker_init = linker_init;
|
|
s->mtx.Unlock();
|
|
}
|
|
|
|
void MutexDestroy(ThreadState *thr, uptr pc, uptr addr) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexDestroy %zx\n", thr->tid, addr);
|
|
StatInc(thr, StatMutexDestroy);
|
|
#ifndef TSAN_GO
|
|
// Global mutexes not marked as LINKER_INITIALIZED
|
|
// cause tons of not interesting reports, so just ignore it.
|
|
if (IsGlobalVar(addr))
|
|
return;
|
|
#endif
|
|
SyncVar *s = ctx->synctab.GetAndRemove(thr, pc, addr);
|
|
if (s == 0)
|
|
return;
|
|
if (flags()->detect_deadlocks) {
|
|
Callback cb(thr, pc);
|
|
ctx->dd->MutexDestroy(&cb, &s->dd);
|
|
}
|
|
if (IsAppMem(addr)) {
|
|
CHECK(!thr->is_freeing);
|
|
thr->is_freeing = true;
|
|
MemoryWrite(thr, pc, addr, kSizeLog1);
|
|
thr->is_freeing = false;
|
|
}
|
|
if (flags()->report_destroy_locked
|
|
&& s->owner_tid != SyncVar::kInvalidTid
|
|
&& !s->is_broken) {
|
|
s->is_broken = true;
|
|
ThreadRegistryLock l(ctx->thread_registry);
|
|
ScopedReport rep(ReportTypeMutexDestroyLocked);
|
|
rep.AddMutex(s);
|
|
StackTrace trace;
|
|
trace.ObtainCurrent(thr, pc);
|
|
rep.AddStack(&trace);
|
|
FastState last(s->last_lock);
|
|
RestoreStack(last.tid(), last.epoch(), &trace, 0);
|
|
rep.AddStack(&trace);
|
|
rep.AddLocation(s->addr, 1);
|
|
OutputReport(ctx, rep);
|
|
}
|
|
thr->mset.Remove(s->GetId());
|
|
DestroyAndFree(s);
|
|
}
|
|
|
|
void MutexLock(ThreadState *thr, uptr pc, uptr addr, int rec, bool try_lock) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexLock %zx rec=%d\n", thr->tid, addr, rec);
|
|
CHECK_GT(rec, 0);
|
|
if (IsAppMem(addr))
|
|
MemoryReadAtomic(thr, pc, addr, kSizeLog1);
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
thr->fast_state.IncrementEpoch();
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeLock, s->GetId());
|
|
if (s->owner_tid == SyncVar::kInvalidTid) {
|
|
CHECK_EQ(s->recursion, 0);
|
|
s->owner_tid = thr->tid;
|
|
s->last_lock = thr->fast_state.raw();
|
|
} else if (s->owner_tid == thr->tid) {
|
|
CHECK_GT(s->recursion, 0);
|
|
} else {
|
|
Printf("ThreadSanitizer WARNING: double lock of mutex %p\n", addr);
|
|
PrintCurrentStack(thr, pc);
|
|
}
|
|
if (s->recursion == 0) {
|
|
StatInc(thr, StatMutexLock);
|
|
AcquireImpl(thr, pc, &s->clock);
|
|
AcquireImpl(thr, pc, &s->read_clock);
|
|
} else if (!s->is_recursive) {
|
|
StatInc(thr, StatMutexRecLock);
|
|
}
|
|
s->recursion += rec;
|
|
thr->mset.Add(s->GetId(), true, thr->fast_state.epoch());
|
|
if (flags()->detect_deadlocks && s->recursion == 1) {
|
|
Callback cb(thr, pc);
|
|
if (!try_lock)
|
|
ctx->dd->MutexBeforeLock(&cb, &s->dd, true);
|
|
ctx->dd->MutexAfterLock(&cb, &s->dd, true, try_lock);
|
|
}
|
|
s->mtx.Unlock();
|
|
if (flags()->detect_deadlocks) {
|
|
Callback cb(thr, pc);
|
|
ReportDeadlock(thr, pc, ctx->dd->GetReport(&cb));
|
|
}
|
|
}
|
|
|
|
int MutexUnlock(ThreadState *thr, uptr pc, uptr addr, bool all) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexUnlock %zx all=%d\n", thr->tid, addr, all);
|
|
if (IsAppMem(addr))
|
|
MemoryReadAtomic(thr, pc, addr, kSizeLog1);
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
thr->fast_state.IncrementEpoch();
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeUnlock, s->GetId());
|
|
int rec = 0;
|
|
if (s->recursion == 0) {
|
|
if (!s->is_broken) {
|
|
s->is_broken = true;
|
|
Printf("ThreadSanitizer WARNING: unlock of unlocked mutex %p\n", addr);
|
|
PrintCurrentStack(thr, pc);
|
|
}
|
|
} else if (s->owner_tid != thr->tid) {
|
|
if (!s->is_broken) {
|
|
s->is_broken = true;
|
|
Printf("ThreadSanitizer WARNING: mutex %p is unlocked by wrong thread\n",
|
|
addr);
|
|
PrintCurrentStack(thr, pc);
|
|
}
|
|
} else {
|
|
rec = all ? s->recursion : 1;
|
|
s->recursion -= rec;
|
|
if (s->recursion == 0) {
|
|
StatInc(thr, StatMutexUnlock);
|
|
s->owner_tid = SyncVar::kInvalidTid;
|
|
ReleaseStoreImpl(thr, pc, &s->clock);
|
|
} else {
|
|
StatInc(thr, StatMutexRecUnlock);
|
|
}
|
|
}
|
|
thr->mset.Del(s->GetId(), true);
|
|
if (flags()->detect_deadlocks && s->recursion == 0) {
|
|
Callback cb(thr, pc);
|
|
ctx->dd->MutexBeforeUnlock(&cb, &s->dd, true);
|
|
}
|
|
s->mtx.Unlock();
|
|
if (flags()->detect_deadlocks) {
|
|
Callback cb(thr, pc);
|
|
ReportDeadlock(thr, pc, ctx->dd->GetReport(&cb));
|
|
}
|
|
return rec;
|
|
}
|
|
|
|
void MutexReadLock(ThreadState *thr, uptr pc, uptr addr, bool trylock) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexReadLock %zx\n", thr->tid, addr);
|
|
StatInc(thr, StatMutexReadLock);
|
|
if (IsAppMem(addr))
|
|
MemoryReadAtomic(thr, pc, addr, kSizeLog1);
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, false);
|
|
thr->fast_state.IncrementEpoch();
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeRLock, s->GetId());
|
|
if (s->owner_tid != SyncVar::kInvalidTid) {
|
|
Printf("ThreadSanitizer WARNING: read lock of a write locked mutex %p\n",
|
|
addr);
|
|
PrintCurrentStack(thr, pc);
|
|
}
|
|
AcquireImpl(thr, pc, &s->clock);
|
|
s->last_lock = thr->fast_state.raw();
|
|
thr->mset.Add(s->GetId(), false, thr->fast_state.epoch());
|
|
if (flags()->detect_deadlocks && s->recursion == 0) {
|
|
Callback cb(thr, pc);
|
|
if (!trylock)
|
|
ctx->dd->MutexBeforeLock(&cb, &s->dd, false);
|
|
ctx->dd->MutexAfterLock(&cb, &s->dd, false, trylock);
|
|
}
|
|
s->mtx.ReadUnlock();
|
|
if (flags()->detect_deadlocks) {
|
|
Callback cb(thr, pc);
|
|
ReportDeadlock(thr, pc, ctx->dd->GetReport(&cb));
|
|
}
|
|
}
|
|
|
|
void MutexReadUnlock(ThreadState *thr, uptr pc, uptr addr) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexReadUnlock %zx\n", thr->tid, addr);
|
|
StatInc(thr, StatMutexReadUnlock);
|
|
if (IsAppMem(addr))
|
|
MemoryReadAtomic(thr, pc, addr, kSizeLog1);
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
thr->fast_state.IncrementEpoch();
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeRUnlock, s->GetId());
|
|
if (s->owner_tid != SyncVar::kInvalidTid) {
|
|
Printf("ThreadSanitizer WARNING: read unlock of a write locked mutex %p\n",
|
|
addr);
|
|
PrintCurrentStack(thr, pc);
|
|
}
|
|
ReleaseImpl(thr, pc, &s->read_clock);
|
|
if (flags()->detect_deadlocks && s->recursion == 0) {
|
|
Callback cb(thr, pc);
|
|
ctx->dd->MutexBeforeUnlock(&cb, &s->dd, false);
|
|
}
|
|
s->mtx.Unlock();
|
|
thr->mset.Del(s->GetId(), false);
|
|
if (flags()->detect_deadlocks) {
|
|
Callback cb(thr, pc);
|
|
ReportDeadlock(thr, pc, ctx->dd->GetReport(&cb));
|
|
}
|
|
}
|
|
|
|
void MutexReadOrWriteUnlock(ThreadState *thr, uptr pc, uptr addr) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexReadOrWriteUnlock %zx\n", thr->tid, addr);
|
|
if (IsAppMem(addr))
|
|
MemoryReadAtomic(thr, pc, addr, kSizeLog1);
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
bool write = true;
|
|
if (s->owner_tid == SyncVar::kInvalidTid) {
|
|
// Seems to be read unlock.
|
|
write = false;
|
|
StatInc(thr, StatMutexReadUnlock);
|
|
thr->fast_state.IncrementEpoch();
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeRUnlock, s->GetId());
|
|
ReleaseImpl(thr, pc, &s->read_clock);
|
|
} else if (s->owner_tid == thr->tid) {
|
|
// Seems to be write unlock.
|
|
thr->fast_state.IncrementEpoch();
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeUnlock, s->GetId());
|
|
CHECK_GT(s->recursion, 0);
|
|
s->recursion--;
|
|
if (s->recursion == 0) {
|
|
StatInc(thr, StatMutexUnlock);
|
|
s->owner_tid = SyncVar::kInvalidTid;
|
|
ReleaseImpl(thr, pc, &s->clock);
|
|
} else {
|
|
StatInc(thr, StatMutexRecUnlock);
|
|
}
|
|
} else if (!s->is_broken) {
|
|
s->is_broken = true;
|
|
Printf("ThreadSanitizer WARNING: mutex %p is unlock by wrong thread\n",
|
|
addr);
|
|
PrintCurrentStack(thr, pc);
|
|
}
|
|
thr->mset.Del(s->GetId(), write);
|
|
if (flags()->detect_deadlocks && s->recursion == 0) {
|
|
Callback cb(thr, pc);
|
|
ctx->dd->MutexBeforeUnlock(&cb, &s->dd, write);
|
|
}
|
|
s->mtx.Unlock();
|
|
if (flags()->detect_deadlocks) {
|
|
Callback cb(thr, pc);
|
|
ReportDeadlock(thr, pc, ctx->dd->GetReport(&cb));
|
|
}
|
|
}
|
|
|
|
void MutexRepair(ThreadState *thr, uptr pc, uptr addr) {
|
|
Context *ctx = CTX();
|
|
DPrintf("#%d: MutexRepair %zx\n", thr->tid, addr);
|
|
SyncVar *s = ctx->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
s->owner_tid = SyncVar::kInvalidTid;
|
|
s->recursion = 0;
|
|
s->mtx.Unlock();
|
|
}
|
|
|
|
void Acquire(ThreadState *thr, uptr pc, uptr addr) {
|
|
DPrintf("#%d: Acquire %zx\n", thr->tid, addr);
|
|
if (thr->ignore_sync)
|
|
return;
|
|
SyncVar *s = CTX()->synctab.GetOrCreateAndLock(thr, pc, addr, false);
|
|
AcquireImpl(thr, pc, &s->clock);
|
|
s->mtx.ReadUnlock();
|
|
}
|
|
|
|
static void UpdateClockCallback(ThreadContextBase *tctx_base, void *arg) {
|
|
ThreadState *thr = reinterpret_cast<ThreadState*>(arg);
|
|
ThreadContext *tctx = static_cast<ThreadContext*>(tctx_base);
|
|
if (tctx->status == ThreadStatusRunning)
|
|
thr->clock.set(tctx->tid, tctx->thr->fast_state.epoch());
|
|
else
|
|
thr->clock.set(tctx->tid, tctx->epoch1);
|
|
}
|
|
|
|
void AcquireGlobal(ThreadState *thr, uptr pc) {
|
|
DPrintf("#%d: AcquireGlobal\n", thr->tid);
|
|
if (thr->ignore_sync)
|
|
return;
|
|
ThreadRegistryLock l(CTX()->thread_registry);
|
|
CTX()->thread_registry->RunCallbackForEachThreadLocked(
|
|
UpdateClockCallback, thr);
|
|
}
|
|
|
|
void Release(ThreadState *thr, uptr pc, uptr addr) {
|
|
DPrintf("#%d: Release %zx\n", thr->tid, addr);
|
|
if (thr->ignore_sync)
|
|
return;
|
|
SyncVar *s = CTX()->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
thr->fast_state.IncrementEpoch();
|
|
// Can't increment epoch w/o writing to the trace as well.
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeMop, 0);
|
|
ReleaseImpl(thr, pc, &s->clock);
|
|
s->mtx.Unlock();
|
|
}
|
|
|
|
void ReleaseStore(ThreadState *thr, uptr pc, uptr addr) {
|
|
DPrintf("#%d: ReleaseStore %zx\n", thr->tid, addr);
|
|
if (thr->ignore_sync)
|
|
return;
|
|
SyncVar *s = CTX()->synctab.GetOrCreateAndLock(thr, pc, addr, true);
|
|
thr->fast_state.IncrementEpoch();
|
|
// Can't increment epoch w/o writing to the trace as well.
|
|
TraceAddEvent(thr, thr->fast_state, EventTypeMop, 0);
|
|
ReleaseStoreImpl(thr, pc, &s->clock);
|
|
s->mtx.Unlock();
|
|
}
|
|
|
|
#ifndef TSAN_GO
|
|
static void UpdateSleepClockCallback(ThreadContextBase *tctx_base, void *arg) {
|
|
ThreadState *thr = reinterpret_cast<ThreadState*>(arg);
|
|
ThreadContext *tctx = static_cast<ThreadContext*>(tctx_base);
|
|
if (tctx->status == ThreadStatusRunning)
|
|
thr->last_sleep_clock.set(tctx->tid, tctx->thr->fast_state.epoch());
|
|
else
|
|
thr->last_sleep_clock.set(tctx->tid, tctx->epoch1);
|
|
}
|
|
|
|
void AfterSleep(ThreadState *thr, uptr pc) {
|
|
DPrintf("#%d: AfterSleep %zx\n", thr->tid);
|
|
if (thr->ignore_sync)
|
|
return;
|
|
thr->last_sleep_stack_id = CurrentStackId(thr, pc);
|
|
ThreadRegistryLock l(CTX()->thread_registry);
|
|
CTX()->thread_registry->RunCallbackForEachThreadLocked(
|
|
UpdateSleepClockCallback, thr);
|
|
}
|
|
#endif
|
|
|
|
void AcquireImpl(ThreadState *thr, uptr pc, SyncClock *c) {
|
|
if (thr->ignore_sync)
|
|
return;
|
|
thr->clock.set(thr->tid, thr->fast_state.epoch());
|
|
thr->clock.acquire(c);
|
|
StatInc(thr, StatSyncAcquire);
|
|
}
|
|
|
|
void ReleaseImpl(ThreadState *thr, uptr pc, SyncClock *c) {
|
|
if (thr->ignore_sync)
|
|
return;
|
|
thr->clock.set(thr->tid, thr->fast_state.epoch());
|
|
thr->fast_synch_epoch = thr->fast_state.epoch();
|
|
thr->clock.release(c);
|
|
StatInc(thr, StatSyncRelease);
|
|
}
|
|
|
|
void ReleaseStoreImpl(ThreadState *thr, uptr pc, SyncClock *c) {
|
|
if (thr->ignore_sync)
|
|
return;
|
|
thr->clock.set(thr->tid, thr->fast_state.epoch());
|
|
thr->fast_synch_epoch = thr->fast_state.epoch();
|
|
thr->clock.ReleaseStore(c);
|
|
StatInc(thr, StatSyncRelease);
|
|
}
|
|
|
|
void AcquireReleaseImpl(ThreadState *thr, uptr pc, SyncClock *c) {
|
|
if (thr->ignore_sync)
|
|
return;
|
|
thr->clock.set(thr->tid, thr->fast_state.epoch());
|
|
thr->fast_synch_epoch = thr->fast_state.epoch();
|
|
thr->clock.acq_rel(c);
|
|
StatInc(thr, StatSyncAcquire);
|
|
StatInc(thr, StatSyncRelease);
|
|
}
|
|
|
|
void ReportDeadlock(ThreadState *thr, uptr pc, DDReport *r) {
|
|
#ifndef TSAN_GO
|
|
if (r == 0)
|
|
return;
|
|
Context *ctx = CTX();
|
|
ThreadRegistryLock l(ctx->thread_registry);
|
|
ScopedReport rep(ReportTypeDeadlock);
|
|
for (int i = 0; i < r->n; i++)
|
|
rep.AddMutex(r->loop[i].mtx_ctx0);
|
|
StackTrace stacks[2 * DDReport::kMaxLoopSize];
|
|
for (int i = 0; i < r->n; i++) {
|
|
uptr size;
|
|
for (int j = 0; j < 2; j++) {
|
|
u32 stk = r->loop[i].stk[j];
|
|
if (!stk) continue;
|
|
const uptr *trace = StackDepotGet(stk, &size);
|
|
stacks[i].Init(const_cast<uptr *>(trace), size);
|
|
rep.AddStack(&stacks[i]);
|
|
}
|
|
}
|
|
OutputReport(ctx, rep);
|
|
#endif // TSAN_GO
|
|
}
|
|
|
|
} // namespace __tsan
|