1*c8dee2aaSAndroid Build Coastguard Worker /* 2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2015 Google Inc. 3*c8dee2aaSAndroid Build Coastguard Worker * 4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be 5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file. 6*c8dee2aaSAndroid Build Coastguard Worker */ 7*c8dee2aaSAndroid Build Coastguard Worker 8*c8dee2aaSAndroid Build Coastguard Worker #ifndef SkSharedLock_DEFINED 9*c8dee2aaSAndroid Build Coastguard Worker #define SkSharedLock_DEFINED 10*c8dee2aaSAndroid Build Coastguard Worker 11*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkDebug.h" 12*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkSemaphore.h" 13*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkThreadAnnotations.h" 14*c8dee2aaSAndroid Build Coastguard Worker 15*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_DEBUG 16*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkMutex.h" 17*c8dee2aaSAndroid Build Coastguard Worker #include <memory> 18*c8dee2aaSAndroid Build Coastguard Worker #endif // SK_DEBUG 19*c8dee2aaSAndroid Build Coastguard Worker 20*c8dee2aaSAndroid Build Coastguard Worker // There are two shared lock implementations one debug the other is high performance. They implement 21*c8dee2aaSAndroid Build Coastguard Worker // an interface similar to pthread's rwlocks. 22*c8dee2aaSAndroid Build Coastguard Worker // This is a shared lock implementation similar to pthreads rwlocks. The high performance 23*c8dee2aaSAndroid Build Coastguard Worker // implementation is cribbed from Preshing's article: 24*c8dee2aaSAndroid Build Coastguard Worker // http://preshing.com/20150316/semaphores-are-surprisingly-versatile/ 25*c8dee2aaSAndroid Build Coastguard Worker // 26*c8dee2aaSAndroid Build Coastguard Worker // This lock does not obey strict queue ordering. It will always alternate between readers and 27*c8dee2aaSAndroid Build Coastguard Worker // a single writer. 28*c8dee2aaSAndroid Build Coastguard Worker class SK_CAPABILITY("mutex") SkSharedMutex { 29*c8dee2aaSAndroid Build Coastguard Worker public: 30*c8dee2aaSAndroid Build Coastguard Worker SkSharedMutex(); 31*c8dee2aaSAndroid Build Coastguard Worker ~SkSharedMutex(); 32*c8dee2aaSAndroid Build Coastguard Worker // Acquire lock for exclusive use. 33*c8dee2aaSAndroid Build Coastguard Worker void acquire() SK_ACQUIRE(); 34*c8dee2aaSAndroid Build Coastguard Worker 35*c8dee2aaSAndroid Build Coastguard Worker // Release lock for exclusive use. 36*c8dee2aaSAndroid Build Coastguard Worker void release() SK_RELEASE_CAPABILITY(); 37*c8dee2aaSAndroid Build Coastguard Worker 38*c8dee2aaSAndroid Build Coastguard Worker // Fail if exclusive is not held. 39*c8dee2aaSAndroid Build Coastguard Worker void assertHeld() const SK_ASSERT_CAPABILITY(this); 40*c8dee2aaSAndroid Build Coastguard Worker 41*c8dee2aaSAndroid Build Coastguard Worker // Acquire lock for shared use. 42*c8dee2aaSAndroid Build Coastguard Worker void acquireShared() SK_ACQUIRE_SHARED(); 43*c8dee2aaSAndroid Build Coastguard Worker 44*c8dee2aaSAndroid Build Coastguard Worker // Release lock for shared use. 45*c8dee2aaSAndroid Build Coastguard Worker void releaseShared() SK_RELEASE_SHARED_CAPABILITY(); 46*c8dee2aaSAndroid Build Coastguard Worker 47*c8dee2aaSAndroid Build Coastguard Worker // Fail if shared lock not held. 48*c8dee2aaSAndroid Build Coastguard Worker void assertHeldShared() const SK_ASSERT_SHARED_CAPABILITY(this); 49*c8dee2aaSAndroid Build Coastguard Worker 50*c8dee2aaSAndroid Build Coastguard Worker private: 51*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_DEBUG 52*c8dee2aaSAndroid Build Coastguard Worker class ThreadIDSet; 53*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<ThreadIDSet> fCurrentShared; 54*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<ThreadIDSet> fWaitingExclusive; 55*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<ThreadIDSet> fWaitingShared; 56*c8dee2aaSAndroid Build Coastguard Worker int fSharedQueueSelect{0}; 57*c8dee2aaSAndroid Build Coastguard Worker mutable SkMutex fMu; 58*c8dee2aaSAndroid Build Coastguard Worker SkSemaphore fSharedQueue[2]; 59*c8dee2aaSAndroid Build Coastguard Worker SkSemaphore fExclusiveQueue; 60*c8dee2aaSAndroid Build Coastguard Worker #else 61*c8dee2aaSAndroid Build Coastguard Worker std::atomic<int32_t> fQueueCounts; 62*c8dee2aaSAndroid Build Coastguard Worker SkSemaphore fSharedQueue; 63*c8dee2aaSAndroid Build Coastguard Worker SkSemaphore fExclusiveQueue; 64*c8dee2aaSAndroid Build Coastguard Worker #endif // SK_DEBUG 65*c8dee2aaSAndroid Build Coastguard Worker }; 66*c8dee2aaSAndroid Build Coastguard Worker 67*c8dee2aaSAndroid Build Coastguard Worker #ifndef SK_DEBUG assertHeld()68*c8dee2aaSAndroid Build Coastguard Workerinline void SkSharedMutex::assertHeld() const {} assertHeldShared()69*c8dee2aaSAndroid Build Coastguard Workerinline void SkSharedMutex::assertHeldShared() const {} 70*c8dee2aaSAndroid Build Coastguard Worker #endif // SK_DEBUG 71*c8dee2aaSAndroid Build Coastguard Worker 72*c8dee2aaSAndroid Build Coastguard Worker class SK_SCOPED_CAPABILITY SkAutoSharedMutexExclusive { 73*c8dee2aaSAndroid Build Coastguard Worker public: SkAutoSharedMutexExclusive(SkSharedMutex & lock)74*c8dee2aaSAndroid Build Coastguard Worker explicit SkAutoSharedMutexExclusive(SkSharedMutex& lock) SK_ACQUIRE(lock) 75*c8dee2aaSAndroid Build Coastguard Worker : fLock(lock) { 76*c8dee2aaSAndroid Build Coastguard Worker lock.acquire(); 77*c8dee2aaSAndroid Build Coastguard Worker } SK_RELEASE_CAPABILITY()78*c8dee2aaSAndroid Build Coastguard Worker ~SkAutoSharedMutexExclusive() SK_RELEASE_CAPABILITY() { fLock.release(); } 79*c8dee2aaSAndroid Build Coastguard Worker 80*c8dee2aaSAndroid Build Coastguard Worker private: 81*c8dee2aaSAndroid Build Coastguard Worker SkSharedMutex& fLock; 82*c8dee2aaSAndroid Build Coastguard Worker }; 83*c8dee2aaSAndroid Build Coastguard Worker 84*c8dee2aaSAndroid Build Coastguard Worker class SK_SCOPED_CAPABILITY SkAutoSharedMutexShared { 85*c8dee2aaSAndroid Build Coastguard Worker public: SkAutoSharedMutexShared(SkSharedMutex & lock)86*c8dee2aaSAndroid Build Coastguard Worker explicit SkAutoSharedMutexShared(SkSharedMutex& lock) SK_ACQUIRE_SHARED(lock) 87*c8dee2aaSAndroid Build Coastguard Worker : fLock(lock) { 88*c8dee2aaSAndroid Build Coastguard Worker lock.acquireShared(); 89*c8dee2aaSAndroid Build Coastguard Worker } 90*c8dee2aaSAndroid Build Coastguard Worker 91*c8dee2aaSAndroid Build Coastguard Worker // You would think this should be SK_RELEASE_SHARED_CAPABILITY, but SK_SCOPED_CAPABILITY 92*c8dee2aaSAndroid Build Coastguard Worker // doesn't fully understand the difference between shared and exclusive. 93*c8dee2aaSAndroid Build Coastguard Worker // Please review https://reviews.llvm.org/D52578 for more information. SK_RELEASE_CAPABILITY()94*c8dee2aaSAndroid Build Coastguard Worker ~SkAutoSharedMutexShared() SK_RELEASE_CAPABILITY() { fLock.releaseShared(); } 95*c8dee2aaSAndroid Build Coastguard Worker 96*c8dee2aaSAndroid Build Coastguard Worker private: 97*c8dee2aaSAndroid Build Coastguard Worker SkSharedMutex& fLock; 98*c8dee2aaSAndroid Build Coastguard Worker }; 99*c8dee2aaSAndroid Build Coastguard Worker 100*c8dee2aaSAndroid Build Coastguard Worker #endif // SkSharedLock_DEFINED 101