linux - 使用 libmnl 和 rtnetlink 将 IPv6 地址添加到接口(interface)

标签 linux kernel netlink

我正在尝试使用 libmnl 将 IPv6 地址添加到以太网接口(interface)。构建消息并发送给内核后,我看到它没有添加到接口(interface)中,即使内核回复的返回码不包含任何错误。请任何人看看,并帮助我纠正它。我应该向 nlmsghdr 或其他东西添加更多属性吗?

#include <assert.h>
#include <string.h>
#include <inttypes.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <libmnl/libmnl.h>
#include <linux/rtnetlink.h>
#include <net/if.h>
#include <time.h>
static struct mnl_socket *nl;
static unsigned int nlportid;

int mnl_init(void);
int mnl_init(){
    nl = mnl_socket_open(NETLINK_ROUTE);
    if(nl == NULL){
        printf("Error: mnl_socket_open\n");
        return 0;
    }
    if(mnl_socket_bind(nl, 0, MNL_SOCKET_AUTOPID) < 0){
        printf("Error: mnl_socket_bind\n");
        return 0;
    }
    nlportid = mnl_socket_get_portid(nl);
    return 1;
}
int add_to_interface(const char* eip){
    char buf[MNL_SOCKET_BUFFER_SIZE];
    struct nlmsghdr *nlh;
    struct ifaddrmsg *ifm;
    int ret;
    uint8_t seq;

    nlh = mnl_nlmsg_put_header(buf);

    nlh->nlmsg_type = RTM_NEWADDR;
    nlh->nlmsg_flags = NLM_F_REQUEST|NLM_F_CREATE|NLM_F_EXCL;
    nlh->nlmsg_seq = seq = time(NULL);

    ifm = mnl_nlmsg_put_extra_header(nlh, sizeof(*ifm));

    ifm->ifa_family = AF_INET6;
    ifm->ifa_prefixlen = 64; 
    ifm->ifa_flags = IFA_F_PERMANENT;
    ifm->ifa_scope = RT_SCOPE_UNIVERSE;
    /* TODO get interaface name from user or configuration*/
    ifm->ifa_index = if_nametoindex("eth0");
    unsigned char eipn[16]; 
    inet_pton(AF_INET6, eip, eipn);
    mnl_attr_put(nlh, IFA_ADDRESS, 16,eipn);

    mnl_nlmsg_fprintf(stdout, nlh, nlh->nlmsg_len, sizeof(struct ifaddrmsg));

    if(mnl_socket_sendto(nl,nlh, nlh->nlmsg_len) < 0){
        printf("Error: mnl_socket_sendto");
        return 0;
    }
    ret = mnl_socket_recvfrom(nl,buf, sizeof(buf));
    if(ret == -1){
        printf("Error: mnl_socket_recvfrom");
        return 0;
    }
    ret = mnl_cb_run(buf, ret, seq, nlportid, NULL, NULL); 
    return 0;
}

int main(int argc, char *argv[]){
    if(mnl_init()){
        add_to_interface("2001::20c:29ff:fe5f:13c7/64"); // for testing
    }
}

最佳答案

我对libmnl一无所知,下面的解决方案并不完美。但如果你仍然卡住了......

请注意,您正在丢弃多个错误代码。这一个,特别是:

inet_pton(AF_INET6, eip, eipn);

应该是这样的:

ret = inet_pton(AF_INET6, eip, eipn);
if (ret != 1) {
    printf("Bad address.\n");
    return -22;
}

而且我想您能看出我的目标是什么。这:

add_to_interface("2001::20c:29ff:fe5f:13c7/64"); // for testing

应该是这样的:

add_to_interface("2001::20c:29ff:fe5f:13c7"); // for testing

这对我来说已经解决了。除了它在 mnl_socket_recvfrom() 处停顿,因为内核显然没有回答。

但是要更加小心这些错误代码;我提到的并不是唯一的。

关于linux - 使用 libmnl 和 rtnetlink 将 IPv6 地址添加到接口(interface),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39183969/

相关文章:

c - 网络链接套接字

linux - 为什么即使包含 "RTMGRP_LINK undeclared"也会收到 "rtnetlink.h"错误?

linux - 有没有办法捕获基于 openssh key 的身份验证失败?

linux - 关于 Gtk 应用程序中模态对话框的问题

c - 软链接(soft link) atime 和 mtime 修改

linux - 禁用滴答中断时如何更新时间

linux - Bash 脚本对于不同的结果获得相同的存在状态

linux-kernel - 前缀 printk/pr_* 调用

linux - 我如何知道发行版使用哪个 Linux 内核版本?

linux - netlink 套接字 : kernel freeze 的问题