|
-
- /*******************************************************************************/
- /* 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. */
- /*******************************************************************************/
-
- /* Wrapper for a JACK audio port */
-
- #include "Port.H"
-
- #include <string.h>
- #include <stdio.h> // sprintf
- #include <errno.h>
-
- #include <assert.h>
- #include "debug.h"
-
- namespace JACK
- {
-
- /* static char *name_for_port ( Port::direction_e dir, const char *base, int n, const char *type ); */
-
- int
- Port::max_name ( void )
- {
- return jack_port_name_size() - jack_client_name_size() - 6;
- }
-
- Port::Port ( const Port &rhs )
- {
- _connections = NULL;
- _terminal = rhs._terminal;
- // _connections = rhs._connections;
- _client = rhs._client;
- _port = rhs._port;
- _direction = rhs._direction;
- _type = rhs._type;
- _name = NULL;
- _name = strdup( rhs._name );
- _trackname = NULL;
- if ( rhs._trackname )
- _trackname = strdup( rhs._trackname );
- _client->port_added( this );
- }
-
- /* nframes is the number of frames to buffer */
- Port::Port ( JACK::Client *client, jack_port_t *port )
- {
- _terminal = 0;
- _connections = NULL;
- _client = client;
- _port = port;
- _name = strdup( jack_port_name( port ) );
- _trackname = NULL;
- _direction = ( jack_port_flags( _port ) & JackPortIsOutput ) ? Output : Input;
- const char *type = jack_port_type( _port );
-
- _type = Audio;
- if ( strstr( type, "MIDI") )
- _type = MIDI;
- else if ( strstr( type, "CV)") )
- _type = CV;
-
- _client->port_added( this );
-
- }
-
- Port::Port ( JACK::Client *client, const char *trackname, const char *name, direction_e dir, type_e type )
- {
- _port = 0;
- _terminal = 0;
- _name = NULL;
- _trackname = NULL;
- _connections = NULL;
- _client = client;
- _direction = dir;
- _type = type;
- _trackname = NULL;
-
- if ( trackname )
- _trackname = strdup( trackname );
-
- _name = strdup( name );
-
- _client->port_added( this );
- }
-
- /* Port::Port ( JACK::Client *client, direction_e dir, type_e type, const char *base, int n, const char *subtype ) */
- /* { */
- /* _port = 0; */
- /* _terminal = 0; */
- /* _name = NULL; */
- /* _connections = NULL; */
- /* _client = client; */
-
- /* _name = name_for_port( dir, base, n, subtype ); */
- /* _direction = dir; */
- /* _type = type; */
-
- /* _client->port_added( this ); */
- /* } */
-
- /* Port::Port ( JACK::Client *client, direction_e dir, type_e type, int n, const char *subtype ) */
- /* { */
- /* _port = 0; */
- /* _terminal = 0; */
- /* _name = NULL; */
- /* _connections = NULL; */
- /* _client = client; */
-
- /* _name = name_for_port( dir, NULL, n, subtype ); */
- /* _direction = dir; */
- /* _type = type; */
-
- /* _client->port_added( this ); */
- /* } */
-
- Port::~Port ( )
- {
- _client->port_removed( this );
-
- if ( _name )
- {
- free( _name );
- _name = NULL;
- }
- if ( _trackname )
- {
- free( _trackname );
- _trackname = NULL;
- }
-
- }
-
- /* sort input before output and then by alpha */
- bool
- Port::operator < ( const Port & rhs ) const
- {
- if ( type() == rhs.type() )
- return strcmp( name(), rhs.name() );
- else
- return direction() == Port::Input;
- }
-
-
- /* static char * */
- /* name_for_port ( Port::direction_e dir, const char *base, int n, const char *type ) */
- /* { */
- /* char *pname; */
-
- /* const char *dir_s = dir == Port::Output ? "out" : "in"; */
-
- /* if ( type ) */
- /* asprintf( &pname, "%s-%s%s%s-%d", type, base ? base : "", base ? "/" : "", dir_s, n + 1 ); */
- /* else */
- /* asprintf( &pname, "%s%s%s-%d", base ? base : "", base ? "/" : "", dir_s, n + 1 ); */
-
- /* return pname; */
- /* } */
-
- bool
- Port::activate ( void )
- {
- /* assert( !_port ); */
-
- int flags = 0;
-
- if ( _direction == Output )
- flags |= JackPortIsOutput;
- else
- flags |= JackPortIsInput;
-
- if ( _terminal )
- flags |= JackPortIsTerminal;
-
- char jackname[max_name()];
-
- snprintf( jackname, sizeof(jackname), "%s%s%s", _trackname ? _trackname : "", _trackname ? "/" : "", _name );
-
- DMESSAGE( "Activating port name %s", jackname );
- _port = jack_port_register( _client->jack_client(), jackname,
- ( _type == Audio ) || ( _type == CV ) ? JACK_DEFAULT_AUDIO_TYPE : JACK_DEFAULT_MIDI_TYPE,
- flags,
- 0 );
-
- #ifdef HAVE_JACK_METADATA
- if ( _type == CV )
- {
- jack_uuid_t uuid = jack_port_uuid( _port );
- jack_set_property( _client->jack_client(), uuid, "http://jackaudio.org/metadata/signal-type", "CV", "text/plain" );
- }
- #endif
-
- DMESSAGE( "Port = %p", _port );
-
- if ( ! _port )
- return false;
-
- _client->port_added( this );
-
- return true;
- }
-
- /** returns the sum of latency of all ports between this one and a
- terminal port. */
- nframes_t
- Port::total_latency ( void ) const
- {
- #ifdef HAVE_JACK_PORT_GET_LATENCY_RANGE
- jack_latency_range_t range;
-
- jack_port_get_latency_range( _port, _direction == Input ? JackPlaybackLatency : JackCaptureLatency, &range );
-
- return range.max;
- #else
- return jack_port_get_total_latency( _client->jack_client() , _port );
- #endif
- }
-
- /** returns the number of frames of latency assigned to this port */
- void
- Port::get_latency ( direction_e dir, nframes_t *min, nframes_t *max ) const
- {
- #ifdef HAVE_JACK_PORT_GET_LATENCY_RANGE
- jack_latency_range_t range;
-
- jack_port_get_latency_range( _port, dir == Output ? JackPlaybackLatency : JackCaptureLatency, &range );
-
- *min = range.min;
- *max = range.max;
- #else
- *min = *max = jack_port_get_latency( _port );
- #endif
-
- }
-
- /** inform JACK that port has /frames/ frames of latency */
- void
- Port::set_latency ( direction_e dir, nframes_t min, nframes_t max )
- {
- #ifdef HAVE_JACK_PORT_GET_LATENCY_RANGE
- jack_latency_range_t range;
- // DMESSAGE( "Setting port latency!" );
-
- range.max = max;
- range.min = min;
-
- jack_port_set_latency_range( _port, dir == Output ? JackPlaybackLatency : JackCaptureLatency, &range );
- #else
- jack_port_set_latency( _port, max );
- #endif
- }
-
- void
- Port::shutdown ( void )
- {
- deactivate();
-
- _client->port_removed( this );
- }
-
- void
- Port::deactivate ( void )
- {
- if ( _port )
- {
- #ifdef HAVE_JACK_METADATA
- if ( _type == CV )
- {
- jack_uuid_t uuid = jack_port_uuid(_port);
- jack_remove_property(_client->jack_client(), uuid, "http://jackaudio.org/metadata/signal-type");
- }
- #endif
- jack_port_unregister( _client->jack_client(), _port );
- }
-
- _port = 0;
- }
-
-
- void
- Port::name ( const char *name )
- {
- if ( _name )
- free( _name );
-
- _name = strdup( name );
- }
-
- void
- Port::trackname ( const char *trackname )
- {
- if ( _trackname )
- free( _trackname );
-
- _trackname = NULL;
-
- if ( trackname )
- _trackname = strdup( trackname );
- }
-
- bool
- Port::rename ( void )
- {
- char jackname[max_name()];
-
- snprintf( jackname, sizeof(jackname), "%s%s%s", _trackname ? _trackname : "", _trackname ? "/" : "", _name );
-
- if ( _port )
- return 0 == jack_port_set_name( _port, jackname );
- else
- return false;
- }
-
- void
- Port::write ( sample_t *buf, nframes_t nframes )
- {
- memcpy( buffer( nframes ), buf, nframes * sizeof( sample_t ) );
- }
-
- void
- Port::read ( sample_t *buf, nframes_t nframes )
- {
- memcpy( buf, buffer( nframes ), nframes * sizeof( sample_t ) );
- }
-
- void *
- Port::buffer ( nframes_t nframes )
- {
- return jack_port_get_buffer( _port, nframes );
- }
-
- void
- Port::silence ( nframes_t nframes )
- {
- memset( buffer( nframes ), 0, nframes * sizeof( sample_t ) );
- }
-
- /** Return a malloc()'d null terminated array of strings
- * representing all ports to which this port is connected. */
- const char **
- Port::connections ( void )
- {
- ASSERT( _port, "Attempt to get connections of null port" );
-
- return jack_port_get_connections( _port );
- }
-
- /** Restore the connections returned by connections() */
- bool
- Port::connections ( const char **port_names )
- {
- if ( ! port_names )
- return true;
-
- for ( const char **port_name = port_names; *port_name; ++port_name )
- {
- printf( "Attempting to reconnect to %s\n", *port_name );
- connect( *port_name );
- }
-
- return true;
- }
-
- int
- Port::connect ( const char *to )
- {
- const char *name = jack_port_name( _port );
-
- /* jack complains when you attempt to connect an already connected port... */
- if ( connected_to( to ) )
- return 0;
-
- if ( _direction == Output )
- {
- DMESSAGE("Connecting jack port %s to %s", name, to );
-
- return jack_connect( _client->jack_client(), name, to );
- }
- else
- {
- DMESSAGE("Connecting jack port %s to %s", to, name );
-
- return jack_connect( _client->jack_client(), to, name );
- }
- }
-
-
- int
- Port::disconnect ( const char *from )
- {
- const char *name = jack_port_name( _port );
-
- if ( _direction == Output )
- {
- DMESSAGE("Disconnecting jack port %s from %s", name, from );
-
- return jack_disconnect( _client->jack_client(), name, from );
- }
- else
- {
- DMESSAGE("Disconnecting jack port %s from %s", from, name );
-
- return jack_disconnect( _client->jack_client(), from, name );
- }
- }
-
- bool
- Port::connected_to ( const char *to )
- {
- return jack_port_connected_to( _port, to );
- }
-
- void
- Port::freeze ( void )
- {
- if ( _connections )
- free( _connections );
-
- // DMESSAGE( "Freezing port %s", _name );
-
- _connections = connections();
-
- // deactivate();
- }
-
- void
- Port::thaw ( void )
- {
- // DMESSAGE( "Thawing port %s", _name );
-
- activate();
-
- if ( _connections )
- {
- connections( _connections );
-
- free( _connections );
-
- _connections = NULL;
- }
- }
- }
|