我正在尝试使用libmnl将IPv6地址添加到以太网接口。在构造消息并发送到内核之后,我看到它没有添加到接口,即使内核回复的返回码没有包含任何错误。请任何人看看,并帮我纠正它。我应该为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
}
}
答案 0 :(得分:0)
我对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()
,因为内核没有回答,显然。
但是要更加小心这些错误代码;我提到的不是唯一的。
答案 1 :(得分:0)
来自Pablo Neira的解决方案,
eipn应该是&#39; struct in6_addr&#39;代替。
mnl_attr_put(nlh, IFA_ADDRESS, 16,eipn);
所以这看起来像:
mnl_attr_put(nlh, IFA_ADDRESS, sizeof(eipn), &eipn);