I'm trying to assign an IPv6 address to an interface using ioctl, but in vain. Here's the code I used:
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stddef.h>
#include <net/if.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <linux/sockios.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#define IFNAME "eth1"
#define HOST "fec2::22"
#define ifreq_offsetof(x) offsetof(struct ifreq, x)
int main(int argc, char **argv) {
struct ifreq ifr;
struct sockaddr_in6 sai;
int sockfd; /* socket fd we use to manipulate stuff with */
int selector;
unsigned char mask;
char *p;
/* Create a channel to the NET kernel. */
sockfd = socket(AF_INET6, SOCK_DGRAM, 0);
if (sockfd == -1) {
printf("Bad fd
");
return -1;
}
/* get interface name */
strncpy(ifr.ifr_name, IFNAME, IFNAMSIZ);
memset(&sai, 0, sizeof(struct sockaddr));
sai.sin6_family = AF_INET6;
sai.sin6_port = 0;
//if(inet_aton(HOST, &sai.sin_addr.s_addr) == 0) {
if(inet_pton(AF_INET6, HOST, (void *)&sai.sin6_addr) <= 0) {
//&((struct sockaddr_in*)&sa)->sin_addr
printf("Bad address
");
return -1;
}
p = (char *) &sai;
memcpy( (((char *)&ifr + ifreq_offsetof(ifr_addr) )),
p, sizeof(struct sockaddr));
int ret = ioctl(sockfd, SIOCSIFADDR, &ifr);
printf("ret: %derrno: %d
", ret, errno);
ioctl(sockfd, SIOCGIFFLAGS, &ifr);
printf("ret: %derrno: %d
", ret, errno);
ifr.ifr_flags |= IFF_UP | IFF_RUNNING;
// ifr.ifr_flags &= ~selector; // unset something
ioctl(sockfd, SIOCSIFFLAGS, &ifr);
printf("ret: %derrno: %d
", ret, errno);
close(sockfd);
return 0;
}
The ioctl calls fail saying ENODEV. When the family of the socket is changed to sockfd = socket(AF_INET, SOCK_DGRAM, 0);
, the calls fail again saying EINVAL.
I was able to assign an IPv4 address to the interface with the above code by using
sockaddr_in
in lieu of sockaddr_in6
.
Is it not possible to assign IPv6 address using ioctl?
See Question&Answers more detail:
os 与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…