/* ******************************************************************************* * * debug.c -- * * Routines to print out packets received from a name server query. * * Modified version of 4.3BSD BIND res_debug.c 5.6 9/14/85 * * Copyright (c) 1985 Regents of the University of California. * All rights reserved. The Berkeley software License Agreement * specifies the terms and conditions for redistribution. * ******************************************************************************* */ #ifndef lint static char sccsid[] = "@(#)debug.c 5.3 (Berkeley) 3/31/86"; #endif not lint #include #include #include #include #include #include "res.h" extern char ctime(); /* * Imported from res_debug.c */ extern char *rcodes[]; extern char *opcodes[]; /* * Used to highlight the start of a record when printing it. */ #define INDENT "-> " /* * Print the contents of a query. * This is intended to be primarily a debugging routine. */ Print_query(msg, eom, printHeader) char *msg, *eom; int printHeader; { Fprint_query(msg, eom, printHeader,stdout); } Fprint_query(msg, eom, printHeader,file) char *msg, *eom; int printHeader; FILE *file; { register char *cp; register HEADER *hp; register int n; short class; short type; /* * Print header fields. */ hp = (HEADER *)msg; cp = msg + sizeof(HEADER); if (printHeader || (_res.options & RES_DEBUG2)) { fprintf(file,"HEADER:\n"); fprintf(file,"\topcode = %s", opcodes[hp->opcode]); fprintf(file,", id = %d", ntohs(hp->id)); fprintf(file,", rcode = %s\n", rcodes[hp->rcode]); fprintf(file,"\theader flags: "); if (hp->qr) { fprintf(file," response"); } else { fprintf(file," query"); } if (hp->aa) fprintf(file,", auth. answer"); if (hp->tc) fprintf(file,", truncation"); if (hp->rd) fprintf(file,", want recursion"); if (hp->ra) fprintf(file,", recursion avail."); if (hp->pr) fprintf(file,", primary"); fprintf(file,"\n\tquestions = %d", ntohs(hp->qdcount)); fprintf(file,", answers = %d", ntohs(hp->ancount)); fprintf(file,", n.s. = %d", ntohs(hp->nscount)); fprintf(file,", additional = %d\n\n", ntohs(hp->arcount)); } /* * Print question records. */ if (n = ntohs(hp->qdcount)) { fprintf(file,"QUESTIONS:\n"); while (--n >= 0) { fprintf(file,"\t"); cp = Print_cdname(cp, msg, eom, file); if (cp == NULL) return; type = getshort(cp); cp += sizeof(u_short); class = getshort(cp); cp += sizeof(u_short); fprintf(file,", type = %s", p_type(type)); fprintf(file,", class = %s\n", p_class(class)); } } /* * Print authoritative answer records */ if (n = ntohs(hp->ancount)) { fprintf(file,"ANSWERS:\n"); while (--n >= 0) { fprintf(file, INDENT); cp = Print_rr(cp, msg, eom, file); if (cp == NULL) return; } } /* * print name server records */ if (n = ntohs(hp->nscount)) { fprintf(file,"NAME SERVERS:\n"); while (--n >= 0) { fprintf(file, INDENT); cp = Print_rr(cp, msg, eom, file); if (cp == NULL) return; } } /* * print additional records */ if (n = ntohs(hp->arcount)) { fprintf(file,"ADDITIONAL RECORDS:\n"); while (--n >= 0) { fprintf(file, INDENT); cp = Print_rr(cp, msg, eom, file); if (cp == NULL) return; } } fprintf(file,"\n"); } char * Print_cdname_sub(cp, msg, eom, file, format) char *cp, *msg, *eom; FILE *file; int format; { int n; char name[MAXDNAME]; extern char *strcpy(); if ((n = dn_expand(msg, eom, cp, name, sizeof(name))) < 0) return (NULL); if (name[0] == '\0') { (void) strcpy(name, "(root)"); } if (format) { fprintf(file, "%-30s", name); } else { fputs(name, file); } return (cp + n); } char * Print_cdname(cp, msg, eom, file) char *cp, *msg, *eom; FILE *file; { return(Print_cdname_sub(cp, msg, eom, file, 0)); } char * Print_cdname2(cp, msg, eom, file) char *cp, *msg, *eom; FILE *file; { return(Print_cdname_sub(cp, msg, eom, file, 1)); } /* * Print resource record fields in human readable form. */ char * Print_rr(cp, msg, eom, file) char *cp, *msg, *eom; FILE *file; { int type, class, dlen, n, c; long ttl; struct in_addr inaddr; char *cp1; if ((cp = Print_cdname(cp, msg, eom, file)) == NULL) return (NULL); /* compression error */ type = getshort(cp); cp += sizeof(u_short); class = getshort(cp); cp += sizeof(u_short); ttl = getlong(cp); cp += sizeof(u_long); dlen = getshort(cp); cp += sizeof(u_short); if (_res.options & RES_DEBUG2) { fprintf(file,"\n\ttype = %s, class = %s, ttl = %u, dlen = %d", p_type(type), p_class(class), ttl, dlen); fprintf(file,"\n"); } cp1 = cp; /* * Print type specific data, if appropriate */ switch (type) { case T_A: switch (class) { case C_IN: bcopy(cp, (char *)&inaddr, sizeof(inaddr)); if (dlen == 4) { fprintf(file,"\tinet address = %s\n", inet_ntoa(inaddr)); cp += dlen; } else if (dlen == 7) { fprintf(file,"\tinet address = %s", inet_ntoa(inaddr)); fprintf(file,", protocol = %d", cp[4]); fprintf(file,", port = %d\n", (cp[5] << 8) + cp[6]); cp += dlen; } break; } break; case T_CNAME: fprintf(file,"\tcanonical name = "); cp = Print_cdname(cp, msg, eom, file); fprintf(file,"\n"); break; case T_MX: fprintf(file,"\tpreference = %d",getshort(cp)); cp += sizeof(u_short); fprintf(file,", mail exchanger = "); cp = Print_cdname(cp,msg, eom, file); fprintf(file,"\n"); break; case T_MG: case T_MB: case T_MR: case T_NS: case T_PTR: fprintf(file,"\tserver name = "); cp = Print_cdname(cp, msg, eom, file); fprintf(file,"\n"); break; case T_HINFO: if (n = *cp++) { fprintf(file,"\tCPU=%.*s", n, cp); cp += n; } if (n = *cp++) { fprintf(file,"\tOS=%.*s\n", n, cp); cp += n; } break; case T_SOA: fprintf(file,"\torigin = "); cp = Print_cdname(cp, msg, eom, file); fprintf(file,"\n\tmail addr = "); cp = Print_cdname(cp, msg, eom, file); fprintf(file,"\n\tserial=%ld", getlong(cp)); cp += sizeof(u_long); fprintf(file,", refresh=%ld", getlong(cp)); cp += sizeof(u_long); fprintf(file,", retry=%ld", getlong(cp)); cp += sizeof(u_long); fprintf(file,", expire=%ld", getlong(cp)); cp += sizeof(u_long); fprintf(file,", min=%ld\n", getlong(cp)); cp += sizeof(u_long); break; case T_MINFO: fprintf(file,"\trequests = "); cp = Print_cdname(cp, msg, eom, file); fprintf(file,"\n\terrors = "); cp = Print_cdname(cp, msg, eom, file); break; case T_UINFO: fprintf(file,"\t%s\n", cp); cp += dlen; break; case T_UID: case T_GID: if (dlen == 4) { fprintf(file,"\t%ld\n", getlong(cp)); cp += sizeof(int); } break; case T_WKS: if (dlen < sizeof(u_long) + 1) break; bcopy(cp, (char *)&inaddr, sizeof(inaddr)); cp += sizeof(u_long); fprintf(file,"\tinet address = %s, protocol = %d\n\t", inet_ntoa(inaddr), *cp++); n = 0; while (cp < cp1 + dlen) { c = *cp++; do { if (c & 0200) fprintf(file," %d", n); c <<= 1; } while (++n & 07); } putc('\n',file); break; case T_NULL: break; default: fprintf(file,"\t???\n"); cp += dlen; } if (cp != cp1 + dlen) fprintf(file,"packet size error (%#x != %#x)\n", cp, cp1+dlen); return (cp); }