реализация канала и перенаправление вместе в C - PullRequest
0 голосов
/ 30 мая 2018

Я работаю над созданием минимальной оболочки на C. Я понимаю, как работают pipe и перенаправление.У меня есть две программы, одна программа, которая обрабатывает перенаправление, например ls -l > outputfile.У меня есть другая программа, которая обрабатывает один канал, например, ls - l | wc.Главная проблема, которую я имею, состоит в том, чтобы соединить их.По сути, я хочу иметь возможность реализовать перенаправление pipe и в одной программе.Ниже приведена часть кода для pipe.

   pipe(p);
      if (fork() == 0) {
      dup2(p[1], 1);
      close(p[0]);
      execvp(upstream[0], upstream );
      exit(1);
    }
    if (fork() == 0) {
    dup2(p[0], 0);

    close(p[1]);
    execvp(downstream[0], downstream);
    exit(1);
    }
    else { /* Parent */
        /*  so Close both ends of pipe */
    close(p[0]);
    close(p[1]);

Я был бы очень признателен, если бы смог получить указатель на этот.Я не уверен, в какой части моего кода я должен реализовать перенаправление.Заранее спасибо

Ответы [ 2 ]

0 голосов
/ 04 июня 2018

Вы хотите выполнить

ls -l   |   wc   >   outputfile
 |           |         |
p-1        p-2      redirect the o/p of ls | wc to outputfile

Для достижения вышеуказанного вам необходимо создать процесс (p-1 и p-2), используя системный вызов fork(), а затем использовать dup() или dup2()системный вызов для перенаправления o / p туда, куда он предназначен.

вот пример кода

int main(int argc,char *argv[]) {
        int fd[2];
        pid_t cpid;
        close(1); /* so that output file should get file descriptor 1 */
        int file_dsptr = open(argv[3],O_CREAT|O_TRUNC |O_WRONLY , 0664); /* file should be exist in current directory, otherwise use O_CREAT | O_TRUNC */
        printf("[%d]\n",file_dsptr);

        if(argc == 4){
                if(pipe(fd) == -1){
                        perror("error in pipe creation");
                        exit(EXIT_FAILURE);
                }
                cpid = fork();
                if( cpid == 0 ){ /*child process */
                        close( fd[0] ); /* close read end of pipe */
                        if( -1 == dup2(fd[1] , file_dsptr)){ /* duplicate fd[1] to fd where data.txt points */
                                exit( EXIT_FAILURE );
                        }
                        if(-1 == execlp(argv[1] , argv[1] , NULL )){ /* executes the commands */
                                exit( EXIT_FAILURE );
                        }
                        close( fd[1] );
                        _exit( EXIT_SUCCESS );
                }
                else if(cpid > 0){
                        wait( NULL ); /* wait for child to completes */
                        close( fd[1] ); /* close write end of pipe */
                        if( -1 == dup2(fd[0] , STDIN_FILENO)) {
                                exit( EXIT_FAILURE );
                        }
                        if(-1 == execlp(argv[2] , argv[2], NULL )) {
                                exit( EXIT_FAILURE );
                        }
                        close( fd[0] );
                        _exit( EXIT_SUCCESS );
                }
                else{
                        fprintf(stderr, "fork failed\n");
                        exit(EXIT_FAILURE);
                }
        }
        else{
                fprintf(stderr, "Usage Msg :. ./a.out ls wc data.txt \n");
                exit(EXIT_FAILURE);
        }
        return 0;
}

Как я уже упоминал в комментариях, запустите приведенный выше код, например

gcc -Wall test.c
./a.out ls  wc  outputfile

И проверьте содержимое outputfile сейчас, оно должно бытьрезультат ls | wc.

0 голосов
/ 30 мая 2018

Так что для аспекта перенаправления вам нужно использовать dup2 () в ответвлении файловых дескрипторов.Чтобы настроить трубу, вам нужно использовать pipe () в главном потоке, чтобы создать дескрипторы, которые вы затем будете dup2 () внутри вилки.

Обратите внимание, что вы должны быть очень осторожны, чтобы pipe ()не дает вам дескриптор файла с низким номером.Вы можете использовать fcntrl (F_DUPFD, ...), чтобы запугивать их до больших чисел.Кроме того, используйте CLOEXEC, чтобы избавиться от всех вторичных дескрипторов мусора, плавающих вокруг.

Псевдокод:

fin = open("input", read)
pipeIO = pipe()
fout = open("output", write)

fin2 = fcntrl(F_DUPFD_CLOEXEC, fin, 3);   close fin
pipeI2 = fcntrl(F_DUPFD_CLOEXEC, pipeI, 3);   close pipeI
pipeO2 = fcntrl(F_DUPFD_CLOEXEC, pipeO, 3);   close pipeO
fout2 = fcntrl(F_DUPFD_CLOEXEC, fout, 3);   close fout

fork 
{
   dup(fin2, 0)
   dup(pipeO2, 1)
  exec Command1
}
close fin2
close pipeO2
fork 
{
   dup(pipeI2, 0)
   dup(out2, 1)
   exec Command2
}
close fout2
close pipeI2

wait

В качестве альтернативы, вторая ветвь может цепочкой-exec:

Псевдокод:

fin = open("input", read)
pipeIO = pipe()
fout = open("output", write)

fin2 = fcntrl(F_DUPFD_CLOEXEC, fin, 3);   close fin
pipeI2 = fcntrl(F_DUPFD_CLOEXEC, pipeI, 3);   close pipeI
pipeO2 = fcntrl(F_DUPFD_CLOEXEC, pipeO, 3);   close pipeO
fout2 = fcntrl(F_DUPFD_CLOEXEC, fout, 3);   close fout

fork 
{
   dup(fin2, 0)
   dup(pipeO2, 1)
  exec Command1
}
dup(pipeI2, 0)
dup(out2, 1)
exec Command2

Вы уже ясно знаете, что когда вы выполняете разветвление, дочерний процесс фактически дублирует все дескрипторы, и в тот момент, когда код фактически выполняется, должна быть открыта только одна версия.Родитель или дочерний элемент должны закрывать каждый дескриптор, который ему не интересен. Используя CLOEXEC, мы уменьшаем эту потребность, хотя, если вы хотите сохранить родительский процесс в качестве монитора, ему необходимо закрыть все свои копии всех этих дескрипторов..

Краткое описание функций dup:

0 = pipe (handle [2])
Это создает пару дескрипторов, которые присоединяются друг к другу в качестве входа и выхода.Классически вы можете связываться с вилкой одним процессом, записывая в один дескриптор вилки, а другой процесс считывая другой дескриптор, в зависимости от того, каким образом вы хотите их использовать.В этом случае мы используем канал для связи между двумя дочерними процессами, а родительский не задействован.

handle = dup (handle)
Создает дублирующий дескриптор в самом низком неиспользуемом слоте.Если ваша программа закрыла stdin / stdout / stderr, она будет использовать те, которые будут проблематичными.Также эти дескрипторы остаются активными в вызовах exec.

handle = dup2 (handle, trghandle)
Это создает дублирующий дескриптор по указанному вами индексу.Он будет молча закрывать любое существующее использование trghandle, что может быть как полезным, так и раздражающим!

handle = dup3 (handle, trghandle, flags)
Это создает дублирующий дескриптор по указанному вами индексу.Это молча закроет любое существующее использование trghandle, которое может быть и полезным и раздражающим!Вы можете указать флаги, такие как CLOEXEC, который автоматически закрывает дескриптор при вызове exec.CLOEXEC действительно полезен во избежание необходимости явно закрывать файлы.

int fcntl (int fd, int cmd, ... / * arg * /);
F_DUPFD (int) Дублируйте дескриптор файла fd, используядескриптор файла с наименьшим номером больше или равен аргументу.
F_DUPFD_CLOEXEC (int; начиная с Linux 2.6.24) Как и в случае F_DUPFD, но дополнительно установите флаг закрытия при выполнении для дескриптора дубликата файла.
... что позволяет легко избежать стандартного ввода / вывода.

...