getaddrinfo在IPv6之前为localhost排序IPv4?

时间:2018-03-28 14:49:23

标签: networking ipv6 getaddrinfo

我为getaddrinfo写了一个非常简单的测试程序:

#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <stdio.h>
#include <string.h>

int main() {
    struct addrinfo hints;
    struct addrinfo *res, *rp;
    char hoststr[64], servstr[8];

    memset(&hints, 0, sizeof(hints));
    hints.ai_family = AF_UNSPEC;
    hints.ai_socktype = SOCK_STREAM;
    hints.ai_flags = AI_PASSIVE;

    getaddrinfo(NULL, "9998", &hints, &res);
    for (rp = res; rp != NULL; rp = rp->ai_next) {
        getnameinfo(rp->ai_addr, rp->ai_addrlen, hoststr, sizeof(hoststr),
                servstr, sizeof(servstr), NI_NUMERICHOST | NI_NUMERICSERV);
        printf("%s:%s\n", hoststr, servstr);
    }
}

当我编译并运行此程序时,它会在IPv6地址之前提供IPv4地址:

# gcc -o getaddrinfo getaddrinfo.c
# ./getaddrinfo 
0.0.0.0:9998
:::9998

据我所知,其他来源的IPv6地址应优先于IPv4。我使用默认的/etc/gai.conf,这意味着IPv6应该优先于IPv4。那么为什么getaddrinfo会这样排序呢?

# ip addr show lo
1: lo: <LOOPBACK,UP,LOWER_UP> mtu 65536 qdisc noqueue state UNKNOWN group default 
    link/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00
    inet 127.0.0.1/8 scope host lo
       valid_lft forever preferred_lft forever
    inet6 ::1/128 scope host 
       valid_lft forever preferred_lft forever

1 个答案:

答案 0 :(得分:1)

从手册页:

  

如果在hints.ai_flags中指定了AI_PASSIVE标志,并且节点是          NULL,则返回的套接字地址将适合          绑定(2)将接受(2)连接的套接字。归来了          套接字地址将包含&#34;通配符地址&#34; (IPv4的INADDR_ANY          地址,IPv6地址IN6ADDR_ANY_INIT)。通配符地址是          由打算接受任何主机网络地址上的连接的应用程序(通常是服务器)使用。如果node不是NULL,          然后忽略AI_PASSIVE标志。

您描述的行为是指您使用getaddrinfo()查找名称以准备传出连接时的行为。因为您正在反向查找本地地址以绑定服务,所以提供地址的顺序无关紧要。