【问题标题】:Transmitting UDP packages with source address 0.0.0.0发送源地址为 0.0.0.0 的 UDP 包
【发布时间】:2012-10-14 04:59:47
【问题描述】:

在 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 标头。这是可能的,但我想避免手动执行此操作。

【问题讨论】:

  • 如果没有接口有 IPv4 地址,则根本无法发送数据包...
  • @glglgl 那么你将如何发送 DHCP 发现(正是需要这个)?
  • 可能使用 RAW 套接字...
  • 啊,你的评论是关于 DGRAM 套接字的,我以为你的意思是一般的,现在有道理了 :)
  • @ALlan:您为什么要避免使用 RAW SOCKET,有什么具体原因吗?

标签: c linux sockets network-programming


【解决方案1】:

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

【讨论】:

    【解决方案2】:

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

    你可以举个例子:https://github.com/fycth/DHCP-server-scanner/blob/master/src/dhcpd-detector.c

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

    【讨论】:

    • 这是我试图避免的......但我想没有其他方法可以做到这一点。
    • 很抱歉让您失望了,但似乎没有办法避免这种激动人心的体验......
    • 不幸的是,即使这样似乎也不起作用。对于原始套接字,源地址 0.0.0.0 被认为是“零”,并由 Linux 内核“有用地”填充另一个任意 IP 地址:man7.org/linux/man-pages/man7/raw.7.html 一个选项可能是使用原始以太网帧来避免这种情况,但是这有其自身的缺点。
    猜你喜欢
    • 1970-01-01
    • 2022-01-19
    • 1970-01-01
    • 1970-01-01
    • 2020-03-11
    • 2017-04-25
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多