C ++ DGRAM сокет получить адрес получателя - PullRequest
3 голосов
/ 15 октября 2010

В C ++,
как я могу получить адрес получателя пакета UDP, который я получил, используя recvfrom.Я знаю, что это должен быть тот же хост, на котором я получаю пакет, но мне нужно извлечь его из полученного пакета, чтобы что-то проверить.Как я могу это сделать?

Я обнаружил, что один из способов сделать это:

int r = getsockopt(receiver_sock, SOL_IP, SO_ORIGINAL_DST, (struct sockaddr *) &sender_addr, (socklen_t *)&addr_len);`

Но я получаю ошибку:

error: ‘SO_ORIGINAL_DST’ was not declared in this scope

Я используюсоответствующие заголовки

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include </usr/src/linux-headers-2.6.32-21/include/linux/netfilter_ipv4.h>
#include <arpa/inet.h>    
#include <linux/netfilter.h>

Использование netfilter_ipv4 дает другие ошибки, такие как INT_MIN, не объявленные.Тем не менее, я думаю, что ошибка является чем-то более фундаментальным, чем включение правильного заголовка.

Пожалуйста, помогите.

Ответы [ 2 ]

3 голосов
/ 15 октября 2010

В Linux вы хотите использовать опцию IP_PKTINFO, см. ip(7) и вызов recvmsg(2).

Стивенс есть примеры этого, но с опциями IP_RECVDSTADDR и IP_RECVIF, которые недоступны в Linux.

2 голосов
/ 15 марта 2011

Я создал пример, который извлекает адреса источника, назначения и интерфейса.Для краткости проверка ошибок не предусмотрена.Посмотрите этот дубликат: Получите адрес назначения полученного пакета UDP .

// sock is bound AF_INET socket, usually SOCK_DGRAM
// include struct in_pktinfo in the message "ancilliary" control data
setsockopt(sock, IPPROTO_IP, IP_PKTINFO, &opt, sizeof(opt));
// the control data is dumped here
char cmbuf[0x100];
// the remote/source sockaddr is put here
struct sockaddr_in peeraddr;
// if you want access to the data you need to init the msg_iovec fields
struct msghdr mh = {
    .msg_name = &peeraddr,
    .msg_namelen = sizeof(peeraddr),
    .msg_control = cmbuf,
    .msg_controllen = sizeof(cmbuf),
};
recvmsg(sock, &mh, 0);
for ( // iterate through all the control headers
    struct cmsghdr *cmsg = CMSG_FIRSTHDR(&mh);
    cmsg != NULL;
    cmsg = CMSG_NXTHDR(&mh, cmsg))
{
    // ignore the control headers that don't match what we want
    if (cmsg->cmsg_level != IPPROTO_IP ||
        cmsg->cmsg_type != IP_PKTINFO)
    {
        continue;
    }
    struct in_pktinfo *pi = CMSG_DATA(cmsg);
    // at this point, peeraddr is the source sockaddr
    // pi->ipi_spec_dst is the destination in_addr
    // pi->ipi_addr is the receiving interface in_addr
}
...