2012-05-07 20:23:27 +00:00
|
|
|
#include "SkTLS.h"
|
|
|
|
|
2012-05-21 15:29:27 +00:00
|
|
|
// enable to help debug TLS storage
|
|
|
|
//#define SK_TRACE_TLS_LIFETIME
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SK_TRACE_TLS_LIFETIME
|
|
|
|
#include "SkThread.h"
|
|
|
|
static int32_t gTLSRecCount;
|
|
|
|
#endif
|
|
|
|
|
2012-05-07 20:23:27 +00:00
|
|
|
struct SkTLSRec {
|
|
|
|
SkTLSRec* fNext;
|
|
|
|
void* fData;
|
|
|
|
SkTLS::CreateProc fCreateProc;
|
|
|
|
SkTLS::DeleteProc fDeleteProc;
|
|
|
|
|
2012-05-21 15:29:27 +00:00
|
|
|
#ifdef SK_TRACE_TLS_LIFETIME
|
|
|
|
SkTLSRec() {
|
|
|
|
int n = sk_atomic_inc(&gTLSRecCount);
|
|
|
|
SkDebugf(" SkTLSRec[%d]\n", n);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-05-07 20:23:27 +00:00
|
|
|
~SkTLSRec() {
|
|
|
|
if (fDeleteProc) {
|
|
|
|
fDeleteProc(fData);
|
|
|
|
}
|
|
|
|
// else we leak fData, or it will be managed by the caller
|
2012-05-21 15:29:27 +00:00
|
|
|
|
|
|
|
#ifdef SK_TRACE_TLS_LIFETIME
|
|
|
|
int n = sk_atomic_dec(&gTLSRecCount);
|
|
|
|
SkDebugf("~SkTLSRec[%d]\n", n - 1);
|
|
|
|
#endif
|
2012-05-07 20:23:27 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void SkTLS::Destructor(void* ptr) {
|
2012-05-21 15:29:27 +00:00
|
|
|
#ifdef SK_TRACE_TLS_LIFETIME
|
|
|
|
SkDebugf("SkTLS::Destructor(%p)\n", ptr);
|
|
|
|
#endif
|
|
|
|
|
2012-05-07 20:23:27 +00:00
|
|
|
SkTLSRec* rec = (SkTLSRec*)ptr;
|
|
|
|
do {
|
|
|
|
SkTLSRec* next = rec->fNext;
|
|
|
|
SkDELETE(rec);
|
|
|
|
rec = next;
|
|
|
|
} while (NULL != rec);
|
|
|
|
}
|
|
|
|
|
|
|
|
void* SkTLS::Get(CreateProc createProc, DeleteProc deleteProc) {
|
|
|
|
if (NULL == createProc) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-05-08 21:45:03 +00:00
|
|
|
void* ptr = SkTLS::PlatformGetSpecific(true);
|
2012-05-07 20:23:27 +00:00
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
const SkTLSRec* rec = (const SkTLSRec*)ptr;
|
|
|
|
do {
|
|
|
|
if (rec->fCreateProc == createProc) {
|
|
|
|
SkASSERT(rec->fDeleteProc == deleteProc);
|
|
|
|
return rec->fData;
|
|
|
|
}
|
|
|
|
} while ((rec = rec->fNext) != NULL);
|
|
|
|
// not found, so create a new one
|
|
|
|
}
|
2012-08-23 18:09:54 +00:00
|
|
|
|
2012-05-07 20:23:27 +00:00
|
|
|
// add a new head of our change
|
|
|
|
SkTLSRec* rec = new SkTLSRec;
|
|
|
|
rec->fNext = (SkTLSRec*)ptr;
|
|
|
|
|
|
|
|
SkTLS::PlatformSetSpecific(rec);
|
|
|
|
|
|
|
|
rec->fData = createProc();
|
|
|
|
rec->fCreateProc = createProc;
|
|
|
|
rec->fDeleteProc = deleteProc;
|
|
|
|
return rec->fData;
|
|
|
|
}
|
|
|
|
|
|
|
|
void* SkTLS::Find(CreateProc createProc) {
|
|
|
|
if (NULL == createProc) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-08-23 18:09:54 +00:00
|
|
|
|
2012-05-08 21:45:03 +00:00
|
|
|
void* ptr = SkTLS::PlatformGetSpecific(false);
|
2012-05-07 20:23:27 +00:00
|
|
|
|
|
|
|
if (ptr) {
|
|
|
|
const SkTLSRec* rec = (const SkTLSRec*)ptr;
|
|
|
|
do {
|
|
|
|
if (rec->fCreateProc == createProc) {
|
|
|
|
return rec->fData;
|
|
|
|
}
|
|
|
|
} while ((rec = rec->fNext) != NULL);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkTLS::Delete(CreateProc createProc) {
|
|
|
|
if (NULL == createProc) {
|
|
|
|
return;
|
|
|
|
}
|
2012-08-23 18:09:54 +00:00
|
|
|
|
2012-05-08 21:45:03 +00:00
|
|
|
void* ptr = SkTLS::PlatformGetSpecific(false);
|
2012-08-23 18:09:54 +00:00
|
|
|
|
2012-05-07 20:23:27 +00:00
|
|
|
SkTLSRec* curr = (SkTLSRec*)ptr;
|
|
|
|
SkTLSRec* prev = NULL;
|
|
|
|
while (curr) {
|
|
|
|
SkTLSRec* next = curr->fNext;
|
|
|
|
if (curr->fCreateProc == createProc) {
|
|
|
|
if (prev) {
|
|
|
|
prev->fNext = next;
|
|
|
|
} else {
|
|
|
|
// we have a new head of our chain
|
|
|
|
SkTLS::PlatformSetSpecific(next);
|
|
|
|
}
|
|
|
|
SkDELETE(curr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
prev = curr;
|
|
|
|
curr = next;
|
|
|
|
}
|
|
|
|
}
|