7

在 linux 中,如何使用 0.0.0.0 作为源地址传输 UDP 数据包。

这是我到目前为止所尝试的。

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

问题是,如果没有接口具有 IPv4 地址,则源地址仅设置为 0.0.0.0。如果只有一个接口有 IPv4 地址,则该地址用作源地址。

我查看了一些现有 DHCP 客户端的源代码,发现它们使用 RAW 套接字并手动构建 IP 和 UDP 标头。这是可能的,但我想避免手动执行此操作。

4

2 回答 2

4

您应该使用 RAW SOCKET 并自己努力构建数据包。

作为示例,您可以这样做:https ://github.com/fycth/DHCP-server-scanner/blob/master/src/dhcpd-detector.c

这是我的教育项目。它很小,你可以看看这个任务是如何在那里解决的。

于 2012-10-24T07:19:53.803 回答
0

我认为如果可以通过 UDP 套接字来实现,他们会这样做而不是手动构建一个。所以我认为它根本不起作用......

于 2012-10-24T07:15:44.483 回答