尝试从服务器向客户端发送消息的分段错误

时间:2013-03-22 07:22:53

标签: c sockets segmentation-fault

我需要向客户端发送消息,然后客户端必须使用选项进行响应。我到达客户端和服务器连接,但两个程序都以“Segmentation Fault”结束。有谁知道这个错误意味着什么?有人可以提出如何创建将使客户端和服务器交互的代码的想法。在收到客户选择的选项后,服务器必须对其进行分析并再次将结果发送给客户端。

我的代码是:

服务器

    int
main(int argc, char **argv)
{
    int                 listenfd, connfd;
    socklen_t           len;
    struct sockaddr_in  servaddr, cliaddr;
    char                buff[MAXLINE];
    time_t              ticks;
    char                message[MAXLINE]="This is the server";
    char                temp_scale[2];
    char                recvdata[MAXLINE + 1];

    listenfd = Socket(AF_INET, SOCK_STREAM, 0);
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family      = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);/*----------------------------------------------------*/
    servaddr.sin_port        = htons(5555); 

    Bind(listenfd, (SA *) &servaddr, sizeof(servaddr));
    Listen(listenfd, LISTENQ);

    for ( ; ; )
    {
        len = sizeof(cliaddr);
        connfd = Accept(listenfd, (SA *) &cliaddr, &len);

        printf("Connection from %s, port %d\n",
               Inet_ntop(AF_INET, &cliaddr.sin_addr, buff, sizeof(buff)),
               ntohs(cliaddr.sin_port));


        snprintf(message, sizeof(message), "%s\r\n");
            Writen(connfd, message, strlen(message));
            while ( (n = read(connfd, recvdata, MAXLINE)) > 0)
            {
             recvdata[n] = 0;   /* null terminate*/
             if (fputs(recvdata, stdout) == EOF)
                 err_sys("fputs error");
            }
             if (n < 0)
          err_sys("read error");

           Close(connfd);
    }
}

客户端

int
main(int argc, char **argv)
{
    int                 sockfd, rd;
    socklen_t           len;
    char                recvline[MAXLINE + 1];
    struct sockaddr_in  servaddr, cliaddr;
    char  scale[2];

    /*if (argc != 2)
        err_quit("usage: a.out <IPaddress>");*/

    if ( (sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
        err_sys("socket error");

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port   = htons(atoi(argv[2])); /*port passed through command line*/
    if (inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0) /*The client translates the server address, passed on the command line*/
        err_quit("inet_pton error for %s", argv[1]);


    if (connect(sockfd, (SA *) &servaddr, sizeof(servaddr)) < 0)
        err_sys("connect error");

    len = sizeof(cliaddr);
    Getsockname(sockfd, (SA *) &cliaddr, &len);
    printf("Local Address is: %s\n",
           Sock_ntop((SA *) &cliaddr, sizeof(cliaddr))); 

    printf("Iniciando read...\n");
    while ( (rd = read(sockfd, recvline, MAXLINE)) > 0)
    {
        recvline[rd] = 0;   /* null terminate*/
        if (fputs(recvline, stdout) == EOF)
            err_sys("fputs error");
    }
    if (rd < 0)
     err_sys("read error");

    printf("Enter option 'A' or 'B'");
    send_scale(sockfd);

    exit(0);
}

由于

3 个答案:

答案 0 :(得分:2)

在调试器中运行您的代码(例如gdb ./a.out)并立即查找。

答案 1 :(得分:2)

您的服务器可能因此而发生故障:

snprintf(message, sizeof(message), "%s\r\n"); // <== no parameters

这是完全错误的。 snprintf()调用有一个格式说明符,期望char *为以null结尾的字符串,并且你绝对没有传递它。因此,它会从堆栈中获取一个随机值,将其视为指针,并将其解除引用以尝试完成格式化请求。

在不知道你正在使用的API的细节的情况下(显然不仅仅是名字的标准BSD套接字),还有很多事情要继续下去。

答案 2 :(得分:0)

我不知道它是否有用,但在C中,当你打印你的回复时,字符串的空终止是'\ 0':

recvdata[n] = 0;   /* null terminate ----> this must be '\0'*/
if (fputs(recvdata, stdout) == EOF)
err_sys("fputs error");

你用“0”填充它,所以当fputs解析你的字符串以便打印它时,它可能会引导你进行段错误。

希望它有所帮助!