gpt4 book ai didi

c - 根据输入字符串查找匹配的 IPv6 地址

转载 作者:太空宇宙 更新时间:2023-11-04 04:07:43 24 4
gpt4 key购买 nike

我需要一些帮助来根据提供的信息查找网络接口(interface)IP地址。我需要一种跨平台的方式来可靠地找到本地具有 IPv4 和 IPv6 格式的给定地址的接口(interface)。我创建附加程序以将 IP 地址作为字符串并搜索 getifaddrs 的结果。

原因是我最终想通过这个in_addr或者in6_addr 结构到套接字上的 IP_MULTICAST_IF ioctl指定应使用哪个接口(interface)发送多播消息。我想在继续之前检查它是否是已知 IP。

虽然它似乎在我的 Linux 机器和 MacBook 上运行良好,但它在我实验室的服务器上失败。 (哦,天哪,这最终必须继续Windows 也是..)这是在 Linux 上运行的示例:

$ ifconfig wlan0 | grep inet6
inet6 addr: fe80::1e4b:d6ff:fe6e:f846/64 Scope:Link

$ ifconfig wlan0 | grep inet
inet addr:192.168.0.13 Bcast:192.168.0.255 Mask:255.255.255.0
inet6 addr: fe80::1e4b:d6ff:fe6e:f846/64 Scope:Link

$ ./findif 192.168.0.13

found: wlan0

$ ./findif fe80::1e4b:d6ff:fe6e:f846

name: lo
ifa: 00000000000000000000000000000001
provided: fe800000000000001e4bd6fffe6ef846

name: wlan0
ifa: fe800000000000001e4bd6fffe6ef846
provided: fe800000000000001e4bd6fffe6ef846

found: wlan0

到目前为止一切顺利。我在运行 OS X 10.6 的 MacBook 上得到了类似的结果。然而,当我在我们的服务器上做类似的事情时,这是一个较旧的基于 PPC 的 OS X 机器运行 10.4.11,我得到以下信息:

$ ifconfig en0 | grep inet
inet6 fe80::20d:XXXX:XXXX:XXXX%en0 prefixlen 64 scopeid 0x4
inet 132.XXX.XX.XX netmask 0xffffff00 broadcast 132.XXX.XX.255

$ ./findif 132.XXX.XX.XX

found: en0

$ ./findif fe80::XXX:XXXX:XXXX:XXXX

name: lo0
ifa: 00000000000000000000000000000001
provided: fe800000000000000XXXXXXXXXXXXXXX

name: lo0
ifa: fe800001000000000000000000000001
provided: fe800000000000000XXXXXXXXXXXXXXX

name: en0
ifa: fe800004000000000XXXXXXXXXXXXXXX
provided: fe800000000000000XXXXXXXXXXXXXXX

name: en1
ifa: fe800005000000000YYYYYYYYYYYYYYY
provided: fe800000000000000YYYYYYYYYYYYYYY

对于 X'ing 出一些地址数字表示歉意,我认为这是最好的不要在这里公开实时服务器的真实 IP。请放心无论您看到 X 还是 Y,它都是匹配的十六进制数字。

无论如何,请注意为 en0 返回的 IPv6 恰好有 1 位与 ifconfig 报告的不同。谁能告诉我为什么是,我应该如何调整我的代码?是一个简单的 memcmp 不是比较 IPv6 地址的正确方法?

感谢您的任何建议。

这是findif.c的代码:

#include <stdio.h>
#include <stdlib.h>
#include <arpa/inet.h>
#include <ifaddrs.h>
#include <netdb.h>
#include <string.h>

void printipv6(const void* ipv6)
{
int i;
for (i=0; i<16; i++)
printf("%02x", ((unsigned char*)ipv6)[i]);
}

int find_iface(const char *ip)
{
union {
struct in_addr addr;
struct in6_addr addr6;
} a;

int rc = inet_pton(AF_INET6, ip, &a.addr6);
int fam = AF_INET6;
if (rc==0) {
rc = inet_pton(AF_INET, ip, &a.addr);
fam = AF_INET;
}
if (rc < 0) {
perror("inet_pton");
return 1;
}

struct ifaddrs *ifa, *ifa_list;
if (getifaddrs(&ifa_list)==-1) {
perror("getifaddrs");
return 1;
}
ifa = ifa_list;

int i=0;
while (ifa) {
if (ifa->ifa_addr) {
if (ifa->ifa_addr->sa_family == AF_INET && fam == AF_INET)
{
if (memcmp(&((struct sockaddr_in*)ifa->ifa_addr)->sin_addr,
&a.addr, sizeof(struct in_addr))==0) {
printf("\nfound: %s\n", ifa->ifa_name);
break;
}
}
else if (ifa->ifa_addr->sa_family == AF_INET6 && fam == AF_INET6)
{
struct in6_addr *p = &((struct sockaddr_in6*)ifa->ifa_addr)->sin6_addr;
printf("\nname: %s\n", ifa->ifa_name);
printf("ifa: "); printipv6(p); printf("\n");
printf("provided: "); printipv6(&a.addr6); printf("\n");
if (memcmp(p, &a.addr6, sizeof(struct in6_addr))==0) {
printf("\nfound: %s\n", ifa->ifa_name);
break;
}
}
}
ifa = ifa->ifa_next;
i++;
}
freeifaddrs(ifa_list);
return 0;
}

int main(int argc, char *argv[])
{
if (argc > 1)
return find_iface(argv[1]);
return 0;
}

最佳答案

您的代码看起来正确 - IPv6 链接本地地址应该在前 10 位之后有 54 个零位,因此 getifaddrs() 返回的地址看起来是错误的。看起来那个版本的 OS X 可能错误地将范围 ID 改成了地址。

关于c - 根据输入字符串查找匹配的 IPv6 地址,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3847930/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com