C套接字编程:使用getaddrinfo()绑定并获取设备IP服务器

时间:2017-05-21 20:16:07

标签: c sockets networking

我使用(因为我读'现代'方法)getaddrinfo()创建被动服务器套接字。使用这种方法并在提示中指定:AI_PASSIVE,AF_UNSPEC,sock_type我循环结果如下:

 for(ai_ptr = addrinfo_res; ai_ptr != NULL; ai_ptr = ai_ptr->ai_next)

然后创建套接字并绑定到第一个addrinfo_res中找到的地址。

 bind(ps_fd, ai_ptr->ai_addr, ai_ptr->ai_addrlen)

我怎么知道我的服务器在局域网中可用的地址(不是0.0.0.0)?

我已经尝试过以下代码来查找它们上的所有接口和IP地址:

 // linked list of structures describing
    // the network interfaces of the local system
    struct ifaddrs *ifaddr, *ifa;
    int n, gai_err;

    if(getifaddrs(&ifaddr) == FAILURE) {
        fprintf(stderr, "getifaddrs: %s\n", strerror(errno));
        return FAILURE;
    }

    // walk through linked list
    for(ifa = ifaddr, n=0; ifa != NULL; ifa = ifa->ifa_next, n++) {

        if(ifa->ifa_addr == NULL)
            continue;

        int family = ifa->ifa_addr->sa_family;

        // display interface name, family
        printf("%-8s %s (%d)\n", ifa->ifa_name,
               (family == AF_INET) ? "AF_INET" :
               (family == AF_INET6) ? "AF_INET6" : "???",
               family);

        // for an AF_INET* interface addresses, display the address
        if(family == AF_INET || family == AF_INET6) {

            char host[NI_MAXHOST];

            if( (gai_err = getnameinfo(ifa->ifa_addr,
                        (family == AF_INET) ? sizeof(struct sockaddr_in) :
                                              sizeof(struct sockaddr_in6),
                        host, NI_MAXHOST,
                        NULL, 0, NI_NUMERICHOST)) != 0) {
                fprintf(stderr, "getnameinfo: %s\n", gai_strerror(gai_err));
            }

            printf("\t\t address: <%s>\n", host);
        }
    }

    freeifaddrs(ifaddr);
    return SUCCESS;

但我怎么能确定我会返回(打印)客户端可以连接的合适IP地址?

我有这样的结果:

lo0      ??? (18)
lo0      AF_INET (2)
         address: <127.0.0.1>
lo0      AF_INET6 (30)
         address: <::1>
lo0      AF_INET6 (30)
         address: <fe80::1%lo0>
gif0     ??? (18)
stf0     ??? (18)
en0      ??? (18)
en0      AF_INET6 (30)
         address: <fe80::1088:e458:89a7:7ee9%en0>
en0      AF_INET (2)
         address: <192.168.8.102>
en1      ??? (18)
bridge0  ??? (18)
p2p0     ??? (18)
awdl0    ??? (18)
awdl0    AF_INET6 (30)
         address: <fe80::dce1:4eff:fef8:f3bd%awdl0>
utun0    ??? (18)
utun0    AF_INET6 (30)
         address: <fe80::695f:8478:f380:9efb%utun0>

当我测试服务器时,仅在192.168.8.102或IPv6 :: ffff:192.168.102上可用。但是在第二个en0 AF_INET6 IP地址fe80 :: 1088:e458:89a7:7ee9。如何确保当其他人在具有不同IP地址和接口的另一台计算机上使用此程序时(例如不是en0但是eth0?)我将可以返回我的服务器可用的正确IP地址,以便他/她知道哪个IP它可以从客户端程序连接? enter image description here

1 个答案:

答案 0 :(得分:0)

您无法确定,因为任何系统都可以拥有多个具有不同网络和地址的接口。即使返回的路线也不必与通往系统的路线相同。

此外,由于客户端和服务器之间的NAT,客户端可能必须连接到不同的地址和端口。

您可以使用IP_PKTINFO套接字选项从recvmsg获取地址,以查看邮件发往哪个地址,但同样,这可能已在网络级别进行了翻译。