Сетевая программа работает в Linux, а не под Cygwin - PullRequest
3 голосов
/ 09 июня 2011

Я написал простую программу, которая читает входные данные из 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".Это должен быть интернет-адрес?

/ UPDATE

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;
}

Ответы [ 2 ]

1 голос
/ 09 июня 2011

Эта строка

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

Неверно.h_addr_list содержит адреса struct in_addr или struct in6_addr, длина указана в h_length, а тип можно определить с помощью h_addrtype.

Чтобы получить адрес, memcpy или назначьтеэто к соответствующему типу

0 голосов
/ 09 июня 2011

Ошибка вызова connect () из-за неправильного адреса.Скорее всего, вызов gethostbyname () возвращает структуру без полезных адресов (т.е. h_addr_list [0] == 0).

Не совсем уверен, как cygwin взаимодействует с DNS-клиентом ...

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...