【问题标题】:Network program works in Linux, not under Cygwin网络程序在 Linux 下工作,在 Cygwin 下不行
【发布时间】:2011-09-11 16:27:07
【问题描述】:

我编写了一个简单的程序,它从stdin 读取输入并通过 TCP 将其发送到侦听端口 15557 的服务器。

当我在 Linux 下编译和运行它时,它工作正常。但是,当我尝试在 Cygwin 下编译和运行它时,它会失败并显示以下错误消息:

$ ./a.out servername.net 15557 < test.dat
Unable to connect: Cannot assign requested address

我已经排除了任何防火墙/网络问题,因为我能够通过 telnet 连接到同一台服务器并发送手动输入的相同数据。

知道这里出了什么问题吗?

更新

按照@Hasturkun 的提示,我在GDB 下启动了程序来检查gethostbyname 的结果是什么。就是在gethostbyname 调用之后:

(gdb) print *serverent
$2 = {h_name = 0x603217 "bilbo.neurobat.net", h_aliases = 0x603030, h_addrtype = 2, h_length = 4, h_addr_list = 0x6031c0}
(gdb) print serverent->h_addr_list[0]
$3 = 0x60322c ">\002V0"
(gdb) print atoi(">\002V0")
$5 = 0

我不明白如何解释“>\002V0”字符串。这应该是一个互联网地址吗?

/更新

FWIW,整个程序如下图:

#include <stdio.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <string.h>
#include <unistd.h>

#include "neurocli.h"

int main(int argc, char* argv[]) {
  char *line = NULL;
  char buf[40] = {0};
  ssize_t write_len = 0, read_len;
  size_t n = 0;
  int neuro_socket;

  if (argc != 3) {
    fprintf(stderr, "Usage: %s host port\n", argv[0]);
    return -1;
  }

  neuro_socket = open_tcp_socket(argv[1], atoi(argv[2]));

  /* Main loop: connect, write, read, re-connect and finally close */
  while ((write_len=getline(&line, &n, stdin)) != -1) {
    printf("# %s", line);
    if (write(neuro_socket, line, write_len) < 0) {
      perror("Unable to write to server");
      exit(EXIT_FAILURE);
    }
    if (*line=='\n') {
      read_len = read(neuro_socket, buf, 40);
      buf[read_len] = '\0';
      printf("%s", buf);
      close(neuro_socket);
      neuro_socket = open_tcp_socket(argv[1], atoi(argv[2]));
    }
    free(line);
    line = NULL; /* getline() needs this */
  }
  free(line);
  close(neuro_socket);
  return 0;
}


int open_tcp_socket(char *server, int port) {
  int result;
  static struct sockaddr_in *sockaddr;

  /* Create socket */
  if ((result = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
    perror("Unable to create socket");
    exit(EXIT_FAILURE);
  }

  if (sockaddr==NULL)
    sockaddr = make_sockaddr(server, port); /* never mind the memory leak */

  /* Connect */
  if (connect(result, (struct sockaddr*)sockaddr, sizeof(*sockaddr)) != 0) {
    perror("Unable to connect");
    exit(EXIT_FAILURE);
  }

  return result;
}

struct sockaddr_in *make_sockaddr(char *name, int port) {
  struct sockaddr_in *sockaddr = malloc(sizeof(struct sockaddr_in));
  struct hostent *serverent;

  memset(sockaddr, 0, sizeof(*sockaddr));
  sockaddr->sin_family = AF_INET;
  sockaddr->sin_port = htons(port);

  if ((serverent = gethostbyname(name)) == NULL) {
    perror("Unable to lookup server IP address");
    exit(EXIT_FAILURE);
  }

  sockaddr->sin_addr.s_addr = atoi(serverent->h_addr_list[0]);

  return sockaddr;
}

【问题讨论】:

  • 是你自己的错误信息被打印出来,还是来自图书馆的东西?如果它是您自己的(即您自己对perror 的调用或类似性质的东西),那么您有相应的代码部分吗?
  • @Jason 我相信是因为调用perror失败,距离程序结束大约6行。
  • 那不是说“无法查找服务器 IP 地址”吗? ...我只是看不到任何地方说“无法建立数据连接”。我假设消息的一部分是您传递给某处的perror 的字符串。如果不是,那么我应该看到您的一个字符串表示对 connect 的调用失败,等等。
  • @Jason 对不起,你是对的。我正在从内存中复制在客户端机器上看到的一行。我现在已经更正了。

标签: c networking cygwin


【解决方案1】:

这一行

sockaddr->sin_addr.s_addr = atoi(serverent->h_addr_list[0]);

不正确。 h_addr_list 包含struct in_addrstruct in6_addr 的地址,长度在h_length 中给出,类型可以通过h_addrtype 确定。

要获取地址,memcpy 或将其分配给适当的类型

【讨论】:

  • 我已经在 GDB 下启动了该程序,以检查该行发生了什么,并用结果更新了问题。我必须说我现在更加困惑,因为该程序将不再在 Linux 下运行:-o
  • @lindelof:它不是一个字符串,它是一个struct in_addr(我的答案以前是错误的,因为已更正)你可以分配它,例如。 sockaddr-&gt;sin_addr = *(struct in_addr*)serverent-&gt;h_addr_list[0];
  • 我大吃一惊,它有效。但是gethostyname(3) 页面说gethostbyname 返回一个结构,其h_addr_list 字段是char**h_addr_list[0] 怎么能转换成struct in_addr*
【解决方案2】:

connect() 调用失败,因为地址无效。对 gethostbyname() 的调用很可能返回一个没有有用地址的结构(即 h_addr_list[0] == 0)。

不太确定 cygwin 如何与 DNS 客户端交互...

【讨论】:

  • 可能是这样。实际上,该程序在 Linux 下也会失败,尽管是以不可预知的方式。查看问题的更新。
最近更新 更多