Sending the query

We start main() with the following code:

/*dns_query.c*/

int main(int argc, char *argv[]) {

if (argc < 3) {
printf("Usage:\n\tdns_query hostname type\n");
printf("Example:\n\tdns_query example.com aaaa\n");
exit(0);
}

if (strlen(argv[1]) > 255) {
fprintf(stderr, "Hostname too long.");
exit(1);
}

The preceding code checks that the user passed in a hostname and record type to query. If they didn't, it prints a helpful message. It also checks that the hostname isn't more than 255 characters long. Hostnames longer than that aren't allowed by the DNS standard, and checking it now ensures that we don't need to allocate too much memory.

We then try to interpret the record type requested by the user. We support the following options – a, aaaa, txt, mx, and any. The code to read in those types and store their corresponding DNS integer value is as follows:

/*dns_query.c*/

unsigned char type;
if (strcmp(argv[2], "a") == 0) {
type = 1;
} else if (strcmp(argv[2], "mx") == 0) {
type = 15;
} else if (strcmp(argv[2], "txt") == 0) {
type = 16;
} else if (strcmp(argv[2], "aaaa") == 0) {
type = 28;
} else if (strcmp(argv[2], "any") == 0) {
type = 255;
} else {
fprintf(stderr, "Unknown type '%s'. Use a, aaaa, txt, mx, or any.",
argv[2]);
exit(1);
}

Like all of our previous programs, we need to initialize Winsock. The code for that is as follows:

/*dns_query.c*/

#if defined(_WIN32)
WSADATA d;
if (WSAStartup(MAKEWORD(2, 2), &d)) {
fprintf(stderr, "Failed to initialize.\n");
return 1;
}
#endif

Our program connects to 8.8.8.8, which is a public DNS server run by Google. Refer to Chapter 1,  An Introduction to Networks and Protocols, Domain Names, for a list of additional public DNS servers you can use.

Recall that we are connecting on UDP port 53. We use getaddrinfo() to set up the required structures for our socket with the following code:

/*dns_query.c*/

printf("Configuring remote address...\n");
struct addrinfo hints;
memset(&hints, 0, sizeof(hints));
hints.ai_socktype = SOCK_DGRAM;
struct addrinfo *peer_address;
if (getaddrinfo("8.8.8.8", "53", &hints, &peer_address)) {
fprintf(stderr, "getaddrinfo() failed. (%d)\n", GETSOCKETERRNO());
return 1;
}

We then create our socket using the data returned from getaddrinfo(). The following code does that:

/*dns_query.c*/

printf("Creating socket...\n");
SOCKET socket_peer;
socket_peer = socket(peer_address->ai_family,
peer_address->ai_socktype, peer_address->ai_protocol);
if (!ISVALIDSOCKET(socket_peer)) {
fprintf(stderr, "socket() failed. (%d)\n", GETSOCKETERRNO());
return 1;
}

Our program then constructs the data for the DNS query message. The first 12 bytes compose the header and are known at compile time. We can store them with the following code:

/*dns_query.c*/

char query[1024] = {0xAB, 0xCD, /* ID */
0x01, 0x00, /* Set recursion */
0x00, 0x01, /* QDCOUNT */
0x00, 0x00, /* ANCOUNT */
0x00, 0x00, /* NSCOUNT */
0x00, 0x00 /* ARCOUNT */};

The preceding code sets our query's ID to 0xABCD, sets a recursion request, and indicates that we are attaching 1 question. As mentioned earlier, 1 is the only number of questions supported by real-world DNS servers.

We then need to encode the user's desired hostname into the query. The following code does that:

/*dns_query.c*/

char *p = query + 12;
char *h = argv[1];

while(*h) {
char *len = p;
p++;
if (h != argv[1]) ++h;

while(*h && *h != '.') *p++ = *h++;
*len = p - len - 1;
}

*p++ = 0;

The preceding code first sets a new pointer, p, to the end of the query header. We will be adding to the query starting at p. We also define a pointer, h, which we use to loop through the hostname.

We can loop while *h != 0 because *h is equal to zero when we've finished reading the hostname. Inside the loop, we use the len variable to store the position of the label beginning. The value in this position needs to be set to indicate the length of the upcoming label. We then copy characters from *h to *p until we find a dot or the end of the hostname. If either is found, the code sets *len equal to the label length. The code then loops into the next label.

Finally, outside the loop, we add a terminating 0 byte to finish the name section of the question.

We then add the question type and question class to the query with the following code:

/*dns_query.c*/

*p++ = 0x00; *p++ = type; /* QTYPE */
*p++ = 0x00; *p++ = 0x01; /* QCLASS */

We can then calculate the query size by comparing p to the query beginning. The code for figuring the total query size is as follows:

/*dns_query.c*/

const int query_size = p - query;

Now, with the query message formed, and its length known, we can use sendto() to transmit the DNS query to the DNS server. The code for sending the query is as follows:

/*dns_query.c*/

int bytes_sent = sendto(socket_peer,
query, query_size,
0,
peer_address->ai_addr, peer_address->ai_addrlen);
printf("Sent %d bytes.\n", bytes_sent);

For debugging purposes, we can also display the query we sent with the following code:

/*dns_query.c*/

print_dns_message(query, query_size);

The preceding code is useful to see whether we've made any mistakes in encoding our query.

Now that the query has been sent, we await a DNS response message using recvfrom(). In a practical program, you may want to use select() here to time out. It could also be wise to listen for additional messages in the case that an invalid message is received first.

The code to receive and display the DNS response is as follows:

/*dns_query.c*/

char read[1024];
int bytes_received = recvfrom(socket_peer,
read, 1024, 0, 0, 0);

printf("Received %d bytes.\n", bytes_received);

print_dns_message(read, bytes_received);
printf("\n");

We can finish our program by freeing the address(es) from getaddrinfo() and cleaning up Winsock. The code to complete the main() function is as follows:

/*dns_query.c*/

freeaddrinfo(peer_address);
CLOSESOCKET(socket_peer);

#if defined(_WIN32)
WSACleanup();
#endif

return 0;
}

That concludes the dns_query program.

You can compile and run dns_query.c on Linux and macOS by running the following command:

gcc dns_query.c -o dns_query
./dns_query example.com a

Compiling and running on Windows with MinGW is done by using the following command:

gcc dns_query.c -o dns_query.exe -lws2_32
dns_query.exe example.com a

Try running dns_query with different domain names and different record types. In particular, try it with mx and txt records. If you're brave, try running it with the any record type. You may find the results interesting.

The following screenshot is an example of using dns_query to query the A record of example.com:

The next screenshot shows dns_query querying the mx record of gmail.com:

Note that UDP is not always reliable. If our DNS query is lost in transit, then dns_query hangs while waiting forever for a reply that never comes. This could be fixed by using the select() function to time out and retry.