Ошибка EPIPE не появляется?

Основываясь на том, что я читал о SIGPIPE, я сделал тест, чтобы попытаться создать проблему SIGPIPE. Вот код для сервера и клиента:

Код сервера:

#include <stdio.h>
#include <stdlib.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <errno.h>
#include <unistd.h>
#include <fcntl.h>

void error(char *msg)
{
    perror(msg);
    exit(1);
}

int main(int argc, char *argv[])
{
     int sockfd, newsockfd, portno, clilen;
     char buffer[256];
     struct sockaddr_in serv_addr, cli_addr;
     int n;

     if (argc < 2) {
         fprintf(stderr,"ERROR, no port provided\n");
         exit(1);
     }

     sockfd = socket(AF_INET, SOCK_STREAM, 0);

     if (sockfd < 0)
        error("ERROR opening socket");

     bzero((char *) &serv_addr, sizeof(serv_addr));
     portno = atoi(argv[1]);
     serv_addr.sin_family = AF_INET;
     serv_addr.sin_addr.s_addr = INADDR_ANY;
     serv_addr.sin_port = htons(portno);

     if (bind(sockfd, (struct sockaddr *) &serv_addr,
              sizeof(serv_addr)) < 0)
              error("ERROR on binding");

     listen(sockfd,5);

     clilen = sizeof(cli_addr);
     newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);

     if (newsockfd < 0)
          error("ERROR on accept");
     bzero(buffer,256);

     n = read(newsockfd,buffer,255);

     if (n < 0)
         error("ERROR reading from socket");

     printf("Here is the message: %s\n",buffer);

     n = write(newsockfd,"I got your message",18);

     if (n < 0)
         error("ERROR writing to socket");
     else
         error("no error");
         return 0;
}

Код клиента:

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <errno.h>
#include <signal.h>
#include <resolv.h>
#include <arpa/inet.h>
#include <unistd.h>

void error(char *msg)
{
    perror(msg);
    exit(0);
}

int main(int argc, char *argv[])
{
    int sockfd, portno, n;

    struct sockaddr_in serv_addr;
    struct hostent *server;

    char buffer[256];
    if (argc < 3) {
       fprintf(stderr,"usage %s hostname port\n", argv[0]);
       exit(0);
    }
    portno = atoi(argv[2]);
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
        error("ERROR opening socket");
    server = gethostbyname(argv[1]);
    if (server == NULL) {
        fprintf(stderr,"ERROR, no such host\n");
        exit(0);
    }
    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    bcopy((char *)server->h_addr,
         (char *)&serv_addr.sin_addr.s_addr,
         server->h_length);
    serv_addr.sin_port = htons(portno);
    if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
        error("ERROR connecting");
    printf("Please enter the message: ");
    bzero(buffer,256);
    fgets(buffer,255,stdin);

    signal(SIGPIPE, SIG_IGN);

    n = write(sockfd,buffer,strlen(buffer));

    printf("n=%d, errno=%d", n, errno);

    bzero(buffer,256);
    n = read(sockfd,buffer,255);

    printf("n=%d, errno=%d", n, errno);

    printf("%s\n",buffer);
    return 0;
}

Я поставил эксперимент следующим образом. Сначала запустил сервер, потом клиент. После того, как клиент распечатал сообщение: «Пожалуйста, введите сообщение:», я остановил сервер. Затем я попытался ввести какой-либо текст (например, «тест») в клиенте для отправки. Поскольку я использовал signal(SIGPIPE, SIG_IGN), я ожидал, что от клиента будет какой-то код ошибки EPIPE, но ничего не получил. Клиент распечатал:

Please enter the message: test
n=5, errno=0
n=0, errno=0

Я что-то упускаю? Как я могу настроить код для операции записи, чтобы он возвращал ошибку EPIPE?


person Hieu Nguyen    schedule 03.02.2014    source источник


Ответы (1)


Не пытайтесь читать, просто запишите больше данных. Может пройти какое-то время, прежде чем отправка завершится ошибкой.

person Steffen Ullrich    schedule 03.02.2014
comment
Почему он не может вернуть ошибку при первом write()? В реальном приложении, если у меня есть только один write(), как мне перехватить ошибку EPIPE? - person Hieu Nguyen; 04.02.2014
comment
потому что еще не знает, что другая сторона закрылась - person Steffen Ullrich; 04.02.2014
comment
В моем приложении я хочу, чтобы сервер зафиксировал ошибку, чтобы выполнить некоторую работу (очистить некоторые данные, сделанные запросами от клиента). Как я мог сделать это в первом вызове write()? - person Hieu Nguyen; 04.02.2014
comment
Я не уверен, какую ошибку вы имеете в виду. В вашем примере вы ожидали, что запись клиентов завершится ошибкой, потому что сервер закрыт, так почему сервер должен что-то очищать, если вы его явно закрыли? В любом случае: запись вернет ошибку, если запись не удалась локально, например. ядро ​​не смогло обработать его, потому что оно было закрыто с другой стороны до того, как пользовательское пространство отправило данные. Он не будет доходить до другой стороны, проверьте, были ли получены данные там, и затем верните. - person Steffen Ullrich; 04.02.2014