Audio-Nama
view release on metacpan or search on metacpan
lib/Audio/Nama/ChainSetup.pm view on Meta::CPAN
@input_chains + @output_chains # to sense empty chain setup
}
{ my ($m,$n,$o,$p,$q,$r);
sub by_chain {
($m,$n,$o) = $a =~ /(\D*)(\d+)(\D*)/ ;
($p,$q,$r) = $b =~ /(\D*)(\d+)(\D*)/ ;
if ($n != $q){ $n <=> $q }
elsif ( $m ne $p){ $m cmp $p }
else { $o cmp $r }
}
}
sub by_index {
my ($i) = $a =~ /(\d+)/;
my ($j) = $b =~ /(\d+)/;
$i <=> $j
}
sub non_track_dispatch {
# loop -> loop
#
# assign chain_id to edge based on chain_id of left-side loop's
# corresponding track:
#
# hihat_out -- J7a -> Main_in
#
# soundcard_in -> wav_out (rec_file)
#
# currently handled using an anonymous track
#
# we expect edge attributes
# to have been provided for handling this.
# loop -> soundcard_out
#
# track7-soundcard_out as aux_send will have chain id S7
# that will be transferred by expand_graph() to
# the new edge, loop-soundcard-out
# we will issue two IO objects, one for the chain input
# fragment, one for the chain output
my $edge = shift;
logpkg(__FILE__,__LINE__,'debug',"non-track IO dispatch:",join ' -> ',@$edge);
my $eattr = $g->get_edge_attributes(@$edge) // {};
logpkg(__FILE__,__LINE__,'debug',"found edge attributes: ",json_out($eattr)) if $eattr;
my $vattr = $g->get_vertex_attributes($edge->[0]) // {};
logpkg(__FILE__,__LINE__,'debug',"found vertex attributes: ",json_out($vattr)) if $vattr;
if ( ! $eattr->{chain_id} and ! $vattr->{chain_id} ){
my $n = $eattr->{n} || $vattr->{n};
$eattr->{chain_id} = jumper_count($n);
}
my @direction = qw(input output);
map{
my $direction = shift @direction;
my $class = Audio::Nama::IO::get_class($_, $direction);
my $attrib = {%$vattr, %$eattr};
$attrib->{endpoint} //= $_ if Audio::Nama::Graph::is_a_loop($_);
logpkg(__FILE__,__LINE__,'debug',"non-track: $_, class: $class, chain_id: $attrib->{chain_id},","device_id: $attrib->{device_id}");
my $io = $class->new($attrib ? %$attrib : () ) ;
$g->set_edge_attribute(@$edge, $direction, $io);
$io;
} @$edge;
}
{
### counter for jumper chains
#
# sequence: J1 J1a J1b J1c, J2, J3, J4, J4d, J4e
my %used;
my $counter;
my $prefix = 'J';
reset_aux_chain_counter();
sub reset_aux_chain_counter {
%used = ();
$counter = 'a';
}
sub jumper_count {
my $track_index = shift;
my $try1 = $prefix . $track_index;
$used{$try1}++, return $try1 unless $used{$try1};
$try1 . $counter++;
}
}
sub dispatch { # creates an IO object from a graph edge
my $edge = shift;
return non_track_dispatch($edge) if not grep{ $tn{$_} } @$edge ;
logpkg(__FILE__,__LINE__,'debug','dispatch: ',join ' -> ', @$edge);
my($name, $endpoint, $direction) = decode_edge($edge);
logpkg(__FILE__,__LINE__,'debug',"name: $name, endpoint: $endpoint, direction: $direction");
my $track = $tn{$name};
my $class = Audio::Nama::IO::get_class( $endpoint, $direction );
# we need the $direction because there can be
# edges to and from loop,Main_in
my @args = (track => $name,
endpoint => massaged_endpoint($track, $endpoint, $direction),
chain_id => $tn{$name}->n, # default
override($name, $edge)); # priority: edge > node
#say "dispatch class: $class";
my $io = $class->new(@args);
$g->set_edge_attribute(@$edge, $direction => $io );
$io
}
sub massaged_endpoint {
my ($track, $endpoint, $direction) = @_;
if ( $endpoint =~ /^(loop_in|loop_out)$/ ){
my $final = ($direction eq 'input' ? $track->source_id : $track->send_id );
$final =~ s/^loop,//;
$final
} else { $endpoint }
}
sub decode_edge {
# assume track-endpoint or endpoint-track
# return track, endpoint
my ($a, $b) = @{$_[0]};
#say "a: $a, b: $b";
my ($name, $endpoint) = $tn{$a} ? @{$_[0]} : reverse @{$_[0]} ;
my $direction = $tn{$a} ? 'output' : 'input';
($name, $endpoint, $direction)
}
sub override {
# data from edges has priority over data from vertexes
# we specify $name, because it could be left or right
# vertex
logsub((caller(0))[3]);
my ($name, $edge) = @_;
(override_from_vertex($name), override_from_edge($edge))
}
sub override_from_vertex {
my $name = shift;
warn("undefined graph\n"), return () unless (ref $g) =~ /Graph/;
my $attr = $g->get_vertex_attributes($name);
$attr ? %$attr : ();
}
sub override_from_edge {
my $edge = shift;
warn("undefined graph\n"), return () unless (ref $g) =~ /Graph/;
my $attr = $g->get_edge_attributes(@$edge);
$attr ? %$attr : ();
}
sub write_chains {
logsub((caller(0))[3]);
## write general options
my @globals;
my $format = signal_format($config->{devices}->{jack}->{signal_format},2); # HARDCODED XXX
push @globals, $config->{ecasound_globals}->{common};
push @globals, "-f:$format", join(',', '-G:jack',
$config->{ecasound_jack_client_name},
$config->{jack_transport_mode}
) if $jack->{jackd_running};
push @globals, "-b", $config->buffersize, $config->globals_realtime;
my $globals = join " ", @globals;
my $ecs_file = join "\n\n",
"# ecasound chainsetup file",
"# general",
$globals,
"# audio inputs",
join("\n", @input_chains), "";
$ecs_file .= join "\n\n",
"# post-input processing",
join("\n", @post_input), "" if @post_input;
$ecs_file .= join "\n\n",
"# pre-output processing",
join("\n", @pre_output), "" if @pre_output;
$ecs_file .= join "\n\n",
"# audio outputs",
join("\n", @output_chains), "";
logpkg(__FILE__,__LINE__,'debug',"Chain setup:\n",$ecs_file);
open(my $fh, ">", $file->chain_setup)
or die("can't open chain setup file ".$file->chain_setup.": $!");
print $fh $ecs_file;
close $fh;
( run in 1.146 second using v1.01-cache-2.11-cpan-97f6503c9c8 )