Xterm寻呼机 - 两个终端输出 - 使用管道和dup2

时间:2016-04-07 11:18:17

标签: c fork dup

我正在尝试在C中实现一个寻呼机,我希望代码打开另一个终端(xterm)并在其中打印一些输出。

所以我首先创建一个管道和fork主程序,子程序将使用tail命令执行xterm,主程序将输出管道中的东西,而子程序在执行xterm之前将dupplicate将管道输出与stdin文件关联孩子的描述。

我可能误解了pipedup2的用法,因为我的代码不起作用。

int p[2];
pipe(p);
char buff[512];
switch (fork()) {
    case -1:
        fprintf(stderr, "Fork error.\n");
        break;
    case 0:
        dup2(p[0], 0);
        close(p[0]);
        close(p[1]);
        execlp("xterm", "xterm", "tail", NULL);
        break; 
    default:
        scanf("%s", buff);
        write(p[1], buff, strlen(buff));
        getchar();
        break;
}

现在我在父母中键入内容,并且在这两个过程中都没有打印。 那么如何在xterm和父进程之间创建通信呢?

编辑:我的程序示例:

#define VERBM_NOVERB 0
#define VERBM_STDOUT 1
#define VERBM_XTERMO 2

static int fd_xterm = -1;

void init_outputxterm() {
    mkfifo("/tmp/mypipe", 0600);
    switch (fork()) {
        case -1:
            fprintf(stderr, "Fork error.\n");
            break;
        case 0:
            execlp("xterm", "xterm", "-e", "/usr/bin/tail -f /tmp/mypipe", NULL);
            printf("FAILURE\n");
            exit(EXIT_FAILURE);
            break; 
        default:
            if ((fd_xterm = open("/tmp/mypipe", O_WRONLY)) == -1) {
                fprintf(stderr, "Can't open pipe");
                exit(1);
            }
            write(fd_xterm, "yayay\n", 6);
            dprintf(fd_xterm, "Hello world\n");
            getchar();
            break;
    }
}

void verbose_xterm(char *format, ...) {
    dprintf(fd_xterm, BOLD UNDERLINED "verbose - " RESET);
    va_list aptr;
    va_start(aptr, format);
    dprintf(fd_xterm, format, aptr);
    va_end(aptr);
}


void verbose_stdout(char *format, ...) {
    printf(BOLD UNDERLINED "verbose - " RESET);
    va_list aptr;
    va_start(aptr, format);
    vprintf(format, aptr);
    va_end(aptr);
}

void verbose_noverb(char *format, ...) {
}

void (*verbose)(char *format, ...) = verbose_stdout;


void (*verbose_mode[])(char *, ...) = { 
    verbose_noverb,
    verbose_stdout,
    verbose_xterm
};

void verbosity(int mode) {
    if (mode == VERBM_XTERMO)
        init_outputxterm();
    verbose = verbose_mode[mode];
}

主要:

verbosity(VERBM_XTERMO);

随后拨打verbose几次。您可以在init_outputxterm中看到一个在xterm创建之后在xterm中编写的tentavie,就像在解决方案中一样。但是只有在我通过关闭主终端而退出程序之后才显示所有内容,使得子进程孤儿(如果我退出Ctrl-C该孩子也被杀死了。)

1 个答案:

答案 0 :(得分:1)

Xterm本身不从stdin读取,因此向其提供数据不会产生影响。但你可以打开一个命名管道,并从中读取尾部-f / tmp / mypipe,例如:

mkfifo("/tmp/mypipe", 0600);
switch (fork()) {
    case -1:
        fprintf(stderr, "Fork error.\n");
        break;
    case 0:
        execlp("xterm", "xterm", "-e", "/usr/bin/tail -f /tmp/mypipe", NULL);
        exit(EXIT_FAILURE);
        break; 
    default:
        ;
        char buff[512];
        int fd = open("/tmp/mypipe", O_WRONLY);
        scanf("%s", buff);
        write(fd, buff, strlen(buff));
        getchar();
        break;
}

这将打开命名管道/tmp/mypipe。父进程将数据写入其中,xterm中的tail -f进程随后将输出它。