如何获取 IPv6 主机的范围?

Posted

技术标签:

【中文标题】如何获取 IPv6 主机的范围?【英文标题】:How to get scope of an IPv6 host? 【发布时间】:2012-07-22 20:30:56 【问题描述】:

我对 IPv6 协议了解不多,如果这个问题听起来很愚蠢,我很抱歉。 当我检索网络中所有 IPv6 地址的列表时,我得到一个名为 scope 的字段,如下所示:

      inet6 addr: 2001:470:1:82::11/64 Scope:Global
      inet6 addr: 2001:470:1:82::10/64 Scope:Global
      inet6 addr: 2001:470:1:82::13/64 Scope:Global
      inet6 addr: fe80::21d:9ff:fe69:2c50/64 Scope:Link
      inet6 addr: 2001:470:1:82::12/64 Scope:Global
      inet6 addr: 2001:470:1:82::15/64 Scope:Global
      inet6 addr: 2001:470:1:82::14/64 Scope:Global
      inet6 addr: 2001:470:1:82::5/64 Scope:Global
      inet6 addr: 2001:470:1:82::17/64 Scope:Global
      inet6 addr: 2001:470:1:82::6/64 Scope:Global
      inet6 addr: 2001:470:1:82::16/64 Scope:Global
      inet6 addr: 2001:470:1:82::7/64 Scope:Global
      inet6 addr: 2001:470:1:82::19/64 Scope:Global
      inet6 addr: 2001:470:1:82::8/64 Scope:Global
      inet6 addr: 2001:470:1:82::18/64 Scope:Global
      inet6 addr: 2001:470:1:82::9/64 Scope:Global
      inet6 addr: 2001:470:1:82::1b/64 Scope:Global
      inet6 addr: 2001:470:1:82::a/64 Scope:Global
      inet6 addr: 2001:470:1:82::1a/64 Scope:Global
      inet6 addr: 2001:470:1:82::b/64 Scope:Global
      inet6 addr: 2001:470:1:82::1d/64 Scope:Global
      inet6 addr: 2001:470:1:82::c/64 Scope:Global
      inet6 addr: 2001:470:1:82::1c/64 Scope:Global
      inet6 addr: 2001:470:1:82::d/64 Scope:Global
      inet6 addr: 2001:470:1:82::1f/64 Scope:Global
      inet6 addr: 2001:470:1:82::e/64 Scope:Global
      inet6 addr: 2001:470:1:82::1e/64 Scope:Global
      inet6 addr: 2001:470:1:82::f/64 Scope:Global
      inet6 addr: ::1/128 Scope:Host

在我的应用程序中,我需要获取范围为“链接”的地址。我本可以对 ifconfig 使用系统调用,然后解析输出以提取相应的地址。但问题是,我正在使用对 getifaddrs() 的调用,它返回结构 ifaddr 的链表,给出为:

       struct ifaddrs 
           struct ifaddrs  *ifa_next;    /* Next item in list */
           char            *ifa_name;    /* Name of interface */
           unsigned int     ifa_flags;   /* Flags from SIOCGIFFLAGS */
           struct sockaddr *ifa_addr;    /* Address of interface */
           struct sockaddr *ifa_netmask; /* Netmask of interface */
           union 
               struct sockaddr *ifu_broadaddr;
                                /* Broadcast address of interface */
               struct sockaddr *ifu_dstaddr;
                                /* Point-to-point destination address */
            ifa_ifu;
       #define              ifa_broadaddr ifa_ifu.ifu_broadaddr
       #define              ifa_dstaddr   ifa_ifu.ifu_dstaddr
           void            *ifa_data;    /* Address-specific data */
       ;

问题是:如何从此列表中获取具有“链接”范围的地址?

【问题讨论】:

为什么票数接近?解释会很好。 @EBGreen 因为问题是询问如何编写例程以提取接口细节。这在 *** 上是最好的,因此最接近的投票是将问题迁移到可以更好地回答它的网站。 @Manish,您无需对此做任何事情,如果合适,迁移将自动发生,您可以继续在那里处理您的问题。 为我工作。我只是认为,如果用户在接近投票的同时获得一些评论,它会帮助用户,尤其是新用户。 @Manish,接受正确答案(单击勾号)或评论为什么没有帮助是有礼貌的。如果您不这样做,您可能会发现将来没有人会回答您的问题! 【参考方案1】:

