|  | 
/*******************************************************************************/
/* 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);
}
 |