Is there a particular reason you've used system instead of an exec? How do you expect data to get into your child if you never read or print? Do you expect the child to do anything after its child returns? I probably would have written this as (untested for your target):
use POSIX qw( mkfifo );
my $svitok_truby;
mkfifo( $svitok_truby, 0700 ) || die "Ошиб
+ка созданиn
+3; трубы $svitok_truby : $!";
my @args = ('/usr/bin/mplayer', '-slave', "-input file=$svitok_tru
+by", '/tmp/1.flac');
local $SIG{CHLD} = "IGNORE";
my $pid = fork();
die "Fork failed\n" unless defined $pid;
if ($pid == 0) {
$| = 1;
exec @args;
}
open (my $FIFO, '>', $svitok_truby) || die "can't open $svitok_tr
+uby: $!";
while(read STDIN, my($char), 1) {
print $FIFO $char;
}
close($FIFO);
See also Named Pipes and note loop terminates on ^D.
#11929 First ask yourself `How would I do this without a computer?' Then have the computer do it the same way.
| [reply] [d/l] |
I did understand your code not fully. -- It would be far better for me (to understand your answer), had you applied only necessary changes to my code structure -- not changing everything by the way. I would not ask you to explain many things because, it could be hard, so, i did apply what is seems me relevant, and can say, that it is works, but worse -- insted of sometimes, i had to press a key twice-thrice, now the number went up to seven! -- If i but write to pipe the PERLs' way instead of Shell's. Here is the code.
my $FIFO=undef;
mkfifo( $svitok_truby, 0700 ) || die 'Ошиб

+82;а создания &
+#1090;рубы '.$svitok_truby.' : '.$!;
local $SIG{CHLD} = "IGNORE";
if( $nomer_vosproizvodyshchego_protsessu=fork ){
$|=1;
open( $FIFO, '>', $svitok_truby ) || die 'Не о

+90;крыть труб&#
+1091; '.$svitok_truby.' на запи&#
+1089;ь : '.$!;
<sniped>
print $FIFO 'pause';
<sniped>
}else{
exec '/usr/bin/mplayer -slave -input file='.$svitok_truby.' /tmp/1.f
+lac';
}
close( $FIFO );
unlink $svitok_truby;
Just one question, is there a flush command -- to execute after each writing to pipe?
I did test more and have noticed weird connection between input in parent and pausing of mplayer, though to pipe was sent nothing at all -- parent just does its own work, w/o sending a thing to pipe, for example:
mkfifo( $svitok_truby, 0700 ) || die 'Ошиб

+82;а создания &
+#1090;рубы '.$svitok_truby.' : '.$!;
# Управляющk
+2;я часть ("род
+итель").
if( $nomer_vosproizvodyshchego_protsessu=fork ){
$|=1;
use Term::ReadKey;
ReadMode 'cbreak';
# Parent
while( 1 ){
$knopka=&ReadKey( 0 );
if( $knopka eq '/' ){
system( '/bin/date' );
}elsif( $knopka eq $knopka_vyhodu ){
system( '/bin/echo quit >'.$svitok_truby );
last;
}
}
}else{
## Child
for( $i=0; $i<=$#svitok_na_vosproizvedenie; $i++ ){
system( '/usr/bin/mplayer -slave -input file='.$svitok_truby.' '.$
+svitok_na_vosproizvedenie_kom[$i] );
}
}
ReadMode 'normal';
unlink $svitok_truby;
So, when i press key / -- parent just outputs date. But many times pressing, once, «mplayer2» will pause playing -- though nothing was sent to pipe! -- And that input-pipe relation i can not understand. Module «Term» does not have influence -- i had the same w/ the «read» function.
Oh! The site terribly corrupts my input!
Thanks for your answers.
| [reply] [d/l] [select] |
Sorry, I neglected to set flush on the pipe, which is why you saw the issue. That can be accomplished by explicitly importing IO::Handle and calling autoflush. If I write a complete, self-contained script wrapping date, it might look like:
#!/usr/bin/perl
use strict;
use warnings;
use POSIX qw( mkfifo );
use IO::Handle;
my $svitok_truby = 'tmp.pp';
mkfifo( $svitok_truby, 0700 ) || die "Pipe fail; $svitok_truby : $!"
+;
local $SIG{CHLD} = "IGNORE";
my @args = ('perl','-MIO::Handle','-E',<<'EOC',$svitok_truby);
$| = 1;
open my $fh, "<", $ARGV[0];
while(read $fh, my($char), 1) {
print $char;
}
EOC
my $pid = fork();
die "Fork failed\n" unless defined $pid;
if ($pid == 0) {
exec @args;
}
open (my $FIFO, '>', $svitok_truby) || die "can't open $svitok_truby:
+ $!";
$FIFO->autoflush(1);
while(read STDIN, my($char), 1) {
print $FIFO $char;
}
close($FIFO);
END {
unlink $svitok_truby;
}
To modify this from piping into the toy echo program, you would swap @args (including the here-doc, which runs until EOC) to whatever is appropriate for your program, probably (untested):
my @args = ('/usr/bin/mplayer', '-slave', '-input', "file=$svitok_trub
+y", $svitok_na_vosproizvedenie_kom[$i]);
It's possible you will still see buffering issues depending on your terminal, because many terminal programs buffer STDIN by default. To test if that is the problem, you can pipe in commands. So, for my toy above,
perl -E'$|=1;print,sleep 1 for 0..9' | perl script.pl
will slowly print 0 - 9 on the screen, one character a second. For your case, you might want to try pausing and unpausing music once per second.
TLDR: use IO::Handle; and $FIFO->autoflush(1); will set the pipe to hot.
#11929 First ask yourself `How would I do this without a computer?' Then have the computer do it the same way.
| [reply] [d/l] [select] |
Nothing in your code interacts with any user, and why are you using system to write to the FIFO? | [reply] [d/l] |
Interaction w/ a user is out of the problem scope. I used system because: tried and it worked, whereas the PERL's documentation is purely written/understood to/by me.
| [reply] |