Отправка и получение пакета UDP от одного и того же процесса - PullRequest
0 голосов
/ 13 января 2012

Вот то, что у меня есть (проверка на наличие ошибок):

struct sockaddr_in addr, ss, dest;
int port, s = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
fcntl(s, F_SETFL, fcntl(s, F_GETFL, 0) | O_NONBLOCK);

memset((char*) &addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
inet_aton("127.0.0.1", &addr.sin_addr);
bind(s, (struct sockaddr*) &addr, sizeof(addr));

unsigned int len = sizeof(ss);
getsockname(s, (struct sockaddr*) &ss, &len);
port = ss.sin_port;

memset((char*) &dest, 0, sizeof(dest));
dest.sin_family = AF_INET;
dest.sin_port = htons(port);
inet_aton("127.0.0.1", &dest.sin_addr);
sendto(s, "test", 5, 0, (struct sockaddr*) &dest, sizeof(dest));

char buf[5];
recv(s, buf, 5, 0);

Последнее предложение завершается с ошибкой Resource temporarily unavailable (из-за флага O_NONBLOCK).

Во фрагменте я разрешаю ОС привязывать произвольный порт, а затем получаю его с getsockname.Если я вместо этого использую фиксированный порт и удаляю вызов на getsockname, он работает.

PS: я на машине с linux.

Ответы [ 2 ]

4 голосов
/ 13 января 2012

port = ss.sin_port должен дать заказанный номер сети порт. Когда вы назначаете порт с dest.sin_port = htons(port), вы применяете htons() к короткому замыканию, которое уже находится в сетевом порядке байтов. Вместо этого используйте dest.sin_port = port, и все должно быть в порядке.

В качестве альтернативы, если вы хотите получить заказанный хостом номер порта из getsockname() результата, вы должны использовать ntohs():

getsockname(s, (struct sockaddr*) &ss, &len);
port = ntohs(ss.sin_port);
/*...*/
dest.sin_port = htons(port);
2 голосов
/ 13 января 2012

Вы забыли использовать ntohs при захвате назначенного ОС порта.Вот что я закончил (я сделал пару небольших вещей, чтобы сделать код более лаконичным):

#include <netinet/in.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>

int main()
{
  struct sockaddr_in addr = {}, ss, dest = {};
  int port, s = socket(AF_INET, SOCK_DGRAM | SOCK_NONBLOCK, 0);

  addr.sin_family = AF_INET;
  addr.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
  bind(s, (struct sockaddr*) &addr, sizeof(addr));

  unsigned int len = sizeof(ss);
  getsockname(s, (struct sockaddr*) &ss, &len);
  port = ntohs(ss.sin_port);

  dest.sin_family = AF_INET;
  dest.sin_port = htons(port);
  dest.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
  sendto(s, "test", 5, 0, (struct sockaddr*) &dest, sizeof(dest));

  char buf[5];
  int got = recv(s, buf, 5, 0);

  printf("got: %d, errno: %s\n", got, strerror(errno));

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