| 
							- /*
 - Copyright (C) 2001 Paul Davis
 - Copyright (C) 2004-2008 Grame
 - 
 - This program is free software; you can redistribute it and/or modify
 - it under the terms of the GNU Lesser General Public License as published by
 - the Free Software Foundation; either version 2.1 of the License, or
 - (at your option) any later version.
 - 
 - This program is distributed in the hope that it will be useful,
 - but WITHOUT ANY WARRANTY; without even the implied warranty of
 - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 - GNU Lesser General Public License for more details.
 - 
 - You should have received a copy of the GNU Lesser General Public License
 - along with this program; if not, write to the Free Software 
 - Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 - 
 - */
 - 
 - #if defined(HAVE_CONFIG_H)
 - #include "config.h"
 - #endif
 - 
 - #include "JackFrameTimer.h"
 - #include "JackError.h"
 - #include <math.h>
 - 
 - namespace Jack
 - {
 - 
 - JackTimer::JackTimer()
 - {
 -     fInitialized = false;
 -     fFrames = 0;
 -     fCurrentWakeup = 0;
 -     fCurrentCallback = 0;
 -     fNextWakeUp = 0;
 -     fFilterCoefficient = 0.01f;
 -     fSecondOrderIntegrator = 0.0f;
 - }
 - 
 - void JackFrameTimer::InitFrameTime()
 - {
 -     fFirstWakeUp = true;
 - }
 - 
 - void JackFrameTimer::IncFrameTime(jack_nframes_t nframes, jack_time_t callback_usecs, jack_time_t period_usecs)
 - {
 -     if (fFirstWakeUp) {
 -         InitFrameTimeAux(callback_usecs, period_usecs);
 -         fFirstWakeUp = false;
 -     } else {
 -         IncFrameTimeAux(nframes, callback_usecs, period_usecs);
 -     }
 - }
 - 
 - void JackFrameTimer::ResetFrameTime(jack_nframes_t frames_rate, jack_time_t callback_usecs, jack_time_t period_usecs)
 - {
 -     if (!fFirstWakeUp) { // ResetFrameTime may be called by a xrun/delayed wakeup on the first cycle
 -         JackTimer* timer = WriteNextStateStart();
 -         jack_nframes_t period_size_guess = (jack_nframes_t)(frames_rate * ((timer->fNextWakeUp - timer->fCurrentWakeup) / 1000000.0));
 -         timer->fFrames += ((callback_usecs - timer->fNextWakeUp) / period_size_guess) * period_size_guess;
 -         timer->fCurrentWakeup = callback_usecs;
 -         timer->fCurrentCallback = callback_usecs;
 -         timer->fNextWakeUp = callback_usecs + period_usecs;
 -         WriteNextStateStop();
 -         TrySwitchState(); // always succeed since there is only one writer
 -     }
 - }
 - 
 - /*
 - 	Use the state returned by ReadCurrentState and check that the state was not changed during the read operation.
 - 	The operation is lock-free since there is no intermediate state in the write operation that could cause the
 - 	read to loop forever.
 - */
 - void JackFrameTimer::ReadFrameTime(JackTimer* timer)
 - {
 -     UInt16 next_index = GetCurrentIndex();
 -     UInt16 cur_index;
 -     do {
 -         cur_index = next_index;
 -         memcpy(timer, ReadCurrentState(), sizeof(JackTimer));
 -         next_index = GetCurrentIndex();
 -     } while (cur_index != next_index); // Until a coherent state has been read
 - }
 - 
 - // Internal
 - 
 - void JackFrameTimer::InitFrameTimeAux(jack_time_t callback_usecs, jack_time_t period_usecs)
 - {
 -     JackTimer* timer = WriteNextStateStart();
 -     timer->fSecondOrderIntegrator = 0.0f;
 -     timer->fCurrentCallback = callback_usecs;
 -     timer->fNextWakeUp = callback_usecs + period_usecs;
 -     WriteNextStateStop();
 -     TrySwitchState(); // always succeed since there is only one writer
 - }
 - 
 - void JackFrameTimer::IncFrameTimeAux(jack_nframes_t nframes, jack_time_t callback_usecs, jack_time_t period_usecs)
 - {
 -     JackTimer* timer = WriteNextStateStart();
 -     float delta = (int64_t)callback_usecs - (int64_t)timer->fNextWakeUp;
 -     timer->fCurrentWakeup = timer->fNextWakeUp;
 -     timer->fCurrentCallback = callback_usecs;
 -     timer->fFrames += nframes;
 -     timer->fSecondOrderIntegrator += 0.5f * timer->fFilterCoefficient * delta;
 -     timer->fNextWakeUp = timer->fCurrentWakeup + period_usecs + (int64_t) floorf((timer->fFilterCoefficient * (delta + timer->fSecondOrderIntegrator)));
 -     timer->fInitialized = true;
 -     WriteNextStateStop();
 -     TrySwitchState(); // always succeed since there is only one writer
 - }
 - 
 - } // end of namespace
 
 
  |