Using getnameinfo()

getnameinfo() can be used to convert an addrinfo structure back into a text format. It works with both IPv4 and IPv6. It also, optionally, converts a port number into a text format number or service name.

The declaration for getnameinfo() can be seen in the following code:

int getnameinfo(const struct sockaddr *addr, socklen_t addrlen,
char *host, socklen_t hostlen,
char *serv, socklen_t servlen, int flags);

The first two parameters are passed in from the ai_addr and ai_addrlen fields of struct addrinfo.

The next two parameters, host and hostlen, specify a character buffer and buffer length to store the hostname or IP address text.

The following two parameters, serv and servlen, specify the buffer and length to store the service name.

If you don't need both the hostname and the service, you can optionally pass in only one of either host or serv.

Flags can be a bitwise OR combination of the following:

For example, we can use getnameinfo() as follows:

char host[100];
char serv[100];
getnameinfo(address->ai_addr, address->ai_addrlen,
host, sizeof(host),
serv, sizeof(serv),
0);

printf("%s %s\n", host, serv);

In the preceding code, getnameinfo() attempts to perform a reverse DNS lookup. This works like the DNS queries we've done in this chapter so far, but backward. A DNS query asks which IP address does this hostname point to? A reverse DNS query asks instead, which hostname does this IP address point to? Keep in mind that this is not a one-to-one relationship. Many hostnames can point to one IP address, but an IP address can store a DNS record for only one hostname. In fact, reverse DNS records are not even set for many IP addresses.

If address is a struct addrinfo with the address for example.com port 80 (http), then the preceding code might print as follows:

example.com http

If the code prints something different for you, it's probably working as intended. It is dependent on which address is in address and how the reverse DNS is set up for that IP address. Try it with a different address.

If, instead of the hostname, we would like the IP address, we can modify our code to the following:

char host[100];
char serv[100];
getnameinfo(address->ai_addr, address->ai_addrlen,
host, sizeof(host),
serv, sizeof(serv),
NI_NUMERICHOST | NI_NUMERICSERV);

printf("%s %s\n", host, serv);

In the case of the previous code, it might print the following:

93.184.216.34 80

Using NI_NUMERICHOST generally runs much faster too, as it doesn't require getnameinfo() to send off any reverse DNS queries.