Передача пакетов UDP с адресом источника 0.0.0.0
В Linux, как я могу передать пакет UDP, используя 0.0.0.0 в качестве адреса источника.
Вот что я пробовал до сих пор.
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <net/if.h>
#include <string.h>
#include <unistd.h>
#include <assert.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <sys/socket.h>
int main(int argc, const char *argv[])
{
struct sockaddr_in dst, src;
struct ifreq ifr;
int sock, tmp;
char payload[128];
memset(payload, 0, 128);
memset(&dst, 0, sizeof(dst));
dst.sin_family = AF_INET;
dst.sin_addr.s_addr = inet_addr("255.255.255.255");
dst.sin_port = htons(67);
memset(&src,0,sizeof(src));
src.sin_family = AF_INET;
src.sin_addr.s_addr = inet_addr("0.0.0.0");
src.sin_port = htons(68);
sock = socket(AF_INET, SOCK_DGRAM, 0);
if (sock < 0)
perror("Failed to create socket");
tmp = 1;
if (setsockopt(sock, SOL_SOCKET, SO_BROADCAST, &tmp, sizeof(tmp)) < 0)
perror("SO_BROADCAST failed");
if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &tmp, sizeof(tmp)) < 0)
perror("SO_REUSEADDR failed");
if (setsockopt(sock, IPPROTO_IP, IP_FREEBIND, &tmp, sizeof(tmp)) < 0)
perror("IP_FREEBIND failed");
memset(&ifr, 0, sizeof(ifr));
strncpy(ifr.ifr_name, "eth1", sizeof(ifr.ifr_name));
if (setsockopt(sock, SOL_SOCKET, SO_BINDTODEVICE, (void *)&ifr, sizeof(ifr)) < 0)
perror("SO_BINDTODEVICE failed");
if (bind(sock, (const struct sockaddr*)&src, sizeof(struct sockaddr_in)) < 0)
perror("bind failed");
if (connect(sock, (const struct sockaddr*)&dst, sizeof(struct sockaddr_in)) < 0)
perror("bind failed");
if (write(sock, payload, 128) < 0 )
perror("Write failed");
close(sock);
return 0;
}
Проблема заключается в том, что адрес источника установлен на 0.0.0.0, только если ни один из интерфейсов не имеет адреса IPv4. Если только один интерфейс имеет адрес IPv4, этот адрес используется в качестве адреса источника.
Я просмотрел исходный код некоторых существующих DHCP-клиентов и обнаружил, что они используют сокеты RAW и создают заголовки IP и UDP вручную. Это возможно, но я бы хотел избежать этого вручную.
2 ответа
Вы должны использовать RAW SOCKET и создать пакет своими силами.
В качестве примера вы можете взять это: https://github.com/fycth/DHCP-server-scanner/blob/master/src/dhcpd-detector.c
Это мой образовательный проект. Он маленький, и вы можете посмотреть, как именно эта задача там решается.
Я думаю, что если бы это было возможно сделать через сокеты UDP, они бы сделали это вместо того, чтобы создавать его вручную. Так что я предполагаю, что это не работает вообще...