当我运行与套接字相关的程序时,出现“ Segmentation Fault(Core dump)”问题

时间:2019-07-11 12:43:49

标签: c sockets

我曾在引起分段错误时用谷歌搜索,但我无法纠正我的代码中的错误。我正在使用select函数来找出可读取的文件描述符。 c,当我们具有用于ipv4地址长度的宏时,我无法理解socklen_t结构的目的。

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<netdb.h>

#define PORT "9034"

int main()
{
    fd_set master,read_fds;
    int file_descriptor,new_fd,server,yes=1,error,fd_max,i;
    char ip[INET_ADDRSTRLEN],data[1024];
    struct addrinfo hints,*p;
    struct sockaddr_in addr;
    socklen_t addrlen; //?
    memset(&hints,0,sizeof(struct addrinfo));
    FD_ZERO(&master);
    FD_ZERO(&read_fds);
    hints.ai_family=AF_INET;
    hints.ai_socktype=SOCK_STREAM;
    hints.ai_protocol=AI_PASSIVE;
    file_descriptor=getaddrinfo(NULL,PORT,&hints,&p);
    printf("%s",inet_ntop(AF_INET,p->ai_addr,ip,addrlen));
    server=socket(p->ai_family,p->ai_socktype,p->ai_protocol);
    setsockopt(server,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(int));
    listen(server,5);
    FD_SET(server,&master);
    fd_max=server;
    while (1)
    {
        read_fds=master;
        select(fd_max+1,&read_fds,NULL,NULL,NULL);
        for(i=0;i<fd_max;i++)
        {
            if(i==server)
            {
                new_fd=accept(server,(struct sockaddr *)&addr,&addrlen);
                FD_SET(new_fd,&master);
                printf("Connected to the client %s on socket %d",inet_ntop(AF_INET,(struct sockaddr *)&addr,ip,INET_ADDRSTRLEN),new_fd);
                if(new_fd>fd_max)
                fd_max=new_fd;
            }
            else
            {
                recv(i,data,sizeof(data),0);
                printf("%s",data);                
            }   
        }
    }
}

2 个答案:

答案 0 :(得分:0)

根据您的评论,似乎您正在使用的某些实用程序可能未正确定义。对于要放入项目的库,请确保使用正确的(和相应的)头文件。即,对于POSIX开发,建议使用以下方法:

/* According to POSIX.1-2001 */
#include <sys/select.h>

More here about that

在您的select.h语句中看不到#include文件。

答案 1 :(得分:0)

崩溃在这里:

file_descriptor=getaddrinfo(NULL,PORT,&hints,&p);
printf("%s",inet_ntop(AF_INET,p->ai_addr,ip,addrlen));

请注意,在第二行中,您正在取消引用指针p,但是p为NULL,因为getaddrinfo()失败并且未将其设置为有效地址,因此请尝试取消引用会调用未定义的行为,在这种情况下会导致崩溃。

再说一次,我可以通过在程序的各个位置插入临时printf()语句并查看打印了(或未打印)的内容来调试此问题。特别是:

 printf("k1\n");
     file_descriptor=getaddrinfo(NULL,PORT,&hints,&p);
 printf("k2 file_descriptor=%i p=%p\n", file_descriptor, p);
     printf("%s",inet_ntop(AF_INET,p->ai_addr,ip,addrlen));
 printf("k3\n");

...当我运行类似的代码时,我看到了以下输出:

 k1
 k2 file_descriptor=12 p=0x0
 Segmentation fault: 11

...这使问题显而易见-p在“ k2”行中为NULL(aka 0x0),并且崩溃发生在“ k3”行被打印之前,因此崩溃肯定在您的{{1 }}调用,并且由于NULL指针取消引用。

要解决崩溃问题,您需要检查printf()的返回值以确保其返回零/成功,并在返回非零/错误时优雅地处理它,而不仅仅是假设将永远成功。 (通常,您需要对返回成功/失败值的所有函数调用执行此操作)