С ребенком читаю даю "ресурс временно недоступен" - PullRequest
6 голосов
/ 05 мая 2010

Итак, у меня есть поток файлов от родительского процесса к дочернему, и в большинстве случаев он работает нормально. Однако при многократном чтении из него функция fgets () вернет NULL, а для ошибки будет установлено «ресурс временно недоступен». Проблема временная: при запуске сценария, выполняющего чтение, fgets возвращает NULL, а иногда - нет.

Может ли кто-нибудь помочь мне предотвратить возникновение этой ошибки? Спасибо!

Редактировать: вот код ... Я не уверен, какой другой код будет полезен? там совсем немного

// this is the bit that gets a line from the child
if( fgets( line, MAX_LINE_LENGTH, fpin ) == NULL ) {
    if( ferror(fpin) ) {
        perror("error on stream fpin");
    }
    free( line );
    return FAIL;
}

По запросу, код, который открывает канал и обрабатывает дочерний процесс.

// set up pipes
int readPipe[2]; // child -> parent communication
int writePipe[2]; // parent -> child communication
int errorPipe[2]; // child -> parent, to check for errors in exec

/* create pipe */
pipe( readPipe ); // error if return val < 1 for any
pipe( writePipe );
pipe( errorPipe );
pid_t pid; /* process id when we fork */
pid = fork(); /* create new child */

if( pid == 0 ) { /* pid == 0 indicates child process */

    // close unused fds
    close( PARENT_READ );
    close( PARENT_WRITE );
    close( errorPipe[0] );

    dup2( CHILD_READ, 0 ); // replace stdin with pipe in
    dup2( CHILD_WRITE, 1 ); // replace stdout with pipe out

    /* replace child process with program to run */
    execvp(args[0], args);

    // if we get here, exec failed so inform the parent
    char *error_message = "exec failed";
    write( errorPipe[1], error_message, strlen(error_message)+1 );
    exit(-1);

} 

1 Ответ

10 голосов
/ 05 мая 2010

Это означает, что кто-то установил стандартный дескриптор входного файла как неблокирующий.

( Ресурс временно недоступен - это сообщение об ошибке, соответствующее EAGAIN / EWOULDBLOCK, которое возвращается read(), только если был выбран неблокирующий ввод-вывод и нет данных для чтения ).

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

Некоторые идеи для дальнейшего исследования:

  • Если вы strace() дочерний процесс, какой системный вызов возвращает EAGAIN? На каком файле дескриптор номер?

  • Какой вывод printf("%d\n", fcntl(fileno(fpin), F_GETFL)); прямо перед ошибкой fgets()?

...