DNS encodes names in a particular way. Normally, each label is indicated by its length, followed by its text. A number of labels can be repeated, and then the name is terminated with a single 0 byte.
If a length has its two highest bits set (that is, 0xc0), then it and the next byte should be interpreted as a pointer instead.
We must also be aware at all times that the DNS response from the DNS server could be ill-formed or corrupted. We must try to write our program in such a way that it won't crash if it receives a bad message. This is easier said than done.
The declaration for our name-printing function looks like this:
/*dns_query.c*/
const unsigned char *print_name(const unsigned char *msg,
const unsigned char *p, const unsigned char *end);
We take msg to be a pointer to the message's beginning, p to be a pointer to the name to print, and end to be a pointer to one past the end of the message. end is required so that we can check that we're not reading past the end of the received message. msg is required for the same reason, but also so that we can interpret name pointers.
Inside the print_name function, our code checks that a proper name is even possible. Because a name should consist of at least a length and some text, we can return an error if p is already within two characters of the end. The code for that check is as follows:
/*dns_query.c*/
if (p + 2 > end) {
fprintf(stderr, "End of message.\n"); exit(1);}
We then check to see if p points to a name pointer. If it does, we interpret the pointer and call print_name recursively to print the name that is pointed to. The code for this is as follows:
/*dns_query.c*/
if ((*p & 0xC0) == 0xC0) {
const int k = ((*p & 0x3F) << 8) + p[1];
p += 2;
printf(" (pointer %d) ", k);
print_name(msg, msg+k, end);
return p;
Note that 0xC0 in binary is 0b11000000. We use (*p & 0xC0) == 0xC0 to check for a name pointer. In that case, we take the lower 6 bits of *p and all 8 bits of p[1] to indicate the pointer. We know that p[1] is still within the message because of our earlier check that p was at least 2 bytes from the end. Knowing the name pointer, we can pass a new value of p to print_name().
If the name is not a pointer, we simply print it one label at a time. The code for printing the name is as follows:
/*dns_query.c*/
} else {
const int len = *p++;
if (p + len + 1 > end) {
fprintf(stderr, "End of message.\n"); exit(1);}
printf("%.*s", len, p);
p += len;
if (*p) {
printf(".");
return print_name(msg, p, end);
} else {
return p+1;
}
}
In the preceding code, *p is read into len to store the length of the current label. We are careful to check that reading len + 1 bytes doesn't put us past the end of the buffer. We can then print the next len characters to the console. If the next byte isn't 0, then the name is continued, and we should print a dot to separate the labels. We call print_name() recursively to print the next label of the name.
If the next byte is 0, then that means the name is finished and we return.
That concludes the print_name() function.
Now we will devise a function that prints an entire DNS message.