在开发一个小型DHCP客户端时,我通过绑定到IFADDR_ANY并设置SO_BROADCAST和SO_BINDTODEVICE来在特定接口上发送广播,我可以看到来自DHCP服务器的答复。
不幸的是,除非该接口配置了地址,否则我的应用程序不会接收任何数据包。这使得编写DHCP客户端变得毫无意义。
问题:如何在没有分配地址的接口上接收IPv4 UDP广播?
我知道原始套接字,但我想避免使用它们。如果发送是不可能的,我就会接受,除非配置了地址,否则不可能使用UDP,但是由于发送工作正常,而仅仅接收不可能,我想知道我是否遗漏了什么。
下面是最低限度的代码:
#define _DEFAULT_SOURCE
#include <sys/types.h>
#include <arpa/inet.h>
#include <assert.h>
#include <errno.h>
#include <string.h>
#include <sys/time.h>
#include <sys/select.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <unistd.h>
#include <stdio.h>
int main(int argc, char *argv[])
{
    if(argc < 2) {
        return 0;
    }
    struct sockaddr_in src = { .sin_family=AF_INET, .sin_addr.s_addr=INADDR_ANY, .sin_port=htons(68) };
    int fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_IP);
    const int on=1;
    setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
    struct ifreq ifr;
    memset(&ifr, '\0', sizeof(ifr));
    snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), argv[1]);
    ioctl(fd, SIOCGIFINDEX, &ifr);
    setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, (void *)&ifr, sizeof(ifr));
    bind(fd, (struct sockaddr *)&src, sizeof(src));
    char buf[512];
    ssize_t res = recvfrom(fd, buf, sizeof(buf), 0, NULL, 0);
    printf("res=%zi\n", res);
    return 0;
}我在"sudo strace“中运行它,它永远停留在recvmsg()中,即使我可以在wireshark的端口68上观看UDP广播。
发布于 2016-02-11 23:05:27
我解决了。“反向路径过滤”被激活,导致所有将采用不同返回路径(或没有路由)的IPv4数据包被丢弃。如果没有为该接口配置地址,则每个数据包都会被丢弃。
echo 0 > /proc/sys/net/ipv4/conf/all/rp_filter成功了。
https://stackoverflow.com/questions/35325189
复制相似问题