系统相关
首页 > 系统相关> > linux中的分段错误(C中的套接字编程(TCP))

linux中的分段错误(C中的套接字编程(TCP))

作者:互联网

我只是通过一些网站学习Linux上的套接字编程,这是使用TCP在服务器端的部分代码:

#define BufferLength 100
#define SERVPORT 3111
int main()
{
  /* Variable and structure definitions. */
  int sd, sd2, rc, length = sizeof(int);
  int totalcnt = 0, on = 1;
  char temp;
  char buffer[BufferLength];
  struct sockaddr_in serveraddr;
  struct sockaddr_in their_addr;
  fd_set read_fd;

  /* Get a socket descriptor */
  if((sd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
  {
    perror("Server-socket() error");
    exit (-1);
  }
  else
    printf("Server-socket() is OK\n");

  /* Allow socket descriptor to be reusable */
  if((rc = setsockopt(sd, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on))) < 0)
  {
    perror("Server-setsockopt() error");
    close(sd);
    exit (-1);
  }
  else
    printf("Server-setsockopt() is OK\n");

  /* bind to an address */
  memset(&serveraddr, 0x00, sizeof(struct sockaddr_in));
  serveraddr.sin_family = AF_INET;
  serveraddr.sin_port = htons(SERVPORT);
  serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);
  printf("Using %s, listening at %d\n", inet_ntoa(serveraddr.sin_addr), SERVPORT);

  /* continue */
}

当我执行最后一行(printf(“ using ……”))时,出现了分段错误,为什么?谢谢.

解决方法:

所示代码缺少#include任何标头,因此由于某些未定义的符号,它无法编译.

但是,如果您错过了代码所引用的任何库函数的原型,它将进行编译,这将导致假定任何函数都返回int.

后一个事实可能是致命的,也可能不是致命的.

至少在64位系统上,将inet_ntoa()用作printf()的参数是致命的,因为在64位系统上,最有可能返回64位(字符指针)值(但32位int ).因此(假设原型未命中)在生成代码时,编译器假定inet_ntoa()返回32位int,这将导致“斩除”返回地址的最高32位.尝试从此类“已损坏”的(因此很可能是)无效地址进行printf()会引发未定义的行为,并且在您的情况下会导致观察到分段冲突.

要解决此问题,请通过添加以下内容添加相关的原型(用于inet_ntoa()):

#include <arpa/inet.h>

编译器应该已经对此发出警告.要为gcc启用所有编译器警告,请使用-Wall -Wextra -pedantic选项.认真对待此类警告.

标签:sockets,tcp,c-3,linux,segmentation-fault
来源: https://codeday.me/bug/20191120/2043200.html