【问题标题】:Assign ipv6 address using ioctl使用 ioctl 分配 ipv6 地址
【发布时间】:2014-02-27 04:35:20
【问题描述】:

我正在尝试使用 ioctl 为接口分配 IPv6 地址,但没有成功。这是我使用的代码:

#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\n");
    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\n");
    return -1;
  }

  p = (char *) &sai;
  memcpy( (((char *)&ifr + ifreq_offsetof(ifr_addr) )),
          p, sizeof(struct sockaddr));

  int ret = ioctl(sockfd, SIOCSIFADDR, &ifr);
  printf("ret: %d\terrno: %d\n", ret, errno);
  ioctl(sockfd, SIOCGIFFLAGS, &ifr);
  printf("ret: %d\terrno: %d\n", ret, errno);

  ifr.ifr_flags |= IFF_UP | IFF_RUNNING;
  // ifr.ifr_flags &= ~selector;  // unset something

  ioctl(sockfd, SIOCSIFFLAGS, &ifr);
  printf("ret: %d\terrno: %d\n", ret, errno);
  close(sockfd);
  return 0;
}

ioctl 调用失败,提示 ENODEV。当套接字的族更改为sockfd = socket(AF_INET, SOCK_DGRAM, 0); 时,调用再次失败并显示 EINVAL。

我能够通过使用上述代码为接口分配 IPv4 地址 sockaddr_in 代替 sockaddr_in6

难道不能用ioctl分配IPv6地址吗?

【问题讨论】:

  • 您可能应该先将您的struct ifreq ifr 清零,然后再对它进行任何操作。

标签: networking ipv6


【解决方案1】:

从 'ifconfig' 命令的 linux 实现中汲取灵感,我能够在接口上设置 IPv6 地址。这是它的代码:

#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>
#if __GLIBC__ >=2 && __GLIBC_MINOR >= 1
#include <netpacket/packet.h>
#include <net/ethernet.h>
#else
#include <asm/types.h>
#include <linux/if_ether.h>
#endif

#define IFNAME "eth0"
#define HOST "fec2::22"
#define ifreq_offsetof(x)  offsetof(struct ifreq, x)

struct in6_ifreq {
    struct in6_addr ifr6_addr;
    __u32 ifr6_prefixlen;
    unsigned int ifr6_ifindex;
};

int main(int argc, char **argv) {

    struct ifreq ifr;
    struct sockaddr_in6 sai;
    int sockfd;                     
    struct in6_ifreq ifr6;

    sockfd = socket(AF_INET6, SOCK_DGRAM, IPPROTO_IP);
    if (sockfd == -1) {
          printf("Bad fd\n");
          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_pton(AF_INET6, HOST, (void *)&sai.sin6_addr) <= 0) {
        printf("Bad address\n");
        return -1;
    }

    memcpy((char *) &ifr6.ifr6_addr, (char *) &sai.sin6_addr,
               sizeof(struct in6_addr));

    if (ioctl(sockfd, SIOGIFINDEX, &ifr) < 0) {
        perror("SIOGIFINDEX");
    }
    ifr6.ifr6_ifindex = ifr.ifr_ifindex;
    ifr6.ifr6_prefixlen = 64;
    if (ioctl(sockfd, SIOCSIFADDR, &ifr6) < 0) {
        perror("SIOCSIFADDR");
    }

    ifr.ifr_flags |= IFF_UP | IFF_RUNNING;

    int ret = ioctl(sockfd, SIOCSIFFLAGS, &ifr);
    printf("ret: %d\terrno: %d\n", ret, errno);

    close(sockfd);
    return 0;
}

【讨论】:

  • 这段代码向接口添加了一个新的IPv6地址而不是替换它,我如何删除以前的??
【解决方案2】:

根据@maddy 的回答,我制作了一个更紧凑的版本,更容易适应。诀窍是必须将struct in6_ifreq 结构传递给ioctl

#include <stdint.h>
#include <string.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <netinet/in.h>

#define IFNAME "eth0"
#define HOST   "2001::22"

struct in6_ifreq {
    struct in6_addr addr;
    uint32_t        prefixlen;
    unsigned int    ifindex;
};

int main(int argc, char **argv) {

    struct ifreq ifr;
    struct in6_ifreq ifr6;
    int sockfd;
    int err;

    // Create IPv6 socket to perform the ioctl operations on
    sockfd = socket(AF_INET6, SOCK_DGRAM, IPPROTO_IP);

    // Copy the interface name to the ifreq struct
    strncpy(ifr.ifr_name, IFNAME, IFNAMSIZ);
    // Get the ifrindex of the interface
    err = ioctl(sockfd, SIOGIFINDEX, &ifr);

    // Prepare the in6_ifreq struct and set the address to the interface
    inet_pton(AF_INET6, HOST, &ifr6.addr);
    ifr6.ifindex = ifr.ifr_ifindex;
    ifr6.prefixlen = 64;
    err = ioctl(sockfd, SIOCSIFADDR, &ifr6);

    close(sockfd);
    return 0;
}

为了可读性,我放弃了所有错误检查,但应该检查命令是否有错误。

【讨论】:

    猜你喜欢
    • 2014-01-11
    • 1970-01-01
    • 2017-01-31
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-06-22
    • 2011-07-15
    • 1970-01-01
    相关资源
    最近更新 更多