blob: f612bb5fca5e6055834bc24cb9b5cfd52c11d2c9 [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef GrSingleOwner_DEFINED
#define GrSingleOwner_DEFINED
#include "include/core/SkTypes.h"
#ifdef SK_DEBUG
#include "include/private/SkMutex.h"
#include "include/private/SkThreadID.h"
#define GR_ASSERT_SINGLE_OWNER(obj) \
GrSingleOwner::AutoEnforce debug_SingleOwner(obj, __FILE__, __LINE__);
// This is a debug tool to verify an object is only being used from one thread at a time.
class GrSingleOwner {
public:
GrSingleOwner() : fOwner(kIllegalThreadID), fReentranceCount(0) {}
struct AutoEnforce {
AutoEnforce(GrSingleOwner* so, const char* file, int line)
: fFile(file), fLine(line), fSO(so) {
fSO->enter(file, line);
}
~AutoEnforce() { fSO->exit(fFile, fLine); }
const char* fFile;
int fLine;
GrSingleOwner* fSO;
};
private:
void enter(const char* file, int line) {
SkAutoMutexExclusive lock(fMutex);
SkThreadID self = SkGetThreadID();
SkASSERTF(fOwner == self || fOwner == kIllegalThreadID, "%s:%d Single owner failure.",
file, line);
fReentranceCount++;
fOwner = self;
}
void exit(const char* file, int line) {
SkAutoMutexExclusive lock(fMutex);
SkASSERTF(fOwner == SkGetThreadID(), "%s:%d Single owner failure.", file, line);
fReentranceCount--;
if (fReentranceCount == 0) {
fOwner = kIllegalThreadID;
}
}
SkMutex fMutex;
SkThreadID fOwner SK_GUARDED_BY(fMutex);
int fReentranceCount SK_GUARDED_BY(fMutex);
};
#else
#define GR_ASSERT_SINGLE_OWNER(obj)
class GrSingleOwner {}; // Provide a no-op implementation so we can pass pointers to constructors
#endif
#endif