| 
							- 
 - /*******************************************************************************/
 - /* Copyright (C) 2008 Jonathan Moore Liles                                     */
 - /*                                                                             */
 - /* This program is free software; you can redistribute it and/or modify it     */
 - /* under the terms of the GNU General Public License as published by the       */
 - /* Free Software Foundation; either version 2 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 General Public License for   */
 - /* more details.                                                               */
 - /*                                                                             */
 - /* You should have received a copy of the GNU General Public License along     */
 - /* with This program; see the file COPYING.  If not,write to the Free Software */
 - /* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
 - /*******************************************************************************/
 - 
 - #include "../Track.H"
 - // #include "Audio_Sequence.H"
 - class Audio_Sequence;
 - 
 - // #include "Port.H"
 - #include "Engine.H" // for locking.
 - 
 - #include "Disk_Stream.H"
 - #include "dsp.h"
 - 
 - #include "const.h"
 - #include "util/debug.h"
 - 
 - 
 - 
 - 
 - /**********/
 - /* Engine */
 - /**********/
 - 
 - /* A Disk_Stream uses a separate I/O thread to stream a track's
 -    regions from disk into a ringbuffer to be processed by the RT
 -    thread (or vice-versa). The I/O thread syncronizes access with the
 -    user thread via the Timeline mutex. The size of the buffer (in
 -    seconds) must be set before any Disk_Stream objects are created;
 -    that is, at startup time. The default is 5 seconds, which may or
 -    may not be excessive depending on various external factors. */
 - 
 - float Disk_Stream::seconds_to_buffer = 2.0f;
 - /* this is really only a rough estimate. The actual amount of data
 -  read depends on many factors.  Overlapping regions, for example, will
 -  require more data to be read from disk, as will varying channel
 -  counts.*/
 - size_t Disk_Stream::disk_io_kbytes = 256;
 - 
 - 
 - 
 - Disk_Stream::Disk_Stream ( Track *track, float frame_rate, nframes_t nframes, int channels ) : _track( track )
 - {
 -     assert( channels );
 - 
 -     _frame = 0;
 -     _terminate = false;
 -     _pending_seek = -1;
 -     _xruns = 0;
 -     _frame_rate = frame_rate;
 - 
 -     _resize_buffers( nframes, channels );
 - 
 -     sem_init( &_blocks, 0, _total_blocks );
 - }
 - 
 - Disk_Stream::~Disk_Stream ( )
 - {
 -     /* it isn't safe to do all this with the RT thread running */
 -     engine->lock();
 - 
 -     shutdown();
 - 
 -      _track = NULL;
 - 
 -     sem_destroy( &_blocks );
 - 
 -     for ( int i = channels(); i--; )
 -         jack_ringbuffer_free( _rb[ i ] );
 - 
 -     engine->unlock();
 - }
 - 
 - 
 - 
 - /** flush buffers and reset. Must only be called from the RT thread. */
 - void
 - Disk_Stream::base_flush ( bool is_output )
 - {
 -     THREAD_ASSERT( RT );
 - 
 -     /* flush buffers */
 -     for ( int i = _rb.size(); i--; )
 -         jack_ringbuffer_read_advance( _rb[ i ], jack_ringbuffer_read_space( _rb[ i ] ) );
 - 
 - /*  sem_destroy( &_blocks ); */
 - 
 - /*     if ( is_output ) */
 - /*         sem_init( &_blocks, 0, _total_blocks ); */
 - /*     else */
 - /*         sem_init( &_blocks, 0, 0 ); */
 - 
 -     if ( is_output )
 -     {
 -         int n;
 -         sem_getvalue( &_blocks, &n );
 - 
 -         n = _total_blocks - n;
 - 
 -         while ( n-- )
 -             sem_post( &_blocks );
 -     }
 -     else
 -     {
 -         sem_destroy( &_blocks );
 - 
 -         sem_init( &_blocks, 0, 0 );
 -     }
 - 
 - 
 - }
 - 
 - /** signal thread to terminate, then detach it */
 - void
 - Disk_Stream::detach ( void )
 - {
 -     _terminate = true;
 - 
 -     block_processed();
 - 
 -     _thread.detach();
 - }
 - 
 - /** stop the IO thread. */
 - void
 - Disk_Stream::shutdown ( void )
 - {
 -     _terminate = true;
 - 
 -     /* try to wake the thread so it'll see that it's time to die */
 -     block_processed();
 - 
 -     if ( _thread.running() )
 -         _thread.join();
 - }
 - 
 - Track *
 - Disk_Stream::track ( void ) const
 - {
 -     return _track;
 - }
 - 
 - Audio_Sequence *
 - Disk_Stream::sequence ( void ) const
 - {
 -     return (Audio_Sequence*)_track->sequence();
 - }
 - 
 - /** start Disk_Stream thread */
 - void
 - Disk_Stream::run ( void )
 - {
 -     ASSERT( ! _thread.running(), "Thread is already running" );
 - 
 -     if ( ! _thread.clone( &Disk_Stream::disk_thread, this ) )
 -         FATAL( "Could not create IO thread!" );
 - }
 - 
 - void
 - Disk_Stream::_resize_buffers ( nframes_t nframes, int channels )
 - {
 -     for ( int i = _rb.size(); i--; )
 -         jack_ringbuffer_free( _rb[ i ] );
 - 
 -     _rb.clear();
 - 
 -     _nframes = nframes;
 - 
 -     _total_blocks = _frame_rate * seconds_to_buffer / nframes;
 - 
 -     size_t bufsize = _total_blocks * nframes * sizeof( sample_t );
 - 
 -     if ( disk_io_kbytes )
 -         _disk_io_blocks = ( bufsize * channels ) / ( disk_io_kbytes * 1024 );
 -     else
 -         _disk_io_blocks = 1;
 - 
 -     for ( int i = channels; i--; )
 -         _rb.push_back( jack_ringbuffer_create( bufsize ) );
 - }
 - 
 - /* THREAD: RT (non-RT)  */
 - /* to be called when the JACK buffer size changes. */
 - void
 - Disk_Stream::resize_buffers ( nframes_t nframes )
 - {
 -     if ( nframes != _nframes )
 -     {
 -         DMESSAGE( "resizing buffers" );
 - 
 -         const bool was_running = _thread.running();
 - 
 -         if ( was_running )
 -             shutdown();
 - 
 -         flush();
 - 
 -         _resize_buffers( nframes, channels() );
 - 
 -         if ( was_running )
 -             run();
 -     }
 - }
 - 
 - 
 - /* static wrapper */
 - void *
 - Disk_Stream::disk_thread ( void *arg )
 - {
 -     ((Disk_Stream*)arg)->disk_thread();
 - 
 -     return NULL;
 - }
 - 
 - int
 - Disk_Stream::buffer_percent ( void )
 - {
 -     int n;
 - 
 -     sem_getvalue( &_blocks, &n );
 - 
 -     return 100 - (n * 100 / _total_blocks);
 - }
 
 
  |