| 
							- /*
 -   ==============================================================================
 - 
 -    This file is part of the JUCE library.
 -    Copyright (c) 2020 - Raw Material Software Limited
 - 
 -    JUCE is an open source library subject to commercial or open-source
 -    licensing.
 - 
 -    The code included in this file is provided under the terms of the ISC license
 -    http://www.isc.org/downloads/software-support-policy/isc-license. Permission
 -    To use, copy, modify, and/or distribute this software for any purpose with or
 -    without fee is hereby granted provided that the above copyright notice and
 -    this permission notice appear in all copies.
 - 
 -    JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
 -    EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
 -    DISCLAIMED.
 - 
 -   ==============================================================================
 - */
 - 
 - namespace juce
 - {
 - 
 - ReadWriteLock::ReadWriteLock() noexcept
 - {
 -     readerThreads.ensureStorageAllocated (16);
 - }
 - 
 - ReadWriteLock::~ReadWriteLock() noexcept
 - {
 -     jassert (readerThreads.size() == 0);
 -     jassert (numWriters == 0);
 - }
 - 
 - //==============================================================================
 - void ReadWriteLock::enterRead() const noexcept
 - {
 -     while (! tryEnterRead())
 -         readWaitEvent.wait (100);
 - }
 - 
 - bool ReadWriteLock::tryEnterRead() const noexcept
 - {
 -     auto threadId = Thread::getCurrentThreadId();
 - 
 -     const SpinLock::ScopedLockType sl (accessLock);
 - 
 -     for (auto& readerThread : readerThreads)
 -     {
 -         if (readerThread.threadID == threadId)
 -         {
 -             readerThread.count++;
 -             return true;
 -         }
 -     }
 - 
 -     if (numWriters + numWaitingWriters == 0
 -          || (threadId == writerThreadId && numWriters > 0))
 -     {
 -         readerThreads.add ({ threadId, 1 });
 -         return true;
 -     }
 - 
 -     return false;
 - }
 - 
 - void ReadWriteLock::exitRead() const noexcept
 - {
 -     auto threadId = Thread::getCurrentThreadId();
 -     const SpinLock::ScopedLockType sl (accessLock);
 - 
 -     for (int i = 0; i < readerThreads.size(); ++i)
 -     {
 -         auto& readerThread = readerThreads.getReference (i);
 - 
 -         if (readerThread.threadID == threadId)
 -         {
 -             if (--(readerThread.count) == 0)
 -             {
 -                 readerThreads.remove (i);
 - 
 -                 readWaitEvent.signal();
 -                 writeWaitEvent.signal();
 -             }
 - 
 -             return;
 -         }
 -     }
 - 
 -     jassertfalse; // unlocking a lock that wasn't locked..
 - }
 - 
 - //==============================================================================
 - void ReadWriteLock::enterWrite() const noexcept
 - {
 -     auto threadId = Thread::getCurrentThreadId();
 -     const SpinLock::ScopedLockType sl (accessLock);
 - 
 -     while (! tryEnterWriteInternal (threadId))
 -     {
 -         ++numWaitingWriters;
 -         accessLock.exit();
 -         writeWaitEvent.wait (100);
 -         accessLock.enter();
 -         --numWaitingWriters;
 -     }
 - }
 - 
 - bool ReadWriteLock::tryEnterWrite() const noexcept
 - {
 -     const SpinLock::ScopedLockType sl (accessLock);
 -     return tryEnterWriteInternal (Thread::getCurrentThreadId());
 - }
 - 
 - bool ReadWriteLock::tryEnterWriteInternal (Thread::ThreadID threadId) const noexcept
 - {
 -     if (readerThreads.size() + numWriters == 0
 -          || threadId == writerThreadId
 -          || (readerThreads.size() == 1 && readerThreads.getReference (0).threadID == threadId))
 -     {
 -         writerThreadId = threadId;
 -         ++numWriters;
 -         return true;
 -     }
 - 
 -     return false;
 - }
 - 
 - void ReadWriteLock::exitWrite() const noexcept
 - {
 -     const SpinLock::ScopedLockType sl (accessLock);
 - 
 -     // check this thread actually had the lock..
 -     jassert (numWriters > 0 && writerThreadId == Thread::getCurrentThreadId());
 - 
 -     if (--numWriters == 0)
 -     {
 -         writerThreadId = {};
 - 
 -         readWaitEvent.signal();
 -         writeWaitEvent.signal();
 -     }
 - }
 - 
 - } // namespace juce
 
 
  |