1 /* 2 * Copyright 2012 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #include "SkTLS.h" 9 10 // enable to help debug TLS storage 11 //#define SK_TRACE_TLS_LIFETIME 12 13 14 #ifdef SK_TRACE_TLS_LIFETIME 15 #include "SkAtomics.h" 16 static int32_t gTLSRecCount; 17 #endif 18 19 struct SkTLSRec { 20 SkTLSRec* fNext; 21 void* fData; 22 SkTLS::CreateProc fCreateProc; 23 SkTLS::DeleteProc fDeleteProc; 24 25 #ifdef SK_TRACE_TLS_LIFETIME 26 SkTLSRec() { 27 int n = sk_atomic_inc(&gTLSRecCount); 28 SkDebugf(" SkTLSRec[%d]\n", n); 29 } 30 #endif 31 32 ~SkTLSRec() { 33 if (fDeleteProc) { 34 fDeleteProc(fData); 35 } 36 // else we leak fData, or it will be managed by the caller 37 38 #ifdef SK_TRACE_TLS_LIFETIME 39 int n = sk_atomic_dec(&gTLSRecCount); 40 SkDebugf("~SkTLSRec[%d]\n", n - 1); 41 #endif 42 } 43 }; 44 45 void SkTLS::Destructor(void* ptr) { 46 #ifdef SK_TRACE_TLS_LIFETIME 47 SkDebugf("SkTLS::Destructor(%p)\n", ptr); 48 #endif 49 50 SkTLSRec* rec = (SkTLSRec*)ptr; 51 do { 52 SkTLSRec* next = rec->fNext; 53 delete rec; 54 rec = next; 55 } while (rec); 56 } 57 58 void* SkTLS::Get(CreateProc createProc, DeleteProc deleteProc) { 59 if (nullptr == createProc) { 60 return nullptr; 61 } 62 63 void* ptr = SkTLS::PlatformGetSpecific(true); 64 65 if (ptr) { 66 const SkTLSRec* rec = (const SkTLSRec*)ptr; 67 do { 68 if (rec->fCreateProc == createProc) { 69 SkASSERT(rec->fDeleteProc == deleteProc); 70 return rec->fData; 71 } 72 } while ((rec = rec->fNext) != nullptr); 73 // not found, so create a new one 74 } 75 76 // add a new head of our change 77 SkTLSRec* rec = new SkTLSRec; 78 rec->fNext = (SkTLSRec*)ptr; 79 80 SkTLS::PlatformSetSpecific(rec); 81 82 rec->fData = createProc(); 83 rec->fCreateProc = createProc; 84 rec->fDeleteProc = deleteProc; 85 return rec->fData; 86 } 87 88 void* SkTLS::Find(CreateProc createProc) { 89 if (nullptr == createProc) { 90 return nullptr; 91 } 92 93 void* ptr = SkTLS::PlatformGetSpecific(false); 94 95 if (ptr) { 96 const SkTLSRec* rec = (const SkTLSRec*)ptr; 97 do { 98 if (rec->fCreateProc == createProc) { 99 return rec->fData; 100 } 101 } while ((rec = rec->fNext) != nullptr); 102 } 103 return nullptr; 104 } 105 106 void SkTLS::Delete(CreateProc createProc) { 107 if (nullptr == createProc) { 108 return; 109 } 110 111 void* ptr = SkTLS::PlatformGetSpecific(false); 112 113 SkTLSRec* curr = (SkTLSRec*)ptr; 114 SkTLSRec* prev = nullptr; 115 while (curr) { 116 SkTLSRec* next = curr->fNext; 117 if (curr->fCreateProc == createProc) { 118 if (prev) { 119 prev->fNext = next; 120 } else { 121 // we have a new head of our chain 122 SkTLS::PlatformSetSpecific(next); 123 } 124 delete curr; 125 break; 126 } 127 prev = curr; 128 curr = next; 129 } 130 } 131