1

私の目的は、FIFO を介して子と親の間で IPC を作成することです。子供は走ればいい

execl ("/bin/cat", "cat", "/etc/passwd", (char *)0);

出力を親の入力にリダイレクトし、親は次のコマンドを実行する必要があります。

cut -l : -f 1

これをコマンドラインに出力します。

現在、FIFO を正常にリンクし、子プロセスの出力を親プロセスの入力にリダイレクトしました。複数のテストを行ったところ、その接続は正しく機能しています。問題は、カットの execl にあり、次のようになります。

execlp("/bin/cut", "cut", "-l:", "-f", "1", NULL);

しかし、そうではないことは確かです。

int cut(){

    //
    int myfifo;     
    char buf[MAX_BUF];


    printf("\nCut opening FIFO");
    if((myfifo = open("/tmp/myfifo", O_RDONLY | O_TRUNC))<0){
        perror("open FIFO at cut");
        quit(EXIT_FAILURE);}
    else{printf("\nCut has FIFO opened and is reading\n");}

    //read(myfifo, buf, MAX_BUF); outputting buf goes as supposed to

    if( dup2(myfifo, 0) < 0 ){
        perror("dup2 at cut");
        quit(EXIT_FAILURE);}

    //read(STDIN_FILENO, buf, MAX_BUF);

    close(myfifo);

    execlp("/bin/cut", "cut", "-l:", "-f", "1", NULL);

    //this is for testing buf, but i guess the program shouldn't even get here
    printf("\nCut has received: %s\nAnd is closing FIFO", buf);

    return -1;

}


int cat(){

    int myfifo;
    //OPEN FIFO
    printf("\nCat opening FIFO");
    if( (myfifo = open("/tmp/myfifo", O_WRONLY | O_TRUNC) )<0){
        perror("open FIFO at cat");
        quit(EXIT_FAILURE);
    }
    else{
        printf("\nCat has opened FIFO");
    //WRITE OUTPUT OF "cat \etc\passwd" TO FIFO
        dup2(myfifo, 1);
        execl ("/bin/cat", "cat", "/etc/passwd", (char *)0);
    }
    close(myfifo);


    return 0;
}

main は現在、fifo (mkfifo)、forks() を作成し、関数を呼び出すだけです。私の問題は、親の stdin (running cut) にある可能性がありますが、そうは思わないか、または execl() が stdin から直接読み取ると想定していて、そうではありません。execl() を介して「カット」を適切に記述していないためだと思います。

コードの修正、またはいくつかのアイデアを表現した方法でさえ、私が何かを正しく理解していないことを示している可能性があるため、非常に高く評価されます。助けてくれてありがとう

4

1 に答える 1

0

コメントに記載されているように、cutGNU および BSD (および POSIX) のコマンドは-lオプションをサポートしていません。必要なオプションは-d区切り文字です。

このコードは私にとってはうまくいきます。私のマシンで/bin/catは、正しいですが、/usr/bin/cut正しい(ではない/bin/cut)ため、次のようにコンパイルする必要がありました。

$ rmk cutcat UFLAGS=-DCUT_CMD=/usr/bin/cut && ./cutcat
    gcc -O3 -g -std=c11 -Wall -Wextra -Werror -DCUT_CMD=/usr/bin/cut cutcat.c -o cutcat
$

makeこれは(と呼ばれるrmk)のカスタム バリアントとカスタム を使用しますが、コマンドmakefileの場所は(ユーザー フラグ) マクロcutを介してコマンド ラインで指定されます。UFLAGSC コードのSTRINGおよびマクロは厄介ですが、 (または) を呼び出すシェルを超えて二重引用符を取得しようとし、次にコンパイラを実行するために呼び出すEXPANDシェルを取得しようとするほど厄介ではありません。makermkmake

コード:

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/stat.h>

#ifndef CUT_CMD
#define CUT_CMD /bin/cut
#endif
#ifndef CAT_CMD
#define CAT_CMD /bin/cat
#endif
#define EXPAND(x)   #x
#define STRING(x)   EXPAND(x)

static const char cut_cmd[] = STRING(CUT_CMD);
static const char cat_cmd[] = STRING(CAT_CMD);

static inline void quit(int status) { exit(status); }

enum { MAX_BUF = 4096 };

static void cut(void)
{
    int myfifo;

    printf("\nCut opening FIFO");
    if ((myfifo = open("/tmp/myfifo", O_RDONLY | O_TRUNC)) < 0)
    {
        perror("open FIFO at cut");
        quit(EXIT_FAILURE);
    }
    printf("\nCut has FIFO opened and is reading\n");

    if (dup2(myfifo, 0) < 0)
    {
        perror("dup2 at cut");
        quit(EXIT_FAILURE);
    }

    close(myfifo);

    execlp(cut_cmd, "cut", "-d:", "-f", "1", NULL);
    fprintf(stderr, "Failed to execute %s\n", cut_cmd);
    exit(1);
}

static
void cat(void)
{
    int myfifo;

    printf("\nCat opening FIFO");
    if ( (myfifo = open("/tmp/myfifo", O_WRONLY | O_TRUNC) ) < 0)
    {
        perror("open FIFO at cat");
        quit(EXIT_FAILURE);
    }
    printf("\nCat has opened FIFO");
    dup2(myfifo, 1);
    close(myfifo);
    execl(cat_cmd, "cat", "/etc/passwd", (char *)0);
    fprintf(stderr, "Failed to execute %s\n", cat_cmd);
    exit(1);
}

int main(void)
{
    mkfifo("/tmp/myfifo", 0600);
    if (fork() == 0)
        cat();
    else
        cut();
    /*NOTREACHED*/
    fprintf(stderr, "You should not see this message\n");
    return 0;
}

出力例:

かなり切り捨てられた

…
nobody
root
daemon
_uucp
_taskgated
_networkd
…

コードは実際にあなたが持っていたものとあまり変わらない. 私はいくつかの混乱を取り除きました。close(myfifo)によって実行されたことを確認しましたcat()。原作にはありませんでした。それは重要かもしれません。

于 2015-05-24T03:32:48.270 に答える