Как мне сказать wc прекратить чтение?

У меня есть программа, которая fork() создает новый процесс, а затем перезаписывает stdin этого нового процесса с помощью канала моего файлового дескриптора fd. В этом процессе я затем использую execvp() для выполнения wc, и он должен прочитать свой ввод от родителя. Затем в родительском элементе я пишу в конец канала для записи и, закончив запись, закрываю канал.

Проблема в том, что wc все еще ожидает ввода, а des не выходит.
Обычно я могу остановить wc, набрав CTRLD, но я не могу отправить это сигнал в C/C++.

Как сказать wc прекратить чтение?

EDIT: я следовал идиоме pipe/fork/dup/exec (думаю, так она и называется).
Потоковая передача работает с другими программами, которым нужен ввод, просто wc нужен специальный EOF, чтобы прекратить чтение.

int in_fd[2];
//parent-child setup
pipe(in_fd);
child = fork();
if(child == 0) {
    close(in_fd[1]);
    dup2(in_fd[0], 0);
    close(in_fd[0]);
    execvp(cmdArg[0], cmdArg);
} else {
    close(in_fd[0]);
    in_pid = fork_in_proc();
    close(in_fd[1]);
}

//writing function
pid_t fork_in_proc() {
    string line;
    pid_t in_pid;
    if((in_pid = fork()) == 0) {
        close(in_fd[0]);
        ifstream file(stream_file[STREAM_IN].c_str(), ios_base::in);
        if (file.bad()) {
            cerr << "File read error\n";
            return 1;
        }

        while(file.good()) {
            getline(file, line);
            if(file.good()) {
                write(in_fd[1], line.c_str(), line.length());
            }
        }
        int end = 3;
        write(in_fd[1], &end, sizeof(end));
        file.close();
        close(in_fd[1]);
        cout << "PIPE IN" << endl;
        exit(0);
    } else {
        return in_pid;
    }
}

Извините, если код кажется немного бессвязным. Мне пришлось собирать его из файла.


person chustar    schedule 22.11.2011    source источник
comment
CTRL+D не отправляет сигнал, но заставляет все, что читается с терминала, читать EOF.   -  person ThiefMaster    schedule 22.11.2011
comment
@JoachimPileborg Я добавил немного кода.   -  person chustar    schedule 22.11.2011
comment
Первое, что бросается мне в глаза, это то, что вы называете execvp неправильно. Вторым аргументом должен быть массив строк с последней записью в массиве, равной NULL.   -  person Some programmer dude    schedule 22.11.2011
comment
@JoachimPileborg, извините, мне пришлось упростить это, чтобы показать. Я передаю массив строк, у меня просто возникает эта проблема при выполнении wc   -  person chustar    schedule 22.11.2011
comment
ios_base::in означает, что это не код C, а код C++. Аналогичные обозначения, такие как line.length().   -  person Jonathan Leffler    schedule 22.11.2011
comment
@chustar Почему вы дважды вызываете fork () в настройке «родитель-потомок» и fork_in_proc ? Если бы я не заметил, что он работает с другими исполняемыми файлами, я бы сказал, что вы поменяли местами индексы 0/1 in_fd в соответствии с cs.uleth.ca/~holzmann/C/system/pipeforkexec.html   -  person stacker    schedule 22.11.2011
comment
В дочернем процессе fork_in_proc есть один путь кода, где вы возвращаете 1; (почему бы не выйти()?) без закрытия in_fd[1]. Вы ввязываетесь в это дело? Если нет, то я не вижу ошибки в вашем коде. Кстати, вы вызываете close(in_fd[0]) в дочернем процессе в fork_in_proc, но in_fd[0] там уже закрыт.   -  person Werner Henze    schedule 22.11.2011
comment
@stacker Я дважды вызываю вилку, потому что это связь между двумя дочерними элементами, которые разветвляются от одного и того же родителя и используют одни и те же каналы.   -  person chustar    schedule 22.11.2011
comment
Я ввел следующее в файл С++, и, похоже, он работает: Единственные изменения, которые я сделал, кроме include и main(), это изменение execvp(...) и istream file(...) для использования некоторых постоянных значений. Какую ОС вы используете? Вы уверены, что указали execvp правильные параметры? (вы можете попробовать cat вместо wc, чтобы увидеть, что происходит.)   -  person DirtY iCE    schedule 23.11.2011


Ответы (1)


wc закрывается, когда вы закрываете stdout.

person jimmy    schedule 25.11.2011