/* * Copyright 2008 The Android Open Source Project * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include #include #include "SkThread.h" #include "SkTLS.h" //MSDN says in order to declare an interlocked function for use as an //intrinsic, include intrin.h and put the function in a #pragma intrinsic //directive. //The pragma appears to be unnecessary, but doesn't hurt. #pragma intrinsic(_InterlockedIncrement, _InterlockedExchangeAdd, _InterlockedDecrement) #pragma intrinsic(_InterlockedCompareExchange) int32_t sk_atomic_inc(int32_t* addr) { // InterlockedIncrement returns the new value, we want to return the old. return _InterlockedIncrement(reinterpret_cast(addr)) - 1; } int32_t sk_atomic_add(int32_t* addr, int32_t inc) { return _InterlockedExchangeAdd(reinterpret_cast(addr), static_cast(inc)); } int32_t sk_atomic_dec(int32_t* addr) { return _InterlockedDecrement(reinterpret_cast(addr)) + 1; } void sk_membar_aquire__after_atomic_dec() { } int32_t sk_atomic_conditional_inc(int32_t* addr) { while (true) { LONG value = static_cast(*addr); if (value == 0) { return 0; } if (_InterlockedCompareExchange(reinterpret_cast(addr), value + 1, value) == value) { return value; } } } void sk_membar_aquire__after_atomic_conditional_inc() { } SkMutex::SkMutex() { SK_COMPILE_ASSERT(sizeof(fStorage) > sizeof(CRITICAL_SECTION), NotEnoughSizeForCriticalSection); InitializeCriticalSection(reinterpret_cast(&fStorage)); } SkMutex::~SkMutex() { DeleteCriticalSection(reinterpret_cast(&fStorage)); } void SkMutex::acquire() { EnterCriticalSection(reinterpret_cast(&fStorage)); } void SkMutex::release() { LeaveCriticalSection(reinterpret_cast(&fStorage)); } /////////////////////////////////////////////////////////////////////////// static bool gOnce; static DWORD gTlsIndex; SK_DECLARE_STATIC_MUTEX(gMutex); void* SkTLS::PlatformGetSpecific(bool forceCreateTheSlot) { if (!forceCreateTheSlot && !gOnce) { return NULL; } if (!gOnce) { SkAutoMutexAcquire tmp(gMutex); if (!gOnce) { gTlsIndex = TlsAlloc(); gOnce = true; } } return TlsGetValue(gTlsIndex); } void SkTLS::PlatformSetSpecific(void* ptr) { SkASSERT(gOnce); (void)TlsSetValue(gTlsIndex, ptr); }