#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
int
main( int argc, char **argv)
{
int pfds[ 2], i;
size_t pbytrd;
pid_t childpid;
char buffer[ 200];
pipe( pfds);
if( ( childpid = fork() ) == -1)
{
perror( "fork error: ");
exit( 1);
}
else if( childpid == 0)
{
close( pfds[ 0]);
dup2( pfds[1], 1);
close( pfds[ 1]);
for( i = 0; i < 10; i++)
printf("Hello...");
execlp( "xterm","xterm","-e","./sample_a", (char *) 0);
exit( 0);
}
else
{
close( pfds[ 1]);
for( ; ; )
{
if( ( pbytrd = read( pfds[ 0], buffer, sizeof( buffer) ) ) == -1)
{
perror(" read error: ");
exit( 1);
}
else if( pbytrd == 0)
{
write( fileno( stdout), "Cannot read from pipe...\n", strlen( "Cannot read from pipe...\n") );
exit( 0);
}
else
{
write( fileno( stdout), "MESSAGE: ", strlen( "MESSAGE: ") );
write( fileno( stdout), buffer, pbytrd);
}
}
}
return( 0);
}
My sample_a.c code is below:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
int
main( int argc, char **argv)
{
int i;
for( i= 0; i< 10; i++)
write( fileno( stdout), "Hello...", strlen( "Hello...") );
return( 0);
}
In the above code, what I really want to do is: 1) Redirect output from the child process to pipe and have parent read from the pipe and print it out to stdout.
I am able to redirect child process output( "printf") from stdout to the pipe but I am not able to redirect execlp's child process, xterm, output to the pipe.
Can anybody help me this?
The pipeFD array stores two file descriptions of in and out, once you created a xterm window. The child process do not know where to write. So you can use dup2:
It maps the old write port to a new port(other numbers also work). And in the xterm you can write to the pipe by
xterm has no output, so there's no way to redirect it anywhere. xterm runs a child and redirects the child's output to itself, so if you want to redirect the child's output somewhere other than the xterm, you'll need to do that yourself in the child, AFTER xterm starts it. The easiest way to do that is probably to have xterm start a shell and use shell redirection. To redirect just stderr for the child, you'd use something like:
replacing
somewhere
with something appropriate. Assuming you want to capture that in your original program, you probably want to create a temporary fifo with tempnam(3) and mkfifo(3) and open it for reading in your parent. You need to create it before forking and open it after forking, as the open will block until the shell runs and opens the write end. Once the open is complete, you can delete the fifo, and read from the now unnamed fifo.xterm
is a terminal emulator. It executes the program you provided (sample_a
), connecting its input and outputs to itself, so that the program receives the user input in its standard input, and prints to the user whatever the it sends to its standard and error outputs.What your program is doing is to connect the
xterm
output to the pipe, and trying to read that. Butxterm
is a GUI program, it does not normally write data to its outputs. Perhaps it is not really clear what exactly you are trying to achieve.If you remove the
xterm
part, it should work as you expect, that is, the parent process will see whatsample_a
writes to the output.I faced a similar situation. My goal was so send some information from child process that runs in a xterm. As also reported earlier, I was not able to use dup2. A workaround for this is to pass the fd to the new binary as xterm args.
A suggested approach (without error checks for brevity) is
On the child process that runs in xterm, extract the fd
I don't know why dup2 doses'nt work with xterm, but this workaround solves the requirement.