我认为范围没有明确存储在该数据结构中。但是,您可以从 IP 地址本身推断范围。见http://en.wikipedia.org/wiki/IPv6_address#Address_scopes

【讨论】:

【参考方案2】:

有辅助宏来辅助:

struct sockaddr_in6 s6;
if (IN6_IS_ADDR_LINKLOCAL(&s6.sin6_addr)) 
  puts ("link-local");
 else if (IN6_IS_ADDR_SITELOCAL(&s6.sin6_addr)) 
  puts ("site-local");
 else if (IN6_IS_ADDR_V4MAPPED(&s6.sin6_addr)) 
  puts ("v4mapped");
 else if (IN6_IS_ADDR_V4COMPAT(&s6.sin6_addr)) 
  puts ("v4compat");
 else if (IN6_IS_ADDR_LOOPBACK(&s6.sin6_addr)) 
  puts ("host");
 else if (IN6_IS_ADDR_UNSPECIFIED(&s6.sin6_addr)) 
  puts ("unspecified");
 else 
  puts ("global");

【讨论】:

【参考方案3】:

一种方法是检查地址是否在fe80::/10 范围内。 IPv6 address space 可从 IANA 获得,其中详细说明了可能的可用范围。

我将源代码下载到net-tools(ifconfig 的源包),看起来他们解析了/proc/net/if_inet6。 (注释是我自己在下面的代码中添加的——下面的代码也非常删节,肯定不会编译。)

/* some defines collected around the place: */

#define _PATH_PROCNET_IFINET6   "/proc/net/if_inet6"
#define IPV6_ADDR_LOOPBACK      0x0010U
#define IPV6_ADDR_LINKLOCAL     0x0020U
#define IPV6_ADDR_SITELOCAL     0x0040U
#define IPV6_ADDR_COMPATv4      0x0080U

int scope; /* among other stuff */

/* looks like here we parse the contents of /proc/net/if_inet6: */

if ((f = fopen(_PATH_PROCNET_IFINET6, "r")) != NULL) 
while (fscanf(f, "%4s%4s%4s%4s%4s%4s%4s%4s %02x %02x %02x %02x %20s\n",
          addr6p[0], addr6p[1], addr6p[2], addr6p[3],
          addr6p[4], addr6p[5], addr6p[6], addr6p[7],
      &if_idx, &plen, &scope, &dad_status, devname) != EOF) 

/* slightly later: */

printf(_(" Scope:"));
switch (scope) 
case 0:
    printf(_("Global"));
    break;
case IPV6_ADDR_LINKLOCAL:
    printf(_("Link"));
    break;
case IPV6_ADDR_SITELOCAL:
    printf(_("Site"));
    break;
case IPV6_ADDR_COMPATv4:
    printf(_("Compat"));
    break;
case IPV6_ADDR_LOOPBACK:
    printf(_("Host"));
    break;
default:
    printf(_("Unknown"));

那么我们来看看上面解析的是什么:

$ cat /proc/net/if_inet6
20010db8000008000000000000000001 03 40 00 00     eth0
fe800000000000000000000000004321 03 40 20 80     eth0
00000000000000000000000000000001 01 80 10 80       lo

所以你可以看到左起第三列(0x00 Global、0x20 Link-Local 和0x10 Loopback)是范围。使用 net-tools 代码中的上述常量,您可以了解它们的含义。需要进一步调查以确定此类常量的更权威来源,以及解析 /proc/net/if_inet6 是否是您的最佳选择。

【讨论】:

非常感谢杰里米,这真的帮助我理解了很多。正如你所说,我只是检查了地址是否在指定的范围内,即fe80::/10,这对我有用。 不用担心。有时最简单的解决方案是最好的! 谁投了反对票,请解释一下。我愿意编辑我的答案以改进它(或相应地自己编辑)。

以上是关于如何获取 IPv6 主机的范围?的主要内容,如果未能解决你的问题,请参考以下文章

按主机名的 IPv6 地址 [关闭]

电信光猫如何以路由模式获取IPV6?

如何使用ip地址和IPv4Mask获取IP地址范围是多少

C# 获取本机IP地址,IPv4,IPv6(保姆级)

如何以编程方式在 iOS 中通过 IPv6 获取远程 MAC 地址

为啥获取了ipv6地址却无法访问纯ipv6网站