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

segmentation fault in linux (socket programming (TCP) in C)

我刚刚通过一些网站在 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......")) 时,出现了分段错误,为什么?谢谢

似乎 inet_ntoa() 可能以某种方式 returning NULL,导致在 printf() 中取消引用时出现段错误。我找不到明确说明 inet_ntoa 的 Linux 版本可以做到这一点的直接参考,但我发现有几个人提出了这种说法,这是该代码中唯一的一点正在取消引用指针。

这个问题底部的答案:segmentation fault for inet_ntoa 声称 inet_ntoa 可以 return NULL。但是,按照他的参考链接,我找不到关于该事实的实际陈述。

有一篇 MSDN 文章(具有启发性,但当然不能直接应用于 Linux 代码)明确指出 inet_ntoa() 可以 return NULL这里:https://msdn.microsoft.com/en-us/library/windows/desktop/ms738564%28v=vs.85%29.aspx

显示的代码遗漏了 #include 任何 headers,因此由于一些未定义的符号,目前无法编译。

但是,如果您错过了对代码引用的任何库函数进行原型化,它会编译,这将导致任何函数被假定为 return int

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

在 64 位系统上,至少在 inet_ntoa() 用作 printf() 的参数的情况下它是致命的,因为在 64 位系统上它很可能是 return 64 位(char-指针)值(但 32 位 int)。因此(假设原型未命中)编译器在生成代码时假设 inet_ntoa() 到 return 一个 32 位 int 这将导致 "chopping-off" 地址的最重要的 32 位 return编辑。尝试从这样的 "crippled" 中 printf() 并因此(很可能)无效地址会引发未定义的行为,并且在您的情况下会导致观察到的分段违规。

要解决此问题,请添加相关原型(针对 inet_ntoa()):

#include <arpa/inet.h>

编译器应该就此警告过您。要启用 gcc 的所有编译器警告,请使用选项 -Wall -Wextra -pedantic。认真对待此类警告。