non/mixer/src/Group.C

323 lines
6.9 KiB
C++
Raw Normal View History

2013-08-06 09:03:19 +02:00
/*******************************************************************************/
/* Copyright (C) 2013 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 <Mixer.H>
#include "Group.H"
#include "Chain.H"
#include "Mixer_Strip.H"
#include "Module.H"
2013-08-09 08:32:44 +02:00
#include <unistd.h>
2013-08-06 09:03:19 +02:00
extern char *instance_name;
Group::Group ( )
{
_single =false;
_name = NULL;
2013-08-09 08:32:44 +02:00
_dsp_load = _load_coef = 0;
2013-08-06 09:03:19 +02:00
}
Group::Group ( const char *name, bool single ) : Loggable ( !single )
{
_single = single;
_name = strdup(name);
2013-08-09 08:32:44 +02:00
_dsp_load = _load_coef = 0;
2013-08-06 09:03:19 +02:00
// this->name( name );
/* FIXME: handle client creation error */
/* if ( ! jack_name ) */
/* { */
/* _engine = NULL; */
/* // fl_alert( "Could not create JACK client. Perhaps the sound device already in use. In any event, now I'll die." ); */
/* exit( 1 ); */
/* // return false; */
/* } */
}
Group::~Group ( )
{
DMESSAGE( "Destroying group" );
for ( std::list<Mixer_Strip*>::iterator i = strips.begin();
i != strips.end();
i++ )
{
/* avoid a use after free during project close when the group
* may be destroyed before its member strips are */
(*i)->clear_group();
}
2013-08-06 09:03:19 +02:00
if ( _name )
free( _name );
2013-08-06 09:03:19 +02:00
deactivate();
}
void
Group::get ( Log_Entry &e ) const
{
e.add( ":name", name() );
}
void
Group::set ( Log_Entry &e )
{
for ( int i = 0; i < e.size(); ++i )
{
const char *s, *v;
e.get( i, &s, &v );
if ( ! ( strcmp( s, ":name" ) ) )
{
bool add = false;
if (!_name )
add = true;
_name = strdup(v);
if ( add )
mixer->add_group(this);
}
}
}
/*************/
/* Callbacks */
/*************/
void
Group::latency ( jack_latency_callback_mode_t mode )
{
if ( trylock() )
{
for ( std::list<Mixer_Strip*>::iterator i = strips.begin();
i != strips.end();
i++ )
{
if ( (*i)->chain() )
(*i)->chain()->set_latency(mode == JackCaptureLatency ? JACK::Port::Input : JACK::Port::Output );
}
unlock();
}
}
2013-08-06 09:03:19 +02:00
/* THREAD: RT */
/** This is the jack xrun callback */
int
Group::xrun ( void )
{
return 0;
}
/* THREAD: RT */
void
Group::freewheel ( bool starting )
{
if ( starting )
DMESSAGE( "entering freewheeling mode" );
else
DMESSAGE( "leaving freewheeling mode" );
}
/* THREAD: RT (non-RT) */
int
Group::buffer_size ( nframes_t nframes )
{
2013-08-09 08:32:44 +02:00
recal_load_coef();
2013-08-06 09:03:19 +02:00
/* JACK calls this in the RT thread, even though it's a
* non-realtime operation. This mucks up our ability to do
* THREAD_ASSERT, so just lie and say this is the UI thread... */
_thread.set( "UI" );
for ( std::list<Mixer_Strip*>::iterator i = strips.begin();
i != strips.end();
i++ )
{
if ( (*i)->chain() )
(*i)->chain()->buffer_size(nframes);
}
_thread.set( "RT" );
return 0;
}
/* THREAD: ?? */
void
Group::port_connect( jack_port_id_t a, jack_port_id_t b, int connect )
{
for ( std::list<Mixer_Strip*>::iterator i = strips.begin();
i != strips.end();
i++ )
{
if ( (*i)->chain() )
(*i)->chain()->port_connect( a, b, connect);
}
}
/* THREAD: RT */
int
Group::process ( nframes_t nframes )
{
2013-08-09 08:32:44 +02:00
jack_time_t then = jack_get_time();
2013-08-06 09:03:19 +02:00
/* FIXME: wrong place for this */
_thread.set( "RT" );
if ( ! trylock() )
{
/* the data structures we need to access here (tracks and
* their ports, but not track contents) may be in an
* inconsistent state at the moment. Just punt and drop this
* buffer. */
++_buffers_dropped;
return 0;
}
2013-08-09 08:32:44 +02:00
2013-08-06 09:03:19 +02:00
/* since feedback loops are forbidden and outputs are
* summed, we don't care what order these are processed
* in */
for ( std::list<Mixer_Strip*>::iterator i = strips.begin();
i != strips.end();
i++ )
{
if ( (*i)->chain() )
(*i)->chain()->process(nframes);
}
unlock();
2013-08-09 08:32:44 +02:00
_dsp_load = (float)(jack_get_time() - then ) * _load_coef;
2013-08-06 09:03:19 +02:00
return 0;
}
2013-08-09 08:32:44 +02:00
void
Group::recal_load_coef ( void )
{
_load_coef = 1.0f / ( nframes() / (float)sample_rate() * 1000000.0f );
2013-08-09 08:32:44 +02:00
}
2013-08-06 09:03:19 +02:00
int
Group::sample_rate_changed ( nframes_t srate )
{
2013-08-09 08:32:44 +02:00
recal_load_coef();
2013-08-06 09:03:19 +02:00
for ( std::list<Mixer_Strip*>::iterator i = strips.begin();
i != strips.end();
i++ )
{
if ( (*i)->chain() )
(*i)->chain()->sample_rate_change(srate);
}
return 0;
}
/* TRHEAD: RT */
void
Group::thread_init ( void )
{
_thread.set( "RT" );
}
/* THREAD: RT */
void
Group::shutdown ( void )
{
}
/*******************/
/* Group interface */
/*******************/
void
Group::name ( const char *n )
{
if ( _name )
free( _name );
char ename[512];
_name = strdup( n );
if ( _single )
snprintf( ename, sizeof(ename), "%s/%s", instance_name, n );
else
snprintf( ename, sizeof(ename), "%s (%s)", instance_name, n );
if ( !active() )
{
2013-08-24 05:41:17 +02:00
Client::init( ename );
Module::sample_rate( sample_rate() );
2013-08-06 09:03:19 +02:00
}
else
{
Client::name( ename );
}
}
void
Group::add ( Mixer_Strip *o )
{
lock();
2013-08-06 09:03:19 +02:00
if ( ! active() )
{
/* to call init */
char *n = strdup(name());
name(n);
free(n);
}
2013-08-06 09:03:19 +02:00
if ( o->chain() )
o->chain()->thaw_ports();
strips.push_back(o);
2013-08-06 09:03:19 +02:00
unlock();
}
void
Group::remove ( Mixer_Strip *o )
{
lock();
2013-08-06 09:03:19 +02:00
strips.remove(o);
2013-08-06 09:03:19 +02:00
if ( o->chain() )
o->chain()->freeze_ports();
2013-08-06 09:03:19 +02:00
if ( strips.size() == 0 && active() )
Client::close();
2013-08-06 09:03:19 +02:00
unlock();
}