Установка исходного IP для сокета UDP - PullRequest
26 голосов
/ 17 июня 2010

У меня есть сокет UDP, связанный с INADDR_ANY для прослушивания пакетов на всех IP-адресах, которые есть на моем сервере.Я отправляю ответы через тот же сокет.

Сейчас сервер автоматически выбирает, какой IP-адрес используется в качестве исходного IP-адреса при отправке пакетов, но я хотел бы иметь возможность установить исходящий IP-адрес источника.себя.

Есть ли способ сделать это без необходимости создавать отдельный сокет для каждого IP?

Ответы [ 4 ]

26 голосов
/ 14 октября 2010

Николай, использование отдельного сокета и bind (2) для каждого адреса или работа с таблицами маршрутизации часто неосуществима, например. с динамическими адресами. Один UDP-сервер, связанный с IP_ADDRANY, должен иметь возможность отвечать на тот же динамически назначенный IP-адрес, на который получен пакет.

К счастью, есть другой способ. В зависимости от поддержки вашей системы вы можете использовать опции сокета IP_PKTINFO для установки или получения вспомогательных данных о сообщении. Вспомогательные данные (через cmsg(3)) доступны во многих местах онлайн, хотя comp.os.linux.development.system содержит полный пример кода, специфичный для IP_PKTINFO.

Код в ссылке использует IP_PKTINFO (или IP_RECVDSTADDR в зависимости от платформы) для получения адреса назначения UDP-сообщения из вспомогательных данных cmsg(3). Перефразируя здесь:

struct msghdr msg;
struct cmsghdr *cmsg;
struct in_addr addr;
// after recvmsg(sd, &msg, flags);
for(cmsg = CMSG_FIRSTHDR(&msg);
    cmsg != NULL;
    cmsg = CMSG_NXTHDR(&msg, cmsg)) {
  if (cmsg->cmsg_level == IPPROTO_IP && cmsg->cmsg_type == IP_PKTINFO) {
    addr = ((struct in_pktinfo*)CMSG_DATA(cmsg))->ipi_addr;
    printf("message received on address %s\n", inet_ntoa(addr));
  }
}

Джин, твой вопрос спросил, как установить исходный адрес для исходящих пакетов. С помощью IP_PKTINFO можно установить поле ipi_spec_dst для struct in_pktinfo во вспомогательных данных, передаваемых на sendmsg(2). См. Пост, на который ссылаются выше, cmsg(3) и sendmsg(2) для получения инструкций о том, как создавать и обрабатывать вспомогательные данные в struct msghdr. Пример (без гарантии здесь) может быть:

struct msghdr msg;
struct cmsghdr *cmsg;
struct in_pktinfo *pktinfo;
// after initializing msghdr & control data to CMSG_SPACE(sizeof(struct in_pktinfo))
cmsg = CMSG_FIRSTHDR(&msg);
cmsg->cmsg_level = IPPROTO_IP;
cmsg->cmsg_type = IP_PKTINFO;
cmsg->cmsg_len = CMSG_LEN(sizeof(struct in_pktinfo));
pktinfo = (struct in_pktinfo*) CMSG_DATA(cmsg);
pktinfo->ipi_ifindex = src_interface_index;
pktinfo->ipi_spec_dst = src_addr;
// bytes_sent = sendmsg(sd, &msg, flags);

Обратите внимание, что в IPv6 все по-другому: используйте struct in6_pktinfo::ipi6_addr в случаях recvmsg и sendmsg.

Обратите внимание также, что Windows не поддерживает эквивалент ipi_spec_dst в структуре in_pktinfo, поэтому вы не можете использовать этот метод для установки адреса источника в исходящем пакете winsock2.

(ссылки на справочные страницы - ограничение по 1 гиперссылке)

http:// linux.die.net/man/2/sendmsg
http:// linux.die.net/man/3/cmsg
18 голосов
/ 13 сентября 2012

Я думал, что расскажу Джереми, как это сделать для IPv6. Джереми пропускает много деталей, и некоторая документация (например, справочная страница Linux для ipv6) просто неверна. Сначала в некоторых дистрибутивах вы должны определить _GNU_SOURCE, иначе некоторые вещи IPv6 не определены:

#define _GNU_SOURCE
#include <netinet/in.h>
#include <sys/types.h>
#include <sys/socket.h>

Затем настройте сокет довольно стандартным способом, который прослушивает все IP-пакеты (т. Е. Как IPv4, так и IPv6) на конкретном порте UDP:

const int on=1, off=0;
int result;
struct sockaddr_in6 sin6;
int soc;

