gethostbyname() return 可以是 IPv6 地址吗?

Can gethostbyname() return an IPv6 address?

我正在阅读 UNIX 网络编程,想出一个关于练习 11.4 的问题来支持 gethostbyname for 的代码IPv6。书中答案如下:

#include    "unp.h"

int
main(int argc, char **argv)
{
    int                 sockfd, n;
    char                recvline[MAXLINE + 1];
    struct sockaddr_in  servaddr;
    struct sockaddr_in6 servaddr6;
    struct sockaddr     *sa;
    socklen_t           salen;
    struct in_addr      **pptr;
    struct hostent      *hp;
    struct servent      *sp;

    if (argc != 3)
        err_quit("usage: daytimetcpcli3 <hostname> <service>");

    if ( (hp = gethostbyname(argv[1])) == NULL)
        err_quit("hostname error for %s: %s", argv[1], hstrerror(h_errno));

    if ( (sp = getservbyname(argv[2], "tcp")) == NULL)
        err_quit("getservbyname error for %s", argv[2]);

    pptr = (struct in_addr **) hp->h_addr_list;
    for ( ; *pptr != NULL; pptr++) {
        sockfd = Socket(hp->h_addrtype, SOCK_STREAM, 0);

        if (hp->h_addrtype == AF_INET) {
            sa = (SA *) &servaddr;
            salen = sizeof(servaddr);
        } else if (hp->h_addrtype == AF_INET6) {
            sa = (SA *) &servaddr6;
            salen = sizeof(servaddr6);
        } else
            err_quit("unknown addrtype %d", hp->h_addrtype);

        bzero(sa, salen);
        sa->sa_family = hp->h_addrtype;
        sock_set_port(sa, salen, sp->s_port);
        sock_set_addr(sa, salen, *pptr);

        printf("trying %s\n", Sock_ntop(sa, salen));

        if (connect(sockfd, sa, salen) == 0)
            break;      /* success */
        err_ret("connect error");
        close(sockfd);
    }
    if (*pptr == NULL)
        err_quit("unable to connect");

    while ( (n = Read(sockfd, recvline, MAXLINE)) > 0) {
        recvline[n] = 0;    /* null terminate */
        Fputs(recvline, stdout);
    }
    exit(0);
}

根据书籍和其他资源的信息,我发现 gethostbyname 不能 return IPv6。所以我的问题是 gethostbyname 是否可以 return 有关 IPV6 的信息,或者只是代码错误?

我觉得这个问题不是要替换gethostbyname,我想知道这段代码的正确性。

gethostbyname() 不再是(当前)POSIX(IEEE Std 1003.1-2008,2016 版)的一部分。

It explicitly mentions:

The obsolescent h_errno external integer, and the obsolescent gethostbyaddr() and gethostbyname() functions are removed, along with the HOST_NOT_FOUND, NO_DATA, NO_RECOVERY, and TRY_AGAIN macros.


最后一个 POSIX 支持 gethostbyname() 的版本是 IEEE Std 1003.1, 2004 Edition, which states:

struct hostent *gethostbyname(const char *name);

[...]

The gethostbyname() function shall return an entry containing addresses of address family AF_INET for the host with name name.

根本没有提到AF_INET6


所以从 POSIX 的角度来看你的问题

the correctness of this code.

可以回答为:

  • 是的,过去是正确的,依赖于实现defined/specific扩展(即支持AF_INET6).
  • 不,从 开始,现在 正确,因为 gethostbyname() 已从 POSIX.