diff options
-rwxr-xr-x | bin/AudioTestServer | 136 | ||||
-rw-r--r-- | etc/arctica/pulseaudio/daemon.conf | 6 | ||||
-rw-r--r-- | etc/arctica/pulseaudio/default.pa | 3 | ||||
-rw-r--r-- | lib/Arctica/Services/Audio/Server/PulseAudio/PAVirtualDevices.pm | 35 | ||||
-rw-r--r-- | lib/Arctica/Services/Audio/Server/PulseAudio/PulseAudio2GST.pm | 360 | ||||
-rw-r--r-- | lib/Arctica/Services/Audio/Server/PulseAudio/ThreadGST_server.pm | 477 |
6 files changed, 975 insertions, 42 deletions
diff --git a/bin/AudioTestServer b/bin/AudioTestServer index 1551f73..2d57286 100755 --- a/bin/AudioTestServer +++ b/bin/AudioTestServer @@ -1,4 +1,4 @@ -#!/usr/bin/perl -T -I /home/testx/arctica/HACK/convergence/perl/modules +#!/usr/bin/perl -T -I /audiotest/perlmodules/ ################################################################################ # _____ _ # |_ _| |_ ___ @@ -62,8 +62,8 @@ use strict; use Data::Dumper; use Arctica::Core::eventInit qw(genARandom BugOUT); use Arctica::Core::JABus::Socket; -use Arctica::Services::Audio::PulseAudio::PAVirtualDevices; -use Arctica::Services::Audio::PulseAudio::PulseAudio2GST; +use Arctica::Services::Audio::Server::PulseAudio::PAVirtualDevices; +use Arctica::Services::Audio::Server::PulseAudio::PulseAudio2GST; #FIXME ADD something that checks for active PulseAudio before we start doing our stuff.... my $ACO = Arctica::Core::eventInit->new({ @@ -71,72 +71,134 @@ my $ACO = Arctica::Core::eventInit->new({ app_class =>'amoduletester', app_version=>'0.0.1.1'}); + my $JABusServer = Arctica::Core::JABus::Socket->new($ACO,{ type => "unix", destination => "local", is_server => 1, handle_in_dispatch => { # heartbeat => \&heartbeat, #FIXME heartbeat will be intergrated in RTT and persistency code in JABus... - gstctl => \&pa2gst_ctl,# Client facing stuff must be NON pulse centric + gstctl => \&pa2gst_ctl,# Client facing stuff must be NON pulse centric init => \&client_init, -# cmd => \&client_init,# JABus runtime control +# cmd => \&client_init,# JABus runtime control # sub => \&subsrvc,# JABus runtime notifications service }, }); -my $PA2GST = Arctica::Services::Audio::PulseAudio::PulseAudio2GST->new($ACO,$JABusServer); +open(SID,">$ACO->{'a_dirs'}{'tmp_adir'}/audiotest_socet_id"); +print SID "$JABusServer->{'_socket_id'}\n"; +close(SID); + +my $PA2GST = Arctica::Services::Audio::Server::PulseAudio::PulseAudio2GST->new($ACO,$JABusServer); + +if (@ARGV) { + foreach my $arg (@ARGV) { + BugOUT(8,"ARG:\t$arg\t:ARG\n"); + + if ($arg =~ /^\-bitrate=([0-9\:]{1,})/) { + $PA2GST->set_bitrate($1); + } elsif ($arg =~ /^\-dgst_soc_port\=([io]\d*)\:(\d*)$/) {# FIXME by the time we're doing something with unix sockets this will be looooooong gone... + $PA2GST->set_device_gst_port($1,$2); + } elsif ($arg =~ /^\-dgst_soc_type\=([a-z]{3,5})/) { + $PA2GST->set_device_socket_type($1); + } + + } +} else { + BugOUT(0,"NO ARGS?"); +} + + -my $PA_VDev = Arctica::Services::Audio::PulseAudio::PAVirtualDevices->new($ACO,{ + +my $PA_VDev = Arctica::Services::Audio::Server::PulseAudio::PAVirtualDevices->new($ACO,{ hook_device_state => \&handle_PA_device_events, -# hook_device_state => sub {$PA2GST->handle_PA_device_events(@_)}, }); +my $wtf = Glib::Timeout->add (500, \&chk_bitrate_file, undef, 1 );# FIXME DIRTY HACK... WILL BE HANDELED ON JABus in the TeKi enabled version -#print Dumper($PA_VDev); $ACO->{'Glib'}{'MainLoop'}->run; + +sub pa2gst_ctl { + my $JDATA = $_[0]; + my (undef,$device_soc_style) = $PA2GST->get_device_socket_type(); +# print "SOCK STYLE: $device_soc_style\n"; +# print "GSTCTL_JSON:\n",Dumper($JDATA); +# print "Action:\t$JDATA->{'action'}\n"; + if ($JDATA->{'action'} eq "ready") { + if ($JDATA->{'type'} eq "output") { + if ($JDATA->{'idnum'} =~ /^(\d{1,})$/) { + my $idnum = $1; + if ($device_soc_style eq "stream") { + $PA2GST->start_output($idnum,$PA_VDev->{'pa_vdev'}{'output'}{$idnum}{'pa_sink_name'}); + } + + } + } + } + return 1; +} + + + sub handle_PA_device_events {# FIXME this has moved into PulseAudio2GST...: But then it came back out here... print "-------------------------------\n"; print Dumper(@_),"\n"; + my $type = $_[0]; my $idnum = $_[1]; my $name = $_[2]; my $new_state = $_[3]; my $clientID = $PA2GST->get_active_client_id(); - print "CID#\t$clientID\n"; + my (undef,$device_soc_style) = $PA2GST->get_device_socket_type(); + if ($clientID) { - + if ($type eq "input") { if ($new_state eq "R") { # START LOCAL THEN CLIENTSIDE - $PA2GST->start_input($idnum,$PA_VDev->{'pa_vdev'}{'input'}{0}{'pa_sink_name'}); - $JABusServer->server_send($clientID,"gstctl",{ - action => "start", - type => $type, - idnum => $idnum, - rate => $PA2GST->get_input_bitrate($idnum), - }); + $PA2GST->start_input($idnum,$PA_VDev->{'pa_vdev'}{$type}{$idnum}{'pa_sink_name'},sub { + $JABusServer->server_send($clientID,"gstctl",{ + action => "start", + type => $type, + idnum => $idnum, + bitrate => $PA2GST->get_bitrate("input"), + }); + }); + + open(HM,">/tmp/hotmic");print HM time;close(HM);# FIXME DIRTY HACK... WILL BE HANDELED ON JABus in the TeKi enabled version + } elsif($new_state eq "S") { + $PA2GST->stop_input($idnum); + $JABusServer->server_send($clientID,"gstctl",{ action => "stop", type => $type, idnum => $idnum, }); - $PA2GST->stop_input($idnum); + + if (-f "/tmp/hotmic") {unlink("/tmp/hotmic");}# FIXME DIRTY HACK... WILL BE HANDELED ON JABus in the TeKi enabled version + } + } elsif ($type eq "output") { if ($new_state eq "R") { - # START CLIENT SIDE THEN LOCAL - $PA2GST->start_output($idnum,$PA_VDev->{'pa_vdev'}{'input'}{0}{'pa_sink_name'}); + + unless ($device_soc_style ne "datagram") { + $PA2GST->start_output($idnum,$PA_VDev->{'pa_vdev'}{$type}{$idnum}{'pa_sink_name'}); + } + $JABusServer->server_send($clientID,"gstctl",{ action => "start", type => $type, idnum => $idnum, }); + } elsif ($new_state eq "S") { $JABusServer->server_send($clientID,"gstctl",{ action => "stop", @@ -145,7 +207,8 @@ sub handle_PA_device_events {# FIXME this has moved into PulseAudio2GST...: But }); $PA2GST->stop_output($idnum); } - } + + } } } @@ -158,14 +221,37 @@ sub client_init { BugOUT(8,"NEW CLIENT! ( $client_ID )"); if ($ACO->{'aobj'}{'AudioServer'}{'client_ID'}) { client_cleanup($ACO->{'aobj'}{'AudioServer'}{'client_ID'}); - } - + } + $ACO->{'aobj'}{'AudioServer'}{'client_ID'} = $client_ID; $PA2GST->set_jbus_client_id($client_ID); # FIXME Add something to force check of pulse vdev status at this point.... - + $PA_VDev->force_chk_dev_state(); # $TheJBUS->server_send($client_ID,'init','GOOD TO GO!'); + return 1; +} + +sub chk_bitrate_file {# FIXME DIRTY HACK... WILL BE HANDELED ON JABus in the TeKi enabled version + if (-f "/tmp/ch_bitrate") { + open(BR,"/tmp/ch_bitrate"); + my ($rate,undef) = <BR>; + close(BR); + if ($rate =~ /^([0-9\:]{1,})/) { + BugOUT(1,"CHBITRATE: $rate"); + $PA2GST->set_bitrate($rate); + my $clientID = $PA2GST->get_active_client_id(); + if ($clientID) { + my $input_rate = $PA2GST->get_bitrate("input"); + $JABusServer->server_send($clientID,"gstctl",{ + action => "ch_input_bitrate", + bitrate => $input_rate , + }); + } + } + unlink("/tmp/ch_bitrate"); + } + return 1; } sub client_cleanup { diff --git a/etc/arctica/pulseaudio/daemon.conf b/etc/arctica/pulseaudio/daemon.conf index 2371057..94a7ae3 100644 --- a/etc/arctica/pulseaudio/daemon.conf +++ b/etc/arctica/pulseaudio/daemon.conf @@ -17,9 +17,9 @@ ## more information. Default values are commented out. Use either ; or # for ## commenting. -daemonize = no +daemonize = yes fail = yes -allow-module-loading = no +allow-module-loading = yes allow-exit = no use-pid-file = no system-instance = no @@ -41,7 +41,7 @@ system-instance = no ; dl-search-path = (depends on architecture) load-default-script-file = yes -default-script-file = /etc/arctica/pulse/default.pa +default-script-file = /audiotest/etc/pulse/default.pa ; log-target = auto ; log-level = notice diff --git a/etc/arctica/pulseaudio/default.pa b/etc/arctica/pulseaudio/default.pa index 464df41..317a08f 100644 --- a/etc/arctica/pulseaudio/default.pa +++ b/etc/arctica/pulseaudio/default.pa @@ -13,7 +13,7 @@ load-module module-native-protocol-unix load-module module-null-sink sink_name=arctica.output0 channels=2 sink_properties=device.description="VirtualOutput" load-module module-null-sink sink_name=arctica.input0 channels=1 sink_properties=device.description=".VirtualInputSink" -load-module module-remap-source master=arctica.input0.monitor source_name=arctica.mic0 source_properties=device.description="VirtualMicrophone" +#load-module module-remap-source master=arctica.input0.monitor source_name=arctica.mic0 source_properties=device.description="VirtualMicrophone" remix=0 ### Honour intended role device property @@ -54,6 +54,7 @@ load-module module-x11-publish .fail .endif + ### Make some devices default set-default-sink arctica.output0 set-default-source arctica.mic0 diff --git a/lib/Arctica/Services/Audio/Server/PulseAudio/PAVirtualDevices.pm b/lib/Arctica/Services/Audio/Server/PulseAudio/PAVirtualDevices.pm index 4c7435f..9e027ca 100644 --- a/lib/Arctica/Services/Audio/Server/PulseAudio/PAVirtualDevices.pm +++ b/lib/Arctica/Services/Audio/Server/PulseAudio/PAVirtualDevices.pm @@ -57,7 +57,7 @@ # Copyright (C) 2015-2017 Mike Gabriel <mike.gabriel@das-netzwerkteam.de> # ################################################################################ -package Arctica::Services::Audio::PulseAudio::PAVirtualDevices; +package Arctica::Services::Audio::Server::PulseAudio::PAVirtualDevices; use strict; use Exporter qw(import); use Arctica::Core::BugOUT::Basics qw( BugOUT ); @@ -99,12 +99,12 @@ sub new { $self->{'pa_vdev'}{'output'}{0}{'gst_thread'} = 0; # "action_map" may go away.... (limited usefullness if any in current itteration of code.) - $self->{'pa_vdev'}{'action_map'}{'by_name'}{'arctica.mic0'} = + $self->{'pa_vdev'}{'action_map'}{'by_name'}{'arctica.mic0'} = { type => "input", idnum => 0, }; - $self->{'pa_vdev'}{'action_map'}{'by_name'}{'arctica.output0'} = + $self->{'pa_vdev'}{'action_map'}{'by_name'}{'arctica.output0'} = { type => "output", idnum => 0, @@ -126,7 +126,7 @@ sub new { exec_path => "/usr/bin/pactl",# FIXME Make this configurable! exec_cl_argv => ["subscribe"], - }); + }); $arctica_core_object->{'aobj'}{'AudioServer'}{'PA_Virtual_Devices'} = \$self; @@ -137,6 +137,7 @@ sub new { return $self; } + sub _suspend_idle {# Cause we can't always rely on PulseAudio for this.... (PA BUG?) my $self = $_[0]; foreach my $idnum (keys %{$self->{'pa_vdev'}{'input'}}) { @@ -202,8 +203,8 @@ sub _set_device_our_state { } - } - + } + BugOUT(9,"set_device_our_state DONE"); } @@ -269,13 +270,13 @@ sub _set_device_pa_state { } - } + } BugOUT(9,"set_device_pa_state DONE"); } sub _pulse_event_handler { -# BugOUT(9,"_pulse_event_handler: ENTER"); + BugOUT(9,"_pulse_event_handler: ENTER"); my $self = $_[0]; if ($_[1] =~ /Event\s*\'change\'\s*on\s*(\w{4,6})\s/) { my $chWhere = $1; @@ -291,11 +292,11 @@ sub _pulse_event_handler { $self->_set_device_pa_state($devices->{$name}{'type'},$devices->{$name}{'idnum'},$name,"I"); } elsif ($2 =~ /(SUSPENDED)$/) { $self->_set_device_pa_state($devices->{$name}{'type'},$devices->{$name}{'idnum'},$name,"S"); - } elsif ($2 =~ /(RUNNING)$/) { + } elsif ($2 =~ /(RUNNING)$/) { $self->_set_device_pa_state($devices->{$name}{'type'},$devices->{$name}{'idnum'},$name,"R"); } - } - } + } + } } close(PACTL); } else { @@ -303,10 +304,18 @@ sub _pulse_event_handler { } } + BugOUT(9,"_pulse_event_handler: DONE"); return 1; -# BugOUT(9,"_pulse_event_handler: DONE"); } +sub force_chk_dev_state { + BugOUT(8,"USE OF/(THE?) FORCE!!!: ENTER"); + my $self = $_[0]; +# (W)HACKY BUT WORKS FINE FOR NOW + $self->_pulse_event_handler("Event 'change' on source "); + $self->_pulse_event_handler("Event 'change' on sink "); + BugOUT(8,"THE LAST JEDI RETIRES..."); +} sub get_list { BugOUT(8,"PAVirtualDevices getLIST"); @@ -316,7 +325,7 @@ sub get_list { return $self->{'pa_vdev'}{'action_map'}{'by_name'}; } } -} +} 1; diff --git a/lib/Arctica/Services/Audio/Server/PulseAudio/PulseAudio2GST.pm b/lib/Arctica/Services/Audio/Server/PulseAudio/PulseAudio2GST.pm new file mode 100644 index 0000000..9ac5de0 --- /dev/null +++ b/lib/Arctica/Services/Audio/Server/PulseAudio/PulseAudio2GST.pm @@ -0,0 +1,360 @@ +################################################################################ +# _____ _ +# |_ _| |_ ___ +# | | | ' \/ -_) +# |_| |_||_\___| +# _ _ ____ _ _ +# / \ _ __ ___| |_(_) ___ __ _ | _ \ _ __ ___ (_) ___ ___| |_ +# / _ \ | '__/ __| __| |/ __/ _` | | |_) | '__/ _ \| |/ _ \/ __| __| +# / ___ \| | | (__| |_| | (_| (_| | | __/| | | (_) | | __/ (__| |_ +# /_/ \_\_| \___|\__|_|\___\__,_| |_| |_| \___// |\___|\___|\__| +# |__/ +# The Arctica Modular Remote Computing Framework +# +################################################################################ +# +# Copyright (C) 2015-2016 The Arctica Project +# http://arctica-project.org/ +# +# This code is dual licensed: strictly GPL-2 or AGPL-3+ +# +# GPL-2 +# ----- +# 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; version 2 of the License. +# +# 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; if not, write to the +# Free Software Foundation, Inc., +# +# 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. +# +# AGPL-3+ +# ------- +# This programm is free software; you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as published by +# the Free Software Foundation; either version 3 of the License, or +# (at your option) any later version. +# +# This programm 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 Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program; if not, write to the +# Free Software Foundation, Inc., +# 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. +# +# Copyright (C) 2015-2016 Guangzhou Nianguan Electronics Technology Co.Ltd. +# <opensource@gznianguan.com> +# Copyright (C) 2015-2016 Mike Gabriel <mike.gabriel@das-netzwerkteam.de> +# +################################################################################ +package Arctica::Services::Audio::Server::PulseAudio::PulseAudio2GST; +use strict; +use Exporter qw(import); +use Arctica::Core::BugOUT::Basics qw( BugOUT ); +use Arctica::Core::Mother::Forker; +use Data::Dumper;# Remove this before release! (unless we're still dependant) + +# Be very selective about what (if any) gets exported by default: +our @EXPORT = qw( ); +# And be mindfull of what we lett the caller request here too: +our @EXPORT_OK = qw( ); + +my $arctica_core_object; + +sub new { + BugOUT(9,"PulseAudio2GST new->ENTER"); + my $class_name = $_[0];# Be EXPLICIT!! DON'T SHIFT OR "@_"; + $arctica_core_object = $_[1]; + my $JBUS_Server = $_[2]; + my $self = { + isArctica => 1, # Declare that this is a Arctica "something" + aobject_name => "PulseAudio2GST", + JBUS_Server => $JBUS_Server, + _defaults => { + output_bitrate => 64, + input_bitrate => 32, + }, + }; + + bless($self, $class_name); + + + + + + $arctica_core_object->{'aobj'}{'AudioServer'}{'PulseAudio2GST'} = \$self; + + BugOUT(9,"PulseAudio2GST new->DONE"); + + return $self; +} + + +sub start_output { + BugOUT(9,"PulseAudio2GST start_output->ENTER"); + my $self = $_[0]; + my $id_num = $_[1]; + my $pa_dev = $_[2]; + BugOUT(8,"Starting OUTPUT:\tAIOD#$id_num\tPA: $pa_dev"); + if ($self->{'vdev'}{'output'}{$id_num}{'port'} and $self->{'_settings'}{'socket_type'}) { + my $pa_dev_monitor = $pa_dev; + unless ($pa_dev =~ /\.monitor$/) { + $pa_dev_monitor = "$pa_dev.monitor"; + BugOUT(9,"Append '.monitor' to $pa_dev_monitor"); + } + my $bitrate = $self->get_bitrate("output"); + $self->{'vdev'}{'output'}{$id_num}{'running'} = 1; + $self->{'vdev'}{'output'}{$id_num}{'gst_thread'} = Arctica::Core::Mother::Forker->new($arctica_core_object,{ + child_name => 'thread_gst', + fork_style => 'interactive_pty', + handle_stdeoc => sub {return 1;}, + return_stdin => 1, + exec_hold => 0, + exec_path => "/audiotest/bin/launch_server_ThreadGST",# FIXME GET FULL PATH FROM CFG OR SOMETHING LIKE THAT + exec_cl_argv => [ + "-src=$self->{'_settings'}{'socket_type'}", + "-port=$self->{'vdev'}{'output'}{$id_num}{'port'}", + "-pa_device_name=$pa_dev_monitor", + "-start_bitrate=$bitrate", + ], + }); + } else { + BugOUT(1,"PulseAudio2GST start_output port and socket type not set?!! WTF?!"); + } + BugOUT(9,"PulseAudio2GST start_output->DONE"); +} + + +sub stop_output { + BugOUT(9,"PulseAudio2GST stop_output->ENTER"); + my $self = $_[0]; + my $id_num = $_[1]; + if ($self->{'vdev'}{'output'}{$id_num}{'gst_thread'}) { + $self->{'vdev'}{'output'}{$id_num}{'gst_thread'}->send("cmd:stop:"); +# FIXME FORCE DESTRUCTION OF Mother::Forker object here? + $self->{'vdev'}{'output'}{$id_num}{'running'} = 0; + } + BugOUT(9,"PulseAudio2GST stop_output->DONE"); +} + + +sub start_input { + BugOUT(9,"PulseAudio2GST start_input->ENTER"); + my $self = $_[0]; + my $id_num = $_[1]; + my $pa_dev = $_[2]; + my $run_on_ready = $_[3]; + BugOUT(8,"Starting INPUT:\tAIOD#$id_num\tPA: $pa_dev"); + if ($self->{'vdev'}{'input'}{$id_num}{'port'} and $self->{'_settings'}{'socket_type'}) { + $self->{'vdev'}{'input'}{$id_num}{'running'} = 1; + $self->{'vdev'}{'input'}{$id_num}{'gst_thread'} = Arctica::Core::Mother::Forker->new($arctica_core_object,{ + child_name => 'thread_gst', + fork_style => 'interactive_pty', + handle_stdeoc => sub { + if ($_[0] =~ /^status:sink_ready:(\d{1,3}):$/) { + $run_on_ready->($1); + } + }, + return_stdin => 1, + exec_hold => 0, + exec_path => "/audiotest/bin/launch_server_ThreadGST",# FIXME GET FULL PATH FROM CFG OR SOMETHING LIKE THAT + exec_cl_argv => [ + "-snk=$self->{'_settings'}{'socket_type'}", + "-port=$self->{'vdev'}{'input'}{$id_num}{'port'}", + "-pa_device_name=$pa_dev", + ], + }); + } else { + BugOUT(1,"PulseAudio2GST start_input port and socket type not set?!! WTF?!"); + } + BugOUT(9,"PulseAudio2GST start_input->DONE"); +} + + +sub stop_input { + BugOUT(9,"PulseAudio2GST stop_input->ENTER"); + my $self = $_[0]; + my $id_num = $_[1]; + if ($self->{'vdev'}{'input'}{$id_num}{'gst_thread'}) { + $self->{'vdev'}{'input'}{$id_num}{'gst_thread'}->send("cmd:stop:"); +# FIXME FORCE DESTRUCTION OF Mother::Forker object here? + $self->{'vdev'}{'input'}{$id_num}{'running'} = 0; + } + BugOUT(9,"PulseAudio2GST stop_input->DONE"); +} + + + +sub thread_cmd { + BugOUT(9,"PulseAudio2GST thread_cmd->ENTER"); + my $self = $_[0]; + my $type = $_[1]; + my $idnum= $_[2]; + my $cmd = $_[3]; + BugOUT(9,"$type\t$idnum\t$cmd\n"); + if ($self->{'vdev'}{$type}{$idnum}{'gst_thread'}) { + BugOUT(9,"sTEP 2; $type\t$idnum\t$cmd\n"); + if ($cmd =~ /^([a-z]{1,10})$/) { + BugOUT(9,"sTEP 3; $type\t$idnum\t$cmd\n"); + $self->{'vdev'}{$type}{$idnum}{'gst_thread'}->send("cmd:$1:"); + BugOUT(8,"PulseAudio2GST: thread_cmd: Sent '$1' to $type #$idnum"); + } + } + BugOUT(9,"PulseAudio2GST thread_cmd->DONE"); + return 1; +} + + + +sub set_jbus_client_id { + my $self = $_[0]; + $self->{'jbus_client_id'} = $_[1]; +} + + +sub set_device_socket_type { + BugOUT(9,"PulseAudio2GST set_device_socket_type->ENTER"); + my $self = $_[0]; + + if (($_[1] eq "tcp") or ($_[1] eq "unixs")) { + $self->{'_settings'}{'socket_type'} = $_[1]; + $self->{'_settings'}{'com_style'} = "stream"; + BugOUT(9,"set_device_socket_type: socket type set to $_[1]/stream"); + } elsif (($_[1] eq "udp") or ($_[1] eq "unixd")) { + $self->{'_settings'}{'socket_type'} = $_[1]; + $self->{'_settings'}{'com_style'} = "datagram"; + BugOUT(9,"set_device_socket_type: socket type set to $_[1]/datagram"); + } else { + BugOUT(0,"set_device_socket_type: '$_[1]' is not a valid socket_type"); + } + + BugOUT(9,"PulseAudio2GST set_device_socket_type->DONE"); +} + + + +sub get_device_socket_type { + my $self = $_[0]; + if ($self->{'_settings'}{'socket_type'} and $self->{'_settings'}{'com_style'}) { + return ($self->{'_settings'}{'socket_type'},$self->{'_settings'}{'com_style'}); + } else { + return (0,0); + } +} + + +sub set_device_gst_port { + BugOUT(9,"PulseAudio2GST set_device_gst_port->ENTER"); + my $self = $_[0]; + my $device = $_[1]; + my $port = $_[2]; + + if ($device =~ /^o(\d{1,})/) { + $self->{'vdev'}{'output'}{$1}{'port'} = $port; + BugOUT(9,"set_device_gst_port: output:$1:$port"); + } elsif ($device =~ /^i(\d{1,})/) { + $self->{'vdev'}{'input'}{$1}{'port'} = $port; + BugOUT(9,"set_device_gst_port: input:$1:$port"); + } else { + BugOUT(2,"set_device_gst_port: Failed to set device '$device' port to '$port'"); + } + + BugOUT(9,"PulseAudio2GST set_device_gst_port->DONE"); +} + + +sub set_bitrate { + BugOUT(9,"PulseAudio2GST set_bitrate->ENTER"); + my $self = $_[0]; + my $new_output_rate = 0; + my $new_input_rate = 0; +# We only check that things are somewhat sane here... If we're above or bellow the accepted range, the closest supported range is used +# Redundant sanity checks are good, but would like to avoid having redundant decission making... (decision is made in the GST thread). + if ($_[1] =~ /^(\d{1,})\:(\d{1,})$/) { + $new_output_rate = $1; + $new_input_rate = $2; + BugOUT(9,"Got asymetrical I/O BW ($1 : $2)"); + } elsif ($1 =~ /^(\d{1,})$/) { + $new_output_rate = $1; + $new_input_rate = $1; + BugOUT(9,"Symetrical I/O BW? ($1)"); + } else { + BugOUT(8,"Weird bitrate format... Using previously set or default BW..."); + } + + if (($new_output_rate > 0) and ($new_input_rate > 0)) { + if (($new_output_rate > 1000) or($new_input_rate > 1000)) { + BugOUT(1,"Bitrates0 ($new_output_rate : $new_input_rate) seem high, expecting KILO bit values so maybe knock of a few zeros?"); + } + + if ($self->{'_settings'}{'output_bitrate'} ne $new_output_rate) { + $self->{'_settings'}{'output_bitrate'} = $new_output_rate; + foreach my $idnum (keys %{$self->{'vdev'}{'output'}}) { + + if ($self->{'vdev'}{'output'}{$idnum}{'running'}) { + if ($self->{'vdev'}{'output'}{$idnum}{'gst_thread'}) { + $self->{'vdev'}{'output'}{$idnum}{'gst_thread'}->send("set:bitrate:$new_output_rate"); + } + } + + } + BugOUT(9,"Output bitrate set to $new_output_rate"); + } else { + BugOUT(9,"Output bitrate is unchanged..."); + } + + if ($self->{'_settings'}{'input_bitrate'} ne $new_input_rate) { + $self->{'_settings'}{'input_bitrate'} = $new_input_rate; +# FIXME Add function to brodcast rate change to "live" input threads + BugOUT(9,"Input bitrate set to $new_input_rate"); + } else { + BugOUT(9,"Input bitrate is unchanged..."); + } + + + } + BugOUT(9,"PulseAudio2GST set_bitrate->DONE"); +} + + +sub get_bitrate { + my $self = $_[0]; + if ($_[1] eq "output") { + if ($self->{'_settings'}{'output_bitrate'}) { + return $self->{'_settings'}{'output_bitrate'}; + } else { + return $self->{'_defaults'}{'output_bitrate'}; + } + } elsif ($_[1] eq "input") { + if ($self->{'_settings'}{'input_bitrate'}) { + return $self->{'_settings'}{'input_bitrate'}; + } else { + return $self->{'_defaults'}{'input_bitrate'}; + } + } else { + BugOUT(2,"And you want bitrate for what? ($_[1])"); + } +} + + +sub get_active_client_id { + my $self = $_[0]; + if ($self->{'jbus_client_id'}) {# FIXME Add stuff to chek if this client is still really truly active. + return $self->{'jbus_client_id'}; + } else { + return 0; + } +} + +1; + diff --git a/lib/Arctica/Services/Audio/Server/PulseAudio/ThreadGST_server.pm b/lib/Arctica/Services/Audio/Server/PulseAudio/ThreadGST_server.pm new file mode 100644 index 0000000..f8b9b37 --- /dev/null +++ b/lib/Arctica/Services/Audio/Server/PulseAudio/ThreadGST_server.pm @@ -0,0 +1,477 @@ +################################################################################ +# _____ _ +# |_ _| |_ ___ +# | | | ' \/ -_) +# |_| |_||_\___| +# _ _ ____ _ _ +# / \ _ __ ___| |_(_) ___ __ _ | _ \ _ __ ___ (_) ___ ___| |_ +# / _ \ | '__/ __| __| |/ __/ _` | | |_) | '__/ _ \| |/ _ \/ __| __| +# / ___ \| | | (__| |_| | (_| (_| | | __/| | | (_) | | __/ (__| |_ +# /_/ \_\_| \___|\__|_|\___\__,_| |_| |_| \___// |\___|\___|\__| +# |__/ +# The Arctica Modular Remote Computing Framework +# +################################################################################ +# +# Copyright (C) 2015-2016 The Arctica Project +# http://arctica-project.org/ +# +# This code is dual licensed: strictly GPL-2 or AGPL-3+ +# +# GPL-2 +# ----- +# 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; version 2 of the License. +# +# 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; if not, write to the +# Free Software Foundation, Inc., +# +# 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. +# +# AGPL-3+ +# ------- +# This programm is free software; you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as published by +# the Free Software Foundation; either version 3 of the License, or +# (at your option) any later version. +# +# This programm 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 Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program; if not, write to the +# Free Software Foundation, Inc., +# 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. +# +# Copyright (C) 2015-2016 Guangzhou Nianguan Electronics Technology Co.Ltd. +# <opensource@gznianguan.com> +# Copyright (C) 2015-2016 Mike Gabriel <mike.gabriel@das-netzwerkteam.de> +# +################################################################################ +package Arctica::Services::Audio::Server::PulseAudio::ThreadGST_server; +use strict; +use Exporter qw(import); +use Arctica::Core::BugOUT::Basics qw( BugOUT ); +#use Arctica::Core::Mother::Forker; +use IO::Handle; +use Time::HiRes qw( usleep ); +use GStreamer1; +use Data::Dumper;# Remove this before release! (unless we're still dependant) + +# Be very selective about what (if any) gets exported by default: +our @EXPORT = qw( ); +# And be mindfull of what we lett the caller request here too: +our @EXPORT_OK = qw( ); + +my $ACO; + +GStreamer1::init([ $0 ]);# Initiate GST + + +sub new { + BugOUT(9,"ThreadGST new->ENTER"); + my $class_name = $_[0];# Be EXPLICIT!! DON'T SHIFT OR "@_"; + $ACO = $_[1]; + + my $self = { + isArctica => 1, # Declare that this is a Arctica "something" + aobject_name => "ThreadGST", + _default_out_rate => 64,# FIXME Look for defaults in a config file somewhere? + _default_in_rate => 32,# FIXME (We have a cfg file handler module somewhere.... Use it soon?) + _default_pasuspender_fpath => "/usr/bin/pasuspender",# FIXME ^^^^^^^^^^ + }; + bless($self, $class_name); + + if ($_[2]) {# FIXME!!!! We got some fancypants module somewhere that handles this.... switch to using that one, some day...!? + foreach my $arg (@{$_[2]}) { + BugOUT(8,"ARGY:\t$arg\t:ARG\n"); + if (($arg =~ /^\-(src)\=([a-z]{3,5})/) or ($arg =~ /^\-(snk)\=([a-z]{3,5})/) ){ + # print "1:\t$1\n2:\t$2\n"; + if ($1 eq "src") { + $self->_set_argv("src_or_snk","src"); + } elsif ($1 eq "snk") { + $self->_set_argv("src_or_snk","snk"); + } else { + BugOUT(9,"Not a server nor a client... what then...?"); + } + if (($2 eq "tcp") or ($2 eq "unixs")) { + $self->_set_argv("socket_type",$2); + $self->_set_argv("com_style","stream"); + } elsif (($2 eq "udp") or ($2 eq "unixd")) { + $self->_set_argv("socket_type",$2); + $self->_set_argv("com_style","datagram"); + } else { + BugOUT(0,"We can't be a '$2' $self->{'s_or_c'}"); + } + + } elsif ($arg =~ /^\-oo_([a-z]{2,10})\=([a-z0-9]*)/) { + $self->_set_argv("opus_$1",$2); + } elsif ($arg =~ /^\-port\=([a-z0-9]*)/) { + $self->_set_argv("port_or_unix-socket",$1); + } elsif ($arg =~ /^\-pa_device_name\=([a-zA-Z0-9\.\_\-]*)/) { + BugOUT(8,"pa_device_name", $1); + $self->_set_argv("pa_device_name", $1); + } elsif ($arg =~ /^\-clientside\=([a-z]*)/) { + if ($1 eq "pulseaudio") { + $self->_set_argv("clientside","pulseaudio"); + } else { + if ($1 ne "autoaudio") { + BugOUT(1,"-clientside=$1 ? wtf? So we're going to try to use autoaudio..."); + } + $self->_set_argv("clientside","autoaudio"); + } + + } elsif ($arg =~ /^\-wait\=1/) { + $self->_set_argv("wait",1); + } elsif ($arg =~ /^\-start_bitrate\=(\d{1,3})/) { + $self->_set_argv("bitrate",$1); + } + } + + } else { + BugOUT(0,"NO ARGS?"); + } + + $ACO->{'aobj'}{'ThreadGST'} = \$self; + + BugOUT(9,"ThreadGST new->DONE"); + return $self; +} + + +sub ch_options { + my $self = $_[0]; + my $o_name = $_[1]; + my $o_value = $_[2]; + if ($o_name eq "bitrate") { + if ($o_value =~ /^(\d{1,})$/) { + $self->_tune_gstpipe($1); + } + } else { + BugOUT(2,"WTF '$o_name'?"); + } +} + +sub _set_argv { + my $self = $_[0]; + my $arg_name = $_[1]; + my $arg_value = $_[2]; + # FIXME DO A BUNCH OF SANETIZING HERE?! + $self->{'_argv'}{$arg_name} = $arg_value; +} + +sub _get_argv { + my $self = $_[0]; + my $arg_name = $_[1]; + if ($self->{'_argv'}{$arg_name}) { + return $self->{'_argv'}{$arg_name}; + } else { + return 0; + } +} + + +sub _tune_gstpipe { + my $self = $_[0]; + my $bitrate = $_[1]; + + if ($self->{'main'}{'elements'}{'opusenc'}) { + my $opusenc_element = $self->{'main'}{'elements'}{'opusenc'}; + $bitrate =~ s/\D//g; + if ($bitrate > 384) { + $bitrate = 384; + } elsif ($bitrate < 4) { + $bitrate = 4; + } + $opusenc_element->set("bitrate" => ($bitrate * 1000)); + BugOUT(1,"\t\tTWEAK IT:\t$bitrate"); + $opusenc_element->set("inband-fec" => 1); + $opusenc_element->set("max-payload-size" => 200); + $opusenc_element->set("frame-size" => 20); + #FIXME INSERT MORE COMPLEX PERFORMANCE TUNING MATRIX STUFF HERE! + } +} + +sub _start_gstsrc { + my $self = $_[0]; + # Cleanup stuff for collecting garbage from serverside pulseaudio + $self->{'garbage'}{'pipeline'} = GStreamer1::Pipeline->new('garbagepipe'); + $self->{'garbage'}{'elements'}{'pasrc'} = GStreamer1::ElementFactory::make( pulsesrc => 'garbage_pasrc' ); + + if ($self->_get_argv("pa_device_name")) { + my $device_name = $self->_get_argv("pa_device_name"); + BugOUT(8,"PA_DEVICE_NAME: $device_name"); + $self->{'garbage'}{'elements'}{'pasrc'}->set('device' => $device_name);# 'arctica.output0.monitor' + + } else { + BugOUT(8,"PA_DEVICE_NAME: USING DEFAULT DEVICE"); + } + + $self->{'garbage'}{'elements'}{'pasrc'}->set('client-name' => 'Arctica Garbage Collector'); + $self->{'garbage'}{'elements'}{'sink'} = GStreamer1::ElementFactory::make( fakesink => 'garbage_sink' ); + $self->{'garbage'}{'pipeline'}->add($self->{'garbage'}{'elements'}{'pasrc'}); + $self->{'garbage'}{'pipeline'}->add($self->{'garbage'}{'elements'}{'sink'}); + $self->{'garbage'}{'elements'}{'pasrc'}->link($self->{'garbage'}{'elements'}{'sink'}); + $self->{'garbage'}{'pipeline'}->set_state( "playing" ); + BugOUT(8,"Garbage collection initiated"); + + my $pasuspender_fpath = $self->{'_default_pasuspender_fpath'};# FIXME FIX THIS AFTER CFG FILES HAVE BEEN PROPERLY IMPLEMENTED + + if (-X $pasuspender_fpath) { + BugOUT(9,"pasuspender full path: $pasuspender_fpath"); + my $tpath = $ENV{'PATH'}; + $ENV{'PATH'} = "/bin:/usr/bin"; + system($pasuspender_fpath,"true");# FIXME Use Mother::Forker::Light. + $ENV{'PATH'} = $tpath; + } else { + BugOUT(1,"NO pasuspender at full path: $pasuspender_fpath");# Not super critical but some audible junk may occur... + } + + + + $self->{'main'}{'pipeline'} = GStreamer1::Pipeline->new('pipeline'); + + $self->{'main'}{'elements'}{'queue1'} = GStreamer1::ElementFactory::make( queue => 'queue1' ); + $self->{'main'}{'elements'}{'queue1'}->set("silent" => 0); + $self->{'main'}{'elements'}{'queue1'}->set("leaky" => "downstream"); + $self->{'main'}{'elements'}{'queue1'}->set("max-size-time" => "30000000"); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'queue1'}); + + + $self->{'main'}{'elements'}{'pasrc'} = GStreamer1::ElementFactory::make( pulsesrc => 'pasrc' ); + + if ($self->_get_argv("pa_device_name")) { + $self->{'main'}{'elements'}{'pasrc'}->set('device' => $self->_get_argv("pa_device_name"));# 'arctica.output0.monitor' + } + + $self->{'main'}{'elements'}{'pasrc'}->set('client-name' => 'Arctica Audio Services'); + + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'pasrc'}); + $self->{'main'}{'elements'}{'pasrc'}->link($self->{'main'}{'elements'}{'queue1'}); + + + $self->{'main'}{'elements'}{'opusenc'} = GStreamer1::ElementFactory::make( opusenc => 'opusenc' ); + $self->_tune_gstpipe($self->_get_argv("bitrate")); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'opusenc'}); + $self->{'main'}{'elements'}{'queue1'}->link($self->{'main'}{'elements'}{'opusenc'}); + + if ($self->_get_argv("com_style") eq "datagram") { + + $self->{'main'}{'elements'}{'rtpopuspay'} = GStreamer1::ElementFactory::make( rtpopuspay => 'rtpopuspay' ); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'rtpopuspay'}); + $self->{'main'}{'elements'}{'opusenc'}->link($self->{'main'}{'elements'}{'rtpopuspay'}); + + if ($self->_get_argv("socket_type") eq "udp") { + + $self->{'main'}{'elements'}{'udpsink'} = GStreamer1::ElementFactory::make( udpsink => 'udpsink' ); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'udpsink'}); + $self->{'main'}{'elements'}{'rtpopuspay'}->link($self->{'main'}{'elements'}{'udpsink'}); + + $self->{'main'}{'elements'}{'udpsink'}->set('port' => $self->_get_argv("port_or_unix-socket")); + $self->{'main'}{'elements'}{'udpsink'}->set('host' => 'localhost'); + + } elsif ($self->_get_argv("socket_type") eq "unixd") { + BugOUT(0,"NOT YET IMPLEMENTED!"); + } else { + BugOUT(0,"This should never happen!"); + } + + } elsif ($self->_get_argv("com_style") eq "stream") { + + $self->{'main'}{'elements'}{'gdppay'} = GStreamer1::ElementFactory::make( gdppay => 'gdppay' ); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'gdppay'}); + $self->{'main'}{'elements'}{'opusenc'}->link($self->{'main'}{'elements'}{'gdppay'}); + + $self->{'main'}{'elements'}{'queue2'} = GStreamer1::ElementFactory::make( queue => 'queue2' ); + $self->{'main'}{'elements'}{'queue2'}->set("silent" => 0); + $self->{'main'}{'elements'}{'queue2'}->set("leaky" => "downstream"); + $self->{'main'}{'elements'}{'queue2'}->set("max-size-time" => "30000000"); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'queue2'}); + + $self->{'main'}{'elements'}{'gdppay'}->link($self->{'main'}{'elements'}{'queue2'}); + + if ($self->_get_argv("socket_type") eq "tcp") { + $self->{'main'}{'elements'}{'tcpclientsink'} = GStreamer1::ElementFactory::make( tcpclientsink => 'tcpclientsink' ); + $self->{'main'}{'elements'}{'tcpclientsink'}->set('port' => $self->_get_argv("port_or_unix-socket")); + $self->{'main'}{'elements'}{'tcpclientsink'}->set('host' => 'localhost'); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'tcpclientsink'}); + $self->{'main'}{'elements'}{'queue2'}->link($self->{'main'}{'elements'}{'tcpclientsink'}); + + } elsif ($self->_get_argv("socket_type") eq "unixs") { + BugOUT(0,"NOT YET IMPLEMENTED!"); + } else { + BugOUT(0,"This should never happen!"); + } + + } else { + BugOUT(0,"com_style missing? WTF this should never happen at this point!"); + } + + + usleep(5000); + $self->{'main'}{'pipeline'}->set_state("paused"); + usleep(1000); + $self->{'main'}{'pipeline'}->set_state("playing"); + BugOUT(8,"Main pipeline initiated"); + usleep(100000); + $self->{'garbage'}{'pipeline'}->set_state("paused"); + $self->{'garbage'}{'pipeline'}->set_state("null"); + BugOUT(8,"Garbage collection done."); + +} + + + +sub _start_gstsnk { + BugOUT(9,"ThreadGST _start_gstsnk->ENTER"); + my $self = $_[0]; +############################## +# + $self->{'garbage'}{'pipeline'} = GStreamer1::Pipeline->new('garbagepipe'); + $self->{'garbage'}{'elements'}{'pasrc'} = GStreamer1::ElementFactory::make( pulsesrc => 'garbage_pasrc' ); + +# if ($self->_get_argv("pa_device_name")) { + + my $device_name = $self->_get_argv("pa_device_name"); + BugOUT(8,"PA_DEVICE_NAME: $device_name"); + print "INPUT: PA_DEVICE_NAME: $device_name"; + $self->{'garbage'}{'elements'}{'pasrc'}->set('device' => "arctica.mic0");# 'arctica.output0.monitor' + +# } else { +# BugOUT(8,"PA_DEVICE_NAME: USING DEFAULT DEVICE"); +# } + + $self->{'garbage'}{'elements'}{'pasrc'}->set('client-name' => 'Arctica Garbage Collector'); + $self->{'garbage'}{'elements'}{'sink'} = GStreamer1::ElementFactory::make( fakesink => 'garbage_sink' ); + $self->{'garbage'}{'pipeline'}->add($self->{'garbage'}{'elements'}{'pasrc'}); + $self->{'garbage'}{'pipeline'}->add($self->{'garbage'}{'elements'}{'sink'}); + $self->{'garbage'}{'elements'}{'pasrc'}->link($self->{'garbage'}{'elements'}{'sink'}); + $self->{'garbage'}{'pipeline'}->set_state( "playing" ); + BugOUT(8,"Garbage collection initiated"); + my $pasuspender_fpath = $self->{'_default_pasuspender_fpath'};# FIXME FIX THIS AFTER CFG FILES HAVE BEEN PROPERLY IMPLEMENTED + + if (-X $pasuspender_fpath) { + BugOUT(9,"pasuspender full path: $pasuspender_fpath"); + my $tpath = $ENV{'PATH'}; + $ENV{'PATH'} = "/bin:/usr/bin"; + system($pasuspender_fpath,"true");# FIXME Use Mother::Forker::Light. + $ENV{'PATH'} = $tpath; + } else { + BugOUT(1,"NO pasuspender at full path: $pasuspender_fpath");# Not super critical but some audible junk may occur... + } +# +############################## + $self->{'main'}{'pipeline'} = GStreamer1::Pipeline->new('pipeline'); + + $self->{'main'}{'elements'}{'opusdec'} = GStreamer1::ElementFactory::make( opusdec => 'opusdec' ); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'opusdec'}); + + if ($self->_get_argv("com_style") eq "datagram") { + if ($self->_get_argv("socket_type") eq "udp") { + + $self->{'main'}{'elements'}{'udpsrc'} = GStreamer1::ElementFactory::make( udpsrc => 'udpsrc' ); + $self->{'main'}{'elements'}{'udpsrc'}->set('port' => $self->_get_argv("port_or_unix-socket")); + $self->{'main'}{'elements'}{'udpsrc'}->set( caps => GStreamer1::Caps::Simple->new( + 'application/x-rtp', + 'media' => 'Glib::String' => 'audio', + 'clock-rate' => 'Glib::Int' => 48000, + 'encoding-name' => 'Glib::String' => 'X-GST-OPUS-DRAFT-SPITTKA-00')); + + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'udpsrc'}); + + $self->{'main'}{'elements'}{'rtpopusdepay'} = GStreamer1::ElementFactory::make( rtpopusdepay => 'rtpopusdepay' ); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'rtpopusdepay'}); + + $self->{'main'}{'elements'}{'udpsrc'}->link($self->{'main'}{'elements'}{'rtpopusdepay'}); + $self->{'main'}{'elements'}{'rtpopusdepay'}->link($self->{'main'}{'elements'}{'opusdec'}); + + } elsif ($self->_get_argv("socket_type") eq "unixd") { + BugOUT(0,"NOT YET IMPLEMENTED!"); + } else { + BugOUT(0,"This should never happen!"); + } + } elsif ($self->_get_argv("com_style") eq "stream") { + if ($self->_get_argv("socket_type") eq "tcp") { + + $self->{'main'}{'elements'}{'tcpserversrc'} = GStreamer1::ElementFactory::make( tcpserversrc => 'tcpserversrc' ); + $self->{'main'}{'elements'}{'tcpserversrc'}->set('port' => $self->_get_argv("port_or_unix-socket")); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'tcpserversrc'}); + + $self->{'main'}{'elements'}{'queue1'} = GStreamer1::ElementFactory::make( queue => 'queue1' ); + $self->{'main'}{'elements'}{'queue1'}->set("silent" => 0); + $self->{'main'}{'elements'}{'queue1'}->set("leaky" => "downstream"); + $self->{'main'}{'elements'}{'queue1'}->set("max-size-time" => "30000000"); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'queue1'}); + + $self->{'main'}{'elements'}{'tcpserversrc'}->link($self->{'main'}{'elements'}{'queue1'}); + + $self->{'main'}{'elements'}{'gdpdepay'} = GStreamer1::ElementFactory::make( gdpdepay => 'gdpdepay' ); + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'gdpdepay'}); + + $self->{'main'}{'elements'}{'queue1'}->link($self->{'main'}{'elements'}{'gdpdepay'}); + $self->{'main'}{'elements'}{'gdpdepay'}->link($self->{'main'}{'elements'}{'opusdec'}); + + } elsif ($self->_get_argv("socket_type") eq "unixs") { + BugOUT(0,"NOT YET IMPLEMENTED!"); + } else { + BugOUT(0,"This should never happen!"); + } + } + + + + $self->{'main'}{'elements'}{'pasink'} = GStreamer1::ElementFactory::make( pulsesink => 'pasink' ); + +# if ($self->_get_argv("pa_device_name")) { +# $self->{'main'}{'elements'}{'pasink'}->set('device' => $self->_get_argv("pa_device_name")); +# } + + $self->{'main'}{'elements'}{'pasink'}->set('device' => "arctica.input0"); + + $self->{'main'}{'elements'}{'pasink'}->set('client-name' => 'Arctica Audio Services'); + + $self->{'main'}{'pipeline'}->add($self->{'main'}{'elements'}{'pasink'}); + $self->{'main'}{'elements'}{'opusdec'}->link($self->{'main'}{'elements'}{'pasink'}); + + usleep(10000); + $self->{'main'}{'pipeline'}->set_state("playing"); + BugOUT(8,"Main pipeline initiated"); + print "status:sink_ready:",$self->_get_argv("idnum"),":\n"; + usleep(100000); + $self->{'garbage'}{'pipeline'}->set_state( "paused" ); + $self->{'garbage'}{'pipeline'}->set_state( "null" ); + + BugOUT(9,"ThreadGST _start_gstsnk->DONE"); +} + +sub start { + my $self = $_[0]; + if ($self->_get_argv("src_or_snk") eq "src") { + $self->_start_gstsrc; + } elsif ($self->_get_argv("src_or_snk") eq "snk") { + $self->_start_gstsnk; + } else { + BugOUT(0,"WTF? (src_or_snk!!)") + } +} + +sub terminate { + my $self = $_[0]; + if ($self->{'main'}{'pipeline'}) { + $self->{'main'}{'pipeline'}->set_state("paused"); + $self->{'main'}{'pipeline'}->set_state("null"); + } + # FIXME! Do something else too?? +} + +1; + |