soc = socket(AF_INET6, SOCK_DGRAM, 0);
setsockopt(soc, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
setsockopt(soc, IPPROTO_IP, IP_PKTINFO, &on, sizeof(on));
setsockopt(soc, IPPROTO_IPV6, IPV6_RECVPKTINFO, &on, sizeof(on));
setsockopt(soc, IPPROTO_IPV6, IPV6_V6ONLY, &off, sizeof(off));
memset(&sin6, '\0', sizeof(sin6));
sin6.sin6_family = htons(AF_INET6);
sin6.sin6_port = htons(MY_UDP_PORT);
result = bind(soc, (struct sockaddr*)&sin6, sizeof(sin6));

Обратите внимание, что приведенный выше код устанавливает параметры IP и IPv6 для сокета IPv6. Оказывается, если пакет приходит на IPv4-адрес, вы получите cmsg IP_PKTINFO (т.е. IPv4), даже если это сокет IPv6, и если вы не включите их, они не будут отправлены. Также обратите внимание, что установлена ​​опция IPV6_RECPKTINFO (которая не упоминается в man 7 ipv6 ), а не IPV6_PKTINFO (что неправильно описано в man 7 ipv6 ). Теперь получите пакет udp:

int bytes_received;
struct sockaddr_in6 from;
struct iovec iovec[1];
struct msghdr msg;
char msg_control[1024];
char udp_packet[1500];

iovec[0].iov_base = udp_packet;
iovec[0].iov_len = sizeof(udp_packet);
msg.msg_name = &from;
msg.msg_namelen = sizeof(from);
msg.msg_iov = iovec;
msg.msg_iovlen = sizeof(iovec) / sizeof(*iovec);
msg.msg_control = msg_control;
msg.msg_controllen = sizeof(msg_control);
msg.msg_flags = 0;
bytes_received = recvmsg(soc, &msg, 0);

Следующим шагом является извлечение интерфейса и адреса, по которому пакет UDP был получен из cmsg:

struct in_pktinfo in_pktinfo;
struct in6_pktinfo in6_pktinfo;
int have_in_pktinfo = 0;
int have_in6_pktinfo = 0;
struct cmsghdr* cmsg;

for (cmsg = CMSG_FIRSTHDR(&msg); cmsg != 0; cmsg = CMSG_NXTHDR(&msg, cmsg))
{
  if (cmsg->cmsg_level == IPPROTO_IP && cmsg->cmsg_type == IP_PKTINFO)
  {
    in_pktinfo = *(struct in_pktinfo*)CMSG_DATA(cmsg);
    have_in_pktinfo = 1;
  }
  if (cmsg->cmsg_level == IPPROTO_IPV6 && cmsg->cmsg_type == IPV6_PKTINFO)
  {
    in6_pktinfo = *(struct in6_pktinfo*)CMSG_DATA(cmsg);
    have_in6_pktinfo = 1;
  }
}

Наконец, мы можем отправить ответ обратно, используя тот же пункт назначения.

int cmsg_space;

iovec[0].iov_base = udp_response;
iovec[0].iov_len = udp_response_length;
msg.msg_name = &from;
msg.msg_namelen = sizeof(from);
msg.msg_iov = iovec;
msg.msg_iovlen = sizeof(iovec) / sizeof(*iovec);
msg.msg_control = msg_control;
msg.msg_controllen = sizeof(msg_control);
msg.msg_flags = 0;
cmsg_space = 0;
cmsg = CMSG_FIRSTHDR(&msg);
if (have_in6_pktinfo)
{
  cmsg->cmsg_level = IPPROTO_IPV6;
  cmsg->cmsg_type = IPV6_PKTINFO;
  cmsg->cmsg_len = CMSG_LEN(sizeof(in6_pktinfo));
  *(struct in6_pktinfo*)CMSG_DATA(cmsg) = in6_pktinfo;
  cmsg_space += CMSG_SPACE(sizeof(in6_pktinfo));
}
if (have_in_pktinfo)
{
  cmsg->cmsg_level = IPPROTO_IP;
  cmsg->cmsg_type = IP_PKTINFO;
  cmsg->cmsg_len = CMSG_LEN(sizeof(in_pktinfo));
  *(struct in_pktinfo*)CMSG_DATA(cmsg) = in_pktinfo;
  cmsg_space += CMSG_SPACE(sizeof(in_pktinfo));
}
msg.msg_controllen = cmsg_space;
ret = sendmsg(soc, &msg, 0);

Снова обратите внимание, как, если пакет поступил через IPv4, мы должны поместить опцию IPv4 в cmsg, даже если это сокет AF_INET6. По крайней мере, это то, что вы должны сделать для Linux.

Это удивительный объем работы, но AFAICT - это минимум, который вам нужно сделать, чтобы создать надежный UDP-сервер, работающий во всех мыслимых средах Linux. Большая часть этого не требуется для TCP, потому что он прозрачно обрабатывает множественную адресацию.

3 голосов
/ 17 июня 2010

Вы либо bind(2) для каждого адреса интерфейса и управляете несколькими сокетами, либо позволяете ядру выполнять неявное назначение IP-адреса источника с помощью INADDR_ANY.Другого пути нет.

Мой вопрос: зачем тебе это?У вас не работает нормальная IP-маршрутизация?

0 голосов
/ 18 августа 2016

Я недавно столкнулся с той же проблемой.

Что я делаю, чтобы решить эту проблему:

  1. получить имя интерфейса из полученного пакета
  2. привязать сокет к определенномуинтерфейс
  3. сокет unbind

Пример:

  struct ifreq ifr;
  ...
  recvmsg(fd, &msg...)
  ...      
  if (msg.msg_controllen >= sizeof(struct cmsghdr))
    for (cmptr = CMSG_FIRSTHDR(&msg); cmptr; cmptr = CMSG_NXTHDR(&msg, cmptr))
      if (cmptr->cmsg_level == SOL_IP && cmptr->cmsg_type == IP_PKTINFO)
      {
        iface_index = ((struct in_pktinfo *)CMSG_DATA(cmptr))->ipi_ifindex;
      }
  if_indextoname(iface_index , ifr.ifr_name);
  mret=setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, &ifr, sizeof(ifr));

  sendmsg(...);

  memset(&ifr, 0, sizeof(ifr));
  snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), "");
  mret=setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, &ifr, sizeof(ifr));
...