xref: /minix3/external/bsd/bind/dist/bin/dig/host.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: host.c,v 1.11 2015/07/08 17:28:54 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004-2007, 2009-2015  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 2000-2003  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /*! \file */
21*00b67f09SDavid van Moolenbroek 
22*00b67f09SDavid van Moolenbroek #include <config.h>
23*00b67f09SDavid van Moolenbroek #include <stdlib.h>
24*00b67f09SDavid van Moolenbroek #include <limits.h>
25*00b67f09SDavid van Moolenbroek 
26*00b67f09SDavid van Moolenbroek #ifdef HAVE_LOCALE_H
27*00b67f09SDavid van Moolenbroek #include <locale.h>
28*00b67f09SDavid van Moolenbroek #endif
29*00b67f09SDavid van Moolenbroek 
30*00b67f09SDavid van Moolenbroek #ifdef WITH_IDN
31*00b67f09SDavid van Moolenbroek #include <idn/result.h>
32*00b67f09SDavid van Moolenbroek #include <idn/log.h>
33*00b67f09SDavid van Moolenbroek #include <idn/resconf.h>
34*00b67f09SDavid van Moolenbroek #include <idn/api.h>
35*00b67f09SDavid van Moolenbroek #endif
36*00b67f09SDavid van Moolenbroek 
37*00b67f09SDavid van Moolenbroek #include <isc/app.h>
38*00b67f09SDavid van Moolenbroek #include <isc/commandline.h>
39*00b67f09SDavid van Moolenbroek #include <isc/netaddr.h>
40*00b67f09SDavid van Moolenbroek #include <isc/print.h>
41*00b67f09SDavid van Moolenbroek #include <isc/string.h>
42*00b67f09SDavid van Moolenbroek #include <isc/util.h>
43*00b67f09SDavid van Moolenbroek #include <isc/task.h>
44*00b67f09SDavid van Moolenbroek #include <isc/stdlib.h>
45*00b67f09SDavid van Moolenbroek #include <isc/timer.h>
46*00b67f09SDavid van Moolenbroek 
47*00b67f09SDavid van Moolenbroek #include <dns/byaddr.h>
48*00b67f09SDavid van Moolenbroek #include <dns/fixedname.h>
49*00b67f09SDavid van Moolenbroek #include <dns/message.h>
50*00b67f09SDavid van Moolenbroek #include <dns/name.h>
51*00b67f09SDavid van Moolenbroek #include <dns/rdata.h>
52*00b67f09SDavid van Moolenbroek #include <dns/rdataclass.h>
53*00b67f09SDavid van Moolenbroek #include <dns/rdataset.h>
54*00b67f09SDavid van Moolenbroek #include <dns/rdatatype.h>
55*00b67f09SDavid van Moolenbroek #include <dns/rdatastruct.h>
56*00b67f09SDavid van Moolenbroek 
57*00b67f09SDavid van Moolenbroek #include <dig/dig.h>
58*00b67f09SDavid van Moolenbroek 
59*00b67f09SDavid van Moolenbroek static isc_boolean_t short_form = ISC_TRUE, listed_server = ISC_FALSE;
60*00b67f09SDavid van Moolenbroek static isc_boolean_t default_lookups = ISC_TRUE;
61*00b67f09SDavid van Moolenbroek static int seen_error = -1;
62*00b67f09SDavid van Moolenbroek static isc_boolean_t list_addresses = ISC_TRUE;
63*00b67f09SDavid van Moolenbroek static dns_rdatatype_t list_type = dns_rdatatype_a;
64*00b67f09SDavid van Moolenbroek static isc_boolean_t printed_server = ISC_FALSE;
65*00b67f09SDavid van Moolenbroek 
66*00b67f09SDavid van Moolenbroek static const char *opcodetext[] = {
67*00b67f09SDavid van Moolenbroek 	"QUERY",
68*00b67f09SDavid van Moolenbroek 	"IQUERY",
69*00b67f09SDavid van Moolenbroek 	"STATUS",
70*00b67f09SDavid van Moolenbroek 	"RESERVED3",
71*00b67f09SDavid van Moolenbroek 	"NOTIFY",
72*00b67f09SDavid van Moolenbroek 	"UPDATE",
73*00b67f09SDavid van Moolenbroek 	"RESERVED6",
74*00b67f09SDavid van Moolenbroek 	"RESERVED7",
75*00b67f09SDavid van Moolenbroek 	"RESERVED8",
76*00b67f09SDavid van Moolenbroek 	"RESERVED9",
77*00b67f09SDavid van Moolenbroek 	"RESERVED10",
78*00b67f09SDavid van Moolenbroek 	"RESERVED11",
79*00b67f09SDavid van Moolenbroek 	"RESERVED12",
80*00b67f09SDavid van Moolenbroek 	"RESERVED13",
81*00b67f09SDavid van Moolenbroek 	"RESERVED14",
82*00b67f09SDavid van Moolenbroek 	"RESERVED15"
83*00b67f09SDavid van Moolenbroek };
84*00b67f09SDavid van Moolenbroek 
85*00b67f09SDavid van Moolenbroek static const char *rcodetext[] = {
86*00b67f09SDavid van Moolenbroek 	"NOERROR",
87*00b67f09SDavid van Moolenbroek 	"FORMERR",
88*00b67f09SDavid van Moolenbroek 	"SERVFAIL",
89*00b67f09SDavid van Moolenbroek 	"NXDOMAIN",
90*00b67f09SDavid van Moolenbroek 	"NOTIMP",
91*00b67f09SDavid van Moolenbroek 	"REFUSED",
92*00b67f09SDavid van Moolenbroek 	"YXDOMAIN",
93*00b67f09SDavid van Moolenbroek 	"YXRRSET",
94*00b67f09SDavid van Moolenbroek 	"NXRRSET",
95*00b67f09SDavid van Moolenbroek 	"NOTAUTH",
96*00b67f09SDavid van Moolenbroek 	"NOTZONE",
97*00b67f09SDavid van Moolenbroek 	"RESERVED11",
98*00b67f09SDavid van Moolenbroek 	"RESERVED12",
99*00b67f09SDavid van Moolenbroek 	"RESERVED13",
100*00b67f09SDavid van Moolenbroek 	"RESERVED14",
101*00b67f09SDavid van Moolenbroek 	"RESERVED15",
102*00b67f09SDavid van Moolenbroek 	"BADVERS"
103*00b67f09SDavid van Moolenbroek };
104*00b67f09SDavid van Moolenbroek 
105*00b67f09SDavid van Moolenbroek struct rtype {
106*00b67f09SDavid van Moolenbroek 	unsigned int type;
107*00b67f09SDavid van Moolenbroek 	const char *text;
108*00b67f09SDavid van Moolenbroek };
109*00b67f09SDavid van Moolenbroek 
110*00b67f09SDavid van Moolenbroek struct rtype rtypes[] = {
111*00b67f09SDavid van Moolenbroek 	{ 1, 	"has address" },
112*00b67f09SDavid van Moolenbroek 	{ 2, 	"name server" },
113*00b67f09SDavid van Moolenbroek 	{ 5, 	"is an alias for" },
114*00b67f09SDavid van Moolenbroek 	{ 11,	"has well known services" },
115*00b67f09SDavid van Moolenbroek 	{ 12,	"domain name pointer" },
116*00b67f09SDavid van Moolenbroek 	{ 13,	"host information" },
117*00b67f09SDavid van Moolenbroek 	{ 15,	"mail is handled by" },
118*00b67f09SDavid van Moolenbroek 	{ 16,	"descriptive text" },
119*00b67f09SDavid van Moolenbroek 	{ 19,	"x25 address" },
120*00b67f09SDavid van Moolenbroek 	{ 20,	"ISDN address" },
121*00b67f09SDavid van Moolenbroek 	{ 24,	"has signature" },
122*00b67f09SDavid van Moolenbroek 	{ 25,	"has key" },
123*00b67f09SDavid van Moolenbroek 	{ 28,	"has IPv6 address" },
124*00b67f09SDavid van Moolenbroek 	{ 29,	"location" },
125*00b67f09SDavid van Moolenbroek 	{ 0, NULL }
126*00b67f09SDavid van Moolenbroek };
127*00b67f09SDavid van Moolenbroek 
128*00b67f09SDavid van Moolenbroek static char *
rcode_totext(dns_rcode_t rcode)129*00b67f09SDavid van Moolenbroek rcode_totext(dns_rcode_t rcode)
130*00b67f09SDavid van Moolenbroek {
131*00b67f09SDavid van Moolenbroek 	static char buf[sizeof("?65535")];
132*00b67f09SDavid van Moolenbroek 	union {
133*00b67f09SDavid van Moolenbroek 		const char *consttext;
134*00b67f09SDavid van Moolenbroek 		char *deconsttext;
135*00b67f09SDavid van Moolenbroek 	} totext;
136*00b67f09SDavid van Moolenbroek 
137*00b67f09SDavid van Moolenbroek 	if (rcode >= (sizeof(rcodetext)/sizeof(rcodetext[0]))) {
138*00b67f09SDavid van Moolenbroek 		snprintf(buf, sizeof(buf), "?%u", rcode);
139*00b67f09SDavid van Moolenbroek 		totext.deconsttext = buf;
140*00b67f09SDavid van Moolenbroek 	} else
141*00b67f09SDavid van Moolenbroek 		totext.consttext = rcodetext[rcode];
142*00b67f09SDavid van Moolenbroek 	return totext.deconsttext;
143*00b67f09SDavid van Moolenbroek }
144*00b67f09SDavid van Moolenbroek 
145*00b67f09SDavid van Moolenbroek ISC_PLATFORM_NORETURN_PRE static void
146*00b67f09SDavid van Moolenbroek show_usage(void) ISC_PLATFORM_NORETURN_POST;
147*00b67f09SDavid van Moolenbroek 
148*00b67f09SDavid van Moolenbroek static void
show_usage(void)149*00b67f09SDavid van Moolenbroek show_usage(void) {
150*00b67f09SDavid van Moolenbroek 	fputs(
151*00b67f09SDavid van Moolenbroek "Usage: host [-aCdlriTwv] [-c class] [-N ndots] [-t type] [-W time]\n"
152*00b67f09SDavid van Moolenbroek "            [-R number] [-m flag] hostname [server]\n"
153*00b67f09SDavid van Moolenbroek "       -a is equivalent to -v -t ANY\n"
154*00b67f09SDavid van Moolenbroek "       -c specifies query class for non-IN data\n"
155*00b67f09SDavid van Moolenbroek "       -C compares SOA records on authoritative nameservers\n"
156*00b67f09SDavid van Moolenbroek "       -d is equivalent to -v\n"
157*00b67f09SDavid van Moolenbroek "       -l lists all hosts in a domain, using AXFR\n"
158*00b67f09SDavid van Moolenbroek "       -i IP6.INT reverse lookups\n"
159*00b67f09SDavid van Moolenbroek "       -N changes the number of dots allowed before root lookup is done\n"
160*00b67f09SDavid van Moolenbroek "       -r disables recursive processing\n"
161*00b67f09SDavid van Moolenbroek "       -R specifies number of retries for UDP packets\n"
162*00b67f09SDavid van Moolenbroek "       -s a SERVFAIL response should stop query\n"
163*00b67f09SDavid van Moolenbroek "       -t specifies the query type\n"
164*00b67f09SDavid van Moolenbroek "       -T enables TCP/IP mode\n"
165*00b67f09SDavid van Moolenbroek "       -v enables verbose output\n"
166*00b67f09SDavid van Moolenbroek "       -w specifies to wait forever for a reply\n"
167*00b67f09SDavid van Moolenbroek "       -W specifies how long to wait for a reply\n"
168*00b67f09SDavid van Moolenbroek "       -4 use IPv4 query transport only\n"
169*00b67f09SDavid van Moolenbroek "       -6 use IPv6 query transport only\n"
170*00b67f09SDavid van Moolenbroek "       -m set memory debugging flag (trace|record|usage)\n"
171*00b67f09SDavid van Moolenbroek "       -V print version number and exit\n", stderr);
172*00b67f09SDavid van Moolenbroek 	exit(1);
173*00b67f09SDavid van Moolenbroek }
174*00b67f09SDavid van Moolenbroek 
175*00b67f09SDavid van Moolenbroek void
dighost_shutdown(void)176*00b67f09SDavid van Moolenbroek dighost_shutdown(void) {
177*00b67f09SDavid van Moolenbroek 	isc_app_shutdown();
178*00b67f09SDavid van Moolenbroek }
179*00b67f09SDavid van Moolenbroek 
180*00b67f09SDavid van Moolenbroek void
received(int bytes,isc_sockaddr_t * from,dig_query_t * query)181*00b67f09SDavid van Moolenbroek received(int bytes, isc_sockaddr_t *from, dig_query_t *query) {
182*00b67f09SDavid van Moolenbroek 	isc_time_t now;
183*00b67f09SDavid van Moolenbroek 	int diff;
184*00b67f09SDavid van Moolenbroek 
185*00b67f09SDavid van Moolenbroek 	if (!short_form) {
186*00b67f09SDavid van Moolenbroek 		char fromtext[ISC_SOCKADDR_FORMATSIZE];
187*00b67f09SDavid van Moolenbroek 		isc_sockaddr_format(from, fromtext, sizeof(fromtext));
188*00b67f09SDavid van Moolenbroek 		TIME_NOW(&now);
189*00b67f09SDavid van Moolenbroek 		diff = (int) isc_time_microdiff(&now, &query->time_sent);
190*00b67f09SDavid van Moolenbroek 		printf("Received %u bytes from %s in %d ms\n",
191*00b67f09SDavid van Moolenbroek 		       bytes, fromtext, diff/1000);
192*00b67f09SDavid van Moolenbroek 	}
193*00b67f09SDavid van Moolenbroek }
194*00b67f09SDavid van Moolenbroek 
195*00b67f09SDavid van Moolenbroek void
trying(char * frm,dig_lookup_t * lookup)196*00b67f09SDavid van Moolenbroek trying(char *frm, dig_lookup_t *lookup) {
197*00b67f09SDavid van Moolenbroek 	UNUSED(lookup);
198*00b67f09SDavid van Moolenbroek 
199*00b67f09SDavid van Moolenbroek 	if (!short_form)
200*00b67f09SDavid van Moolenbroek 		printf("Trying \"%s\"\n", frm);
201*00b67f09SDavid van Moolenbroek }
202*00b67f09SDavid van Moolenbroek 
203*00b67f09SDavid van Moolenbroek static void
say_message(dns_name_t * name,const char * msg,dns_rdata_t * rdata,dig_query_t * query)204*00b67f09SDavid van Moolenbroek say_message(dns_name_t *name, const char *msg, dns_rdata_t *rdata,
205*00b67f09SDavid van Moolenbroek 	    dig_query_t *query)
206*00b67f09SDavid van Moolenbroek {
207*00b67f09SDavid van Moolenbroek 	isc_buffer_t *b = NULL;
208*00b67f09SDavid van Moolenbroek 	char namestr[DNS_NAME_FORMATSIZE];
209*00b67f09SDavid van Moolenbroek 	isc_region_t r;
210*00b67f09SDavid van Moolenbroek 	isc_result_t result;
211*00b67f09SDavid van Moolenbroek 	unsigned int bufsize = BUFSIZ;
212*00b67f09SDavid van Moolenbroek 
213*00b67f09SDavid van Moolenbroek 	dns_name_format(name, namestr, sizeof(namestr));
214*00b67f09SDavid van Moolenbroek  retry:
215*00b67f09SDavid van Moolenbroek 	result = isc_buffer_allocate(mctx, &b, bufsize);
216*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_buffer_allocate");
217*00b67f09SDavid van Moolenbroek 	result = dns_rdata_totext(rdata, NULL, b);
218*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_NOSPACE) {
219*00b67f09SDavid van Moolenbroek 		isc_buffer_free(&b);
220*00b67f09SDavid van Moolenbroek 		bufsize *= 2;
221*00b67f09SDavid van Moolenbroek 		goto retry;
222*00b67f09SDavid van Moolenbroek 	}
223*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_rdata_totext");
224*00b67f09SDavid van Moolenbroek 	isc_buffer_usedregion(b, &r);
225*00b67f09SDavid van Moolenbroek 	if (query->lookup->identify_previous_line) {
226*00b67f09SDavid van Moolenbroek 		printf("Nameserver %s:\n\t",
227*00b67f09SDavid van Moolenbroek 			query->servname);
228*00b67f09SDavid van Moolenbroek 	}
229*00b67f09SDavid van Moolenbroek 	printf("%s %s %.*s", namestr,
230*00b67f09SDavid van Moolenbroek 	       msg, (int)r.length, (char *)r.base);
231*00b67f09SDavid van Moolenbroek 	if (query->lookup->identify) {
232*00b67f09SDavid van Moolenbroek 		printf(" on server %s", query->servname);
233*00b67f09SDavid van Moolenbroek 	}
234*00b67f09SDavid van Moolenbroek 	printf("\n");
235*00b67f09SDavid van Moolenbroek 	isc_buffer_free(&b);
236*00b67f09SDavid van Moolenbroek }
237*00b67f09SDavid van Moolenbroek #ifdef DIG_SIGCHASE
238*00b67f09SDavid van Moolenbroek /* Just for compatibility : not use in host program */
239*00b67f09SDavid van Moolenbroek isc_result_t
printrdataset(dns_name_t * owner_name,dns_rdataset_t * rdataset,isc_buffer_t * target)240*00b67f09SDavid van Moolenbroek printrdataset(dns_name_t *owner_name, dns_rdataset_t *rdataset,
241*00b67f09SDavid van Moolenbroek 	      isc_buffer_t *target)
242*00b67f09SDavid van Moolenbroek {
243*00b67f09SDavid van Moolenbroek   UNUSED(owner_name);
244*00b67f09SDavid van Moolenbroek   UNUSED(rdataset);
245*00b67f09SDavid van Moolenbroek   UNUSED(target);
246*00b67f09SDavid van Moolenbroek   return(ISC_FALSE);
247*00b67f09SDavid van Moolenbroek }
248*00b67f09SDavid van Moolenbroek #endif
249*00b67f09SDavid van Moolenbroek static isc_result_t
printsection(dns_message_t * msg,dns_section_t sectionid,const char * section_name,isc_boolean_t headers,dig_query_t * query)250*00b67f09SDavid van Moolenbroek printsection(dns_message_t *msg, dns_section_t sectionid,
251*00b67f09SDavid van Moolenbroek 	     const char *section_name, isc_boolean_t headers,
252*00b67f09SDavid van Moolenbroek 	     dig_query_t *query)
253*00b67f09SDavid van Moolenbroek {
254*00b67f09SDavid van Moolenbroek 	dns_name_t *name, *print_name;
255*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *rdataset;
256*00b67f09SDavid van Moolenbroek 	dns_rdata_t rdata = DNS_RDATA_INIT;
257*00b67f09SDavid van Moolenbroek 	isc_buffer_t target;
258*00b67f09SDavid van Moolenbroek 	isc_result_t result, loopresult;
259*00b67f09SDavid van Moolenbroek 	isc_region_t r;
260*00b67f09SDavid van Moolenbroek 	dns_name_t empty_name;
261*00b67f09SDavid van Moolenbroek 	char tbuf[4096];
262*00b67f09SDavid van Moolenbroek 	isc_boolean_t first;
263*00b67f09SDavid van Moolenbroek 	isc_boolean_t no_rdata;
264*00b67f09SDavid van Moolenbroek 
265*00b67f09SDavid van Moolenbroek 	if (sectionid == DNS_SECTION_QUESTION)
266*00b67f09SDavid van Moolenbroek 		no_rdata = ISC_TRUE;
267*00b67f09SDavid van Moolenbroek 	else
268*00b67f09SDavid van Moolenbroek 		no_rdata = ISC_FALSE;
269*00b67f09SDavid van Moolenbroek 
270*00b67f09SDavid van Moolenbroek 	if (headers)
271*00b67f09SDavid van Moolenbroek 		printf(";; %s SECTION:\n", section_name);
272*00b67f09SDavid van Moolenbroek 
273*00b67f09SDavid van Moolenbroek 	dns_name_init(&empty_name, NULL);
274*00b67f09SDavid van Moolenbroek 
275*00b67f09SDavid van Moolenbroek 	result = dns_message_firstname(msg, sectionid);
276*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_NOMORE)
277*00b67f09SDavid van Moolenbroek 		return (ISC_R_SUCCESS);
278*00b67f09SDavid van Moolenbroek 	else if (result != ISC_R_SUCCESS)
279*00b67f09SDavid van Moolenbroek 		return (result);
280*00b67f09SDavid van Moolenbroek 
281*00b67f09SDavid van Moolenbroek 	for (;;) {
282*00b67f09SDavid van Moolenbroek 		name = NULL;
283*00b67f09SDavid van Moolenbroek 		dns_message_currentname(msg, sectionid, &name);
284*00b67f09SDavid van Moolenbroek 
285*00b67f09SDavid van Moolenbroek 		isc_buffer_init(&target, tbuf, sizeof(tbuf));
286*00b67f09SDavid van Moolenbroek 		first = ISC_TRUE;
287*00b67f09SDavid van Moolenbroek 		print_name = name;
288*00b67f09SDavid van Moolenbroek 
289*00b67f09SDavid van Moolenbroek 		for (rdataset = ISC_LIST_HEAD(name->list);
290*00b67f09SDavid van Moolenbroek 		     rdataset != NULL;
291*00b67f09SDavid van Moolenbroek 		     rdataset = ISC_LIST_NEXT(rdataset, link)) {
292*00b67f09SDavid van Moolenbroek 			if (query->lookup->rdtype == dns_rdatatype_axfr &&
293*00b67f09SDavid van Moolenbroek 			    !((!list_addresses &&
294*00b67f09SDavid van Moolenbroek 			       (list_type == dns_rdatatype_any ||
295*00b67f09SDavid van Moolenbroek 				rdataset->type == list_type)) ||
296*00b67f09SDavid van Moolenbroek 			      (list_addresses &&
297*00b67f09SDavid van Moolenbroek 			       (rdataset->type == dns_rdatatype_a ||
298*00b67f09SDavid van Moolenbroek 				rdataset->type == dns_rdatatype_aaaa ||
299*00b67f09SDavid van Moolenbroek 				rdataset->type == dns_rdatatype_ns ||
300*00b67f09SDavid van Moolenbroek 				rdataset->type == dns_rdatatype_ptr))))
301*00b67f09SDavid van Moolenbroek 				continue;
302*00b67f09SDavid van Moolenbroek 			if (!short_form) {
303*00b67f09SDavid van Moolenbroek 				result = dns_rdataset_totext(rdataset,
304*00b67f09SDavid van Moolenbroek 							     print_name,
305*00b67f09SDavid van Moolenbroek 							     ISC_FALSE,
306*00b67f09SDavid van Moolenbroek 							     no_rdata,
307*00b67f09SDavid van Moolenbroek 							     &target);
308*00b67f09SDavid van Moolenbroek 				if (result != ISC_R_SUCCESS)
309*00b67f09SDavid van Moolenbroek 					return (result);
310*00b67f09SDavid van Moolenbroek #ifdef USEINITALWS
311*00b67f09SDavid van Moolenbroek 				if (first) {
312*00b67f09SDavid van Moolenbroek 					print_name = &empty_name;
313*00b67f09SDavid van Moolenbroek 					first = ISC_FALSE;
314*00b67f09SDavid van Moolenbroek 				}
315*00b67f09SDavid van Moolenbroek #else
316*00b67f09SDavid van Moolenbroek 				UNUSED(first); /* Shut up compiler. */
317*00b67f09SDavid van Moolenbroek #endif
318*00b67f09SDavid van Moolenbroek 			} else {
319*00b67f09SDavid van Moolenbroek 				loopresult = dns_rdataset_first(rdataset);
320*00b67f09SDavid van Moolenbroek 				while (loopresult == ISC_R_SUCCESS) {
321*00b67f09SDavid van Moolenbroek 					struct rtype *t;
322*00b67f09SDavid van Moolenbroek 					const char *rtt;
323*00b67f09SDavid van Moolenbroek 					char typebuf[DNS_RDATATYPE_FORMATSIZE];
324*00b67f09SDavid van Moolenbroek 					char typebuf2[DNS_RDATATYPE_FORMATSIZE
325*00b67f09SDavid van Moolenbroek 						     + 20];
326*00b67f09SDavid van Moolenbroek 					dns_rdataset_current(rdataset, &rdata);
327*00b67f09SDavid van Moolenbroek 
328*00b67f09SDavid van Moolenbroek 					for (t = rtypes; t->text != NULL; t++) {
329*00b67f09SDavid van Moolenbroek 						if (t->type == rdata.type) {
330*00b67f09SDavid van Moolenbroek 							rtt = t->text;
331*00b67f09SDavid van Moolenbroek 							goto found;
332*00b67f09SDavid van Moolenbroek 						}
333*00b67f09SDavid van Moolenbroek 					}
334*00b67f09SDavid van Moolenbroek 
335*00b67f09SDavid van Moolenbroek 					dns_rdatatype_format(rdata.type,
336*00b67f09SDavid van Moolenbroek 							     typebuf,
337*00b67f09SDavid van Moolenbroek 							     sizeof(typebuf));
338*00b67f09SDavid van Moolenbroek 					snprintf(typebuf2, sizeof(typebuf2),
339*00b67f09SDavid van Moolenbroek 						 "has %s record", typebuf);
340*00b67f09SDavid van Moolenbroek 					rtt = typebuf2;
341*00b67f09SDavid van Moolenbroek 				found:
342*00b67f09SDavid van Moolenbroek 					say_message(print_name, rtt,
343*00b67f09SDavid van Moolenbroek 						    &rdata, query);
344*00b67f09SDavid van Moolenbroek 					dns_rdata_reset(&rdata);
345*00b67f09SDavid van Moolenbroek 					loopresult =
346*00b67f09SDavid van Moolenbroek 						dns_rdataset_next(rdataset);
347*00b67f09SDavid van Moolenbroek 				}
348*00b67f09SDavid van Moolenbroek 			}
349*00b67f09SDavid van Moolenbroek 		}
350*00b67f09SDavid van Moolenbroek 		if (!short_form) {
351*00b67f09SDavid van Moolenbroek 			isc_buffer_usedregion(&target, &r);
352*00b67f09SDavid van Moolenbroek 			if (no_rdata)
353*00b67f09SDavid van Moolenbroek 				printf(";%.*s", (int)r.length,
354*00b67f09SDavid van Moolenbroek 				       (char *)r.base);
355*00b67f09SDavid van Moolenbroek 			else
356*00b67f09SDavid van Moolenbroek 				printf("%.*s", (int)r.length, (char *)r.base);
357*00b67f09SDavid van Moolenbroek 		}
358*00b67f09SDavid van Moolenbroek 
359*00b67f09SDavid van Moolenbroek 		result = dns_message_nextname(msg, sectionid);
360*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_NOMORE)
361*00b67f09SDavid van Moolenbroek 			break;
362*00b67f09SDavid van Moolenbroek 		else if (result != ISC_R_SUCCESS)
363*00b67f09SDavid van Moolenbroek 			return (result);
364*00b67f09SDavid van Moolenbroek 	}
365*00b67f09SDavid van Moolenbroek 
366*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
367*00b67f09SDavid van Moolenbroek }
368*00b67f09SDavid van Moolenbroek 
369*00b67f09SDavid van Moolenbroek static isc_result_t
printrdata(dns_message_t * msg,dns_rdataset_t * rdataset,dns_name_t * owner,const char * set_name,isc_boolean_t headers)370*00b67f09SDavid van Moolenbroek printrdata(dns_message_t *msg, dns_rdataset_t *rdataset, dns_name_t *owner,
371*00b67f09SDavid van Moolenbroek 	   const char *set_name, isc_boolean_t headers)
372*00b67f09SDavid van Moolenbroek {
373*00b67f09SDavid van Moolenbroek 	isc_buffer_t target;
374*00b67f09SDavid van Moolenbroek 	isc_result_t result;
375*00b67f09SDavid van Moolenbroek 	isc_region_t r;
376*00b67f09SDavid van Moolenbroek 	char tbuf[4096];
377*00b67f09SDavid van Moolenbroek 
378*00b67f09SDavid van Moolenbroek 	UNUSED(msg);
379*00b67f09SDavid van Moolenbroek 	if (headers)
380*00b67f09SDavid van Moolenbroek 		printf(";; %s SECTION:\n", set_name);
381*00b67f09SDavid van Moolenbroek 
382*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&target, tbuf, sizeof(tbuf));
383*00b67f09SDavid van Moolenbroek 
384*00b67f09SDavid van Moolenbroek 	result = dns_rdataset_totext(rdataset, owner, ISC_FALSE, ISC_FALSE,
385*00b67f09SDavid van Moolenbroek 				     &target);
386*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
387*00b67f09SDavid van Moolenbroek 		return (result);
388*00b67f09SDavid van Moolenbroek 	isc_buffer_usedregion(&target, &r);
389*00b67f09SDavid van Moolenbroek 	printf("%.*s", (int)r.length, (char *)r.base);
390*00b67f09SDavid van Moolenbroek 
391*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
392*00b67f09SDavid van Moolenbroek }
393*00b67f09SDavid van Moolenbroek 
394*00b67f09SDavid van Moolenbroek static void
chase_cnamechain(dns_message_t * msg,dns_name_t * qname)395*00b67f09SDavid van Moolenbroek chase_cnamechain(dns_message_t *msg, dns_name_t *qname) {
396*00b67f09SDavid van Moolenbroek 	isc_result_t result;
397*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *rdataset;
398*00b67f09SDavid van Moolenbroek 	dns_rdata_cname_t cname;
399*00b67f09SDavid van Moolenbroek 	dns_rdata_t rdata = DNS_RDATA_INIT;
400*00b67f09SDavid van Moolenbroek 	unsigned int i = msg->counts[DNS_SECTION_ANSWER];
401*00b67f09SDavid van Moolenbroek 
402*00b67f09SDavid van Moolenbroek 	while (i-- > 0) {
403*00b67f09SDavid van Moolenbroek 		rdataset = NULL;
404*00b67f09SDavid van Moolenbroek 		result = dns_message_findname(msg, DNS_SECTION_ANSWER, qname,
405*00b67f09SDavid van Moolenbroek 					      dns_rdatatype_cname, 0, NULL,
406*00b67f09SDavid van Moolenbroek 					      &rdataset);
407*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
408*00b67f09SDavid van Moolenbroek 			return;
409*00b67f09SDavid van Moolenbroek 		result = dns_rdataset_first(rdataset);
410*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_rdataset_first");
411*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&rdata);
412*00b67f09SDavid van Moolenbroek 		dns_rdataset_current(rdataset, &rdata);
413*00b67f09SDavid van Moolenbroek 		result = dns_rdata_tostruct(&rdata, &cname, NULL);
414*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_rdata_tostruct");
415*00b67f09SDavid van Moolenbroek 		dns_name_copy(&cname.cname, qname, NULL);
416*00b67f09SDavid van Moolenbroek 		dns_rdata_freestruct(&cname);
417*00b67f09SDavid van Moolenbroek 	}
418*00b67f09SDavid van Moolenbroek }
419*00b67f09SDavid van Moolenbroek 
420*00b67f09SDavid van Moolenbroek isc_result_t
printmessage(dig_query_t * query,dns_message_t * msg,isc_boolean_t headers)421*00b67f09SDavid van Moolenbroek printmessage(dig_query_t *query, dns_message_t *msg, isc_boolean_t headers) {
422*00b67f09SDavid van Moolenbroek 	isc_boolean_t did_flag = ISC_FALSE;
423*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *opt, *tsig = NULL;
424*00b67f09SDavid van Moolenbroek 	dns_name_t *tsigname;
425*00b67f09SDavid van Moolenbroek 	isc_result_t result = ISC_R_SUCCESS;
426*00b67f09SDavid van Moolenbroek 	int force_error;
427*00b67f09SDavid van Moolenbroek 
428*00b67f09SDavid van Moolenbroek 	UNUSED(headers);
429*00b67f09SDavid van Moolenbroek 
430*00b67f09SDavid van Moolenbroek 	/*
431*00b67f09SDavid van Moolenbroek 	 * We get called multiple times.
432*00b67f09SDavid van Moolenbroek 	 * Preserve any existing error status.
433*00b67f09SDavid van Moolenbroek 	 */
434*00b67f09SDavid van Moolenbroek 	force_error = (seen_error == 1) ? 1 : 0;
435*00b67f09SDavid van Moolenbroek 	seen_error = 1;
436*00b67f09SDavid van Moolenbroek 	if (listed_server && !printed_server) {
437*00b67f09SDavid van Moolenbroek 		char sockstr[ISC_SOCKADDR_FORMATSIZE];
438*00b67f09SDavid van Moolenbroek 
439*00b67f09SDavid van Moolenbroek 		printf("Using domain server:\n");
440*00b67f09SDavid van Moolenbroek 		printf("Name: %s\n", query->userarg);
441*00b67f09SDavid van Moolenbroek 		isc_sockaddr_format(&query->sockaddr, sockstr,
442*00b67f09SDavid van Moolenbroek 				    sizeof(sockstr));
443*00b67f09SDavid van Moolenbroek 		printf("Address: %s\n", sockstr);
444*00b67f09SDavid van Moolenbroek 		printf("Aliases: \n\n");
445*00b67f09SDavid van Moolenbroek 		printed_server = ISC_TRUE;
446*00b67f09SDavid van Moolenbroek 	}
447*00b67f09SDavid van Moolenbroek 
448*00b67f09SDavid van Moolenbroek 	if (msg->rcode != 0) {
449*00b67f09SDavid van Moolenbroek 		char namestr[DNS_NAME_FORMATSIZE];
450*00b67f09SDavid van Moolenbroek 		dns_name_format(query->lookup->name, namestr, sizeof(namestr));
451*00b67f09SDavid van Moolenbroek 
452*00b67f09SDavid van Moolenbroek 		if (query->lookup->identify_previous_line)
453*00b67f09SDavid van Moolenbroek 			printf("Nameserver %s:\n\t%s not found: %d(%s)\n",
454*00b67f09SDavid van Moolenbroek 			       query->servname,
455*00b67f09SDavid van Moolenbroek 			       (msg->rcode != dns_rcode_nxdomain) ? namestr :
456*00b67f09SDavid van Moolenbroek 			       query->lookup->textname, msg->rcode,
457*00b67f09SDavid van Moolenbroek 			       rcode_totext(msg->rcode));
458*00b67f09SDavid van Moolenbroek 		else
459*00b67f09SDavid van Moolenbroek 			printf("Host %s not found: %d(%s)\n",
460*00b67f09SDavid van Moolenbroek 			       (msg->rcode != dns_rcode_nxdomain) ? namestr :
461*00b67f09SDavid van Moolenbroek 			       query->lookup->textname, msg->rcode,
462*00b67f09SDavid van Moolenbroek 			       rcode_totext(msg->rcode));
463*00b67f09SDavid van Moolenbroek 		return (ISC_R_SUCCESS);
464*00b67f09SDavid van Moolenbroek 	}
465*00b67f09SDavid van Moolenbroek 
466*00b67f09SDavid van Moolenbroek 	if (default_lookups && query->lookup->rdtype == dns_rdatatype_a) {
467*00b67f09SDavid van Moolenbroek 		char namestr[DNS_NAME_FORMATSIZE];
468*00b67f09SDavid van Moolenbroek 		dig_lookup_t *lookup;
469*00b67f09SDavid van Moolenbroek 		dns_fixedname_t fixed;
470*00b67f09SDavid van Moolenbroek 		dns_name_t *name;
471*00b67f09SDavid van Moolenbroek 
472*00b67f09SDavid van Moolenbroek 		/* Add AAAA and MX lookups. */
473*00b67f09SDavid van Moolenbroek 		dns_fixedname_init(&fixed);
474*00b67f09SDavid van Moolenbroek 		name = dns_fixedname_name(&fixed);
475*00b67f09SDavid van Moolenbroek 		dns_name_copy(query->lookup->name, name, NULL);
476*00b67f09SDavid van Moolenbroek 		chase_cnamechain(msg, name);
477*00b67f09SDavid van Moolenbroek 		dns_name_format(name, namestr, sizeof(namestr));
478*00b67f09SDavid van Moolenbroek 		lookup = clone_lookup(query->lookup, ISC_FALSE);
479*00b67f09SDavid van Moolenbroek 		if (lookup != NULL) {
480*00b67f09SDavid van Moolenbroek 			strncpy(lookup->textname, namestr,
481*00b67f09SDavid van Moolenbroek 				sizeof(lookup->textname));
482*00b67f09SDavid van Moolenbroek 			lookup->textname[sizeof(lookup->textname)-1] = 0;
483*00b67f09SDavid van Moolenbroek 			lookup->rdtype = dns_rdatatype_aaaa;
484*00b67f09SDavid van Moolenbroek 			lookup->rdtypeset = ISC_TRUE;
485*00b67f09SDavid van Moolenbroek 			lookup->origin = NULL;
486*00b67f09SDavid van Moolenbroek 			lookup->retries = tries;
487*00b67f09SDavid van Moolenbroek 			ISC_LIST_APPEND(lookup_list, lookup, link);
488*00b67f09SDavid van Moolenbroek 		}
489*00b67f09SDavid van Moolenbroek 		lookup = clone_lookup(query->lookup, ISC_FALSE);
490*00b67f09SDavid van Moolenbroek 		if (lookup != NULL) {
491*00b67f09SDavid van Moolenbroek 			strncpy(lookup->textname, namestr,
492*00b67f09SDavid van Moolenbroek 				sizeof(lookup->textname));
493*00b67f09SDavid van Moolenbroek 			lookup->textname[sizeof(lookup->textname)-1] = 0;
494*00b67f09SDavid van Moolenbroek 			lookup->rdtype = dns_rdatatype_mx;
495*00b67f09SDavid van Moolenbroek 			lookup->rdtypeset = ISC_TRUE;
496*00b67f09SDavid van Moolenbroek 			lookup->origin = NULL;
497*00b67f09SDavid van Moolenbroek 			lookup->retries = tries;
498*00b67f09SDavid van Moolenbroek 			ISC_LIST_APPEND(lookup_list, lookup, link);
499*00b67f09SDavid van Moolenbroek 		}
500*00b67f09SDavid van Moolenbroek 	}
501*00b67f09SDavid van Moolenbroek 
502*00b67f09SDavid van Moolenbroek 	if (!short_form) {
503*00b67f09SDavid van Moolenbroek 		printf(";; ->>HEADER<<- opcode: %s, status: %s, id: %u\n",
504*00b67f09SDavid van Moolenbroek 		       opcodetext[msg->opcode], rcode_totext(msg->rcode),
505*00b67f09SDavid van Moolenbroek 		       msg->id);
506*00b67f09SDavid van Moolenbroek 		printf(";; flags: ");
507*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_QR) != 0) {
508*00b67f09SDavid van Moolenbroek 			printf("qr");
509*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
510*00b67f09SDavid van Moolenbroek 		}
511*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_AA) != 0) {
512*00b67f09SDavid van Moolenbroek 			printf("%saa", did_flag ? " " : "");
513*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
514*00b67f09SDavid van Moolenbroek 		}
515*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_TC) != 0) {
516*00b67f09SDavid van Moolenbroek 			printf("%stc", did_flag ? " " : "");
517*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
518*00b67f09SDavid van Moolenbroek 		}
519*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_RD) != 0) {
520*00b67f09SDavid van Moolenbroek 			printf("%srd", did_flag ? " " : "");
521*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
522*00b67f09SDavid van Moolenbroek 		}
523*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_RA) != 0) {
524*00b67f09SDavid van Moolenbroek 			printf("%sra", did_flag ? " " : "");
525*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
526*00b67f09SDavid van Moolenbroek 		}
527*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_AD) != 0) {
528*00b67f09SDavid van Moolenbroek 			printf("%sad", did_flag ? " " : "");
529*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
530*00b67f09SDavid van Moolenbroek 		}
531*00b67f09SDavid van Moolenbroek 		if ((msg->flags & DNS_MESSAGEFLAG_CD) != 0) {
532*00b67f09SDavid van Moolenbroek 			printf("%scd", did_flag ? " " : "");
533*00b67f09SDavid van Moolenbroek 			did_flag = ISC_TRUE;
534*00b67f09SDavid van Moolenbroek 			POST(did_flag);
535*00b67f09SDavid van Moolenbroek 		}
536*00b67f09SDavid van Moolenbroek 		printf("; QUERY: %u, ANSWER: %u, "
537*00b67f09SDavid van Moolenbroek 		       "AUTHORITY: %u, ADDITIONAL: %u\n",
538*00b67f09SDavid van Moolenbroek 		       msg->counts[DNS_SECTION_QUESTION],
539*00b67f09SDavid van Moolenbroek 		       msg->counts[DNS_SECTION_ANSWER],
540*00b67f09SDavid van Moolenbroek 		       msg->counts[DNS_SECTION_AUTHORITY],
541*00b67f09SDavid van Moolenbroek 		       msg->counts[DNS_SECTION_ADDITIONAL]);
542*00b67f09SDavid van Moolenbroek 		opt = dns_message_getopt(msg);
543*00b67f09SDavid van Moolenbroek 		if (opt != NULL)
544*00b67f09SDavid van Moolenbroek 			printf(";; EDNS: version: %u, udp=%u\n",
545*00b67f09SDavid van Moolenbroek 			       (unsigned int)((opt->ttl & 0x00ff0000) >> 16),
546*00b67f09SDavid van Moolenbroek 			       (unsigned int)opt->rdclass);
547*00b67f09SDavid van Moolenbroek 		tsigname = NULL;
548*00b67f09SDavid van Moolenbroek 		tsig = dns_message_gettsig(msg, &tsigname);
549*00b67f09SDavid van Moolenbroek 		if (tsig != NULL)
550*00b67f09SDavid van Moolenbroek 			printf(";; PSEUDOSECTIONS: TSIG\n");
551*00b67f09SDavid van Moolenbroek 	}
552*00b67f09SDavid van Moolenbroek 	if (! ISC_LIST_EMPTY(msg->sections[DNS_SECTION_QUESTION]) &&
553*00b67f09SDavid van Moolenbroek 	    !short_form) {
554*00b67f09SDavid van Moolenbroek 		printf("\n");
555*00b67f09SDavid van Moolenbroek 		result = printsection(msg, DNS_SECTION_QUESTION, "QUESTION",
556*00b67f09SDavid van Moolenbroek 				      ISC_TRUE, query);
557*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
558*00b67f09SDavid van Moolenbroek 			return (result);
559*00b67f09SDavid van Moolenbroek 	}
560*00b67f09SDavid van Moolenbroek 	if (! ISC_LIST_EMPTY(msg->sections[DNS_SECTION_ANSWER])) {
561*00b67f09SDavid van Moolenbroek 		if (!short_form)
562*00b67f09SDavid van Moolenbroek 			printf("\n");
563*00b67f09SDavid van Moolenbroek 		result = printsection(msg, DNS_SECTION_ANSWER, "ANSWER",
564*00b67f09SDavid van Moolenbroek 				      ISC_TF(!short_form), query);
565*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
566*00b67f09SDavid van Moolenbroek 			return (result);
567*00b67f09SDavid van Moolenbroek 	}
568*00b67f09SDavid van Moolenbroek 
569*00b67f09SDavid van Moolenbroek 	if (! ISC_LIST_EMPTY(msg->sections[DNS_SECTION_AUTHORITY]) &&
570*00b67f09SDavid van Moolenbroek 	    !short_form) {
571*00b67f09SDavid van Moolenbroek 		printf("\n");
572*00b67f09SDavid van Moolenbroek 		result = printsection(msg, DNS_SECTION_AUTHORITY, "AUTHORITY",
573*00b67f09SDavid van Moolenbroek 				      ISC_TRUE, query);
574*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
575*00b67f09SDavid van Moolenbroek 			return (result);
576*00b67f09SDavid van Moolenbroek 	}
577*00b67f09SDavid van Moolenbroek 	if (! ISC_LIST_EMPTY(msg->sections[DNS_SECTION_ADDITIONAL]) &&
578*00b67f09SDavid van Moolenbroek 	    !short_form) {
579*00b67f09SDavid van Moolenbroek 		printf("\n");
580*00b67f09SDavid van Moolenbroek 		result = printsection(msg, DNS_SECTION_ADDITIONAL,
581*00b67f09SDavid van Moolenbroek 				      "ADDITIONAL", ISC_TRUE, query);
582*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
583*00b67f09SDavid van Moolenbroek 			return (result);
584*00b67f09SDavid van Moolenbroek 	}
585*00b67f09SDavid van Moolenbroek 	if ((tsig != NULL) && !short_form) {
586*00b67f09SDavid van Moolenbroek 		printf("\n");
587*00b67f09SDavid van Moolenbroek 		result = printrdata(msg, tsig, tsigname,
588*00b67f09SDavid van Moolenbroek 				    "PSEUDOSECTION TSIG", ISC_TRUE);
589*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
590*00b67f09SDavid van Moolenbroek 			return (result);
591*00b67f09SDavid van Moolenbroek 	}
592*00b67f09SDavid van Moolenbroek 	if (!short_form)
593*00b67f09SDavid van Moolenbroek 		printf("\n");
594*00b67f09SDavid van Moolenbroek 
595*00b67f09SDavid van Moolenbroek 	if (short_form && !default_lookups &&
596*00b67f09SDavid van Moolenbroek 	    ISC_LIST_EMPTY(msg->sections[DNS_SECTION_ANSWER])) {
597*00b67f09SDavid van Moolenbroek 		char namestr[DNS_NAME_FORMATSIZE];
598*00b67f09SDavid van Moolenbroek 		char typestr[DNS_RDATATYPE_FORMATSIZE];
599*00b67f09SDavid van Moolenbroek 		dns_name_format(query->lookup->name, namestr, sizeof(namestr));
600*00b67f09SDavid van Moolenbroek 		dns_rdatatype_format(query->lookup->rdtype, typestr,
601*00b67f09SDavid van Moolenbroek 				     sizeof(typestr));
602*00b67f09SDavid van Moolenbroek 		printf("%s has no %s record\n", namestr, typestr);
603*00b67f09SDavid van Moolenbroek 	}
604*00b67f09SDavid van Moolenbroek 	seen_error = force_error;
605*00b67f09SDavid van Moolenbroek 	return (result);
606*00b67f09SDavid van Moolenbroek }
607*00b67f09SDavid van Moolenbroek 
608*00b67f09SDavid van Moolenbroek static const char * optstring = "46ac:dilnm:rst:vVwCDN:R:TW:";
609*00b67f09SDavid van Moolenbroek 
610*00b67f09SDavid van Moolenbroek /*% version */
611*00b67f09SDavid van Moolenbroek static void
version(void)612*00b67f09SDavid van Moolenbroek version(void) {
613*00b67f09SDavid van Moolenbroek 	fputs("host " VERSION "\n", stderr);
614*00b67f09SDavid van Moolenbroek }
615*00b67f09SDavid van Moolenbroek 
616*00b67f09SDavid van Moolenbroek static void
pre_parse_args(int argc,char ** argv)617*00b67f09SDavid van Moolenbroek pre_parse_args(int argc, char **argv) {
618*00b67f09SDavid van Moolenbroek 	int c;
619*00b67f09SDavid van Moolenbroek 
620*00b67f09SDavid van Moolenbroek 	while ((c = isc_commandline_parse(argc, argv, optstring)) != -1) {
621*00b67f09SDavid van Moolenbroek 		switch (c) {
622*00b67f09SDavid van Moolenbroek 		case 'm':
623*00b67f09SDavid van Moolenbroek 			memdebugging = ISC_TRUE;
624*00b67f09SDavid van Moolenbroek 			if (strcasecmp("trace", isc_commandline_argument) == 0)
625*00b67f09SDavid van Moolenbroek 				isc_mem_debugging |= ISC_MEM_DEBUGTRACE;
626*00b67f09SDavid van Moolenbroek 			else if (!strcasecmp("record",
627*00b67f09SDavid van Moolenbroek 					     isc_commandline_argument) == 0)
628*00b67f09SDavid van Moolenbroek 				isc_mem_debugging |= ISC_MEM_DEBUGRECORD;
629*00b67f09SDavid van Moolenbroek 			else if (strcasecmp("usage",
630*00b67f09SDavid van Moolenbroek 					    isc_commandline_argument) == 0)
631*00b67f09SDavid van Moolenbroek 				isc_mem_debugging |= ISC_MEM_DEBUGUSAGE;
632*00b67f09SDavid van Moolenbroek 			break;
633*00b67f09SDavid van Moolenbroek 
634*00b67f09SDavid van Moolenbroek 		case '4': break;
635*00b67f09SDavid van Moolenbroek 		case '6': break;
636*00b67f09SDavid van Moolenbroek 		case 'a': break;
637*00b67f09SDavid van Moolenbroek 		case 'c': break;
638*00b67f09SDavid van Moolenbroek 		case 'd': break;
639*00b67f09SDavid van Moolenbroek 		case 'i': break;
640*00b67f09SDavid van Moolenbroek 		case 'l': break;
641*00b67f09SDavid van Moolenbroek 		case 'n': break;
642*00b67f09SDavid van Moolenbroek 		case 'r': break;
643*00b67f09SDavid van Moolenbroek 		case 's': break;
644*00b67f09SDavid van Moolenbroek 		case 't': break;
645*00b67f09SDavid van Moolenbroek 		case 'v': break;
646*00b67f09SDavid van Moolenbroek 		case 'V':
647*00b67f09SDavid van Moolenbroek 			  version();
648*00b67f09SDavid van Moolenbroek 			  exit(0);
649*00b67f09SDavid van Moolenbroek 			  break;
650*00b67f09SDavid van Moolenbroek 		case 'w': break;
651*00b67f09SDavid van Moolenbroek 		case 'C': break;
652*00b67f09SDavid van Moolenbroek 		case 'D':
653*00b67f09SDavid van Moolenbroek 			if (debugging)
654*00b67f09SDavid van Moolenbroek 				debugtiming = ISC_TRUE;
655*00b67f09SDavid van Moolenbroek 			debugging = ISC_TRUE;
656*00b67f09SDavid van Moolenbroek 			break;
657*00b67f09SDavid van Moolenbroek 		case 'N': break;
658*00b67f09SDavid van Moolenbroek 		case 'R': break;
659*00b67f09SDavid van Moolenbroek 		case 'T': break;
660*00b67f09SDavid van Moolenbroek 		case 'W': break;
661*00b67f09SDavid van Moolenbroek 		default:
662*00b67f09SDavid van Moolenbroek 			show_usage();
663*00b67f09SDavid van Moolenbroek 		}
664*00b67f09SDavid van Moolenbroek 	}
665*00b67f09SDavid van Moolenbroek 	isc_commandline_reset = ISC_TRUE;
666*00b67f09SDavid van Moolenbroek 	isc_commandline_index = 1;
667*00b67f09SDavid van Moolenbroek }
668*00b67f09SDavid van Moolenbroek 
669*00b67f09SDavid van Moolenbroek static void
parse_args(isc_boolean_t is_batchfile,int argc,char ** argv)670*00b67f09SDavid van Moolenbroek parse_args(isc_boolean_t is_batchfile, int argc, char **argv) {
671*00b67f09SDavid van Moolenbroek 	char hostname[MXNAME];
672*00b67f09SDavid van Moolenbroek 	dig_lookup_t *lookup;
673*00b67f09SDavid van Moolenbroek 	int c;
674*00b67f09SDavid van Moolenbroek 	char store[MXNAME];
675*00b67f09SDavid van Moolenbroek 	isc_textregion_t tr;
676*00b67f09SDavid van Moolenbroek 	isc_result_t result = ISC_R_SUCCESS;
677*00b67f09SDavid van Moolenbroek 	dns_rdatatype_t rdtype;
678*00b67f09SDavid van Moolenbroek 	dns_rdataclass_t rdclass;
679*00b67f09SDavid van Moolenbroek 	isc_uint32_t serial = 0;
680*00b67f09SDavid van Moolenbroek 
681*00b67f09SDavid van Moolenbroek 	UNUSED(is_batchfile);
682*00b67f09SDavid van Moolenbroek 
683*00b67f09SDavid van Moolenbroek 	lookup = make_empty_lookup();
684*00b67f09SDavid van Moolenbroek 
685*00b67f09SDavid van Moolenbroek 	lookup->servfail_stops = ISC_FALSE;
686*00b67f09SDavid van Moolenbroek 	lookup->comments = ISC_FALSE;
687*00b67f09SDavid van Moolenbroek 
688*00b67f09SDavid van Moolenbroek 	while ((c = isc_commandline_parse(argc, argv, optstring)) != -1) {
689*00b67f09SDavid van Moolenbroek 		switch (c) {
690*00b67f09SDavid van Moolenbroek 		case 'l':
691*00b67f09SDavid van Moolenbroek 			lookup->tcp_mode = ISC_TRUE;
692*00b67f09SDavid van Moolenbroek 			lookup->rdtype = dns_rdatatype_axfr;
693*00b67f09SDavid van Moolenbroek 			lookup->rdtypeset = ISC_TRUE;
694*00b67f09SDavid van Moolenbroek 			fatalexit = 3;
695*00b67f09SDavid van Moolenbroek 			break;
696*00b67f09SDavid van Moolenbroek 		case 'v':
697*00b67f09SDavid van Moolenbroek 		case 'd':
698*00b67f09SDavid van Moolenbroek 			short_form = ISC_FALSE;
699*00b67f09SDavid van Moolenbroek 			break;
700*00b67f09SDavid van Moolenbroek 		case 'r':
701*00b67f09SDavid van Moolenbroek 			lookup->recurse = ISC_FALSE;
702*00b67f09SDavid van Moolenbroek 			break;
703*00b67f09SDavid van Moolenbroek 		case 't':
704*00b67f09SDavid van Moolenbroek 			if (strncasecmp(isc_commandline_argument,
705*00b67f09SDavid van Moolenbroek 					"ixfr=", 5) == 0) {
706*00b67f09SDavid van Moolenbroek 				rdtype = dns_rdatatype_ixfr;
707*00b67f09SDavid van Moolenbroek 				/* XXXMPA add error checking */
708*00b67f09SDavid van Moolenbroek 				serial = strtoul(isc_commandline_argument + 5,
709*00b67f09SDavid van Moolenbroek 						 NULL, 10);
710*00b67f09SDavid van Moolenbroek 				result = ISC_R_SUCCESS;
711*00b67f09SDavid van Moolenbroek 			} else {
712*00b67f09SDavid van Moolenbroek 				tr.base = isc_commandline_argument;
713*00b67f09SDavid van Moolenbroek 				tr.length = strlen(isc_commandline_argument);
714*00b67f09SDavid van Moolenbroek 				result = dns_rdatatype_fromtext(&rdtype,
715*00b67f09SDavid van Moolenbroek 						   (isc_textregion_t *)&tr);
716*00b67f09SDavid van Moolenbroek 			}
717*00b67f09SDavid van Moolenbroek 
718*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
719*00b67f09SDavid van Moolenbroek 				fatalexit = 2;
720*00b67f09SDavid van Moolenbroek 				fatal("invalid type: %s\n",
721*00b67f09SDavid van Moolenbroek 				      isc_commandline_argument);
722*00b67f09SDavid van Moolenbroek 			}
723*00b67f09SDavid van Moolenbroek 			if (!lookup->rdtypeset ||
724*00b67f09SDavid van Moolenbroek 			    lookup->rdtype != dns_rdatatype_axfr)
725*00b67f09SDavid van Moolenbroek 				lookup->rdtype = rdtype;
726*00b67f09SDavid van Moolenbroek 			lookup->rdtypeset = ISC_TRUE;
727*00b67f09SDavid van Moolenbroek #ifdef WITH_IDN
728*00b67f09SDavid van Moolenbroek 			idnoptions = 0;
729*00b67f09SDavid van Moolenbroek #endif
730*00b67f09SDavid van Moolenbroek 			if (rdtype == dns_rdatatype_axfr) {
731*00b67f09SDavid van Moolenbroek 				/* -l -t any -v */
732*00b67f09SDavid van Moolenbroek 				list_type = dns_rdatatype_any;
733*00b67f09SDavid van Moolenbroek 				short_form = ISC_FALSE;
734*00b67f09SDavid van Moolenbroek 				lookup->tcp_mode = ISC_TRUE;
735*00b67f09SDavid van Moolenbroek 			} else if (rdtype == dns_rdatatype_ixfr) {
736*00b67f09SDavid van Moolenbroek 				lookup->ixfr_serial = serial;
737*00b67f09SDavid van Moolenbroek 				lookup->tcp_mode = ISC_TRUE;
738*00b67f09SDavid van Moolenbroek 				list_type = rdtype;
739*00b67f09SDavid van Moolenbroek #ifdef WITH_IDN
740*00b67f09SDavid van Moolenbroek 			} else if (rdtype == dns_rdatatype_a ||
741*00b67f09SDavid van Moolenbroek 				   rdtype == dns_rdatatype_aaaa ||
742*00b67f09SDavid van Moolenbroek 				   rdtype == dns_rdatatype_mx) {
743*00b67f09SDavid van Moolenbroek 				idnoptions = IDN_ASCCHECK;
744*00b67f09SDavid van Moolenbroek 				list_type = rdtype;
745*00b67f09SDavid van Moolenbroek #endif
746*00b67f09SDavid van Moolenbroek 			} else
747*00b67f09SDavid van Moolenbroek 				list_type = rdtype;
748*00b67f09SDavid van Moolenbroek 			list_addresses = ISC_FALSE;
749*00b67f09SDavid van Moolenbroek 			default_lookups = ISC_FALSE;
750*00b67f09SDavid van Moolenbroek 			break;
751*00b67f09SDavid van Moolenbroek 		case 'c':
752*00b67f09SDavid van Moolenbroek 			tr.base = isc_commandline_argument;
753*00b67f09SDavid van Moolenbroek 			tr.length = strlen(isc_commandline_argument);
754*00b67f09SDavid van Moolenbroek 			result = dns_rdataclass_fromtext(&rdclass,
755*00b67f09SDavid van Moolenbroek 						   (isc_textregion_t *)&tr);
756*00b67f09SDavid van Moolenbroek 
757*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
758*00b67f09SDavid van Moolenbroek 				fatalexit = 2;
759*00b67f09SDavid van Moolenbroek 				fatal("invalid class: %s\n",
760*00b67f09SDavid van Moolenbroek 				      isc_commandline_argument);
761*00b67f09SDavid van Moolenbroek 			} else {
762*00b67f09SDavid van Moolenbroek 				lookup->rdclass = rdclass;
763*00b67f09SDavid van Moolenbroek 				lookup->rdclassset = ISC_TRUE;
764*00b67f09SDavid van Moolenbroek 			}
765*00b67f09SDavid van Moolenbroek 			default_lookups = ISC_FALSE;
766*00b67f09SDavid van Moolenbroek 			break;
767*00b67f09SDavid van Moolenbroek 		case 'a':
768*00b67f09SDavid van Moolenbroek 			if (!lookup->rdtypeset ||
769*00b67f09SDavid van Moolenbroek 			    lookup->rdtype != dns_rdatatype_axfr)
770*00b67f09SDavid van Moolenbroek 				lookup->rdtype = dns_rdatatype_any;
771*00b67f09SDavid van Moolenbroek #ifdef WITH_IDN
772*00b67f09SDavid van Moolenbroek 			idnoptions = 0;
773*00b67f09SDavid van Moolenbroek #endif
774*00b67f09SDavid van Moolenbroek 			list_type = dns_rdatatype_any;
775*00b67f09SDavid van Moolenbroek 			list_addresses = ISC_FALSE;
776*00b67f09SDavid van Moolenbroek 			lookup->rdtypeset = ISC_TRUE;
777*00b67f09SDavid van Moolenbroek 			short_form = ISC_FALSE;
778*00b67f09SDavid van Moolenbroek 			default_lookups = ISC_FALSE;
779*00b67f09SDavid van Moolenbroek 			break;
780*00b67f09SDavid van Moolenbroek 		case 'i':
781*00b67f09SDavid van Moolenbroek 			lookup->ip6_int = ISC_TRUE;
782*00b67f09SDavid van Moolenbroek 			break;
783*00b67f09SDavid van Moolenbroek 		case 'n':
784*00b67f09SDavid van Moolenbroek 			/* deprecated */
785*00b67f09SDavid van Moolenbroek 			break;
786*00b67f09SDavid van Moolenbroek 		case 'm':
787*00b67f09SDavid van Moolenbroek 			/* Handled by pre_parse_args(). */
788*00b67f09SDavid van Moolenbroek 			break;
789*00b67f09SDavid van Moolenbroek 		case 'w':
790*00b67f09SDavid van Moolenbroek 			/*
791*00b67f09SDavid van Moolenbroek 			 * The timer routines are coded such that
792*00b67f09SDavid van Moolenbroek 			 * timeout==MAXINT doesn't enable the timer
793*00b67f09SDavid van Moolenbroek 			 */
794*00b67f09SDavid van Moolenbroek 			timeout = INT_MAX;
795*00b67f09SDavid van Moolenbroek 			break;
796*00b67f09SDavid van Moolenbroek 		case 'W':
797*00b67f09SDavid van Moolenbroek 			timeout = atoi(isc_commandline_argument);
798*00b67f09SDavid van Moolenbroek 			if (timeout < 1)
799*00b67f09SDavid van Moolenbroek 				timeout = 1;
800*00b67f09SDavid van Moolenbroek 			break;
801*00b67f09SDavid van Moolenbroek 		case 'R':
802*00b67f09SDavid van Moolenbroek 			tries = atoi(isc_commandline_argument) + 1;
803*00b67f09SDavid van Moolenbroek 			if (tries < 2)
804*00b67f09SDavid van Moolenbroek 				tries = 2;
805*00b67f09SDavid van Moolenbroek 			break;
806*00b67f09SDavid van Moolenbroek 		case 'T':
807*00b67f09SDavid van Moolenbroek 			lookup->tcp_mode = ISC_TRUE;
808*00b67f09SDavid van Moolenbroek 			break;
809*00b67f09SDavid van Moolenbroek 		case 'C':
810*00b67f09SDavid van Moolenbroek 			debug("showing all SOAs");
811*00b67f09SDavid van Moolenbroek 			lookup->rdtype = dns_rdatatype_ns;
812*00b67f09SDavid van Moolenbroek 			lookup->rdtypeset = ISC_TRUE;
813*00b67f09SDavid van Moolenbroek 			lookup->rdclass = dns_rdataclass_in;
814*00b67f09SDavid van Moolenbroek 			lookup->rdclassset = ISC_TRUE;
815*00b67f09SDavid van Moolenbroek 			lookup->ns_search_only = ISC_TRUE;
816*00b67f09SDavid van Moolenbroek 			lookup->trace_root = ISC_TRUE;
817*00b67f09SDavid van Moolenbroek 			lookup->identify_previous_line = ISC_TRUE;
818*00b67f09SDavid van Moolenbroek 			default_lookups = ISC_FALSE;
819*00b67f09SDavid van Moolenbroek 			break;
820*00b67f09SDavid van Moolenbroek 		case 'N':
821*00b67f09SDavid van Moolenbroek 			debug("setting NDOTS to %s",
822*00b67f09SDavid van Moolenbroek 			      isc_commandline_argument);
823*00b67f09SDavid van Moolenbroek 			ndots = atoi(isc_commandline_argument);
824*00b67f09SDavid van Moolenbroek 			break;
825*00b67f09SDavid van Moolenbroek 		case 'D':
826*00b67f09SDavid van Moolenbroek 			/* Handled by pre_parse_args(). */
827*00b67f09SDavid van Moolenbroek 			break;
828*00b67f09SDavid van Moolenbroek 		case '4':
829*00b67f09SDavid van Moolenbroek 			if (have_ipv4) {
830*00b67f09SDavid van Moolenbroek 				isc_net_disableipv6();
831*00b67f09SDavid van Moolenbroek 				have_ipv6 = ISC_FALSE;
832*00b67f09SDavid van Moolenbroek 			} else
833*00b67f09SDavid van Moolenbroek 				fatal("can't find IPv4 networking");
834*00b67f09SDavid van Moolenbroek 			break;
835*00b67f09SDavid van Moolenbroek 		case '6':
836*00b67f09SDavid van Moolenbroek 			if (have_ipv6) {
837*00b67f09SDavid van Moolenbroek 				isc_net_disableipv4();
838*00b67f09SDavid van Moolenbroek 				have_ipv4 = ISC_FALSE;
839*00b67f09SDavid van Moolenbroek 			} else
840*00b67f09SDavid van Moolenbroek 				fatal("can't find IPv6 networking");
841*00b67f09SDavid van Moolenbroek 			break;
842*00b67f09SDavid van Moolenbroek 		case 's':
843*00b67f09SDavid van Moolenbroek 			lookup->servfail_stops = ISC_TRUE;
844*00b67f09SDavid van Moolenbroek 			break;
845*00b67f09SDavid van Moolenbroek 		}
846*00b67f09SDavid van Moolenbroek 	}
847*00b67f09SDavid van Moolenbroek 
848*00b67f09SDavid van Moolenbroek 	lookup->retries = tries;
849*00b67f09SDavid van Moolenbroek 
850*00b67f09SDavid van Moolenbroek 	if (isc_commandline_index >= argc)
851*00b67f09SDavid van Moolenbroek 		show_usage();
852*00b67f09SDavid van Moolenbroek 
853*00b67f09SDavid van Moolenbroek 	strlcpy(hostname, argv[isc_commandline_index], sizeof(hostname));
854*00b67f09SDavid van Moolenbroek 
855*00b67f09SDavid van Moolenbroek 	if (argc > isc_commandline_index + 1) {
856*00b67f09SDavid van Moolenbroek 		set_nameserver(argv[isc_commandline_index+1]);
857*00b67f09SDavid van Moolenbroek 		debug("server is %s", argv[isc_commandline_index+1]);
858*00b67f09SDavid van Moolenbroek 		listed_server = ISC_TRUE;
859*00b67f09SDavid van Moolenbroek 	} else
860*00b67f09SDavid van Moolenbroek 		check_ra = ISC_TRUE;
861*00b67f09SDavid van Moolenbroek 
862*00b67f09SDavid van Moolenbroek 	lookup->pending = ISC_FALSE;
863*00b67f09SDavid van Moolenbroek 	if (get_reverse(store, sizeof(store), hostname,
864*00b67f09SDavid van Moolenbroek 			lookup->ip6_int, ISC_TRUE) == ISC_R_SUCCESS) {
865*00b67f09SDavid van Moolenbroek 		strncpy(lookup->textname, store, sizeof(lookup->textname));
866*00b67f09SDavid van Moolenbroek 		lookup->textname[sizeof(lookup->textname)-1] = 0;
867*00b67f09SDavid van Moolenbroek 		lookup->rdtype = dns_rdatatype_ptr;
868*00b67f09SDavid van Moolenbroek 		lookup->rdtypeset = ISC_TRUE;
869*00b67f09SDavid van Moolenbroek 		default_lookups = ISC_FALSE;
870*00b67f09SDavid van Moolenbroek 	} else {
871*00b67f09SDavid van Moolenbroek 		strncpy(lookup->textname, hostname, sizeof(lookup->textname));
872*00b67f09SDavid van Moolenbroek 		lookup->textname[sizeof(lookup->textname)-1]=0;
873*00b67f09SDavid van Moolenbroek 		usesearch = ISC_TRUE;
874*00b67f09SDavid van Moolenbroek 	}
875*00b67f09SDavid van Moolenbroek 	lookup->new_search = ISC_TRUE;
876*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(lookup_list, lookup, link);
877*00b67f09SDavid van Moolenbroek }
878*00b67f09SDavid van Moolenbroek 
879*00b67f09SDavid van Moolenbroek int
main(int argc,char ** argv)880*00b67f09SDavid van Moolenbroek main(int argc, char **argv) {
881*00b67f09SDavid van Moolenbroek 	isc_result_t result;
882*00b67f09SDavid van Moolenbroek 
883*00b67f09SDavid van Moolenbroek 	tries = 2;
884*00b67f09SDavid van Moolenbroek 
885*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(lookup_list);
886*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(server_list);
887*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(search_list);
888*00b67f09SDavid van Moolenbroek 
889*00b67f09SDavid van Moolenbroek 	fatalexit = 1;
890*00b67f09SDavid van Moolenbroek #ifdef WITH_IDN
891*00b67f09SDavid van Moolenbroek 	idnoptions = IDN_ASCCHECK;
892*00b67f09SDavid van Moolenbroek #endif
893*00b67f09SDavid van Moolenbroek 
894*00b67f09SDavid van Moolenbroek 	debug("main()");
895*00b67f09SDavid van Moolenbroek 	progname = argv[0];
896*00b67f09SDavid van Moolenbroek 	pre_parse_args(argc, argv);
897*00b67f09SDavid van Moolenbroek 	result = isc_app_start();
898*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_app_start");
899*00b67f09SDavid van Moolenbroek 	setup_libs();
900*00b67f09SDavid van Moolenbroek 	parse_args(ISC_FALSE, argc, argv);
901*00b67f09SDavid van Moolenbroek 	setup_system();
902*00b67f09SDavid van Moolenbroek 	result = isc_app_onrun(mctx, global_task, onrun_callback, NULL);
903*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_app_onrun");
904*00b67f09SDavid van Moolenbroek 	isc_app_run();
905*00b67f09SDavid van Moolenbroek 	cancel_all();
906*00b67f09SDavid van Moolenbroek 	destroy_libs();
907*00b67f09SDavid van Moolenbroek 	isc_app_finish();
908*00b67f09SDavid van Moolenbroek 	return ((seen_error == 0) ? 0 : 1);
909*00b67f09SDavid van Moolenbroek }
910