123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185 |
- /*
- ==============================================================================
- This file is part of the JUCE library.
- Copyright (c) 2017 - ROLI Ltd.
- 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
- {
- TimeSliceThread::TimeSliceThread (const String& name) : Thread (name)
- {
- }
- TimeSliceThread::~TimeSliceThread()
- {
- stopThread (2000);
- }
- //==============================================================================
- void TimeSliceThread::addTimeSliceClient (TimeSliceClient* const client, int millisecondsBeforeStarting)
- {
- if (client != nullptr)
- {
- const ScopedLock sl (listLock);
- client->nextCallTime = Time::getCurrentTime() + RelativeTime::milliseconds (millisecondsBeforeStarting);
- clients.addIfNotAlreadyThere (client);
- notify();
- }
- }
- void TimeSliceThread::removeTimeSliceClient (TimeSliceClient* const client)
- {
- const ScopedLock sl1 (listLock);
- // if there's a chance we're in the middle of calling this client, we need to
- // also lock the outer lock..
- if (clientBeingCalled == client)
- {
- const ScopedUnlock ul (listLock); // unlock first to get the order right..
- const ScopedLock sl2 (callbackLock);
- const ScopedLock sl3 (listLock);
- clients.removeFirstMatchingValue (client);
- }
- else
- {
- clients.removeFirstMatchingValue (client);
- }
- }
- void TimeSliceThread::removeAllClients()
- {
- for (;;)
- {
- if (auto* c = getClient (0))
- removeTimeSliceClient (c);
- else
- break;
- }
- }
- void TimeSliceThread::moveToFrontOfQueue (TimeSliceClient* client)
- {
- const ScopedLock sl (listLock);
- if (clients.contains (client))
- {
- client->nextCallTime = Time::getCurrentTime();
- notify();
- }
- }
- int TimeSliceThread::getNumClients() const
- {
- return clients.size();
- }
- TimeSliceClient* TimeSliceThread::getClient (const int i) const
- {
- const ScopedLock sl (listLock);
- return clients[i];
- }
- //==============================================================================
- TimeSliceClient* TimeSliceThread::getNextClient (int index) const
- {
- Time soonest;
- TimeSliceClient* client = nullptr;
- for (int i = clients.size(); --i >= 0;)
- {
- auto* c = clients.getUnchecked ((i + index) % clients.size());
- if (client == nullptr || c->nextCallTime < soonest)
- {
- client = c;
- soonest = c->nextCallTime;
- }
- }
- return client;
- }
- void TimeSliceThread::run()
- {
- int index = 0;
- while (! threadShouldExit())
- {
- int timeToWait = 500;
- {
- Time nextClientTime;
- int numClients = 0;
- {
- const ScopedLock sl2 (listLock);
- numClients = clients.size();
- index = numClients > 0 ? ((index + 1) % numClients) : 0;
- if (auto* firstClient = getNextClient (index))
- nextClientTime = firstClient->nextCallTime;
- }
- if (numClients > 0)
- {
- auto now = Time::getCurrentTime();
- if (nextClientTime > now)
- {
- timeToWait = (int) jmin ((int64) 500, (nextClientTime - now).inMilliseconds());
- }
- else
- {
- timeToWait = index == 0 ? 1 : 0;
- const ScopedLock sl (callbackLock);
- {
- const ScopedLock sl2 (listLock);
- clientBeingCalled = getNextClient (index);
- }
- if (clientBeingCalled != nullptr)
- {
- const int msUntilNextCall = clientBeingCalled->useTimeSlice();
- const ScopedLock sl2 (listLock);
- if (msUntilNextCall >= 0)
- clientBeingCalled->nextCallTime = now + RelativeTime::milliseconds (msUntilNextCall);
- else
- clients.removeFirstMatchingValue (clientBeingCalled);
- clientBeingCalled = nullptr;
- }
- }
- }
- }
- if (timeToWait > 0)
- wait (timeToWait);
- }
- }
- } // namespace juce
|