xref: /minix3/external/bsd/bind/dist/bin/nsupdate/nsupdate.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: nsupdate.c,v 1.13 2015/07/08 17:28:55 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004-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 
24*00b67f09SDavid van Moolenbroek #include <ctype.h>
25*00b67f09SDavid van Moolenbroek #include <errno.h>
26*00b67f09SDavid van Moolenbroek #include <limits.h>
27*00b67f09SDavid van Moolenbroek #include <stdlib.h>
28*00b67f09SDavid van Moolenbroek #include <unistd.h>
29*00b67f09SDavid van Moolenbroek 
30*00b67f09SDavid van Moolenbroek #include <isc/app.h>
31*00b67f09SDavid van Moolenbroek #include <isc/base64.h>
32*00b67f09SDavid van Moolenbroek #include <isc/buffer.h>
33*00b67f09SDavid van Moolenbroek #include <isc/commandline.h>
34*00b67f09SDavid van Moolenbroek #include <isc/entropy.h>
35*00b67f09SDavid van Moolenbroek #include <isc/event.h>
36*00b67f09SDavid van Moolenbroek #include <isc/file.h>
37*00b67f09SDavid van Moolenbroek #include <isc/hash.h>
38*00b67f09SDavid van Moolenbroek #include <isc/lex.h>
39*00b67f09SDavid van Moolenbroek #include <isc/log.h>
40*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
41*00b67f09SDavid van Moolenbroek #include <isc/parseint.h>
42*00b67f09SDavid van Moolenbroek #include <isc/print.h>
43*00b67f09SDavid van Moolenbroek #include <isc/random.h>
44*00b67f09SDavid van Moolenbroek #include <isc/region.h>
45*00b67f09SDavid van Moolenbroek #include <isc/sockaddr.h>
46*00b67f09SDavid van Moolenbroek #include <isc/socket.h>
47*00b67f09SDavid van Moolenbroek #include <isc/stdio.h>
48*00b67f09SDavid van Moolenbroek #include <isc/string.h>
49*00b67f09SDavid van Moolenbroek #include <isc/task.h>
50*00b67f09SDavid van Moolenbroek #include <isc/timer.h>
51*00b67f09SDavid van Moolenbroek #include <isc/types.h>
52*00b67f09SDavid van Moolenbroek #include <isc/util.h>
53*00b67f09SDavid van Moolenbroek 
54*00b67f09SDavid van Moolenbroek #include <isccfg/namedconf.h>
55*00b67f09SDavid van Moolenbroek 
56*00b67f09SDavid van Moolenbroek #include <dns/callbacks.h>
57*00b67f09SDavid van Moolenbroek #include <dns/dispatch.h>
58*00b67f09SDavid van Moolenbroek #include <dns/dnssec.h>
59*00b67f09SDavid van Moolenbroek #include <dns/events.h>
60*00b67f09SDavid van Moolenbroek #include <dns/fixedname.h>
61*00b67f09SDavid van Moolenbroek #include <dns/log.h>
62*00b67f09SDavid van Moolenbroek #include <dns/masterdump.h>
63*00b67f09SDavid van Moolenbroek #include <dns/message.h>
64*00b67f09SDavid van Moolenbroek #include <dns/name.h>
65*00b67f09SDavid van Moolenbroek #include <dns/rcode.h>
66*00b67f09SDavid van Moolenbroek #include <dns/rdata.h>
67*00b67f09SDavid van Moolenbroek #include <dns/rdataclass.h>
68*00b67f09SDavid van Moolenbroek #include <dns/rdatalist.h>
69*00b67f09SDavid van Moolenbroek #include <dns/rdataset.h>
70*00b67f09SDavid van Moolenbroek #include <dns/rdatastruct.h>
71*00b67f09SDavid van Moolenbroek #include <dns/rdatatype.h>
72*00b67f09SDavid van Moolenbroek #include <dns/request.h>
73*00b67f09SDavid van Moolenbroek #include <dns/result.h>
74*00b67f09SDavid van Moolenbroek #include <dns/tkey.h>
75*00b67f09SDavid van Moolenbroek #include <dns/tsig.h>
76*00b67f09SDavid van Moolenbroek 
77*00b67f09SDavid van Moolenbroek #include <dst/dst.h>
78*00b67f09SDavid van Moolenbroek 
79*00b67f09SDavid van Moolenbroek #include <lwres/lwres.h>
80*00b67f09SDavid van Moolenbroek #include <lwres/net.h>
81*00b67f09SDavid van Moolenbroek 
82*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
83*00b67f09SDavid van Moolenbroek #include <dst/gssapi.h>
84*00b67f09SDavid van Moolenbroek #ifdef WIN32
85*00b67f09SDavid van Moolenbroek #include <krb5/krb5.h>
86*00b67f09SDavid van Moolenbroek #else
87*00b67f09SDavid van Moolenbroek #include ISC_PLATFORM_KRB5HEADER
88*00b67f09SDavid van Moolenbroek #endif
89*00b67f09SDavid van Moolenbroek #endif
90*00b67f09SDavid van Moolenbroek #include <bind9/getaddresses.h>
91*00b67f09SDavid van Moolenbroek 
92*00b67f09SDavid van Moolenbroek #if defined(HAVE_READLINE)
93*00b67f09SDavid van Moolenbroek #include <readline/readline.h>
94*00b67f09SDavid van Moolenbroek #include <readline/history.h>
95*00b67f09SDavid van Moolenbroek #endif
96*00b67f09SDavid van Moolenbroek 
97*00b67f09SDavid van Moolenbroek #ifdef HAVE_ADDRINFO
98*00b67f09SDavid van Moolenbroek #ifdef HAVE_GETADDRINFO
99*00b67f09SDavid van Moolenbroek #ifdef HAVE_GAISTRERROR
100*00b67f09SDavid van Moolenbroek #define USE_GETADDRINFO
101*00b67f09SDavid van Moolenbroek #endif
102*00b67f09SDavid van Moolenbroek #endif
103*00b67f09SDavid van Moolenbroek #endif
104*00b67f09SDavid van Moolenbroek 
105*00b67f09SDavid van Moolenbroek #ifndef USE_GETADDRINFO
106*00b67f09SDavid van Moolenbroek #ifndef ISC_PLATFORM_NONSTDHERRNO
107*00b67f09SDavid van Moolenbroek extern int h_errno;
108*00b67f09SDavid van Moolenbroek #endif
109*00b67f09SDavid van Moolenbroek #endif
110*00b67f09SDavid van Moolenbroek 
111*00b67f09SDavid van Moolenbroek #define MAXCMD (128 * 1024)
112*00b67f09SDavid van Moolenbroek #define MAXWIRE (64 * 1024)
113*00b67f09SDavid van Moolenbroek #define PACKETSIZE ((64 * 1024) - 1)
114*00b67f09SDavid van Moolenbroek #define INITTEXT (2 * 1024)
115*00b67f09SDavid van Moolenbroek #define MAXTEXT (128 * 1024)
116*00b67f09SDavid van Moolenbroek #define FIND_TIMEOUT 5
117*00b67f09SDavid van Moolenbroek #define TTL_MAX 2147483647U	/* Maximum signed 32 bit integer. */
118*00b67f09SDavid van Moolenbroek 
119*00b67f09SDavid van Moolenbroek #define DNSDEFAULTPORT 53
120*00b67f09SDavid van Moolenbroek 
121*00b67f09SDavid van Moolenbroek /* Number of addresses to request from bind9_getaddresses() */
122*00b67f09SDavid van Moolenbroek #define MAX_SERVERADDRS 4
123*00b67f09SDavid van Moolenbroek 
124*00b67f09SDavid van Moolenbroek static isc_uint16_t dnsport = DNSDEFAULTPORT;
125*00b67f09SDavid van Moolenbroek 
126*00b67f09SDavid van Moolenbroek #ifndef RESOLV_CONF
127*00b67f09SDavid van Moolenbroek #define RESOLV_CONF "/etc/resolv.conf"
128*00b67f09SDavid van Moolenbroek #endif
129*00b67f09SDavid van Moolenbroek 
130*00b67f09SDavid van Moolenbroek static isc_boolean_t debugging = ISC_FALSE, ddebugging = ISC_FALSE;
131*00b67f09SDavid van Moolenbroek static isc_boolean_t memdebugging = ISC_FALSE;
132*00b67f09SDavid van Moolenbroek static isc_boolean_t have_ipv4 = ISC_FALSE;
133*00b67f09SDavid van Moolenbroek static isc_boolean_t have_ipv6 = ISC_FALSE;
134*00b67f09SDavid van Moolenbroek static isc_boolean_t is_dst_up = ISC_FALSE;
135*00b67f09SDavid van Moolenbroek static isc_boolean_t usevc = ISC_FALSE;
136*00b67f09SDavid van Moolenbroek static isc_boolean_t usegsstsig = ISC_FALSE;
137*00b67f09SDavid van Moolenbroek static isc_boolean_t use_win2k_gsstsig = ISC_FALSE;
138*00b67f09SDavid van Moolenbroek static isc_boolean_t tried_other_gsstsig = ISC_FALSE;
139*00b67f09SDavid van Moolenbroek static isc_boolean_t local_only = ISC_FALSE;
140*00b67f09SDavid van Moolenbroek static isc_taskmgr_t *taskmgr = NULL;
141*00b67f09SDavid van Moolenbroek static isc_task_t *global_task = NULL;
142*00b67f09SDavid van Moolenbroek static isc_event_t *global_event = NULL;
143*00b67f09SDavid van Moolenbroek static isc_log_t *glctx = NULL;
144*00b67f09SDavid van Moolenbroek static isc_mem_t *gmctx = NULL;
145*00b67f09SDavid van Moolenbroek static dns_dispatchmgr_t *dispatchmgr = NULL;
146*00b67f09SDavid van Moolenbroek static dns_requestmgr_t *requestmgr = NULL;
147*00b67f09SDavid van Moolenbroek static isc_socketmgr_t *socketmgr = NULL;
148*00b67f09SDavid van Moolenbroek static isc_timermgr_t *timermgr = NULL;
149*00b67f09SDavid van Moolenbroek static dns_dispatch_t *dispatchv4 = NULL;
150*00b67f09SDavid van Moolenbroek static dns_dispatch_t *dispatchv6 = NULL;
151*00b67f09SDavid van Moolenbroek static dns_message_t *updatemsg = NULL;
152*00b67f09SDavid van Moolenbroek static dns_fixedname_t fuserzone;
153*00b67f09SDavid van Moolenbroek static dns_name_t *userzone = NULL;
154*00b67f09SDavid van Moolenbroek static dns_name_t *zname = NULL;
155*00b67f09SDavid van Moolenbroek static dns_name_t tmpzonename;
156*00b67f09SDavid van Moolenbroek static dns_name_t restart_master;
157*00b67f09SDavid van Moolenbroek static dns_tsig_keyring_t *gssring = NULL;
158*00b67f09SDavid van Moolenbroek static dns_tsigkey_t *tsigkey = NULL;
159*00b67f09SDavid van Moolenbroek static dst_key_t *sig0key = NULL;
160*00b67f09SDavid van Moolenbroek static lwres_context_t *lwctx = NULL;
161*00b67f09SDavid van Moolenbroek static lwres_conf_t *lwconf;
162*00b67f09SDavid van Moolenbroek static isc_sockaddr_t *servers = NULL;
163*00b67f09SDavid van Moolenbroek static isc_sockaddr_t *master_servers = NULL;
164*00b67f09SDavid van Moolenbroek static isc_boolean_t default_servers = ISC_TRUE;
165*00b67f09SDavid van Moolenbroek static int ns_inuse = 0;
166*00b67f09SDavid van Moolenbroek static int master_inuse = 0;
167*00b67f09SDavid van Moolenbroek static int ns_total = 0;
168*00b67f09SDavid van Moolenbroek static int master_total = 0;
169*00b67f09SDavid van Moolenbroek static isc_sockaddr_t *localaddr4 = NULL;
170*00b67f09SDavid van Moolenbroek static isc_sockaddr_t *localaddr6 = NULL;
171*00b67f09SDavid van Moolenbroek static const char *keyfile = NULL;
172*00b67f09SDavid van Moolenbroek static char *keystr = NULL;
173*00b67f09SDavid van Moolenbroek static isc_entropy_t *entropy = NULL;
174*00b67f09SDavid van Moolenbroek static isc_boolean_t shuttingdown = ISC_FALSE;
175*00b67f09SDavid van Moolenbroek static FILE *input;
176*00b67f09SDavid van Moolenbroek static isc_boolean_t interactive = ISC_TRUE;
177*00b67f09SDavid van Moolenbroek static isc_boolean_t seenerror = ISC_FALSE;
178*00b67f09SDavid van Moolenbroek static const dns_master_style_t *style;
179*00b67f09SDavid van Moolenbroek static int requests = 0;
180*00b67f09SDavid van Moolenbroek static unsigned int logdebuglevel = 0;
181*00b67f09SDavid van Moolenbroek static unsigned int timeout = 300;
182*00b67f09SDavid van Moolenbroek static unsigned int udp_timeout = 3;
183*00b67f09SDavid van Moolenbroek static unsigned int udp_retries = 3;
184*00b67f09SDavid van Moolenbroek static dns_rdataclass_t defaultclass = dns_rdataclass_in;
185*00b67f09SDavid van Moolenbroek static dns_rdataclass_t zoneclass = dns_rdataclass_none;
186*00b67f09SDavid van Moolenbroek static dns_message_t *answer = NULL;
187*00b67f09SDavid van Moolenbroek static isc_uint32_t default_ttl = 0;
188*00b67f09SDavid van Moolenbroek static isc_boolean_t default_ttl_set = ISC_FALSE;
189*00b67f09SDavid van Moolenbroek 
190*00b67f09SDavid van Moolenbroek typedef struct nsu_requestinfo {
191*00b67f09SDavid van Moolenbroek 	dns_message_t *msg;
192*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *addr;
193*00b67f09SDavid van Moolenbroek } nsu_requestinfo_t;
194*00b67f09SDavid van Moolenbroek 
195*00b67f09SDavid van Moolenbroek static void
196*00b67f09SDavid van Moolenbroek sendrequest(isc_sockaddr_t *destaddr, dns_message_t *msg,
197*00b67f09SDavid van Moolenbroek 	    dns_request_t **request);
198*00b67f09SDavid van Moolenbroek static void
199*00b67f09SDavid van Moolenbroek send_update(dns_name_t *zonename, isc_sockaddr_t *master);
200*00b67f09SDavid van Moolenbroek 
201*00b67f09SDavid van Moolenbroek ISC_PLATFORM_NORETURN_PRE static void
202*00b67f09SDavid van Moolenbroek fatal(const char *format, ...)
203*00b67f09SDavid van Moolenbroek ISC_FORMAT_PRINTF(1, 2) ISC_PLATFORM_NORETURN_POST;
204*00b67f09SDavid van Moolenbroek 
205*00b67f09SDavid van Moolenbroek static void
206*00b67f09SDavid van Moolenbroek debug(const char *format, ...) ISC_FORMAT_PRINTF(1, 2);
207*00b67f09SDavid van Moolenbroek 
208*00b67f09SDavid van Moolenbroek static void
209*00b67f09SDavid van Moolenbroek ddebug(const char *format, ...) ISC_FORMAT_PRINTF(1, 2);
210*00b67f09SDavid van Moolenbroek 
211*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
212*00b67f09SDavid van Moolenbroek static dns_fixedname_t fkname;
213*00b67f09SDavid van Moolenbroek static isc_sockaddr_t *kserver = NULL;
214*00b67f09SDavid van Moolenbroek static char *realm = NULL;
215*00b67f09SDavid van Moolenbroek static char servicename[DNS_NAME_FORMATSIZE];
216*00b67f09SDavid van Moolenbroek static dns_name_t *keyname;
217*00b67f09SDavid van Moolenbroek typedef struct nsu_gssinfo {
218*00b67f09SDavid van Moolenbroek 	dns_message_t *msg;
219*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *addr;
220*00b67f09SDavid van Moolenbroek 	gss_ctx_id_t context;
221*00b67f09SDavid van Moolenbroek } nsu_gssinfo_t;
222*00b67f09SDavid van Moolenbroek 
223*00b67f09SDavid van Moolenbroek static void
224*00b67f09SDavid van Moolenbroek start_gssrequest(dns_name_t *master);
225*00b67f09SDavid van Moolenbroek static void
226*00b67f09SDavid van Moolenbroek send_gssrequest(isc_sockaddr_t *destaddr, dns_message_t *msg,
227*00b67f09SDavid van Moolenbroek 		dns_request_t **request, gss_ctx_id_t context);
228*00b67f09SDavid van Moolenbroek static void
229*00b67f09SDavid van Moolenbroek recvgss(isc_task_t *task, isc_event_t *event);
230*00b67f09SDavid van Moolenbroek #endif /* GSSAPI */
231*00b67f09SDavid van Moolenbroek 
232*00b67f09SDavid van Moolenbroek static void
233*00b67f09SDavid van Moolenbroek error(const char *format, ...) ISC_FORMAT_PRINTF(1, 2);
234*00b67f09SDavid van Moolenbroek 
235*00b67f09SDavid van Moolenbroek #define STATUS_MORE	(isc_uint16_t)0
236*00b67f09SDavid van Moolenbroek #define STATUS_SEND	(isc_uint16_t)1
237*00b67f09SDavid van Moolenbroek #define STATUS_QUIT	(isc_uint16_t)2
238*00b67f09SDavid van Moolenbroek #define STATUS_SYNTAX	(isc_uint16_t)3
239*00b67f09SDavid van Moolenbroek 
240*00b67f09SDavid van Moolenbroek typedef struct entropysource entropysource_t;
241*00b67f09SDavid van Moolenbroek 
242*00b67f09SDavid van Moolenbroek struct entropysource {
243*00b67f09SDavid van Moolenbroek 	isc_entropysource_t *source;
244*00b67f09SDavid van Moolenbroek 	isc_mem_t *mctx;
245*00b67f09SDavid van Moolenbroek 	ISC_LINK(entropysource_t) link;
246*00b67f09SDavid van Moolenbroek };
247*00b67f09SDavid van Moolenbroek 
248*00b67f09SDavid van Moolenbroek static ISC_LIST(entropysource_t) sources;
249*00b67f09SDavid van Moolenbroek 
250*00b67f09SDavid van Moolenbroek static void
setup_entropy(isc_mem_t * mctx,const char * randomfile,isc_entropy_t ** ectx)251*00b67f09SDavid van Moolenbroek setup_entropy(isc_mem_t *mctx, const char *randomfile, isc_entropy_t **ectx) {
252*00b67f09SDavid van Moolenbroek 	isc_result_t result;
253*00b67f09SDavid van Moolenbroek 	isc_entropysource_t *source = NULL;
254*00b67f09SDavid van Moolenbroek 	entropysource_t *elt;
255*00b67f09SDavid van Moolenbroek 	int usekeyboard = ISC_ENTROPY_KEYBOARDMAYBE;
256*00b67f09SDavid van Moolenbroek 
257*00b67f09SDavid van Moolenbroek 	REQUIRE(ectx != NULL);
258*00b67f09SDavid van Moolenbroek 
259*00b67f09SDavid van Moolenbroek 	if (*ectx == NULL) {
260*00b67f09SDavid van Moolenbroek 		result = isc_entropy_create(mctx, ectx);
261*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
262*00b67f09SDavid van Moolenbroek 			fatal("could not create entropy object");
263*00b67f09SDavid van Moolenbroek 		ISC_LIST_INIT(sources);
264*00b67f09SDavid van Moolenbroek 	}
265*00b67f09SDavid van Moolenbroek 
266*00b67f09SDavid van Moolenbroek 	if (randomfile != NULL && strcmp(randomfile, "keyboard") == 0) {
267*00b67f09SDavid van Moolenbroek 		usekeyboard = ISC_ENTROPY_KEYBOARDYES;
268*00b67f09SDavid van Moolenbroek 		randomfile = NULL;
269*00b67f09SDavid van Moolenbroek 	}
270*00b67f09SDavid van Moolenbroek 
271*00b67f09SDavid van Moolenbroek 	result = isc_entropy_usebestsource(*ectx, &source, randomfile,
272*00b67f09SDavid van Moolenbroek 					   usekeyboard);
273*00b67f09SDavid van Moolenbroek 
274*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
275*00b67f09SDavid van Moolenbroek 		fatal("could not initialize entropy source: %s",
276*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
277*00b67f09SDavid van Moolenbroek 
278*00b67f09SDavid van Moolenbroek 	if (source != NULL) {
279*00b67f09SDavid van Moolenbroek 		elt = isc_mem_get(mctx, sizeof(*elt));
280*00b67f09SDavid van Moolenbroek 		if (elt == NULL)
281*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
282*00b67f09SDavid van Moolenbroek 		elt->source = source;
283*00b67f09SDavid van Moolenbroek 		elt->mctx = mctx;
284*00b67f09SDavid van Moolenbroek 		ISC_LINK_INIT(elt, link);
285*00b67f09SDavid van Moolenbroek 		ISC_LIST_APPEND(sources, elt, link);
286*00b67f09SDavid van Moolenbroek 	}
287*00b67f09SDavid van Moolenbroek }
288*00b67f09SDavid van Moolenbroek 
289*00b67f09SDavid van Moolenbroek static void
cleanup_entropy(isc_entropy_t ** ectx)290*00b67f09SDavid van Moolenbroek cleanup_entropy(isc_entropy_t **ectx) {
291*00b67f09SDavid van Moolenbroek 	entropysource_t *source;
292*00b67f09SDavid van Moolenbroek 	while (!ISC_LIST_EMPTY(sources)) {
293*00b67f09SDavid van Moolenbroek 		source = ISC_LIST_HEAD(sources);
294*00b67f09SDavid van Moolenbroek 		ISC_LIST_UNLINK(sources, source, link);
295*00b67f09SDavid van Moolenbroek 		isc_entropy_destroysource(&source->source);
296*00b67f09SDavid van Moolenbroek 		isc_mem_put(source->mctx, source, sizeof(*source));
297*00b67f09SDavid van Moolenbroek 	}
298*00b67f09SDavid van Moolenbroek 	isc_entropy_detach(ectx);
299*00b67f09SDavid van Moolenbroek }
300*00b67f09SDavid van Moolenbroek 
301*00b67f09SDavid van Moolenbroek static void
master_from_servers(void)302*00b67f09SDavid van Moolenbroek master_from_servers(void) {
303*00b67f09SDavid van Moolenbroek 
304*00b67f09SDavid van Moolenbroek 	if (master_servers != NULL && master_servers != servers)
305*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, master_servers,
306*00b67f09SDavid van Moolenbroek 			    master_total * sizeof(isc_sockaddr_t));
307*00b67f09SDavid van Moolenbroek 	master_servers = servers;
308*00b67f09SDavid van Moolenbroek 	master_total = ns_total;
309*00b67f09SDavid van Moolenbroek 	master_inuse = ns_inuse;
310*00b67f09SDavid van Moolenbroek }
311*00b67f09SDavid van Moolenbroek 
312*00b67f09SDavid van Moolenbroek static dns_rdataclass_t
getzoneclass(void)313*00b67f09SDavid van Moolenbroek getzoneclass(void) {
314*00b67f09SDavid van Moolenbroek 	if (zoneclass == dns_rdataclass_none)
315*00b67f09SDavid van Moolenbroek 		zoneclass = defaultclass;
316*00b67f09SDavid van Moolenbroek 	return (zoneclass);
317*00b67f09SDavid van Moolenbroek }
318*00b67f09SDavid van Moolenbroek 
319*00b67f09SDavid van Moolenbroek static isc_boolean_t
setzoneclass(dns_rdataclass_t rdclass)320*00b67f09SDavid van Moolenbroek setzoneclass(dns_rdataclass_t rdclass) {
321*00b67f09SDavid van Moolenbroek 	if (zoneclass == dns_rdataclass_none ||
322*00b67f09SDavid van Moolenbroek 	    rdclass == dns_rdataclass_none)
323*00b67f09SDavid van Moolenbroek 		zoneclass = rdclass;
324*00b67f09SDavid van Moolenbroek 	if (zoneclass != rdclass)
325*00b67f09SDavid van Moolenbroek 		return (ISC_FALSE);
326*00b67f09SDavid van Moolenbroek 	return (ISC_TRUE);
327*00b67f09SDavid van Moolenbroek }
328*00b67f09SDavid van Moolenbroek 
329*00b67f09SDavid van Moolenbroek static void
fatal(const char * format,...)330*00b67f09SDavid van Moolenbroek fatal(const char *format, ...) {
331*00b67f09SDavid van Moolenbroek 	va_list args;
332*00b67f09SDavid van Moolenbroek 
333*00b67f09SDavid van Moolenbroek 	va_start(args, format);
334*00b67f09SDavid van Moolenbroek 	vfprintf(stderr, format, args);
335*00b67f09SDavid van Moolenbroek 	va_end(args);
336*00b67f09SDavid van Moolenbroek 	fprintf(stderr, "\n");
337*00b67f09SDavid van Moolenbroek 	exit(1);
338*00b67f09SDavid van Moolenbroek }
339*00b67f09SDavid van Moolenbroek 
340*00b67f09SDavid van Moolenbroek static void
error(const char * format,...)341*00b67f09SDavid van Moolenbroek error(const char *format, ...) {
342*00b67f09SDavid van Moolenbroek 	va_list args;
343*00b67f09SDavid van Moolenbroek 
344*00b67f09SDavid van Moolenbroek 	va_start(args, format);
345*00b67f09SDavid van Moolenbroek 	vfprintf(stderr, format, args);
346*00b67f09SDavid van Moolenbroek 	va_end(args);
347*00b67f09SDavid van Moolenbroek 	fprintf(stderr, "\n");
348*00b67f09SDavid van Moolenbroek }
349*00b67f09SDavid van Moolenbroek 
350*00b67f09SDavid van Moolenbroek static void
debug(const char * format,...)351*00b67f09SDavid van Moolenbroek debug(const char *format, ...) {
352*00b67f09SDavid van Moolenbroek 	va_list args;
353*00b67f09SDavid van Moolenbroek 
354*00b67f09SDavid van Moolenbroek 	if (debugging) {
355*00b67f09SDavid van Moolenbroek 		va_start(args, format);
356*00b67f09SDavid van Moolenbroek 		vfprintf(stderr, format, args);
357*00b67f09SDavid van Moolenbroek 		va_end(args);
358*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "\n");
359*00b67f09SDavid van Moolenbroek 	}
360*00b67f09SDavid van Moolenbroek }
361*00b67f09SDavid van Moolenbroek 
362*00b67f09SDavid van Moolenbroek static void
ddebug(const char * format,...)363*00b67f09SDavid van Moolenbroek ddebug(const char *format, ...) {
364*00b67f09SDavid van Moolenbroek 	va_list args;
365*00b67f09SDavid van Moolenbroek 
366*00b67f09SDavid van Moolenbroek 	if (ddebugging) {
367*00b67f09SDavid van Moolenbroek 		va_start(args, format);
368*00b67f09SDavid van Moolenbroek 		vfprintf(stderr, format, args);
369*00b67f09SDavid van Moolenbroek 		va_end(args);
370*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "\n");
371*00b67f09SDavid van Moolenbroek 	}
372*00b67f09SDavid van Moolenbroek }
373*00b67f09SDavid van Moolenbroek 
374*00b67f09SDavid van Moolenbroek static inline void
check_result(isc_result_t result,const char * msg)375*00b67f09SDavid van Moolenbroek check_result(isc_result_t result, const char *msg) {
376*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
377*00b67f09SDavid van Moolenbroek 		fatal("%s: %s", msg, isc_result_totext(result));
378*00b67f09SDavid van Moolenbroek }
379*00b67f09SDavid van Moolenbroek 
380*00b67f09SDavid van Moolenbroek static void *
mem_alloc(void * arg,size_t size)381*00b67f09SDavid van Moolenbroek mem_alloc(void *arg, size_t size) {
382*00b67f09SDavid van Moolenbroek 	return (isc_mem_get(arg, size));
383*00b67f09SDavid van Moolenbroek }
384*00b67f09SDavid van Moolenbroek 
385*00b67f09SDavid van Moolenbroek static void
mem_free(void * arg,void * mem,size_t size)386*00b67f09SDavid van Moolenbroek mem_free(void *arg, void *mem, size_t size) {
387*00b67f09SDavid van Moolenbroek 	isc_mem_put(arg, mem, size);
388*00b67f09SDavid van Moolenbroek }
389*00b67f09SDavid van Moolenbroek 
390*00b67f09SDavid van Moolenbroek static char *
nsu_strsep(char ** stringp,const char * delim)391*00b67f09SDavid van Moolenbroek nsu_strsep(char **stringp, const char *delim) {
392*00b67f09SDavid van Moolenbroek 	char *string = *stringp;
393*00b67f09SDavid van Moolenbroek 	char *s;
394*00b67f09SDavid van Moolenbroek 	const char *d;
395*00b67f09SDavid van Moolenbroek 	char sc, dc;
396*00b67f09SDavid van Moolenbroek 
397*00b67f09SDavid van Moolenbroek 	if (string == NULL)
398*00b67f09SDavid van Moolenbroek 		return (NULL);
399*00b67f09SDavid van Moolenbroek 
400*00b67f09SDavid van Moolenbroek 	for (; *string != '\0'; string++) {
401*00b67f09SDavid van Moolenbroek 		sc = *string;
402*00b67f09SDavid van Moolenbroek 		for (d = delim; (dc = *d) != '\0'; d++) {
403*00b67f09SDavid van Moolenbroek 			if (sc == dc)
404*00b67f09SDavid van Moolenbroek 				break;
405*00b67f09SDavid van Moolenbroek 		}
406*00b67f09SDavid van Moolenbroek 		if (dc == 0)
407*00b67f09SDavid van Moolenbroek 			break;
408*00b67f09SDavid van Moolenbroek 	}
409*00b67f09SDavid van Moolenbroek 
410*00b67f09SDavid van Moolenbroek 	for (s = string; *s != '\0'; s++) {
411*00b67f09SDavid van Moolenbroek 		sc = *s;
412*00b67f09SDavid van Moolenbroek 		for (d = delim; (dc = *d) != '\0'; d++) {
413*00b67f09SDavid van Moolenbroek 			if (sc == dc) {
414*00b67f09SDavid van Moolenbroek 				*s++ = '\0';
415*00b67f09SDavid van Moolenbroek 				*stringp = s;
416*00b67f09SDavid van Moolenbroek 				return (string);
417*00b67f09SDavid van Moolenbroek 			}
418*00b67f09SDavid van Moolenbroek 		}
419*00b67f09SDavid van Moolenbroek 	}
420*00b67f09SDavid van Moolenbroek 	*stringp = NULL;
421*00b67f09SDavid van Moolenbroek 	return (string);
422*00b67f09SDavid van Moolenbroek }
423*00b67f09SDavid van Moolenbroek 
424*00b67f09SDavid van Moolenbroek static void
reset_system(void)425*00b67f09SDavid van Moolenbroek reset_system(void) {
426*00b67f09SDavid van Moolenbroek 	isc_result_t result;
427*00b67f09SDavid van Moolenbroek 
428*00b67f09SDavid van Moolenbroek 	ddebug("reset_system()");
429*00b67f09SDavid van Moolenbroek 	/* If the update message is still around, destroy it */
430*00b67f09SDavid van Moolenbroek 	if (updatemsg != NULL)
431*00b67f09SDavid van Moolenbroek 		dns_message_reset(updatemsg, DNS_MESSAGE_INTENTRENDER);
432*00b67f09SDavid van Moolenbroek 	else {
433*00b67f09SDavid van Moolenbroek 		result = dns_message_create(gmctx, DNS_MESSAGE_INTENTRENDER,
434*00b67f09SDavid van Moolenbroek 					    &updatemsg);
435*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_message_create");
436*00b67f09SDavid van Moolenbroek 	}
437*00b67f09SDavid van Moolenbroek 	updatemsg->opcode = dns_opcode_update;
438*00b67f09SDavid van Moolenbroek 	if (usegsstsig) {
439*00b67f09SDavid van Moolenbroek 		if (tsigkey != NULL)
440*00b67f09SDavid van Moolenbroek 			dns_tsigkey_detach(&tsigkey);
441*00b67f09SDavid van Moolenbroek 		if (gssring != NULL)
442*00b67f09SDavid van Moolenbroek 			dns_tsigkeyring_detach(&gssring);
443*00b67f09SDavid van Moolenbroek 		tried_other_gsstsig = ISC_FALSE;
444*00b67f09SDavid van Moolenbroek 	}
445*00b67f09SDavid van Moolenbroek }
446*00b67f09SDavid van Moolenbroek 
447*00b67f09SDavid van Moolenbroek static isc_uint16_t
parse_hmac(dns_name_t ** hmac,const char * hmacstr,size_t len)448*00b67f09SDavid van Moolenbroek parse_hmac(dns_name_t **hmac, const char *hmacstr, size_t len) {
449*00b67f09SDavid van Moolenbroek 	isc_uint16_t digestbits = 0;
450*00b67f09SDavid van Moolenbroek 	isc_result_t result;
451*00b67f09SDavid van Moolenbroek 	char buf[20];
452*00b67f09SDavid van Moolenbroek 
453*00b67f09SDavid van Moolenbroek 	REQUIRE(hmac != NULL && *hmac == NULL);
454*00b67f09SDavid van Moolenbroek 	REQUIRE(hmacstr != NULL);
455*00b67f09SDavid van Moolenbroek 
456*00b67f09SDavid van Moolenbroek 	if (len >= sizeof(buf))
457*00b67f09SDavid van Moolenbroek 		fatal("unknown key type '%.*s'", (int)(len), hmacstr);
458*00b67f09SDavid van Moolenbroek 
459*00b67f09SDavid van Moolenbroek 	strncpy(buf, hmacstr, len);
460*00b67f09SDavid van Moolenbroek 	buf[len] = 0;
461*00b67f09SDavid van Moolenbroek 
462*00b67f09SDavid van Moolenbroek 	if (strcasecmp(buf, "hmac-md5") == 0) {
463*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACMD5_NAME;
464*00b67f09SDavid van Moolenbroek 	} else if (strncasecmp(buf, "hmac-md5-", 9) == 0) {
465*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACMD5_NAME;
466*00b67f09SDavid van Moolenbroek 		result = isc_parse_uint16(&digestbits, &buf[9], 10);
467*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS || digestbits > 128)
468*00b67f09SDavid van Moolenbroek 			fatal("digest-bits out of range [0..128]");
469*00b67f09SDavid van Moolenbroek 		digestbits = (digestbits +7) & ~0x7U;
470*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(buf, "hmac-sha1") == 0) {
471*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA1_NAME;
472*00b67f09SDavid van Moolenbroek 	} else if (strncasecmp(buf, "hmac-sha1-", 10) == 0) {
473*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA1_NAME;
474*00b67f09SDavid van Moolenbroek 		result = isc_parse_uint16(&digestbits, &buf[10], 10);
475*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS || digestbits > 160)
476*00b67f09SDavid van Moolenbroek 			fatal("digest-bits out of range [0..160]");
477*00b67f09SDavid van Moolenbroek 		digestbits = (digestbits +7) & ~0x7U;
478*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(buf, "hmac-sha224") == 0) {
479*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA224_NAME;
480*00b67f09SDavid van Moolenbroek 	} else if (strncasecmp(buf, "hmac-sha224-", 12) == 0) {
481*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA224_NAME;
482*00b67f09SDavid van Moolenbroek 		result = isc_parse_uint16(&digestbits, &buf[12], 10);
483*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS || digestbits > 224)
484*00b67f09SDavid van Moolenbroek 			fatal("digest-bits out of range [0..224]");
485*00b67f09SDavid van Moolenbroek 		digestbits = (digestbits +7) & ~0x7U;
486*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(buf, "hmac-sha256") == 0) {
487*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA256_NAME;
488*00b67f09SDavid van Moolenbroek 	} else if (strncasecmp(buf, "hmac-sha256-", 12) == 0) {
489*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA256_NAME;
490*00b67f09SDavid van Moolenbroek 		result = isc_parse_uint16(&digestbits, &buf[12], 10);
491*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS || digestbits > 256)
492*00b67f09SDavid van Moolenbroek 			fatal("digest-bits out of range [0..256]");
493*00b67f09SDavid van Moolenbroek 		digestbits = (digestbits +7) & ~0x7U;
494*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(buf, "hmac-sha384") == 0) {
495*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA384_NAME;
496*00b67f09SDavid van Moolenbroek 	} else if (strncasecmp(buf, "hmac-sha384-", 12) == 0) {
497*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA384_NAME;
498*00b67f09SDavid van Moolenbroek 		result = isc_parse_uint16(&digestbits, &buf[12], 10);
499*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS || digestbits > 384)
500*00b67f09SDavid van Moolenbroek 			fatal("digest-bits out of range [0..384]");
501*00b67f09SDavid van Moolenbroek 		digestbits = (digestbits +7) & ~0x7U;
502*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(buf, "hmac-sha512") == 0) {
503*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA512_NAME;
504*00b67f09SDavid van Moolenbroek 	} else if (strncasecmp(buf, "hmac-sha512-", 12) == 0) {
505*00b67f09SDavid van Moolenbroek 		*hmac = DNS_TSIG_HMACSHA512_NAME;
506*00b67f09SDavid van Moolenbroek 		result = isc_parse_uint16(&digestbits, &buf[12], 10);
507*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS || digestbits > 512)
508*00b67f09SDavid van Moolenbroek 			fatal("digest-bits out of range [0..512]");
509*00b67f09SDavid van Moolenbroek 		digestbits = (digestbits +7) & ~0x7U;
510*00b67f09SDavid van Moolenbroek 	} else
511*00b67f09SDavid van Moolenbroek 		fatal("unknown key type '%s'", buf);
512*00b67f09SDavid van Moolenbroek 	return (digestbits);
513*00b67f09SDavid van Moolenbroek }
514*00b67f09SDavid van Moolenbroek 
515*00b67f09SDavid van Moolenbroek static int
basenamelen(const char * file)516*00b67f09SDavid van Moolenbroek basenamelen(const char *file) {
517*00b67f09SDavid van Moolenbroek 	int len = strlen(file);
518*00b67f09SDavid van Moolenbroek 
519*00b67f09SDavid van Moolenbroek 	if (len > 1 && file[len - 1] == '.')
520*00b67f09SDavid van Moolenbroek 		len -= 1;
521*00b67f09SDavid van Moolenbroek 	else if (len > 8 && strcmp(file + len - 8, ".private") == 0)
522*00b67f09SDavid van Moolenbroek 		len -= 8;
523*00b67f09SDavid van Moolenbroek 	else if (len > 4 && strcmp(file + len - 4, ".key") == 0)
524*00b67f09SDavid van Moolenbroek 		len -= 4;
525*00b67f09SDavid van Moolenbroek 	return (len);
526*00b67f09SDavid van Moolenbroek }
527*00b67f09SDavid van Moolenbroek 
528*00b67f09SDavid van Moolenbroek static void
setup_keystr(void)529*00b67f09SDavid van Moolenbroek setup_keystr(void) {
530*00b67f09SDavid van Moolenbroek 	unsigned char *secret = NULL;
531*00b67f09SDavid van Moolenbroek 	int secretlen;
532*00b67f09SDavid van Moolenbroek 	isc_buffer_t secretbuf;
533*00b67f09SDavid van Moolenbroek 	isc_result_t result;
534*00b67f09SDavid van Moolenbroek 	isc_buffer_t keynamesrc;
535*00b67f09SDavid van Moolenbroek 	char *secretstr;
536*00b67f09SDavid van Moolenbroek 	char *s, *n;
537*00b67f09SDavid van Moolenbroek 	dns_fixedname_t fkeyname;
538*00b67f09SDavid van Moolenbroek 	dns_name_t *mykeyname;
539*00b67f09SDavid van Moolenbroek 	char *name;
540*00b67f09SDavid van Moolenbroek 	dns_name_t *hmacname = NULL;
541*00b67f09SDavid van Moolenbroek 	isc_uint16_t digestbits = 0;
542*00b67f09SDavid van Moolenbroek 
543*00b67f09SDavid van Moolenbroek 	dns_fixedname_init(&fkeyname);
544*00b67f09SDavid van Moolenbroek 	mykeyname = dns_fixedname_name(&fkeyname);
545*00b67f09SDavid van Moolenbroek 
546*00b67f09SDavid van Moolenbroek 	debug("Creating key...");
547*00b67f09SDavid van Moolenbroek 
548*00b67f09SDavid van Moolenbroek 	s = strchr(keystr, ':');
549*00b67f09SDavid van Moolenbroek 	if (s == NULL || s == keystr || s[1] == 0)
550*00b67f09SDavid van Moolenbroek 		fatal("key option must specify [hmac:]keyname:secret");
551*00b67f09SDavid van Moolenbroek 	secretstr = s + 1;
552*00b67f09SDavid van Moolenbroek 	n = strchr(secretstr, ':');
553*00b67f09SDavid van Moolenbroek 	if (n != NULL) {
554*00b67f09SDavid van Moolenbroek 		if (n == secretstr || n[1] == 0)
555*00b67f09SDavid van Moolenbroek 			fatal("key option must specify [hmac:]keyname:secret");
556*00b67f09SDavid van Moolenbroek 		name = secretstr;
557*00b67f09SDavid van Moolenbroek 		secretstr = n + 1;
558*00b67f09SDavid van Moolenbroek 		digestbits = parse_hmac(&hmacname, keystr, s - keystr);
559*00b67f09SDavid van Moolenbroek 	} else {
560*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACMD5_NAME;
561*00b67f09SDavid van Moolenbroek 		name = keystr;
562*00b67f09SDavid van Moolenbroek 		n = s;
563*00b67f09SDavid van Moolenbroek 	}
564*00b67f09SDavid van Moolenbroek 
565*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&keynamesrc, name, (unsigned int)(n - name));
566*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&keynamesrc, (unsigned int)(n - name));
567*00b67f09SDavid van Moolenbroek 
568*00b67f09SDavid van Moolenbroek 	debug("namefromtext");
569*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(mykeyname, &keynamesrc, dns_rootname, 0,
570*00b67f09SDavid van Moolenbroek 				   NULL);
571*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_name_fromtext");
572*00b67f09SDavid van Moolenbroek 
573*00b67f09SDavid van Moolenbroek 	secretlen = strlen(secretstr) * 3 / 4;
574*00b67f09SDavid van Moolenbroek 	secret = isc_mem_allocate(gmctx, secretlen);
575*00b67f09SDavid van Moolenbroek 	if (secret == NULL)
576*00b67f09SDavid van Moolenbroek 		fatal("out of memory");
577*00b67f09SDavid van Moolenbroek 
578*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&secretbuf, secret, secretlen);
579*00b67f09SDavid van Moolenbroek 	result = isc_base64_decodestring(secretstr, &secretbuf);
580*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
581*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not create key from %s: %s\n",
582*00b67f09SDavid van Moolenbroek 			keystr, isc_result_totext(result));
583*00b67f09SDavid van Moolenbroek 		goto failure;
584*00b67f09SDavid van Moolenbroek 	}
585*00b67f09SDavid van Moolenbroek 
586*00b67f09SDavid van Moolenbroek 	secretlen = isc_buffer_usedlength(&secretbuf);
587*00b67f09SDavid van Moolenbroek 
588*00b67f09SDavid van Moolenbroek 	debug("keycreate");
589*00b67f09SDavid van Moolenbroek 	result = dns_tsigkey_create(mykeyname, hmacname, secret, secretlen,
590*00b67f09SDavid van Moolenbroek 				    ISC_FALSE, NULL, 0, 0, gmctx, NULL,
591*00b67f09SDavid van Moolenbroek 				    &tsigkey);
592*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
593*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not create key from %s: %s\n",
594*00b67f09SDavid van Moolenbroek 			keystr, dns_result_totext(result));
595*00b67f09SDavid van Moolenbroek 	else
596*00b67f09SDavid van Moolenbroek 		dst_key_setbits(tsigkey->key, digestbits);
597*00b67f09SDavid van Moolenbroek  failure:
598*00b67f09SDavid van Moolenbroek 	if (secret != NULL)
599*00b67f09SDavid van Moolenbroek 		isc_mem_free(gmctx, secret);
600*00b67f09SDavid van Moolenbroek }
601*00b67f09SDavid van Moolenbroek 
602*00b67f09SDavid van Moolenbroek /*
603*00b67f09SDavid van Moolenbroek  * Get a key from a named.conf format keyfile
604*00b67f09SDavid van Moolenbroek  */
605*00b67f09SDavid van Moolenbroek static isc_result_t
read_sessionkey(isc_mem_t * mctx,isc_log_t * lctx)606*00b67f09SDavid van Moolenbroek read_sessionkey(isc_mem_t *mctx, isc_log_t *lctx) {
607*00b67f09SDavid van Moolenbroek 	cfg_parser_t *pctx = NULL;
608*00b67f09SDavid van Moolenbroek 	cfg_obj_t *sessionkey = NULL;
609*00b67f09SDavid van Moolenbroek 	const cfg_obj_t *key = NULL;
610*00b67f09SDavid van Moolenbroek 	const cfg_obj_t *secretobj = NULL;
611*00b67f09SDavid van Moolenbroek 	const cfg_obj_t *algorithmobj = NULL;
612*00b67f09SDavid van Moolenbroek 	const char *mykeyname;
613*00b67f09SDavid van Moolenbroek 	const char *secretstr;
614*00b67f09SDavid van Moolenbroek 	const char *algorithm;
615*00b67f09SDavid van Moolenbroek 	isc_result_t result;
616*00b67f09SDavid van Moolenbroek 	int len;
617*00b67f09SDavid van Moolenbroek 
618*00b67f09SDavid van Moolenbroek 	if (! isc_file_exists(keyfile))
619*00b67f09SDavid van Moolenbroek 		return (ISC_R_FILENOTFOUND);
620*00b67f09SDavid van Moolenbroek 
621*00b67f09SDavid van Moolenbroek 	result = cfg_parser_create(mctx, lctx, &pctx);
622*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
623*00b67f09SDavid van Moolenbroek 		goto cleanup;
624*00b67f09SDavid van Moolenbroek 
625*00b67f09SDavid van Moolenbroek 	result = cfg_parse_file(pctx, keyfile, &cfg_type_sessionkey,
626*00b67f09SDavid van Moolenbroek 				&sessionkey);
627*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
628*00b67f09SDavid van Moolenbroek 		goto cleanup;
629*00b67f09SDavid van Moolenbroek 
630*00b67f09SDavid van Moolenbroek 	result = cfg_map_get(sessionkey, "key", &key);
631*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
632*00b67f09SDavid van Moolenbroek 		goto cleanup;
633*00b67f09SDavid van Moolenbroek 
634*00b67f09SDavid van Moolenbroek 	(void) cfg_map_get(key, "secret", &secretobj);
635*00b67f09SDavid van Moolenbroek 	(void) cfg_map_get(key, "algorithm", &algorithmobj);
636*00b67f09SDavid van Moolenbroek 	if (secretobj == NULL || algorithmobj == NULL)
637*00b67f09SDavid van Moolenbroek 		fatal("key must have algorithm and secret");
638*00b67f09SDavid van Moolenbroek 
639*00b67f09SDavid van Moolenbroek 	mykeyname = cfg_obj_asstring(cfg_map_getname(key));
640*00b67f09SDavid van Moolenbroek 	secretstr = cfg_obj_asstring(secretobj);
641*00b67f09SDavid van Moolenbroek 	algorithm = cfg_obj_asstring(algorithmobj);
642*00b67f09SDavid van Moolenbroek 
643*00b67f09SDavid van Moolenbroek 	len = strlen(algorithm) + strlen(mykeyname) + strlen(secretstr) + 3;
644*00b67f09SDavid van Moolenbroek 	keystr = isc_mem_allocate(mctx, len);
645*00b67f09SDavid van Moolenbroek 	snprintf(keystr, len, "%s:%s:%s", algorithm, mykeyname, secretstr);
646*00b67f09SDavid van Moolenbroek 	setup_keystr();
647*00b67f09SDavid van Moolenbroek 
648*00b67f09SDavid van Moolenbroek  cleanup:
649*00b67f09SDavid van Moolenbroek 	if (pctx != NULL) {
650*00b67f09SDavid van Moolenbroek 		if (sessionkey != NULL)
651*00b67f09SDavid van Moolenbroek 			cfg_obj_destroy(pctx, &sessionkey);
652*00b67f09SDavid van Moolenbroek 		cfg_parser_destroy(&pctx);
653*00b67f09SDavid van Moolenbroek 	}
654*00b67f09SDavid van Moolenbroek 
655*00b67f09SDavid van Moolenbroek 	if (keystr != NULL)
656*00b67f09SDavid van Moolenbroek 		isc_mem_free(mctx, keystr);
657*00b67f09SDavid van Moolenbroek 
658*00b67f09SDavid van Moolenbroek 	return (result);
659*00b67f09SDavid van Moolenbroek }
660*00b67f09SDavid van Moolenbroek 
661*00b67f09SDavid van Moolenbroek static void
setup_keyfile(isc_mem_t * mctx,isc_log_t * lctx)662*00b67f09SDavid van Moolenbroek setup_keyfile(isc_mem_t *mctx, isc_log_t *lctx) {
663*00b67f09SDavid van Moolenbroek 	dst_key_t *dstkey = NULL;
664*00b67f09SDavid van Moolenbroek 	isc_result_t result;
665*00b67f09SDavid van Moolenbroek 	dns_name_t *hmacname = NULL;
666*00b67f09SDavid van Moolenbroek 
667*00b67f09SDavid van Moolenbroek 	debug("Creating key...");
668*00b67f09SDavid van Moolenbroek 
669*00b67f09SDavid van Moolenbroek 	if (sig0key != NULL)
670*00b67f09SDavid van Moolenbroek 		dst_key_free(&sig0key);
671*00b67f09SDavid van Moolenbroek 
672*00b67f09SDavid van Moolenbroek 	/* Try reading the key from a K* pair */
673*00b67f09SDavid van Moolenbroek 	result = dst_key_fromnamedfile(keyfile, NULL,
674*00b67f09SDavid van Moolenbroek 				       DST_TYPE_PRIVATE | DST_TYPE_KEY, mctx,
675*00b67f09SDavid van Moolenbroek 				       &dstkey);
676*00b67f09SDavid van Moolenbroek 
677*00b67f09SDavid van Moolenbroek 	/* If that didn't work, try reading it as a session.key keyfile */
678*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
679*00b67f09SDavid van Moolenbroek 		result = read_sessionkey(mctx, lctx);
680*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_SUCCESS)
681*00b67f09SDavid van Moolenbroek 			return;
682*00b67f09SDavid van Moolenbroek 	}
683*00b67f09SDavid van Moolenbroek 
684*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
685*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read key from %.*s.{private,key}: "
686*00b67f09SDavid van Moolenbroek 				"%s\n", basenamelen(keyfile), keyfile,
687*00b67f09SDavid van Moolenbroek 				isc_result_totext(result));
688*00b67f09SDavid van Moolenbroek 		return;
689*00b67f09SDavid van Moolenbroek 	}
690*00b67f09SDavid van Moolenbroek 
691*00b67f09SDavid van Moolenbroek 	switch (dst_key_alg(dstkey)) {
692*00b67f09SDavid van Moolenbroek 	case DST_ALG_HMACMD5:
693*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACMD5_NAME;
694*00b67f09SDavid van Moolenbroek 		break;
695*00b67f09SDavid van Moolenbroek 	case DST_ALG_HMACSHA1:
696*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACSHA1_NAME;
697*00b67f09SDavid van Moolenbroek 		break;
698*00b67f09SDavid van Moolenbroek 	case DST_ALG_HMACSHA224:
699*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACSHA224_NAME;
700*00b67f09SDavid van Moolenbroek 		break;
701*00b67f09SDavid van Moolenbroek 	case DST_ALG_HMACSHA256:
702*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACSHA256_NAME;
703*00b67f09SDavid van Moolenbroek 		break;
704*00b67f09SDavid van Moolenbroek 	case DST_ALG_HMACSHA384:
705*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACSHA384_NAME;
706*00b67f09SDavid van Moolenbroek 		break;
707*00b67f09SDavid van Moolenbroek 	case DST_ALG_HMACSHA512:
708*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACSHA512_NAME;
709*00b67f09SDavid van Moolenbroek 		break;
710*00b67f09SDavid van Moolenbroek 	}
711*00b67f09SDavid van Moolenbroek 	if (hmacname != NULL) {
712*00b67f09SDavid van Moolenbroek 		result = dns_tsigkey_createfromkey(dst_key_name(dstkey),
713*00b67f09SDavid van Moolenbroek 						   hmacname, dstkey, ISC_FALSE,
714*00b67f09SDavid van Moolenbroek 						   NULL, 0, 0, mctx, NULL,
715*00b67f09SDavid van Moolenbroek 						   &tsigkey);
716*00b67f09SDavid van Moolenbroek 		dst_key_free(&dstkey);
717*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS) {
718*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "could not create key from %s: %s\n",
719*00b67f09SDavid van Moolenbroek 				keyfile, isc_result_totext(result));
720*00b67f09SDavid van Moolenbroek 			return;
721*00b67f09SDavid van Moolenbroek 		}
722*00b67f09SDavid van Moolenbroek 	} else {
723*00b67f09SDavid van Moolenbroek 		dst_key_attach(dstkey, &sig0key);
724*00b67f09SDavid van Moolenbroek 		dst_key_free(&dstkey);
725*00b67f09SDavid van Moolenbroek 	}
726*00b67f09SDavid van Moolenbroek }
727*00b67f09SDavid van Moolenbroek 
728*00b67f09SDavid van Moolenbroek static void
doshutdown(void)729*00b67f09SDavid van Moolenbroek doshutdown(void) {
730*00b67f09SDavid van Moolenbroek 	isc_task_detach(&global_task);
731*00b67f09SDavid van Moolenbroek 
732*00b67f09SDavid van Moolenbroek 	/*
733*00b67f09SDavid van Moolenbroek 	 * The isc_mem_put of master_servers must be before the
734*00b67f09SDavid van Moolenbroek 	 * isc_mem_put of servers as it sets the servers pointer
735*00b67f09SDavid van Moolenbroek 	 * to NULL.
736*00b67f09SDavid van Moolenbroek 	 */
737*00b67f09SDavid van Moolenbroek 	if (master_servers != NULL && master_servers != servers)
738*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, master_servers,
739*00b67f09SDavid van Moolenbroek 			    master_total * sizeof(isc_sockaddr_t));
740*00b67f09SDavid van Moolenbroek 
741*00b67f09SDavid van Moolenbroek 	if (servers != NULL)
742*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, servers, ns_total * sizeof(isc_sockaddr_t));
743*00b67f09SDavid van Moolenbroek 
744*00b67f09SDavid van Moolenbroek 	if (localaddr4 != NULL)
745*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, localaddr4, sizeof(isc_sockaddr_t));
746*00b67f09SDavid van Moolenbroek 
747*00b67f09SDavid van Moolenbroek 	if (localaddr6 != NULL)
748*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, localaddr6, sizeof(isc_sockaddr_t));
749*00b67f09SDavid van Moolenbroek 
750*00b67f09SDavid van Moolenbroek 	if (tsigkey != NULL) {
751*00b67f09SDavid van Moolenbroek 		ddebug("Freeing TSIG key");
752*00b67f09SDavid van Moolenbroek 		dns_tsigkey_detach(&tsigkey);
753*00b67f09SDavid van Moolenbroek 	}
754*00b67f09SDavid van Moolenbroek 
755*00b67f09SDavid van Moolenbroek 	if (sig0key != NULL) {
756*00b67f09SDavid van Moolenbroek 		ddebug("Freeing SIG(0) key");
757*00b67f09SDavid van Moolenbroek 		dst_key_free(&sig0key);
758*00b67f09SDavid van Moolenbroek 	}
759*00b67f09SDavid van Moolenbroek 
760*00b67f09SDavid van Moolenbroek 	if (updatemsg != NULL)
761*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&updatemsg);
762*00b67f09SDavid van Moolenbroek 
763*00b67f09SDavid van Moolenbroek 	if (is_dst_up) {
764*00b67f09SDavid van Moolenbroek 		ddebug("Destroy DST lib");
765*00b67f09SDavid van Moolenbroek 		dst_lib_destroy();
766*00b67f09SDavid van Moolenbroek 		is_dst_up = ISC_FALSE;
767*00b67f09SDavid van Moolenbroek 	}
768*00b67f09SDavid van Moolenbroek 
769*00b67f09SDavid van Moolenbroek 	cleanup_entropy(&entropy);
770*00b67f09SDavid van Moolenbroek 
771*00b67f09SDavid van Moolenbroek 	lwres_conf_clear(lwctx);
772*00b67f09SDavid van Moolenbroek 	lwres_context_destroy(&lwctx);
773*00b67f09SDavid van Moolenbroek 
774*00b67f09SDavid van Moolenbroek 	ddebug("Destroying request manager");
775*00b67f09SDavid van Moolenbroek 	dns_requestmgr_detach(&requestmgr);
776*00b67f09SDavid van Moolenbroek 
777*00b67f09SDavid van Moolenbroek 	ddebug("Freeing the dispatchers");
778*00b67f09SDavid van Moolenbroek 	if (have_ipv4)
779*00b67f09SDavid van Moolenbroek 		dns_dispatch_detach(&dispatchv4);
780*00b67f09SDavid van Moolenbroek 	if (have_ipv6)
781*00b67f09SDavid van Moolenbroek 		dns_dispatch_detach(&dispatchv6);
782*00b67f09SDavid van Moolenbroek 
783*00b67f09SDavid van Moolenbroek 	ddebug("Shutting down dispatch manager");
784*00b67f09SDavid van Moolenbroek 	dns_dispatchmgr_destroy(&dispatchmgr);
785*00b67f09SDavid van Moolenbroek 
786*00b67f09SDavid van Moolenbroek }
787*00b67f09SDavid van Moolenbroek 
788*00b67f09SDavid van Moolenbroek static void
maybeshutdown(void)789*00b67f09SDavid van Moolenbroek maybeshutdown(void) {
790*00b67f09SDavid van Moolenbroek 	ddebug("Shutting down request manager");
791*00b67f09SDavid van Moolenbroek 	dns_requestmgr_shutdown(requestmgr);
792*00b67f09SDavid van Moolenbroek 
793*00b67f09SDavid van Moolenbroek 	if (requests != 0)
794*00b67f09SDavid van Moolenbroek 		return;
795*00b67f09SDavid van Moolenbroek 
796*00b67f09SDavid van Moolenbroek 	doshutdown();
797*00b67f09SDavid van Moolenbroek }
798*00b67f09SDavid van Moolenbroek 
799*00b67f09SDavid van Moolenbroek static void
shutdown_program(isc_task_t * task,isc_event_t * event)800*00b67f09SDavid van Moolenbroek shutdown_program(isc_task_t *task, isc_event_t *event) {
801*00b67f09SDavid van Moolenbroek 	REQUIRE(task == global_task);
802*00b67f09SDavid van Moolenbroek 	UNUSED(task);
803*00b67f09SDavid van Moolenbroek 
804*00b67f09SDavid van Moolenbroek 	ddebug("shutdown_program()");
805*00b67f09SDavid van Moolenbroek 	isc_event_free(&event);
806*00b67f09SDavid van Moolenbroek 
807*00b67f09SDavid van Moolenbroek 	shuttingdown = ISC_TRUE;
808*00b67f09SDavid van Moolenbroek 	maybeshutdown();
809*00b67f09SDavid van Moolenbroek }
810*00b67f09SDavid van Moolenbroek 
811*00b67f09SDavid van Moolenbroek static void
setup_system(void)812*00b67f09SDavid van Moolenbroek setup_system(void) {
813*00b67f09SDavid van Moolenbroek 	isc_result_t result;
814*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t bind_any, bind_any6;
815*00b67f09SDavid van Moolenbroek 	lwres_result_t lwresult;
816*00b67f09SDavid van Moolenbroek 	unsigned int attrs, attrmask;
817*00b67f09SDavid van Moolenbroek 	int i;
818*00b67f09SDavid van Moolenbroek 	isc_logconfig_t *logconfig = NULL;
819*00b67f09SDavid van Moolenbroek 
820*00b67f09SDavid van Moolenbroek 	ddebug("setup_system()");
821*00b67f09SDavid van Moolenbroek 
822*00b67f09SDavid van Moolenbroek 	dns_result_register();
823*00b67f09SDavid van Moolenbroek 
824*00b67f09SDavid van Moolenbroek 	result = isc_net_probeipv4();
825*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_SUCCESS)
826*00b67f09SDavid van Moolenbroek 		have_ipv4 = ISC_TRUE;
827*00b67f09SDavid van Moolenbroek 
828*00b67f09SDavid van Moolenbroek 	result = isc_net_probeipv6();
829*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_SUCCESS)
830*00b67f09SDavid van Moolenbroek 		have_ipv6 = ISC_TRUE;
831*00b67f09SDavid van Moolenbroek 
832*00b67f09SDavid van Moolenbroek 	if (!have_ipv4 && !have_ipv6)
833*00b67f09SDavid van Moolenbroek 		fatal("could not find either IPv4 or IPv6");
834*00b67f09SDavid van Moolenbroek 
835*00b67f09SDavid van Moolenbroek 	result = isc_log_create(gmctx, &glctx, &logconfig);
836*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_log_create");
837*00b67f09SDavid van Moolenbroek 
838*00b67f09SDavid van Moolenbroek 	isc_log_setcontext(glctx);
839*00b67f09SDavid van Moolenbroek 	dns_log_init(glctx);
840*00b67f09SDavid van Moolenbroek 	dns_log_setcontext(glctx);
841*00b67f09SDavid van Moolenbroek 
842*00b67f09SDavid van Moolenbroek 	result = isc_log_usechannel(logconfig, "default_debug", NULL, NULL);
843*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_log_usechannel");
844*00b67f09SDavid van Moolenbroek 
845*00b67f09SDavid van Moolenbroek 	isc_log_setdebuglevel(glctx, logdebuglevel);
846*00b67f09SDavid van Moolenbroek 
847*00b67f09SDavid van Moolenbroek 	lwresult = lwres_context_create(&lwctx, gmctx, mem_alloc, mem_free, 1);
848*00b67f09SDavid van Moolenbroek 	if (lwresult != LWRES_R_SUCCESS)
849*00b67f09SDavid van Moolenbroek 		fatal("lwres_context_create failed");
850*00b67f09SDavid van Moolenbroek 
851*00b67f09SDavid van Moolenbroek 	(void)lwres_conf_parse(lwctx, RESOLV_CONF);
852*00b67f09SDavid van Moolenbroek 	lwconf = lwres_conf_get(lwctx);
853*00b67f09SDavid van Moolenbroek 
854*00b67f09SDavid van Moolenbroek 	if (servers != NULL) {
855*00b67f09SDavid van Moolenbroek 		if (master_servers == servers)
856*00b67f09SDavid van Moolenbroek 			master_servers = NULL;
857*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, servers, ns_total * sizeof(isc_sockaddr_t));
858*00b67f09SDavid van Moolenbroek 	}
859*00b67f09SDavid van Moolenbroek 
860*00b67f09SDavid van Moolenbroek 	ns_inuse = 0;
861*00b67f09SDavid van Moolenbroek 	if (local_only || lwconf->nsnext <= 0) {
862*00b67f09SDavid van Moolenbroek 		struct in_addr in;
863*00b67f09SDavid van Moolenbroek 		struct in6_addr in6;
864*00b67f09SDavid van Moolenbroek 
865*00b67f09SDavid van Moolenbroek 		if (local_only && keyfile == NULL)
866*00b67f09SDavid van Moolenbroek 			keyfile = SESSION_KEYFILE;
867*00b67f09SDavid van Moolenbroek 
868*00b67f09SDavid van Moolenbroek 		default_servers = !local_only;
869*00b67f09SDavid van Moolenbroek 
870*00b67f09SDavid van Moolenbroek 		ns_total = (have_ipv4 ? 1 : 0) + (have_ipv6 ? 1 : 0);
871*00b67f09SDavid van Moolenbroek 		servers = isc_mem_get(gmctx, ns_total * sizeof(isc_sockaddr_t));
872*00b67f09SDavid van Moolenbroek 		if (servers == NULL)
873*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
874*00b67f09SDavid van Moolenbroek 
875*00b67f09SDavid van Moolenbroek 		if (have_ipv4) {
876*00b67f09SDavid van Moolenbroek 			in.s_addr = htonl(INADDR_LOOPBACK);
877*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin(&servers[0], &in, dnsport);
878*00b67f09SDavid van Moolenbroek 		}
879*00b67f09SDavid van Moolenbroek 		if (have_ipv6) {
880*00b67f09SDavid van Moolenbroek 			memset(&in6, 0, sizeof(in6));
881*00b67f09SDavid van Moolenbroek 			in6.s6_addr[15] = 1;
882*00b67f09SDavid van Moolenbroek 			isc_sockaddr_fromin6(&servers[(have_ipv4 ? 1 : 0)],
883*00b67f09SDavid van Moolenbroek 					     &in6, dnsport);
884*00b67f09SDavid van Moolenbroek 		}
885*00b67f09SDavid van Moolenbroek 	} else {
886*00b67f09SDavid van Moolenbroek 		ns_total = lwconf->nsnext;
887*00b67f09SDavid van Moolenbroek 		servers = isc_mem_get(gmctx, ns_total * sizeof(isc_sockaddr_t));
888*00b67f09SDavid van Moolenbroek 		if (servers == NULL)
889*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
890*00b67f09SDavid van Moolenbroek 		for (i = 0; i < ns_total; i++) {
891*00b67f09SDavid van Moolenbroek 			if (lwconf->nameservers[i].family == LWRES_ADDRTYPE_V4)
892*00b67f09SDavid van Moolenbroek 			{
893*00b67f09SDavid van Moolenbroek 				struct in_addr in4;
894*00b67f09SDavid van Moolenbroek 				memmove(&in4,
895*00b67f09SDavid van Moolenbroek 					lwconf->nameservers[i].address, 4);
896*00b67f09SDavid van Moolenbroek 				isc_sockaddr_fromin(&servers[i],
897*00b67f09SDavid van Moolenbroek 						    &in4, dnsport);
898*00b67f09SDavid van Moolenbroek 			} else {
899*00b67f09SDavid van Moolenbroek 				struct in6_addr in6;
900*00b67f09SDavid van Moolenbroek 				memmove(&in6,
901*00b67f09SDavid van Moolenbroek 					lwconf->nameservers[i].address, 16);
902*00b67f09SDavid van Moolenbroek 				isc_sockaddr_fromin6(&servers[i],
903*00b67f09SDavid van Moolenbroek 						     &in6, dnsport);
904*00b67f09SDavid van Moolenbroek 			}
905*00b67f09SDavid van Moolenbroek 		}
906*00b67f09SDavid van Moolenbroek 	}
907*00b67f09SDavid van Moolenbroek 
908*00b67f09SDavid van Moolenbroek 	setup_entropy(gmctx, NULL, &entropy);
909*00b67f09SDavid van Moolenbroek 
910*00b67f09SDavid van Moolenbroek 	result = isc_hash_create(gmctx, entropy, DNS_NAME_MAXWIRE);
911*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_hash_create");
912*00b67f09SDavid van Moolenbroek 	isc_hash_init();
913*00b67f09SDavid van Moolenbroek 
914*00b67f09SDavid van Moolenbroek 	result = dns_dispatchmgr_create(gmctx, entropy, &dispatchmgr);
915*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_dispatchmgr_create");
916*00b67f09SDavid van Moolenbroek 
917*00b67f09SDavid van Moolenbroek 	result = isc_socketmgr_create(gmctx, &socketmgr);
918*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_socketmgr_create");
919*00b67f09SDavid van Moolenbroek 
920*00b67f09SDavid van Moolenbroek 	result = isc_timermgr_create(gmctx, &timermgr);
921*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_timermgr_create");
922*00b67f09SDavid van Moolenbroek 
923*00b67f09SDavid van Moolenbroek 	result = isc_taskmgr_create(gmctx, 1, 0, &taskmgr);
924*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_taskmgr_create");
925*00b67f09SDavid van Moolenbroek 
926*00b67f09SDavid van Moolenbroek 	result = isc_task_create(taskmgr, 0, &global_task);
927*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_task_create");
928*00b67f09SDavid van Moolenbroek 
929*00b67f09SDavid van Moolenbroek 	result = isc_task_onshutdown(global_task, shutdown_program, NULL);
930*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_task_onshutdown");
931*00b67f09SDavid van Moolenbroek 
932*00b67f09SDavid van Moolenbroek 	result = dst_lib_init(gmctx, entropy, 0);
933*00b67f09SDavid van Moolenbroek 	check_result(result, "dst_lib_init");
934*00b67f09SDavid van Moolenbroek 	is_dst_up = ISC_TRUE;
935*00b67f09SDavid van Moolenbroek 
936*00b67f09SDavid van Moolenbroek 	attrmask = DNS_DISPATCHATTR_UDP | DNS_DISPATCHATTR_TCP;
937*00b67f09SDavid van Moolenbroek 	attrmask |= DNS_DISPATCHATTR_IPV4 | DNS_DISPATCHATTR_IPV6;
938*00b67f09SDavid van Moolenbroek 
939*00b67f09SDavid van Moolenbroek 	if (have_ipv6) {
940*00b67f09SDavid van Moolenbroek 		attrs = DNS_DISPATCHATTR_UDP;
941*00b67f09SDavid van Moolenbroek 		attrs |= DNS_DISPATCHATTR_MAKEQUERY;
942*00b67f09SDavid van Moolenbroek 		attrs |= DNS_DISPATCHATTR_IPV6;
943*00b67f09SDavid van Moolenbroek 		isc_sockaddr_any6(&bind_any6);
944*00b67f09SDavid van Moolenbroek 		result = dns_dispatch_getudp(dispatchmgr, socketmgr, taskmgr,
945*00b67f09SDavid van Moolenbroek 					     &bind_any6, PACKETSIZE,
946*00b67f09SDavid van Moolenbroek 					     4, 2, 3, 5,
947*00b67f09SDavid van Moolenbroek 					     attrs, attrmask, &dispatchv6);
948*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_dispatch_getudp (v6)");
949*00b67f09SDavid van Moolenbroek 	}
950*00b67f09SDavid van Moolenbroek 
951*00b67f09SDavid van Moolenbroek 	if (have_ipv4) {
952*00b67f09SDavid van Moolenbroek 		attrs = DNS_DISPATCHATTR_UDP;
953*00b67f09SDavid van Moolenbroek 		attrs |= DNS_DISPATCHATTR_MAKEQUERY;
954*00b67f09SDavid van Moolenbroek 		attrs |= DNS_DISPATCHATTR_IPV4;
955*00b67f09SDavid van Moolenbroek 		isc_sockaddr_any(&bind_any);
956*00b67f09SDavid van Moolenbroek 		result = dns_dispatch_getudp(dispatchmgr, socketmgr, taskmgr,
957*00b67f09SDavid van Moolenbroek 					     &bind_any, PACKETSIZE,
958*00b67f09SDavid van Moolenbroek 					     4, 2, 3, 5,
959*00b67f09SDavid van Moolenbroek 					     attrs, attrmask, &dispatchv4);
960*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_dispatch_getudp (v4)");
961*00b67f09SDavid van Moolenbroek 	}
962*00b67f09SDavid van Moolenbroek 
963*00b67f09SDavid van Moolenbroek 	result = dns_requestmgr_create(gmctx, timermgr,
964*00b67f09SDavid van Moolenbroek 				       socketmgr, taskmgr, dispatchmgr,
965*00b67f09SDavid van Moolenbroek 				       dispatchv4, dispatchv6, &requestmgr);
966*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_requestmgr_create");
967*00b67f09SDavid van Moolenbroek 
968*00b67f09SDavid van Moolenbroek 	if (keystr != NULL)
969*00b67f09SDavid van Moolenbroek 		setup_keystr();
970*00b67f09SDavid van Moolenbroek 	else if (local_only) {
971*00b67f09SDavid van Moolenbroek 		result = read_sessionkey(gmctx, glctx);
972*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
973*00b67f09SDavid van Moolenbroek 			fatal("can't read key from %s: %s\n",
974*00b67f09SDavid van Moolenbroek 			      keyfile, isc_result_totext(result));
975*00b67f09SDavid van Moolenbroek 	} else if (keyfile != NULL)
976*00b67f09SDavid van Moolenbroek 		setup_keyfile(gmctx, glctx);
977*00b67f09SDavid van Moolenbroek }
978*00b67f09SDavid van Moolenbroek 
979*00b67f09SDavid van Moolenbroek static void
get_addresses(char * host,in_port_t port,isc_sockaddr_t * sockaddr,int naddrs)980*00b67f09SDavid van Moolenbroek get_addresses(char *host, in_port_t port,
981*00b67f09SDavid van Moolenbroek 	      isc_sockaddr_t *sockaddr, int naddrs)
982*00b67f09SDavid van Moolenbroek {
983*00b67f09SDavid van Moolenbroek 	int count;
984*00b67f09SDavid van Moolenbroek 	isc_result_t result;
985*00b67f09SDavid van Moolenbroek 
986*00b67f09SDavid van Moolenbroek 	isc_app_block();
987*00b67f09SDavid van Moolenbroek 	result = bind9_getaddresses(host, port, sockaddr, naddrs, &count);
988*00b67f09SDavid van Moolenbroek 	isc_app_unblock();
989*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
990*00b67f09SDavid van Moolenbroek 		fatal("couldn't get address for '%s': %s",
991*00b67f09SDavid van Moolenbroek 		      host, isc_result_totext(result));
992*00b67f09SDavid van Moolenbroek }
993*00b67f09SDavid van Moolenbroek 
994*00b67f09SDavid van Moolenbroek static void
version(void)995*00b67f09SDavid van Moolenbroek version(void) {
996*00b67f09SDavid van Moolenbroek 	fputs("nsupdate " VERSION "\n", stderr);
997*00b67f09SDavid van Moolenbroek }
998*00b67f09SDavid van Moolenbroek 
999*00b67f09SDavid van Moolenbroek #define PARSE_ARGS_FMT "dDML:y:ghlovk:p:Pr:R::t:Tu:V"
1000*00b67f09SDavid van Moolenbroek 
1001*00b67f09SDavid van Moolenbroek static void
pre_parse_args(int argc,char ** argv)1002*00b67f09SDavid van Moolenbroek pre_parse_args(int argc, char **argv) {
1003*00b67f09SDavid van Moolenbroek 	dns_rdatatype_t t;
1004*00b67f09SDavid van Moolenbroek 	int ch;
1005*00b67f09SDavid van Moolenbroek 	char buf[100];
1006*00b67f09SDavid van Moolenbroek 	isc_boolean_t doexit = ISC_FALSE;
1007*00b67f09SDavid van Moolenbroek 
1008*00b67f09SDavid van Moolenbroek 	while ((ch = isc_commandline_parse(argc, argv, PARSE_ARGS_FMT)) != -1) {
1009*00b67f09SDavid van Moolenbroek 		switch (ch) {
1010*00b67f09SDavid van Moolenbroek 		case 'M': /* was -dm */
1011*00b67f09SDavid van Moolenbroek 			debugging = ISC_TRUE;
1012*00b67f09SDavid van Moolenbroek 			ddebugging = ISC_TRUE;
1013*00b67f09SDavid van Moolenbroek 			memdebugging = ISC_TRUE;
1014*00b67f09SDavid van Moolenbroek 			isc_mem_debugging = ISC_MEM_DEBUGTRACE |
1015*00b67f09SDavid van Moolenbroek 					    ISC_MEM_DEBUGRECORD;
1016*00b67f09SDavid van Moolenbroek 			break;
1017*00b67f09SDavid van Moolenbroek 
1018*00b67f09SDavid van Moolenbroek 		case '?':
1019*00b67f09SDavid van Moolenbroek 		case 'h':
1020*00b67f09SDavid van Moolenbroek 			if (isc_commandline_option != '?')
1021*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "%s: invalid argument -%c\n",
1022*00b67f09SDavid van Moolenbroek 					argv[0], isc_commandline_option);
1023*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "usage: nsupdate [-dD] [-L level] [-l]"
1024*00b67f09SDavid van Moolenbroek 				"[-g | -o | -y keyname:secret | -k keyfile] "
1025*00b67f09SDavid van Moolenbroek 				"[-v] [-V] [filename]\n");
1026*00b67f09SDavid van Moolenbroek 			exit(1);
1027*00b67f09SDavid van Moolenbroek 
1028*00b67f09SDavid van Moolenbroek 		case 'P':
1029*00b67f09SDavid van Moolenbroek 			for (t = 0xff00; t <= 0xfffe; t++) {
1030*00b67f09SDavid van Moolenbroek 				if (dns_rdatatype_ismeta(t))
1031*00b67f09SDavid van Moolenbroek 					continue;
1032*00b67f09SDavid van Moolenbroek 				dns_rdatatype_format(t, buf, sizeof(buf));
1033*00b67f09SDavid van Moolenbroek 				if (strncmp(buf, "TYPE", 4) != 0)
1034*00b67f09SDavid van Moolenbroek 					fprintf(stdout, "%s\n", buf);
1035*00b67f09SDavid van Moolenbroek 			}
1036*00b67f09SDavid van Moolenbroek 			doexit = ISC_TRUE;
1037*00b67f09SDavid van Moolenbroek 			break;
1038*00b67f09SDavid van Moolenbroek 
1039*00b67f09SDavid van Moolenbroek 		case 'T':
1040*00b67f09SDavid van Moolenbroek 			for (t = 1; t <= 0xfeff; t++) {
1041*00b67f09SDavid van Moolenbroek 				if (dns_rdatatype_ismeta(t))
1042*00b67f09SDavid van Moolenbroek 					continue;
1043*00b67f09SDavid van Moolenbroek 				dns_rdatatype_format(t, buf, sizeof(buf));
1044*00b67f09SDavid van Moolenbroek 				if (strncmp(buf, "TYPE", 4) != 0)
1045*00b67f09SDavid van Moolenbroek 					fprintf(stdout, "%s\n", buf);
1046*00b67f09SDavid van Moolenbroek 			}
1047*00b67f09SDavid van Moolenbroek 			doexit = ISC_TRUE;
1048*00b67f09SDavid van Moolenbroek 			break;
1049*00b67f09SDavid van Moolenbroek 
1050*00b67f09SDavid van Moolenbroek 		case 'V':
1051*00b67f09SDavid van Moolenbroek 			version();
1052*00b67f09SDavid van Moolenbroek 			doexit = ISC_TRUE;
1053*00b67f09SDavid van Moolenbroek 			break;
1054*00b67f09SDavid van Moolenbroek 
1055*00b67f09SDavid van Moolenbroek 		default:
1056*00b67f09SDavid van Moolenbroek 			break;
1057*00b67f09SDavid van Moolenbroek 		}
1058*00b67f09SDavid van Moolenbroek 	}
1059*00b67f09SDavid van Moolenbroek 	if (doexit)
1060*00b67f09SDavid van Moolenbroek 		exit(0);
1061*00b67f09SDavid van Moolenbroek 	isc_commandline_reset = ISC_TRUE;
1062*00b67f09SDavid van Moolenbroek 	isc_commandline_index = 1;
1063*00b67f09SDavid van Moolenbroek }
1064*00b67f09SDavid van Moolenbroek 
1065*00b67f09SDavid van Moolenbroek static void
parse_args(int argc,char ** argv,isc_mem_t * mctx,isc_entropy_t ** ectx)1066*00b67f09SDavid van Moolenbroek parse_args(int argc, char **argv, isc_mem_t *mctx, isc_entropy_t **ectx) {
1067*00b67f09SDavid van Moolenbroek 	int ch;
1068*00b67f09SDavid van Moolenbroek 	isc_uint32_t i;
1069*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1070*00b67f09SDavid van Moolenbroek 
1071*00b67f09SDavid van Moolenbroek 	debug("parse_args");
1072*00b67f09SDavid van Moolenbroek 	while ((ch = isc_commandline_parse(argc, argv, PARSE_ARGS_FMT)) != -1) {
1073*00b67f09SDavid van Moolenbroek 		switch (ch) {
1074*00b67f09SDavid van Moolenbroek 		case 'd':
1075*00b67f09SDavid van Moolenbroek 			debugging = ISC_TRUE;
1076*00b67f09SDavid van Moolenbroek 			break;
1077*00b67f09SDavid van Moolenbroek 		case 'D': /* was -dd */
1078*00b67f09SDavid van Moolenbroek 			debugging = ISC_TRUE;
1079*00b67f09SDavid van Moolenbroek 			ddebugging = ISC_TRUE;
1080*00b67f09SDavid van Moolenbroek 			break;
1081*00b67f09SDavid van Moolenbroek 		case 'M':
1082*00b67f09SDavid van Moolenbroek 			break;
1083*00b67f09SDavid van Moolenbroek 		case 'l':
1084*00b67f09SDavid van Moolenbroek 			local_only = ISC_TRUE;
1085*00b67f09SDavid van Moolenbroek 			break;
1086*00b67f09SDavid van Moolenbroek 		case 'L':
1087*00b67f09SDavid van Moolenbroek 			result = isc_parse_uint32(&i, isc_commandline_argument,
1088*00b67f09SDavid van Moolenbroek 						  10);
1089*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1090*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "bad library debug value "
1091*00b67f09SDavid van Moolenbroek 					"'%s'\n", isc_commandline_argument);
1092*00b67f09SDavid van Moolenbroek 				exit(1);
1093*00b67f09SDavid van Moolenbroek 			}
1094*00b67f09SDavid van Moolenbroek 			logdebuglevel = i;
1095*00b67f09SDavid van Moolenbroek 			break;
1096*00b67f09SDavid van Moolenbroek 		case 'y':
1097*00b67f09SDavid van Moolenbroek 			keystr = isc_commandline_argument;
1098*00b67f09SDavid van Moolenbroek 			break;
1099*00b67f09SDavid van Moolenbroek 		case 'v':
1100*00b67f09SDavid van Moolenbroek 			usevc = ISC_TRUE;
1101*00b67f09SDavid van Moolenbroek 			break;
1102*00b67f09SDavid van Moolenbroek 		case 'k':
1103*00b67f09SDavid van Moolenbroek 			keyfile = isc_commandline_argument;
1104*00b67f09SDavid van Moolenbroek 			break;
1105*00b67f09SDavid van Moolenbroek 		case 'g':
1106*00b67f09SDavid van Moolenbroek 			usegsstsig = ISC_TRUE;
1107*00b67f09SDavid van Moolenbroek 			use_win2k_gsstsig = ISC_FALSE;
1108*00b67f09SDavid van Moolenbroek 			break;
1109*00b67f09SDavid van Moolenbroek 		case 'o':
1110*00b67f09SDavid van Moolenbroek 			usegsstsig = ISC_TRUE;
1111*00b67f09SDavid van Moolenbroek 			use_win2k_gsstsig = ISC_TRUE;
1112*00b67f09SDavid van Moolenbroek 			break;
1113*00b67f09SDavid van Moolenbroek 		case 'p':
1114*00b67f09SDavid van Moolenbroek 			result = isc_parse_uint16(&dnsport,
1115*00b67f09SDavid van Moolenbroek 						  isc_commandline_argument, 10);
1116*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1117*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "bad port number "
1118*00b67f09SDavid van Moolenbroek 					"'%s'\n", isc_commandline_argument);
1119*00b67f09SDavid van Moolenbroek 				exit(1);
1120*00b67f09SDavid van Moolenbroek 			}
1121*00b67f09SDavid van Moolenbroek 			break;
1122*00b67f09SDavid van Moolenbroek 		case 't':
1123*00b67f09SDavid van Moolenbroek 			result = isc_parse_uint32(&timeout,
1124*00b67f09SDavid van Moolenbroek 						  isc_commandline_argument, 10);
1125*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1126*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "bad timeout '%s'\n",						isc_commandline_argument);
1127*00b67f09SDavid van Moolenbroek 				exit(1);
1128*00b67f09SDavid van Moolenbroek 			}
1129*00b67f09SDavid van Moolenbroek 			if (timeout == 0)
1130*00b67f09SDavid van Moolenbroek 				timeout = UINT_MAX;
1131*00b67f09SDavid van Moolenbroek 			break;
1132*00b67f09SDavid van Moolenbroek 		case 'u':
1133*00b67f09SDavid van Moolenbroek 			result = isc_parse_uint32(&udp_timeout,
1134*00b67f09SDavid van Moolenbroek 						  isc_commandline_argument, 10);
1135*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1136*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "bad udp timeout '%s'\n",						isc_commandline_argument);
1137*00b67f09SDavid van Moolenbroek 				exit(1);
1138*00b67f09SDavid van Moolenbroek 			}
1139*00b67f09SDavid van Moolenbroek 			if (udp_timeout == 0)
1140*00b67f09SDavid van Moolenbroek 				udp_timeout = UINT_MAX;
1141*00b67f09SDavid van Moolenbroek 			break;
1142*00b67f09SDavid van Moolenbroek 		case 'r':
1143*00b67f09SDavid van Moolenbroek 			result = isc_parse_uint32(&udp_retries,
1144*00b67f09SDavid van Moolenbroek 						  isc_commandline_argument, 10);
1145*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1146*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "bad udp retries '%s'\n",						isc_commandline_argument);
1147*00b67f09SDavid van Moolenbroek 				exit(1);
1148*00b67f09SDavid van Moolenbroek 			}
1149*00b67f09SDavid van Moolenbroek 			break;
1150*00b67f09SDavid van Moolenbroek 
1151*00b67f09SDavid van Moolenbroek 		case 'R':
1152*00b67f09SDavid van Moolenbroek 			setup_entropy(mctx, isc_commandline_argument, ectx);
1153*00b67f09SDavid van Moolenbroek 			break;
1154*00b67f09SDavid van Moolenbroek 
1155*00b67f09SDavid van Moolenbroek 		default:
1156*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "%s: unhandled option: %c\n",
1157*00b67f09SDavid van Moolenbroek 				argv[0], isc_commandline_option);
1158*00b67f09SDavid van Moolenbroek 			exit(1);
1159*00b67f09SDavid van Moolenbroek 		}
1160*00b67f09SDavid van Moolenbroek 	}
1161*00b67f09SDavid van Moolenbroek 	if (keyfile != NULL && keystr != NULL) {
1162*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "%s: cannot specify both -k and -y\n",
1163*00b67f09SDavid van Moolenbroek 			argv[0]);
1164*00b67f09SDavid van Moolenbroek 		exit(1);
1165*00b67f09SDavid van Moolenbroek 	}
1166*00b67f09SDavid van Moolenbroek 
1167*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
1168*00b67f09SDavid van Moolenbroek 	if (usegsstsig && (keyfile != NULL || keystr != NULL)) {
1169*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "%s: cannot specify -g with -k or -y\n",
1170*00b67f09SDavid van Moolenbroek 			argv[0]);
1171*00b67f09SDavid van Moolenbroek 		exit(1);
1172*00b67f09SDavid van Moolenbroek 	}
1173*00b67f09SDavid van Moolenbroek #else
1174*00b67f09SDavid van Moolenbroek 	if (usegsstsig) {
1175*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "%s: cannot specify -g	or -o, " \
1176*00b67f09SDavid van Moolenbroek 			"program not linked with GSS API Library\n",
1177*00b67f09SDavid van Moolenbroek 			argv[0]);
1178*00b67f09SDavid van Moolenbroek 		exit(1);
1179*00b67f09SDavid van Moolenbroek 	}
1180*00b67f09SDavid van Moolenbroek #endif
1181*00b67f09SDavid van Moolenbroek 
1182*00b67f09SDavid van Moolenbroek 	if (argv[isc_commandline_index] != NULL) {
1183*00b67f09SDavid van Moolenbroek 		if (strcmp(argv[isc_commandline_index], "-") == 0) {
1184*00b67f09SDavid van Moolenbroek 			input = stdin;
1185*00b67f09SDavid van Moolenbroek 		} else {
1186*00b67f09SDavid van Moolenbroek 			result = isc_stdio_open(argv[isc_commandline_index],
1187*00b67f09SDavid van Moolenbroek 						"r", &input);
1188*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1189*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "could not open '%s': %s\n",
1190*00b67f09SDavid van Moolenbroek 					argv[isc_commandline_index],
1191*00b67f09SDavid van Moolenbroek 					isc_result_totext(result));
1192*00b67f09SDavid van Moolenbroek 				exit(1);
1193*00b67f09SDavid van Moolenbroek 			}
1194*00b67f09SDavid van Moolenbroek 		}
1195*00b67f09SDavid van Moolenbroek 		interactive = ISC_FALSE;
1196*00b67f09SDavid van Moolenbroek 	}
1197*00b67f09SDavid van Moolenbroek }
1198*00b67f09SDavid van Moolenbroek 
1199*00b67f09SDavid van Moolenbroek static isc_uint16_t
parse_name(char ** cmdlinep,dns_message_t * msg,dns_name_t ** namep)1200*00b67f09SDavid van Moolenbroek parse_name(char **cmdlinep, dns_message_t *msg, dns_name_t **namep) {
1201*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1202*00b67f09SDavid van Moolenbroek 	char *word;
1203*00b67f09SDavid van Moolenbroek 	isc_buffer_t *namebuf = NULL;
1204*00b67f09SDavid van Moolenbroek 	isc_buffer_t source;
1205*00b67f09SDavid van Moolenbroek 
1206*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(cmdlinep, " \t\r\n");
1207*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1208*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read owner name\n");
1209*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1210*00b67f09SDavid van Moolenbroek 	}
1211*00b67f09SDavid van Moolenbroek 
1212*00b67f09SDavid van Moolenbroek 	result = dns_message_gettempname(msg, namep);
1213*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettempname");
1214*00b67f09SDavid van Moolenbroek 	result = isc_buffer_allocate(gmctx, &namebuf, DNS_NAME_MAXWIRE);
1215*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_buffer_allocate");
1216*00b67f09SDavid van Moolenbroek 	dns_name_init(*namep, NULL);
1217*00b67f09SDavid van Moolenbroek 	dns_name_setbuffer(*namep, namebuf);
1218*00b67f09SDavid van Moolenbroek 	dns_message_takebuffer(msg, &namebuf);
1219*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&source, word, strlen(word));
1220*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&source, strlen(word));
1221*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(*namep, &source, dns_rootname, 0, NULL);
1222*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_name_fromtext");
1223*00b67f09SDavid van Moolenbroek 	isc_buffer_invalidate(&source);
1224*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1225*00b67f09SDavid van Moolenbroek }
1226*00b67f09SDavid van Moolenbroek 
1227*00b67f09SDavid van Moolenbroek static isc_uint16_t
parse_rdata(char ** cmdlinep,dns_rdataclass_t rdataclass,dns_rdatatype_t rdatatype,dns_message_t * msg,dns_rdata_t * rdata)1228*00b67f09SDavid van Moolenbroek parse_rdata(char **cmdlinep, dns_rdataclass_t rdataclass,
1229*00b67f09SDavid van Moolenbroek 	    dns_rdatatype_t rdatatype, dns_message_t *msg,
1230*00b67f09SDavid van Moolenbroek 	    dns_rdata_t *rdata)
1231*00b67f09SDavid van Moolenbroek {
1232*00b67f09SDavid van Moolenbroek 	char *cmdline = *cmdlinep;
1233*00b67f09SDavid van Moolenbroek 	isc_buffer_t source, *buf = NULL, *newbuf = NULL;
1234*00b67f09SDavid van Moolenbroek 	isc_region_t r;
1235*00b67f09SDavid van Moolenbroek 	isc_lex_t *lex = NULL;
1236*00b67f09SDavid van Moolenbroek 	dns_rdatacallbacks_t callbacks;
1237*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1238*00b67f09SDavid van Moolenbroek 
1239*00b67f09SDavid van Moolenbroek 	if (cmdline == NULL) {
1240*00b67f09SDavid van Moolenbroek 		rdata->flags = DNS_RDATA_UPDATE;
1241*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
1242*00b67f09SDavid van Moolenbroek 	}
1243*00b67f09SDavid van Moolenbroek 
1244*00b67f09SDavid van Moolenbroek 	while (*cmdline != 0 && isspace((unsigned char)*cmdline))
1245*00b67f09SDavid van Moolenbroek 		cmdline++;
1246*00b67f09SDavid van Moolenbroek 
1247*00b67f09SDavid van Moolenbroek 	if (*cmdline != 0) {
1248*00b67f09SDavid van Moolenbroek 		dns_rdatacallbacks_init(&callbacks);
1249*00b67f09SDavid van Moolenbroek 		result = isc_lex_create(gmctx, strlen(cmdline), &lex);
1250*00b67f09SDavid van Moolenbroek 		check_result(result, "isc_lex_create");
1251*00b67f09SDavid van Moolenbroek 		isc_buffer_init(&source, cmdline, strlen(cmdline));
1252*00b67f09SDavid van Moolenbroek 		isc_buffer_add(&source, strlen(cmdline));
1253*00b67f09SDavid van Moolenbroek 		result = isc_lex_openbuffer(lex, &source);
1254*00b67f09SDavid van Moolenbroek 		check_result(result, "isc_lex_openbuffer");
1255*00b67f09SDavid van Moolenbroek 		result = isc_buffer_allocate(gmctx, &buf, MAXWIRE);
1256*00b67f09SDavid van Moolenbroek 		check_result(result, "isc_buffer_allocate");
1257*00b67f09SDavid van Moolenbroek 		result = dns_rdata_fromtext(NULL, rdataclass, rdatatype, lex,
1258*00b67f09SDavid van Moolenbroek 					    dns_rootname, 0, gmctx, buf,
1259*00b67f09SDavid van Moolenbroek 					    &callbacks);
1260*00b67f09SDavid van Moolenbroek 		isc_lex_destroy(&lex);
1261*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_SUCCESS) {
1262*00b67f09SDavid van Moolenbroek 			isc_buffer_usedregion(buf, &r);
1263*00b67f09SDavid van Moolenbroek 			result = isc_buffer_allocate(gmctx, &newbuf, r.length);
1264*00b67f09SDavid van Moolenbroek 			check_result(result, "isc_buffer_allocate");
1265*00b67f09SDavid van Moolenbroek 			isc_buffer_putmem(newbuf, r.base, r.length);
1266*00b67f09SDavid van Moolenbroek 			isc_buffer_usedregion(newbuf, &r);
1267*00b67f09SDavid van Moolenbroek 			dns_rdata_fromregion(rdata, rdataclass, rdatatype, &r);
1268*00b67f09SDavid van Moolenbroek 			isc_buffer_free(&buf);
1269*00b67f09SDavid van Moolenbroek 			dns_message_takebuffer(msg, &newbuf);
1270*00b67f09SDavid van Moolenbroek 		} else {
1271*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "invalid rdata format: %s\n",
1272*00b67f09SDavid van Moolenbroek 				isc_result_totext(result));
1273*00b67f09SDavid van Moolenbroek 			isc_buffer_free(&buf);
1274*00b67f09SDavid van Moolenbroek 			return (STATUS_SYNTAX);
1275*00b67f09SDavid van Moolenbroek 		}
1276*00b67f09SDavid van Moolenbroek 	} else {
1277*00b67f09SDavid van Moolenbroek 		rdata->flags = DNS_RDATA_UPDATE;
1278*00b67f09SDavid van Moolenbroek 	}
1279*00b67f09SDavid van Moolenbroek 	*cmdlinep = cmdline;
1280*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1281*00b67f09SDavid van Moolenbroek }
1282*00b67f09SDavid van Moolenbroek 
1283*00b67f09SDavid van Moolenbroek static isc_uint16_t
make_prereq(char * cmdline,isc_boolean_t ispositive,isc_boolean_t isrrset)1284*00b67f09SDavid van Moolenbroek make_prereq(char *cmdline, isc_boolean_t ispositive, isc_boolean_t isrrset) {
1285*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1286*00b67f09SDavid van Moolenbroek 	char *word;
1287*00b67f09SDavid van Moolenbroek 	dns_name_t *name = NULL;
1288*00b67f09SDavid van Moolenbroek 	isc_textregion_t region;
1289*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *rdataset = NULL;
1290*00b67f09SDavid van Moolenbroek 	dns_rdatalist_t *rdatalist = NULL;
1291*00b67f09SDavid van Moolenbroek 	dns_rdataclass_t rdataclass;
1292*00b67f09SDavid van Moolenbroek 	dns_rdatatype_t rdatatype;
1293*00b67f09SDavid van Moolenbroek 	dns_rdata_t *rdata = NULL;
1294*00b67f09SDavid van Moolenbroek 	isc_uint16_t retval;
1295*00b67f09SDavid van Moolenbroek 
1296*00b67f09SDavid van Moolenbroek 	ddebug("make_prereq()");
1297*00b67f09SDavid van Moolenbroek 
1298*00b67f09SDavid van Moolenbroek 	/*
1299*00b67f09SDavid van Moolenbroek 	 * Read the owner name
1300*00b67f09SDavid van Moolenbroek 	 */
1301*00b67f09SDavid van Moolenbroek 	retval = parse_name(&cmdline, updatemsg, &name);
1302*00b67f09SDavid van Moolenbroek 	if (retval != STATUS_MORE)
1303*00b67f09SDavid van Moolenbroek 		return (retval);
1304*00b67f09SDavid van Moolenbroek 
1305*00b67f09SDavid van Moolenbroek 	/*
1306*00b67f09SDavid van Moolenbroek 	 * If this is an rrset prereq, read the class or type.
1307*00b67f09SDavid van Moolenbroek 	 */
1308*00b67f09SDavid van Moolenbroek 	if (isrrset) {
1309*00b67f09SDavid van Moolenbroek 		word = nsu_strsep(&cmdline, " \t\r\n");
1310*00b67f09SDavid van Moolenbroek 		if (word == NULL || *word == 0) {
1311*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "could not read class or type\n");
1312*00b67f09SDavid van Moolenbroek 			goto failure;
1313*00b67f09SDavid van Moolenbroek 		}
1314*00b67f09SDavid van Moolenbroek 		region.base = word;
1315*00b67f09SDavid van Moolenbroek 		region.length = strlen(word);
1316*00b67f09SDavid van Moolenbroek 		result = dns_rdataclass_fromtext(&rdataclass, &region);
1317*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_SUCCESS) {
1318*00b67f09SDavid van Moolenbroek 			if (!setzoneclass(rdataclass)) {
1319*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "class mismatch: %s\n", word);
1320*00b67f09SDavid van Moolenbroek 				goto failure;
1321*00b67f09SDavid van Moolenbroek 			}
1322*00b67f09SDavid van Moolenbroek 			/*
1323*00b67f09SDavid van Moolenbroek 			 * Now read the type.
1324*00b67f09SDavid van Moolenbroek 			 */
1325*00b67f09SDavid van Moolenbroek 			word = nsu_strsep(&cmdline, " \t\r\n");
1326*00b67f09SDavid van Moolenbroek 			if (word == NULL || *word == 0) {
1327*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "could not read type\n");
1328*00b67f09SDavid van Moolenbroek 				goto failure;
1329*00b67f09SDavid van Moolenbroek 			}
1330*00b67f09SDavid van Moolenbroek 			region.base = word;
1331*00b67f09SDavid van Moolenbroek 			region.length = strlen(word);
1332*00b67f09SDavid van Moolenbroek 			result = dns_rdatatype_fromtext(&rdatatype, &region);
1333*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1334*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "invalid type: %s\n", word);
1335*00b67f09SDavid van Moolenbroek 				goto failure;
1336*00b67f09SDavid van Moolenbroek 			}
1337*00b67f09SDavid van Moolenbroek 		} else {
1338*00b67f09SDavid van Moolenbroek 			rdataclass = getzoneclass();
1339*00b67f09SDavid van Moolenbroek 			result = dns_rdatatype_fromtext(&rdatatype, &region);
1340*00b67f09SDavid van Moolenbroek 			if (result != ISC_R_SUCCESS) {
1341*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "invalid type: %s\n", word);
1342*00b67f09SDavid van Moolenbroek 				goto failure;
1343*00b67f09SDavid van Moolenbroek 			}
1344*00b67f09SDavid van Moolenbroek 		}
1345*00b67f09SDavid van Moolenbroek 	} else
1346*00b67f09SDavid van Moolenbroek 		rdatatype = dns_rdatatype_any;
1347*00b67f09SDavid van Moolenbroek 
1348*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdata(updatemsg, &rdata);
1349*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdata");
1350*00b67f09SDavid van Moolenbroek 
1351*00b67f09SDavid van Moolenbroek 	dns_rdata_init(rdata);
1352*00b67f09SDavid van Moolenbroek 
1353*00b67f09SDavid van Moolenbroek 	if (isrrset && ispositive) {
1354*00b67f09SDavid van Moolenbroek 		retval = parse_rdata(&cmdline, rdataclass, rdatatype,
1355*00b67f09SDavid van Moolenbroek 				     updatemsg, rdata);
1356*00b67f09SDavid van Moolenbroek 		if (retval != STATUS_MORE)
1357*00b67f09SDavid van Moolenbroek 			goto failure;
1358*00b67f09SDavid van Moolenbroek 	} else
1359*00b67f09SDavid van Moolenbroek 		rdata->flags = DNS_RDATA_UPDATE;
1360*00b67f09SDavid van Moolenbroek 
1361*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdatalist(updatemsg, &rdatalist);
1362*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdatalist");
1363*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdataset(updatemsg, &rdataset);
1364*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdataset");
1365*00b67f09SDavid van Moolenbroek 	dns_rdatalist_init(rdatalist);
1366*00b67f09SDavid van Moolenbroek 	rdatalist->type = rdatatype;
1367*00b67f09SDavid van Moolenbroek 	if (ispositive) {
1368*00b67f09SDavid van Moolenbroek 		if (isrrset && rdata->data != NULL)
1369*00b67f09SDavid van Moolenbroek 			rdatalist->rdclass = rdataclass;
1370*00b67f09SDavid van Moolenbroek 		else
1371*00b67f09SDavid van Moolenbroek 			rdatalist->rdclass = dns_rdataclass_any;
1372*00b67f09SDavid van Moolenbroek 	} else
1373*00b67f09SDavid van Moolenbroek 		rdatalist->rdclass = dns_rdataclass_none;
1374*00b67f09SDavid van Moolenbroek 	rdatalist->covers = 0;
1375*00b67f09SDavid van Moolenbroek 	rdatalist->ttl = 0;
1376*00b67f09SDavid van Moolenbroek 	rdata->rdclass = rdatalist->rdclass;
1377*00b67f09SDavid van Moolenbroek 	rdata->type = rdatatype;
1378*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(rdatalist->rdata);
1379*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(rdatalist->rdata, rdata, link);
1380*00b67f09SDavid van Moolenbroek 	dns_rdatalist_tordataset(rdatalist, rdataset);
1381*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(name->list);
1382*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(name->list, rdataset, link);
1383*00b67f09SDavid van Moolenbroek 	dns_message_addname(updatemsg, name, DNS_SECTION_PREREQUISITE);
1384*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1385*00b67f09SDavid van Moolenbroek 
1386*00b67f09SDavid van Moolenbroek  failure:
1387*00b67f09SDavid van Moolenbroek 	if (name != NULL)
1388*00b67f09SDavid van Moolenbroek 		dns_message_puttempname(updatemsg, &name);
1389*00b67f09SDavid van Moolenbroek 	return (STATUS_SYNTAX);
1390*00b67f09SDavid van Moolenbroek }
1391*00b67f09SDavid van Moolenbroek 
1392*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_prereq(char * cmdline)1393*00b67f09SDavid van Moolenbroek evaluate_prereq(char *cmdline) {
1394*00b67f09SDavid van Moolenbroek 	char *word;
1395*00b67f09SDavid van Moolenbroek 	isc_boolean_t ispositive, isrrset;
1396*00b67f09SDavid van Moolenbroek 
1397*00b67f09SDavid van Moolenbroek 	ddebug("evaluate_prereq()");
1398*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1399*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1400*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read operation code\n");
1401*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1402*00b67f09SDavid van Moolenbroek 	}
1403*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "nxdomain") == 0) {
1404*00b67f09SDavid van Moolenbroek 		ispositive = ISC_FALSE;
1405*00b67f09SDavid van Moolenbroek 		isrrset = ISC_FALSE;
1406*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(word, "yxdomain") == 0) {
1407*00b67f09SDavid van Moolenbroek 		ispositive = ISC_TRUE;
1408*00b67f09SDavid van Moolenbroek 		isrrset = ISC_FALSE;
1409*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(word, "nxrrset") == 0) {
1410*00b67f09SDavid van Moolenbroek 		ispositive = ISC_FALSE;
1411*00b67f09SDavid van Moolenbroek 		isrrset = ISC_TRUE;
1412*00b67f09SDavid van Moolenbroek 	} else if (strcasecmp(word, "yxrrset") == 0) {
1413*00b67f09SDavid van Moolenbroek 		ispositive = ISC_TRUE;
1414*00b67f09SDavid van Moolenbroek 		isrrset = ISC_TRUE;
1415*00b67f09SDavid van Moolenbroek 	} else {
1416*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "incorrect operation code: %s\n", word);
1417*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1418*00b67f09SDavid van Moolenbroek 	}
1419*00b67f09SDavid van Moolenbroek 	return (make_prereq(cmdline, ispositive, isrrset));
1420*00b67f09SDavid van Moolenbroek }
1421*00b67f09SDavid van Moolenbroek 
1422*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_server(char * cmdline)1423*00b67f09SDavid van Moolenbroek evaluate_server(char *cmdline) {
1424*00b67f09SDavid van Moolenbroek 	char *word, *server;
1425*00b67f09SDavid van Moolenbroek 	long port;
1426*00b67f09SDavid van Moolenbroek 
1427*00b67f09SDavid van Moolenbroek 	if (local_only) {
1428*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "cannot reset server in localhost-only mode\n");
1429*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1430*00b67f09SDavid van Moolenbroek 	}
1431*00b67f09SDavid van Moolenbroek 
1432*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1433*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1434*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read server name\n");
1435*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1436*00b67f09SDavid van Moolenbroek 	}
1437*00b67f09SDavid van Moolenbroek 	server = word;
1438*00b67f09SDavid van Moolenbroek 
1439*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1440*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0)
1441*00b67f09SDavid van Moolenbroek 		port = dnsport;
1442*00b67f09SDavid van Moolenbroek 	else {
1443*00b67f09SDavid van Moolenbroek 		char *endp;
1444*00b67f09SDavid van Moolenbroek 		port = strtol(word, &endp, 10);
1445*00b67f09SDavid van Moolenbroek 		if (*endp != 0) {
1446*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "port '%s' is not numeric\n", word);
1447*00b67f09SDavid van Moolenbroek 			return (STATUS_SYNTAX);
1448*00b67f09SDavid van Moolenbroek 		} else if (port < 1 || port > 65535) {
1449*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "port '%s' is out of range "
1450*00b67f09SDavid van Moolenbroek 				"(1 to 65535)\n", word);
1451*00b67f09SDavid van Moolenbroek 			return (STATUS_SYNTAX);
1452*00b67f09SDavid van Moolenbroek 		}
1453*00b67f09SDavid van Moolenbroek 	}
1454*00b67f09SDavid van Moolenbroek 
1455*00b67f09SDavid van Moolenbroek 	if (servers != NULL) {
1456*00b67f09SDavid van Moolenbroek 		if (master_servers == servers)
1457*00b67f09SDavid van Moolenbroek 			master_servers = NULL;
1458*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, servers, ns_total * sizeof(isc_sockaddr_t));
1459*00b67f09SDavid van Moolenbroek 	}
1460*00b67f09SDavid van Moolenbroek 
1461*00b67f09SDavid van Moolenbroek 	default_servers = ISC_FALSE;
1462*00b67f09SDavid van Moolenbroek 
1463*00b67f09SDavid van Moolenbroek 	ns_total = MAX_SERVERADDRS;
1464*00b67f09SDavid van Moolenbroek 	ns_inuse = 0;
1465*00b67f09SDavid van Moolenbroek 	servers = isc_mem_get(gmctx, ns_total * sizeof(isc_sockaddr_t));
1466*00b67f09SDavid van Moolenbroek 	if (servers == NULL)
1467*00b67f09SDavid van Moolenbroek 		fatal("out of memory");
1468*00b67f09SDavid van Moolenbroek 
1469*00b67f09SDavid van Moolenbroek 	memset(servers, 0, ns_total * sizeof(isc_sockaddr_t));
1470*00b67f09SDavid van Moolenbroek 	get_addresses(server, (in_port_t)port, servers, ns_total);
1471*00b67f09SDavid van Moolenbroek 
1472*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1473*00b67f09SDavid van Moolenbroek }
1474*00b67f09SDavid van Moolenbroek 
1475*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_local(char * cmdline)1476*00b67f09SDavid van Moolenbroek evaluate_local(char *cmdline) {
1477*00b67f09SDavid van Moolenbroek 	char *word, *local;
1478*00b67f09SDavid van Moolenbroek 	long port;
1479*00b67f09SDavid van Moolenbroek 	struct in_addr in4;
1480*00b67f09SDavid van Moolenbroek 	struct in6_addr in6;
1481*00b67f09SDavid van Moolenbroek 
1482*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1483*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1484*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read server name\n");
1485*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1486*00b67f09SDavid van Moolenbroek 	}
1487*00b67f09SDavid van Moolenbroek 	local = word;
1488*00b67f09SDavid van Moolenbroek 
1489*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1490*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0)
1491*00b67f09SDavid van Moolenbroek 		port = 0;
1492*00b67f09SDavid van Moolenbroek 	else {
1493*00b67f09SDavid van Moolenbroek 		char *endp;
1494*00b67f09SDavid van Moolenbroek 		port = strtol(word, &endp, 10);
1495*00b67f09SDavid van Moolenbroek 		if (*endp != 0) {
1496*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "port '%s' is not numeric\n", word);
1497*00b67f09SDavid van Moolenbroek 			return (STATUS_SYNTAX);
1498*00b67f09SDavid van Moolenbroek 		} else if (port < 1 || port > 65535) {
1499*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "port '%s' is out of range "
1500*00b67f09SDavid van Moolenbroek 				"(1 to 65535)\n", word);
1501*00b67f09SDavid van Moolenbroek 			return (STATUS_SYNTAX);
1502*00b67f09SDavid van Moolenbroek 		}
1503*00b67f09SDavid van Moolenbroek 	}
1504*00b67f09SDavid van Moolenbroek 
1505*00b67f09SDavid van Moolenbroek 	if (have_ipv6 && inet_pton(AF_INET6, local, &in6) == 1) {
1506*00b67f09SDavid van Moolenbroek 		if (localaddr6 == NULL)
1507*00b67f09SDavid van Moolenbroek 			localaddr6 = isc_mem_get(gmctx, sizeof(isc_sockaddr_t));
1508*00b67f09SDavid van Moolenbroek 		if (localaddr6 == NULL)
1509*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
1510*00b67f09SDavid van Moolenbroek 		isc_sockaddr_fromin6(localaddr6, &in6, (in_port_t)port);
1511*00b67f09SDavid van Moolenbroek 	} else if (have_ipv4 && inet_pton(AF_INET, local, &in4) == 1) {
1512*00b67f09SDavid van Moolenbroek 		if (localaddr4 == NULL)
1513*00b67f09SDavid van Moolenbroek 			localaddr4 = isc_mem_get(gmctx, sizeof(isc_sockaddr_t));
1514*00b67f09SDavid van Moolenbroek 		if (localaddr4 == NULL)
1515*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
1516*00b67f09SDavid van Moolenbroek 		isc_sockaddr_fromin(localaddr4, &in4, (in_port_t)port);
1517*00b67f09SDavid van Moolenbroek 	} else {
1518*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "invalid address %s", local);
1519*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1520*00b67f09SDavid van Moolenbroek 	}
1521*00b67f09SDavid van Moolenbroek 
1522*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1523*00b67f09SDavid van Moolenbroek }
1524*00b67f09SDavid van Moolenbroek 
1525*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_key(char * cmdline)1526*00b67f09SDavid van Moolenbroek evaluate_key(char *cmdline) {
1527*00b67f09SDavid van Moolenbroek 	char *namestr;
1528*00b67f09SDavid van Moolenbroek 	char *secretstr;
1529*00b67f09SDavid van Moolenbroek 	isc_buffer_t b;
1530*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1531*00b67f09SDavid van Moolenbroek 	dns_fixedname_t fkeyname;
1532*00b67f09SDavid van Moolenbroek 	dns_name_t *mykeyname;
1533*00b67f09SDavid van Moolenbroek 	int secretlen;
1534*00b67f09SDavid van Moolenbroek 	unsigned char *secret = NULL;
1535*00b67f09SDavid van Moolenbroek 	isc_buffer_t secretbuf;
1536*00b67f09SDavid van Moolenbroek 	dns_name_t *hmacname = NULL;
1537*00b67f09SDavid van Moolenbroek 	isc_uint16_t digestbits = 0;
1538*00b67f09SDavid van Moolenbroek 	char *n;
1539*00b67f09SDavid van Moolenbroek 
1540*00b67f09SDavid van Moolenbroek 	namestr = nsu_strsep(&cmdline, " \t\r\n");
1541*00b67f09SDavid van Moolenbroek 	if (namestr == NULL || *namestr == 0) {
1542*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read key name\n");
1543*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1544*00b67f09SDavid van Moolenbroek 	}
1545*00b67f09SDavid van Moolenbroek 
1546*00b67f09SDavid van Moolenbroek 	dns_fixedname_init(&fkeyname);
1547*00b67f09SDavid van Moolenbroek 	mykeyname = dns_fixedname_name(&fkeyname);
1548*00b67f09SDavid van Moolenbroek 
1549*00b67f09SDavid van Moolenbroek 	n = strchr(namestr, ':');
1550*00b67f09SDavid van Moolenbroek 	if (n != NULL) {
1551*00b67f09SDavid van Moolenbroek 		digestbits = parse_hmac(&hmacname, namestr, n - namestr);
1552*00b67f09SDavid van Moolenbroek 		namestr = n + 1;
1553*00b67f09SDavid van Moolenbroek 	} else
1554*00b67f09SDavid van Moolenbroek 		hmacname = DNS_TSIG_HMACMD5_NAME;
1555*00b67f09SDavid van Moolenbroek 
1556*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&b, namestr, strlen(namestr));
1557*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&b, strlen(namestr));
1558*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(mykeyname, &b, dns_rootname, 0, NULL);
1559*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1560*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not parse key name\n");
1561*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1562*00b67f09SDavid van Moolenbroek 	}
1563*00b67f09SDavid van Moolenbroek 
1564*00b67f09SDavid van Moolenbroek 	secretstr = nsu_strsep(&cmdline, "\r\n");
1565*00b67f09SDavid van Moolenbroek 	if (secretstr == NULL || *secretstr == 0) {
1566*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read key secret\n");
1567*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1568*00b67f09SDavid van Moolenbroek 	}
1569*00b67f09SDavid van Moolenbroek 	secretlen = strlen(secretstr) * 3 / 4;
1570*00b67f09SDavid van Moolenbroek 	secret = isc_mem_allocate(gmctx, secretlen);
1571*00b67f09SDavid van Moolenbroek 	if (secret == NULL)
1572*00b67f09SDavid van Moolenbroek 		fatal("out of memory");
1573*00b67f09SDavid van Moolenbroek 
1574*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&secretbuf, secret, secretlen);
1575*00b67f09SDavid van Moolenbroek 	result = isc_base64_decodestring(secretstr, &secretbuf);
1576*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1577*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not create key from %s: %s\n",
1578*00b67f09SDavid van Moolenbroek 			secretstr, isc_result_totext(result));
1579*00b67f09SDavid van Moolenbroek 		isc_mem_free(gmctx, secret);
1580*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1581*00b67f09SDavid van Moolenbroek 	}
1582*00b67f09SDavid van Moolenbroek 	secretlen = isc_buffer_usedlength(&secretbuf);
1583*00b67f09SDavid van Moolenbroek 
1584*00b67f09SDavid van Moolenbroek 	if (tsigkey != NULL)
1585*00b67f09SDavid van Moolenbroek 		dns_tsigkey_detach(&tsigkey);
1586*00b67f09SDavid van Moolenbroek 	result = dns_tsigkey_create(mykeyname, hmacname, secret, secretlen,
1587*00b67f09SDavid van Moolenbroek 				    ISC_FALSE, NULL, 0, 0, gmctx, NULL,
1588*00b67f09SDavid van Moolenbroek 				    &tsigkey);
1589*00b67f09SDavid van Moolenbroek 	isc_mem_free(gmctx, secret);
1590*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1591*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not create key from %s %s: %s\n",
1592*00b67f09SDavid van Moolenbroek 			namestr, secretstr, dns_result_totext(result));
1593*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1594*00b67f09SDavid van Moolenbroek 	}
1595*00b67f09SDavid van Moolenbroek 	dst_key_setbits(tsigkey->key, digestbits);
1596*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1597*00b67f09SDavid van Moolenbroek }
1598*00b67f09SDavid van Moolenbroek 
1599*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_zone(char * cmdline)1600*00b67f09SDavid van Moolenbroek evaluate_zone(char *cmdline) {
1601*00b67f09SDavid van Moolenbroek 	char *word;
1602*00b67f09SDavid van Moolenbroek 	isc_buffer_t b;
1603*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1604*00b67f09SDavid van Moolenbroek 
1605*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1606*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1607*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read zone name\n");
1608*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1609*00b67f09SDavid van Moolenbroek 	}
1610*00b67f09SDavid van Moolenbroek 
1611*00b67f09SDavid van Moolenbroek 	dns_fixedname_init(&fuserzone);
1612*00b67f09SDavid van Moolenbroek 	userzone = dns_fixedname_name(&fuserzone);
1613*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&b, word, strlen(word));
1614*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&b, strlen(word));
1615*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(userzone, &b, dns_rootname, 0, NULL);
1616*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1617*00b67f09SDavid van Moolenbroek 		userzone = NULL; /* Lest it point to an invalid name */
1618*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not parse zone name\n");
1619*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1620*00b67f09SDavid van Moolenbroek 	}
1621*00b67f09SDavid van Moolenbroek 
1622*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1623*00b67f09SDavid van Moolenbroek }
1624*00b67f09SDavid van Moolenbroek 
1625*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_realm(char * cmdline)1626*00b67f09SDavid van Moolenbroek evaluate_realm(char *cmdline) {
1627*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
1628*00b67f09SDavid van Moolenbroek 	char *word;
1629*00b67f09SDavid van Moolenbroek 	char buf[1024];
1630*00b67f09SDavid van Moolenbroek 	int n;
1631*00b67f09SDavid van Moolenbroek 
1632*00b67f09SDavid van Moolenbroek 	if (realm != NULL) {
1633*00b67f09SDavid van Moolenbroek 		isc_mem_free(gmctx, realm);
1634*00b67f09SDavid van Moolenbroek 		realm = NULL;
1635*00b67f09SDavid van Moolenbroek 	}
1636*00b67f09SDavid van Moolenbroek 
1637*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1638*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0)
1639*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
1640*00b67f09SDavid van Moolenbroek 
1641*00b67f09SDavid van Moolenbroek 	n = snprintf(buf, sizeof(buf), "@%s", word);
1642*00b67f09SDavid van Moolenbroek 	if (n < 0 || (size_t)n >= sizeof(buf))
1643*00b67f09SDavid van Moolenbroek 		fatal("realm is too long");
1644*00b67f09SDavid van Moolenbroek 	realm = isc_mem_strdup(gmctx, buf);
1645*00b67f09SDavid van Moolenbroek 	if (realm == NULL)
1646*00b67f09SDavid van Moolenbroek 		fatal("out of memory");
1647*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1648*00b67f09SDavid van Moolenbroek #else
1649*00b67f09SDavid van Moolenbroek 	UNUSED(cmdline);
1650*00b67f09SDavid van Moolenbroek 	return (STATUS_SYNTAX);
1651*00b67f09SDavid van Moolenbroek #endif
1652*00b67f09SDavid van Moolenbroek }
1653*00b67f09SDavid van Moolenbroek 
1654*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_ttl(char * cmdline)1655*00b67f09SDavid van Moolenbroek evaluate_ttl(char *cmdline) {
1656*00b67f09SDavid van Moolenbroek 	char *word;
1657*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1658*00b67f09SDavid van Moolenbroek 	isc_uint32_t ttl;
1659*00b67f09SDavid van Moolenbroek 
1660*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1661*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1662*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not ttl\n");
1663*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1664*00b67f09SDavid van Moolenbroek 	}
1665*00b67f09SDavid van Moolenbroek 
1666*00b67f09SDavid van Moolenbroek 	if (!strcasecmp(word, "none")) {
1667*00b67f09SDavid van Moolenbroek 		default_ttl = 0;
1668*00b67f09SDavid van Moolenbroek 		default_ttl_set = ISC_FALSE;
1669*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
1670*00b67f09SDavid van Moolenbroek 	}
1671*00b67f09SDavid van Moolenbroek 
1672*00b67f09SDavid van Moolenbroek 	result = isc_parse_uint32(&ttl, word, 10);
1673*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
1674*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1675*00b67f09SDavid van Moolenbroek 
1676*00b67f09SDavid van Moolenbroek 	if (ttl > TTL_MAX) {
1677*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "ttl '%s' is out of range (0 to %u)\n",
1678*00b67f09SDavid van Moolenbroek 			word, TTL_MAX);
1679*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1680*00b67f09SDavid van Moolenbroek 	}
1681*00b67f09SDavid van Moolenbroek 	default_ttl = ttl;
1682*00b67f09SDavid van Moolenbroek 	default_ttl_set = ISC_TRUE;
1683*00b67f09SDavid van Moolenbroek 
1684*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1685*00b67f09SDavid van Moolenbroek }
1686*00b67f09SDavid van Moolenbroek 
1687*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_class(char * cmdline)1688*00b67f09SDavid van Moolenbroek evaluate_class(char *cmdline) {
1689*00b67f09SDavid van Moolenbroek 	char *word;
1690*00b67f09SDavid van Moolenbroek 	isc_textregion_t r;
1691*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1692*00b67f09SDavid van Moolenbroek 	dns_rdataclass_t rdclass;
1693*00b67f09SDavid van Moolenbroek 
1694*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1695*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1696*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read class name\n");
1697*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1698*00b67f09SDavid van Moolenbroek 	}
1699*00b67f09SDavid van Moolenbroek 
1700*00b67f09SDavid van Moolenbroek 	r.base = word;
1701*00b67f09SDavid van Moolenbroek 	r.length = strlen(word);
1702*00b67f09SDavid van Moolenbroek 	result = dns_rdataclass_fromtext(&rdclass, &r);
1703*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1704*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not parse class name: %s\n", word);
1705*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1706*00b67f09SDavid van Moolenbroek 	}
1707*00b67f09SDavid van Moolenbroek 	switch (rdclass) {
1708*00b67f09SDavid van Moolenbroek 	case dns_rdataclass_none:
1709*00b67f09SDavid van Moolenbroek 	case dns_rdataclass_any:
1710*00b67f09SDavid van Moolenbroek 	case dns_rdataclass_reserved0:
1711*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "bad default class: %s\n", word);
1712*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1713*00b67f09SDavid van Moolenbroek 	default:
1714*00b67f09SDavid van Moolenbroek 		defaultclass = rdclass;
1715*00b67f09SDavid van Moolenbroek 	}
1716*00b67f09SDavid van Moolenbroek 
1717*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1718*00b67f09SDavid van Moolenbroek }
1719*00b67f09SDavid van Moolenbroek 
1720*00b67f09SDavid van Moolenbroek static isc_uint16_t
update_addordelete(char * cmdline,isc_boolean_t isdelete)1721*00b67f09SDavid van Moolenbroek update_addordelete(char *cmdline, isc_boolean_t isdelete) {
1722*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1723*00b67f09SDavid van Moolenbroek 	dns_name_t *name = NULL;
1724*00b67f09SDavid van Moolenbroek 	isc_uint32_t ttl;
1725*00b67f09SDavid van Moolenbroek 	char *word;
1726*00b67f09SDavid van Moolenbroek 	dns_rdataclass_t rdataclass;
1727*00b67f09SDavid van Moolenbroek 	dns_rdatatype_t rdatatype;
1728*00b67f09SDavid van Moolenbroek 	dns_rdata_t *rdata = NULL;
1729*00b67f09SDavid van Moolenbroek 	dns_rdatalist_t *rdatalist = NULL;
1730*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *rdataset = NULL;
1731*00b67f09SDavid van Moolenbroek 	isc_textregion_t region;
1732*00b67f09SDavid van Moolenbroek 	isc_uint16_t retval;
1733*00b67f09SDavid van Moolenbroek 
1734*00b67f09SDavid van Moolenbroek 	ddebug("update_addordelete()");
1735*00b67f09SDavid van Moolenbroek 
1736*00b67f09SDavid van Moolenbroek 	/*
1737*00b67f09SDavid van Moolenbroek 	 * Read the owner name.
1738*00b67f09SDavid van Moolenbroek 	 */
1739*00b67f09SDavid van Moolenbroek 	retval = parse_name(&cmdline, updatemsg, &name);
1740*00b67f09SDavid van Moolenbroek 	if (retval != STATUS_MORE)
1741*00b67f09SDavid van Moolenbroek 		return (retval);
1742*00b67f09SDavid van Moolenbroek 
1743*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdata(updatemsg, &rdata);
1744*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdata");
1745*00b67f09SDavid van Moolenbroek 
1746*00b67f09SDavid van Moolenbroek 	dns_rdata_init(rdata);
1747*00b67f09SDavid van Moolenbroek 
1748*00b67f09SDavid van Moolenbroek 	/*
1749*00b67f09SDavid van Moolenbroek 	 * If this is an add, read the TTL and verify that it's in range.
1750*00b67f09SDavid van Moolenbroek 	 * If it's a delete, ignore a TTL if present (for compatibility).
1751*00b67f09SDavid van Moolenbroek 	 */
1752*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1753*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1754*00b67f09SDavid van Moolenbroek 		if (!isdelete) {
1755*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "could not read owner ttl\n");
1756*00b67f09SDavid van Moolenbroek 			goto failure;
1757*00b67f09SDavid van Moolenbroek 		}
1758*00b67f09SDavid van Moolenbroek 		else {
1759*00b67f09SDavid van Moolenbroek 			ttl = 0;
1760*00b67f09SDavid van Moolenbroek 			rdataclass = dns_rdataclass_any;
1761*00b67f09SDavid van Moolenbroek 			rdatatype = dns_rdatatype_any;
1762*00b67f09SDavid van Moolenbroek 			rdata->flags = DNS_RDATA_UPDATE;
1763*00b67f09SDavid van Moolenbroek 			goto doneparsing;
1764*00b67f09SDavid van Moolenbroek 		}
1765*00b67f09SDavid van Moolenbroek 	}
1766*00b67f09SDavid van Moolenbroek 	result = isc_parse_uint32(&ttl, word, 10);
1767*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1768*00b67f09SDavid van Moolenbroek 		if (isdelete) {
1769*00b67f09SDavid van Moolenbroek 			ttl = 0;
1770*00b67f09SDavid van Moolenbroek 			goto parseclass;
1771*00b67f09SDavid van Moolenbroek 		} else if (default_ttl_set) {
1772*00b67f09SDavid van Moolenbroek 			ttl = default_ttl;
1773*00b67f09SDavid van Moolenbroek 			goto parseclass;
1774*00b67f09SDavid van Moolenbroek 		} else {
1775*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "ttl '%s': %s\n", word,
1776*00b67f09SDavid van Moolenbroek 				isc_result_totext(result));
1777*00b67f09SDavid van Moolenbroek 			goto failure;
1778*00b67f09SDavid van Moolenbroek 		}
1779*00b67f09SDavid van Moolenbroek 	}
1780*00b67f09SDavid van Moolenbroek 
1781*00b67f09SDavid van Moolenbroek 	if (isdelete)
1782*00b67f09SDavid van Moolenbroek 		ttl = 0;
1783*00b67f09SDavid van Moolenbroek 	else if (ttl > TTL_MAX) {
1784*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "ttl '%s' is out of range (0 to %u)\n",
1785*00b67f09SDavid van Moolenbroek 			word, TTL_MAX);
1786*00b67f09SDavid van Moolenbroek 		goto failure;
1787*00b67f09SDavid van Moolenbroek 	}
1788*00b67f09SDavid van Moolenbroek 
1789*00b67f09SDavid van Moolenbroek 	/*
1790*00b67f09SDavid van Moolenbroek 	 * Read the class or type.
1791*00b67f09SDavid van Moolenbroek 	 */
1792*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1793*00b67f09SDavid van Moolenbroek  parseclass:
1794*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1795*00b67f09SDavid van Moolenbroek 		if (isdelete) {
1796*00b67f09SDavid van Moolenbroek 			rdataclass = dns_rdataclass_any;
1797*00b67f09SDavid van Moolenbroek 			rdatatype = dns_rdatatype_any;
1798*00b67f09SDavid van Moolenbroek 			rdata->flags = DNS_RDATA_UPDATE;
1799*00b67f09SDavid van Moolenbroek 			goto doneparsing;
1800*00b67f09SDavid van Moolenbroek 		} else {
1801*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "could not read class or type\n");
1802*00b67f09SDavid van Moolenbroek 			goto failure;
1803*00b67f09SDavid van Moolenbroek 		}
1804*00b67f09SDavid van Moolenbroek 	}
1805*00b67f09SDavid van Moolenbroek 	region.base = word;
1806*00b67f09SDavid van Moolenbroek 	region.length = strlen(word);
1807*00b67f09SDavid van Moolenbroek 	rdataclass = dns_rdataclass_any;
1808*00b67f09SDavid van Moolenbroek 	result = dns_rdataclass_fromtext(&rdataclass, &region);
1809*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_SUCCESS && rdataclass != dns_rdataclass_any) {
1810*00b67f09SDavid van Moolenbroek 		if (!setzoneclass(rdataclass)) {
1811*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "class mismatch: %s\n", word);
1812*00b67f09SDavid van Moolenbroek 			goto failure;
1813*00b67f09SDavid van Moolenbroek 		}
1814*00b67f09SDavid van Moolenbroek 		/*
1815*00b67f09SDavid van Moolenbroek 		 * Now read the type.
1816*00b67f09SDavid van Moolenbroek 		 */
1817*00b67f09SDavid van Moolenbroek 		word = nsu_strsep(&cmdline, " \t\r\n");
1818*00b67f09SDavid van Moolenbroek 		if (word == NULL || *word == 0) {
1819*00b67f09SDavid van Moolenbroek 			if (isdelete) {
1820*00b67f09SDavid van Moolenbroek 				rdataclass = dns_rdataclass_any;
1821*00b67f09SDavid van Moolenbroek 				rdatatype = dns_rdatatype_any;
1822*00b67f09SDavid van Moolenbroek 				rdata->flags = DNS_RDATA_UPDATE;
1823*00b67f09SDavid van Moolenbroek 				goto doneparsing;
1824*00b67f09SDavid van Moolenbroek 			} else {
1825*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "could not read type\n");
1826*00b67f09SDavid van Moolenbroek 				goto failure;
1827*00b67f09SDavid van Moolenbroek 			}
1828*00b67f09SDavid van Moolenbroek 		}
1829*00b67f09SDavid van Moolenbroek 		region.base = word;
1830*00b67f09SDavid van Moolenbroek 		region.length = strlen(word);
1831*00b67f09SDavid van Moolenbroek 		result = dns_rdatatype_fromtext(&rdatatype, &region);
1832*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS) {
1833*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "'%s' is not a valid type: %s\n",
1834*00b67f09SDavid van Moolenbroek 				word, isc_result_totext(result));
1835*00b67f09SDavid van Moolenbroek 			goto failure;
1836*00b67f09SDavid van Moolenbroek 		}
1837*00b67f09SDavid van Moolenbroek 	} else {
1838*00b67f09SDavid van Moolenbroek 		rdataclass = getzoneclass();
1839*00b67f09SDavid van Moolenbroek 		result = dns_rdatatype_fromtext(&rdatatype, &region);
1840*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS) {
1841*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "'%s' is not a valid class or type: "
1842*00b67f09SDavid van Moolenbroek 				"%s\n", word, isc_result_totext(result));
1843*00b67f09SDavid van Moolenbroek 			goto failure;
1844*00b67f09SDavid van Moolenbroek 		}
1845*00b67f09SDavid van Moolenbroek 	}
1846*00b67f09SDavid van Moolenbroek 
1847*00b67f09SDavid van Moolenbroek 	retval = parse_rdata(&cmdline, rdataclass, rdatatype, updatemsg,
1848*00b67f09SDavid van Moolenbroek 			     rdata);
1849*00b67f09SDavid van Moolenbroek 	if (retval != STATUS_MORE)
1850*00b67f09SDavid van Moolenbroek 		goto failure;
1851*00b67f09SDavid van Moolenbroek 
1852*00b67f09SDavid van Moolenbroek 	if (isdelete) {
1853*00b67f09SDavid van Moolenbroek 		if ((rdata->flags & DNS_RDATA_UPDATE) != 0)
1854*00b67f09SDavid van Moolenbroek 			rdataclass = dns_rdataclass_any;
1855*00b67f09SDavid van Moolenbroek 		else
1856*00b67f09SDavid van Moolenbroek 			rdataclass = dns_rdataclass_none;
1857*00b67f09SDavid van Moolenbroek 	} else {
1858*00b67f09SDavid van Moolenbroek 		if ((rdata->flags & DNS_RDATA_UPDATE) != 0) {
1859*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "could not read rdata\n");
1860*00b67f09SDavid van Moolenbroek 			goto failure;
1861*00b67f09SDavid van Moolenbroek 		}
1862*00b67f09SDavid van Moolenbroek 	}
1863*00b67f09SDavid van Moolenbroek 
1864*00b67f09SDavid van Moolenbroek  doneparsing:
1865*00b67f09SDavid van Moolenbroek 
1866*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdatalist(updatemsg, &rdatalist);
1867*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdatalist");
1868*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdataset(updatemsg, &rdataset);
1869*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdataset");
1870*00b67f09SDavid van Moolenbroek 	dns_rdatalist_init(rdatalist);
1871*00b67f09SDavid van Moolenbroek 	rdatalist->type = rdatatype;
1872*00b67f09SDavid van Moolenbroek 	rdatalist->rdclass = rdataclass;
1873*00b67f09SDavid van Moolenbroek 	rdatalist->covers = rdatatype;
1874*00b67f09SDavid van Moolenbroek 	rdatalist->ttl = (dns_ttl_t)ttl;
1875*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(rdatalist->rdata);
1876*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(rdatalist->rdata, rdata, link);
1877*00b67f09SDavid van Moolenbroek 	dns_rdatalist_tordataset(rdatalist, rdataset);
1878*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(name->list);
1879*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(name->list, rdataset, link);
1880*00b67f09SDavid van Moolenbroek 	dns_message_addname(updatemsg, name, DNS_SECTION_UPDATE);
1881*00b67f09SDavid van Moolenbroek 	return (STATUS_MORE);
1882*00b67f09SDavid van Moolenbroek 
1883*00b67f09SDavid van Moolenbroek  failure:
1884*00b67f09SDavid van Moolenbroek 	if (name != NULL)
1885*00b67f09SDavid van Moolenbroek 		dns_message_puttempname(updatemsg, &name);
1886*00b67f09SDavid van Moolenbroek 	dns_message_puttemprdata(updatemsg, &rdata);
1887*00b67f09SDavid van Moolenbroek 	return (STATUS_SYNTAX);
1888*00b67f09SDavid van Moolenbroek }
1889*00b67f09SDavid van Moolenbroek 
1890*00b67f09SDavid van Moolenbroek static isc_uint16_t
evaluate_update(char * cmdline)1891*00b67f09SDavid van Moolenbroek evaluate_update(char *cmdline) {
1892*00b67f09SDavid van Moolenbroek 	char *word;
1893*00b67f09SDavid van Moolenbroek 	isc_boolean_t isdelete;
1894*00b67f09SDavid van Moolenbroek 
1895*00b67f09SDavid van Moolenbroek 	ddebug("evaluate_update()");
1896*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1897*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0) {
1898*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not read operation code\n");
1899*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1900*00b67f09SDavid van Moolenbroek 	}
1901*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "delete") == 0)
1902*00b67f09SDavid van Moolenbroek 		isdelete = ISC_TRUE;
1903*00b67f09SDavid van Moolenbroek 	else if (strcasecmp(word, "del") == 0)
1904*00b67f09SDavid van Moolenbroek 		isdelete = ISC_TRUE;
1905*00b67f09SDavid van Moolenbroek 	else if (strcasecmp(word, "add") == 0)
1906*00b67f09SDavid van Moolenbroek 		isdelete = ISC_FALSE;
1907*00b67f09SDavid van Moolenbroek 	else {
1908*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "incorrect operation code: %s\n", word);
1909*00b67f09SDavid van Moolenbroek 		return (STATUS_SYNTAX);
1910*00b67f09SDavid van Moolenbroek 	}
1911*00b67f09SDavid van Moolenbroek 	return (update_addordelete(cmdline, isdelete));
1912*00b67f09SDavid van Moolenbroek }
1913*00b67f09SDavid van Moolenbroek 
1914*00b67f09SDavid van Moolenbroek static void
setzone(dns_name_t * zonename)1915*00b67f09SDavid van Moolenbroek setzone(dns_name_t *zonename) {
1916*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1917*00b67f09SDavid van Moolenbroek 	dns_name_t *name = NULL;
1918*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *rdataset = NULL;
1919*00b67f09SDavid van Moolenbroek 
1920*00b67f09SDavid van Moolenbroek 	result = dns_message_firstname(updatemsg, DNS_SECTION_ZONE);
1921*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_SUCCESS) {
1922*00b67f09SDavid van Moolenbroek 		dns_message_currentname(updatemsg, DNS_SECTION_ZONE, &name);
1923*00b67f09SDavid van Moolenbroek 		dns_message_removename(updatemsg, name, DNS_SECTION_ZONE);
1924*00b67f09SDavid van Moolenbroek 		for (rdataset = ISC_LIST_HEAD(name->list);
1925*00b67f09SDavid van Moolenbroek 		     rdataset != NULL;
1926*00b67f09SDavid van Moolenbroek 		     rdataset = ISC_LIST_HEAD(name->list)) {
1927*00b67f09SDavid van Moolenbroek 			ISC_LIST_UNLINK(name->list, rdataset, link);
1928*00b67f09SDavid van Moolenbroek 			dns_rdataset_disassociate(rdataset);
1929*00b67f09SDavid van Moolenbroek 			dns_message_puttemprdataset(updatemsg, &rdataset);
1930*00b67f09SDavid van Moolenbroek 		}
1931*00b67f09SDavid van Moolenbroek 		dns_message_puttempname(updatemsg, &name);
1932*00b67f09SDavid van Moolenbroek 	}
1933*00b67f09SDavid van Moolenbroek 
1934*00b67f09SDavid van Moolenbroek 	if (zonename != NULL) {
1935*00b67f09SDavid van Moolenbroek 		result = dns_message_gettempname(updatemsg, &name);
1936*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_message_gettempname");
1937*00b67f09SDavid van Moolenbroek 		dns_name_init(name, NULL);
1938*00b67f09SDavid van Moolenbroek 		dns_name_clone(zonename, name);
1939*00b67f09SDavid van Moolenbroek 		result = dns_message_gettemprdataset(updatemsg, &rdataset);
1940*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_message_gettemprdataset");
1941*00b67f09SDavid van Moolenbroek 		dns_rdataset_makequestion(rdataset, getzoneclass(),
1942*00b67f09SDavid van Moolenbroek 					  dns_rdatatype_soa);
1943*00b67f09SDavid van Moolenbroek 		ISC_LIST_INIT(name->list);
1944*00b67f09SDavid van Moolenbroek 		ISC_LIST_APPEND(name->list, rdataset, link);
1945*00b67f09SDavid van Moolenbroek 		dns_message_addname(updatemsg, name, DNS_SECTION_ZONE);
1946*00b67f09SDavid van Moolenbroek 	}
1947*00b67f09SDavid van Moolenbroek }
1948*00b67f09SDavid van Moolenbroek 
1949*00b67f09SDavid van Moolenbroek static void
show_message(FILE * stream,dns_message_t * msg,const char * description)1950*00b67f09SDavid van Moolenbroek show_message(FILE *stream, dns_message_t *msg, const char *description) {
1951*00b67f09SDavid van Moolenbroek 	isc_result_t result;
1952*00b67f09SDavid van Moolenbroek 	isc_buffer_t *buf = NULL;
1953*00b67f09SDavid van Moolenbroek 	int bufsz;
1954*00b67f09SDavid van Moolenbroek 
1955*00b67f09SDavid van Moolenbroek 	ddebug("show_message()");
1956*00b67f09SDavid van Moolenbroek 
1957*00b67f09SDavid van Moolenbroek 	setzone(userzone);
1958*00b67f09SDavid van Moolenbroek 
1959*00b67f09SDavid van Moolenbroek 	bufsz = INITTEXT;
1960*00b67f09SDavid van Moolenbroek 	do {
1961*00b67f09SDavid van Moolenbroek 		if (bufsz > MAXTEXT) {
1962*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "could not allocate large enough "
1963*00b67f09SDavid van Moolenbroek 				"buffer to display message\n");
1964*00b67f09SDavid van Moolenbroek 			exit(1);
1965*00b67f09SDavid van Moolenbroek 		}
1966*00b67f09SDavid van Moolenbroek 		if (buf != NULL)
1967*00b67f09SDavid van Moolenbroek 			isc_buffer_free(&buf);
1968*00b67f09SDavid van Moolenbroek 		result = isc_buffer_allocate(gmctx, &buf, bufsz);
1969*00b67f09SDavid van Moolenbroek 		check_result(result, "isc_buffer_allocate");
1970*00b67f09SDavid van Moolenbroek 		result = dns_message_totext(msg, style, 0, buf);
1971*00b67f09SDavid van Moolenbroek 		bufsz *= 2;
1972*00b67f09SDavid van Moolenbroek 	} while (result == ISC_R_NOSPACE);
1973*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
1974*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "could not convert message to text format.\n");
1975*00b67f09SDavid van Moolenbroek 		isc_buffer_free(&buf);
1976*00b67f09SDavid van Moolenbroek 		return;
1977*00b67f09SDavid van Moolenbroek 	}
1978*00b67f09SDavid van Moolenbroek 	fprintf(stream, "%s\n%.*s", description,
1979*00b67f09SDavid van Moolenbroek 	       (int)isc_buffer_usedlength(buf), (char*)isc_buffer_base(buf));
1980*00b67f09SDavid van Moolenbroek 	isc_buffer_free(&buf);
1981*00b67f09SDavid van Moolenbroek }
1982*00b67f09SDavid van Moolenbroek 
1983*00b67f09SDavid van Moolenbroek static isc_uint16_t
do_next_command(char * cmdline)1984*00b67f09SDavid van Moolenbroek do_next_command(char *cmdline) {
1985*00b67f09SDavid van Moolenbroek 	char *word;
1986*00b67f09SDavid van Moolenbroek 
1987*00b67f09SDavid van Moolenbroek 	ddebug("do_next_command()");
1988*00b67f09SDavid van Moolenbroek 	word = nsu_strsep(&cmdline, " \t\r\n");
1989*00b67f09SDavid van Moolenbroek 
1990*00b67f09SDavid van Moolenbroek 	if (word == NULL || *word == 0)
1991*00b67f09SDavid van Moolenbroek 		return (STATUS_SEND);
1992*00b67f09SDavid van Moolenbroek 	if (word[0] == ';')
1993*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
1994*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "quit") == 0)
1995*00b67f09SDavid van Moolenbroek 		return (STATUS_QUIT);
1996*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "prereq") == 0)
1997*00b67f09SDavid van Moolenbroek 		return (evaluate_prereq(cmdline));
1998*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "nxdomain") == 0)
1999*00b67f09SDavid van Moolenbroek 		return (make_prereq(cmdline, ISC_FALSE, ISC_FALSE));
2000*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "yxdomain") == 0)
2001*00b67f09SDavid van Moolenbroek 		return (make_prereq(cmdline, ISC_TRUE, ISC_FALSE));
2002*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "nxrrset") == 0)
2003*00b67f09SDavid van Moolenbroek 		return (make_prereq(cmdline, ISC_FALSE, ISC_TRUE));
2004*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "yxrrset") == 0)
2005*00b67f09SDavid van Moolenbroek 		return (make_prereq(cmdline, ISC_TRUE, ISC_TRUE));
2006*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "update") == 0)
2007*00b67f09SDavid van Moolenbroek 		return (evaluate_update(cmdline));
2008*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "delete") == 0)
2009*00b67f09SDavid van Moolenbroek 		return (update_addordelete(cmdline, ISC_TRUE));
2010*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "del") == 0)
2011*00b67f09SDavid van Moolenbroek 		return (update_addordelete(cmdline, ISC_TRUE));
2012*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "add") == 0)
2013*00b67f09SDavid van Moolenbroek 		return (update_addordelete(cmdline, ISC_FALSE));
2014*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "server") == 0)
2015*00b67f09SDavid van Moolenbroek 		return (evaluate_server(cmdline));
2016*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "local") == 0)
2017*00b67f09SDavid van Moolenbroek 		return (evaluate_local(cmdline));
2018*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "zone") == 0)
2019*00b67f09SDavid van Moolenbroek 		return (evaluate_zone(cmdline));
2020*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "class") == 0)
2021*00b67f09SDavid van Moolenbroek 		return (evaluate_class(cmdline));
2022*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "send") == 0)
2023*00b67f09SDavid van Moolenbroek 		return (STATUS_SEND);
2024*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "debug") == 0) {
2025*00b67f09SDavid van Moolenbroek 		if (debugging)
2026*00b67f09SDavid van Moolenbroek 			ddebugging = ISC_TRUE;
2027*00b67f09SDavid van Moolenbroek 		else
2028*00b67f09SDavid van Moolenbroek 			debugging = ISC_TRUE;
2029*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2030*00b67f09SDavid van Moolenbroek 	}
2031*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "ttl") == 0)
2032*00b67f09SDavid van Moolenbroek 		return (evaluate_ttl(cmdline));
2033*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "show") == 0) {
2034*00b67f09SDavid van Moolenbroek 		show_message(stdout, updatemsg, "Outgoing update query:");
2035*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2036*00b67f09SDavid van Moolenbroek 	}
2037*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "answer") == 0) {
2038*00b67f09SDavid van Moolenbroek 		if (answer != NULL)
2039*00b67f09SDavid van Moolenbroek 			show_message(stdout, answer, "Answer:");
2040*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2041*00b67f09SDavid van Moolenbroek 	}
2042*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "key") == 0) {
2043*00b67f09SDavid van Moolenbroek 		usegsstsig = ISC_FALSE;
2044*00b67f09SDavid van Moolenbroek 		return (evaluate_key(cmdline));
2045*00b67f09SDavid van Moolenbroek 	}
2046*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "realm") == 0)
2047*00b67f09SDavid van Moolenbroek 		return (evaluate_realm(cmdline));
2048*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "gsstsig") == 0) {
2049*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
2050*00b67f09SDavid van Moolenbroek 		usegsstsig = ISC_TRUE;
2051*00b67f09SDavid van Moolenbroek 		use_win2k_gsstsig = ISC_FALSE;
2052*00b67f09SDavid van Moolenbroek #else
2053*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "gsstsig not supported\n");
2054*00b67f09SDavid van Moolenbroek #endif
2055*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2056*00b67f09SDavid van Moolenbroek 	}
2057*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "oldgsstsig") == 0) {
2058*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
2059*00b67f09SDavid van Moolenbroek 		usegsstsig = ISC_TRUE;
2060*00b67f09SDavid van Moolenbroek 		use_win2k_gsstsig = ISC_TRUE;
2061*00b67f09SDavid van Moolenbroek #else
2062*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "gsstsig not supported\n");
2063*00b67f09SDavid van Moolenbroek #endif
2064*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2065*00b67f09SDavid van Moolenbroek 	}
2066*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "help") == 0) {
2067*00b67f09SDavid van Moolenbroek 		fprintf(stdout,
2068*00b67f09SDavid van Moolenbroek "nsupdate " VERSION ":\n"
2069*00b67f09SDavid van Moolenbroek "local address [port]      (set local resolver)\n"
2070*00b67f09SDavid van Moolenbroek "server address [port]     (set master server for zone)\n"
2071*00b67f09SDavid van Moolenbroek "send                      (send the update request)\n"
2072*00b67f09SDavid van Moolenbroek "show                      (show the update request)\n"
2073*00b67f09SDavid van Moolenbroek "answer                    (show the answer to the last request)\n"
2074*00b67f09SDavid van Moolenbroek "quit                      (quit, any pending update is not sent\n"
2075*00b67f09SDavid van Moolenbroek "help                      (display this message_\n"
2076*00b67f09SDavid van Moolenbroek "key [hmac:]keyname secret (use TSIG to sign the request)\n"
2077*00b67f09SDavid van Moolenbroek "gsstsig                   (use GSS_TSIG to sign the request)\n"
2078*00b67f09SDavid van Moolenbroek "oldgsstsig                (use Microsoft's GSS_TSIG to sign the request)\n"
2079*00b67f09SDavid van Moolenbroek "zone name                 (set the zone to be updated)\n"
2080*00b67f09SDavid van Moolenbroek "class CLASS               (set the zone's DNS class, e.g. IN (default), CH)\n"
2081*00b67f09SDavid van Moolenbroek "[prereq] nxdomain name    (does this name not exist)\n"
2082*00b67f09SDavid van Moolenbroek "[prereq] yxdomain name    (does this name exist)\n"
2083*00b67f09SDavid van Moolenbroek "[prereq] nxrrset ....     (does this RRset exist)\n"
2084*00b67f09SDavid van Moolenbroek "[prereq] yxrrset ....     (does this RRset not exist)\n"
2085*00b67f09SDavid van Moolenbroek "[update] add ....         (add the given record to the zone)\n"
2086*00b67f09SDavid van Moolenbroek "[update] del[ete] ....    (remove the given record(s) from the zone)\n");
2087*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2088*00b67f09SDavid van Moolenbroek 	}
2089*00b67f09SDavid van Moolenbroek 	if (strcasecmp(word, "version") == 0) {
2090*00b67f09SDavid van Moolenbroek 		fprintf(stdout, "nsupdate " VERSION "\n");
2091*00b67f09SDavid van Moolenbroek 		return (STATUS_MORE);
2092*00b67f09SDavid van Moolenbroek 	}
2093*00b67f09SDavid van Moolenbroek 	fprintf(stderr, "incorrect section name: %s\n", word);
2094*00b67f09SDavid van Moolenbroek 	return (STATUS_SYNTAX);
2095*00b67f09SDavid van Moolenbroek }
2096*00b67f09SDavid van Moolenbroek 
2097*00b67f09SDavid van Moolenbroek static isc_uint16_t
get_next_command(void)2098*00b67f09SDavid van Moolenbroek get_next_command(void) {
2099*00b67f09SDavid van Moolenbroek 	isc_uint16_t result = STATUS_QUIT;
2100*00b67f09SDavid van Moolenbroek 	char cmdlinebuf[MAXCMD];
2101*00b67f09SDavid van Moolenbroek 	char *cmdline;
2102*00b67f09SDavid van Moolenbroek 
2103*00b67f09SDavid van Moolenbroek 	isc_app_block();
2104*00b67f09SDavid van Moolenbroek 	if (interactive) {
2105*00b67f09SDavid van Moolenbroek #ifdef HAVE_READLINE
2106*00b67f09SDavid van Moolenbroek 		cmdline = readline("> ");
2107*00b67f09SDavid van Moolenbroek 		if (cmdline != NULL)
2108*00b67f09SDavid van Moolenbroek 			add_history(cmdline);
2109*00b67f09SDavid van Moolenbroek #else
2110*00b67f09SDavid van Moolenbroek 		fprintf(stdout, "> ");
2111*00b67f09SDavid van Moolenbroek 		fflush(stdout);
2112*00b67f09SDavid van Moolenbroek 		cmdline = fgets(cmdlinebuf, MAXCMD, input);
2113*00b67f09SDavid van Moolenbroek #endif
2114*00b67f09SDavid van Moolenbroek 	} else
2115*00b67f09SDavid van Moolenbroek 		cmdline = fgets(cmdlinebuf, MAXCMD, input);
2116*00b67f09SDavid van Moolenbroek 	isc_app_unblock();
2117*00b67f09SDavid van Moolenbroek 
2118*00b67f09SDavid van Moolenbroek 	if (cmdline != NULL) {
2119*00b67f09SDavid van Moolenbroek 		char *tmp = cmdline;
2120*00b67f09SDavid van Moolenbroek 
2121*00b67f09SDavid van Moolenbroek 		/*
2122*00b67f09SDavid van Moolenbroek 		 * Normalize input by removing any eol as readline()
2123*00b67f09SDavid van Moolenbroek 		 * removes eol but fgets doesn't.
2124*00b67f09SDavid van Moolenbroek 		 */
2125*00b67f09SDavid van Moolenbroek 		(void)nsu_strsep(&tmp, "\r\n");
2126*00b67f09SDavid van Moolenbroek 		result = do_next_command(cmdline);
2127*00b67f09SDavid van Moolenbroek 	}
2128*00b67f09SDavid van Moolenbroek #ifdef HAVE_READLINE
2129*00b67f09SDavid van Moolenbroek 	if (interactive)
2130*00b67f09SDavid van Moolenbroek 		free(cmdline);
2131*00b67f09SDavid van Moolenbroek #endif
2132*00b67f09SDavid van Moolenbroek 	return (result);
2133*00b67f09SDavid van Moolenbroek }
2134*00b67f09SDavid van Moolenbroek 
2135*00b67f09SDavid van Moolenbroek static isc_boolean_t
user_interaction(void)2136*00b67f09SDavid van Moolenbroek user_interaction(void) {
2137*00b67f09SDavid van Moolenbroek 	isc_uint16_t result = STATUS_MORE;
2138*00b67f09SDavid van Moolenbroek 
2139*00b67f09SDavid van Moolenbroek 	ddebug("user_interaction()");
2140*00b67f09SDavid van Moolenbroek 	while ((result == STATUS_MORE) || (result == STATUS_SYNTAX)) {
2141*00b67f09SDavid van Moolenbroek 		result = get_next_command();
2142*00b67f09SDavid van Moolenbroek 		if (!interactive && result == STATUS_SYNTAX)
2143*00b67f09SDavid van Moolenbroek 			fatal("syntax error");
2144*00b67f09SDavid van Moolenbroek 	}
2145*00b67f09SDavid van Moolenbroek 	if (result == STATUS_SEND)
2146*00b67f09SDavid van Moolenbroek 		return (ISC_TRUE);
2147*00b67f09SDavid van Moolenbroek 	return (ISC_FALSE);
2148*00b67f09SDavid van Moolenbroek 
2149*00b67f09SDavid van Moolenbroek }
2150*00b67f09SDavid van Moolenbroek 
2151*00b67f09SDavid van Moolenbroek static void
done_update(void)2152*00b67f09SDavid van Moolenbroek done_update(void) {
2153*00b67f09SDavid van Moolenbroek 	isc_event_t *event = global_event;
2154*00b67f09SDavid van Moolenbroek 	ddebug("done_update()");
2155*00b67f09SDavid van Moolenbroek 	isc_task_send(global_task, &event);
2156*00b67f09SDavid van Moolenbroek }
2157*00b67f09SDavid van Moolenbroek 
2158*00b67f09SDavid van Moolenbroek static void
check_tsig_error(dns_rdataset_t * rdataset,isc_buffer_t * b)2159*00b67f09SDavid van Moolenbroek check_tsig_error(dns_rdataset_t *rdataset, isc_buffer_t *b) {
2160*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2161*00b67f09SDavid van Moolenbroek 	dns_rdata_t rdata = DNS_RDATA_INIT;
2162*00b67f09SDavid van Moolenbroek 	dns_rdata_any_tsig_t tsig;
2163*00b67f09SDavid van Moolenbroek 
2164*00b67f09SDavid van Moolenbroek 	result = dns_rdataset_first(rdataset);
2165*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_rdataset_first");
2166*00b67f09SDavid van Moolenbroek 	dns_rdataset_current(rdataset, &rdata);
2167*00b67f09SDavid van Moolenbroek 	result = dns_rdata_tostruct(&rdata, &tsig, NULL);
2168*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_rdata_tostruct");
2169*00b67f09SDavid van Moolenbroek 	if (tsig.error != 0) {
2170*00b67f09SDavid van Moolenbroek 		if (isc_buffer_remaininglength(b) < 1)
2171*00b67f09SDavid van Moolenbroek 		      check_result(ISC_R_NOSPACE, "isc_buffer_remaininglength");
2172*00b67f09SDavid van Moolenbroek 		isc_buffer_putstr(b, "(" /*)*/);
2173*00b67f09SDavid van Moolenbroek 		result = dns_tsigrcode_totext(tsig.error, b);
2174*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_tsigrcode_totext");
2175*00b67f09SDavid van Moolenbroek 		if (isc_buffer_remaininglength(b) < 1)
2176*00b67f09SDavid van Moolenbroek 		      check_result(ISC_R_NOSPACE, "isc_buffer_remaininglength");
2177*00b67f09SDavid van Moolenbroek 		isc_buffer_putstr(b,  /*(*/ ")");
2178*00b67f09SDavid van Moolenbroek 	}
2179*00b67f09SDavid van Moolenbroek }
2180*00b67f09SDavid van Moolenbroek 
2181*00b67f09SDavid van Moolenbroek static isc_boolean_t
next_master(const char * caller,isc_sockaddr_t * addr,isc_result_t eresult)2182*00b67f09SDavid van Moolenbroek next_master(const char *caller, isc_sockaddr_t *addr, isc_result_t eresult) {
2183*00b67f09SDavid van Moolenbroek 	char addrbuf[ISC_SOCKADDR_FORMATSIZE];
2184*00b67f09SDavid van Moolenbroek 
2185*00b67f09SDavid van Moolenbroek 	isc_sockaddr_format(addr, addrbuf, sizeof(addrbuf));
2186*00b67f09SDavid van Moolenbroek 	fprintf(stderr, "; Communication with %s failed: %s\n",
2187*00b67f09SDavid van Moolenbroek 		addrbuf, isc_result_totext(eresult));
2188*00b67f09SDavid van Moolenbroek 	if (++master_inuse >= master_total)
2189*00b67f09SDavid van Moolenbroek 		return (ISC_FALSE);
2190*00b67f09SDavid van Moolenbroek 	ddebug("%s: trying next server", caller);
2191*00b67f09SDavid van Moolenbroek 	return (ISC_TRUE);
2192*00b67f09SDavid van Moolenbroek }
2193*00b67f09SDavid van Moolenbroek 
2194*00b67f09SDavid van Moolenbroek static void
update_completed(isc_task_t * task,isc_event_t * event)2195*00b67f09SDavid van Moolenbroek update_completed(isc_task_t *task, isc_event_t *event) {
2196*00b67f09SDavid van Moolenbroek 	dns_requestevent_t *reqev = NULL;
2197*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2198*00b67f09SDavid van Moolenbroek 	dns_request_t *request;
2199*00b67f09SDavid van Moolenbroek 
2200*00b67f09SDavid van Moolenbroek 	UNUSED(task);
2201*00b67f09SDavid van Moolenbroek 
2202*00b67f09SDavid van Moolenbroek 	ddebug("update_completed()");
2203*00b67f09SDavid van Moolenbroek 
2204*00b67f09SDavid van Moolenbroek 	requests--;
2205*00b67f09SDavid van Moolenbroek 
2206*00b67f09SDavid van Moolenbroek 	REQUIRE(event->ev_type == DNS_EVENT_REQUESTDONE);
2207*00b67f09SDavid van Moolenbroek 	reqev = (dns_requestevent_t *)event;
2208*00b67f09SDavid van Moolenbroek 	request = reqev->request;
2209*00b67f09SDavid van Moolenbroek 
2210*00b67f09SDavid van Moolenbroek 	if (shuttingdown) {
2211*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2212*00b67f09SDavid van Moolenbroek 		isc_event_free(&event);
2213*00b67f09SDavid van Moolenbroek 		maybeshutdown();
2214*00b67f09SDavid van Moolenbroek 		return;
2215*00b67f09SDavid van Moolenbroek 	}
2216*00b67f09SDavid van Moolenbroek 
2217*00b67f09SDavid van Moolenbroek 	if (reqev->result != ISC_R_SUCCESS) {
2218*00b67f09SDavid van Moolenbroek 		if (!next_master("recvsoa", &master_servers[master_inuse],
2219*00b67f09SDavid van Moolenbroek 				 reqev->result)) {
2220*00b67f09SDavid van Moolenbroek 			seenerror = ISC_TRUE;
2221*00b67f09SDavid van Moolenbroek 			goto done;
2222*00b67f09SDavid van Moolenbroek 		}
2223*00b67f09SDavid van Moolenbroek 
2224*00b67f09SDavid van Moolenbroek 		ddebug("Destroying request [%p]", request);
2225*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2226*00b67f09SDavid van Moolenbroek 		dns_message_renderreset(updatemsg);
2227*00b67f09SDavid van Moolenbroek 		dns_message_settsigkey(updatemsg, NULL);
2228*00b67f09SDavid van Moolenbroek 		send_update(zname, &master_servers[master_inuse]);
2229*00b67f09SDavid van Moolenbroek 		isc_event_free(&event);
2230*00b67f09SDavid van Moolenbroek 		return;
2231*00b67f09SDavid van Moolenbroek 	}
2232*00b67f09SDavid van Moolenbroek 
2233*00b67f09SDavid van Moolenbroek 	result = dns_message_create(gmctx, DNS_MESSAGE_INTENTPARSE, &answer);
2234*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_create");
2235*00b67f09SDavid van Moolenbroek 	result = dns_request_getresponse(request, answer,
2236*00b67f09SDavid van Moolenbroek 					 DNS_MESSAGEPARSE_PRESERVEORDER);
2237*00b67f09SDavid van Moolenbroek 	switch (result) {
2238*00b67f09SDavid van Moolenbroek 	case ISC_R_SUCCESS:
2239*00b67f09SDavid van Moolenbroek 		if (answer->verify_attempted)
2240*00b67f09SDavid van Moolenbroek 			ddebug("tsig verification successful");
2241*00b67f09SDavid van Moolenbroek 		break;
2242*00b67f09SDavid van Moolenbroek 	case DNS_R_CLOCKSKEW:
2243*00b67f09SDavid van Moolenbroek 	case DNS_R_EXPECTEDTSIG:
2244*00b67f09SDavid van Moolenbroek 	case DNS_R_TSIGERRORSET:
2245*00b67f09SDavid van Moolenbroek 	case DNS_R_TSIGVERIFYFAILURE:
2246*00b67f09SDavid van Moolenbroek 	case DNS_R_UNEXPECTEDTSIG:
2247*00b67f09SDavid van Moolenbroek 	case ISC_R_FAILURE:
2248*00b67f09SDavid van Moolenbroek #if 0
2249*00b67f09SDavid van Moolenbroek 		if (usegsstsig && answer->rcode == dns_rcode_noerror) {
2250*00b67f09SDavid van Moolenbroek 			/*
2251*00b67f09SDavid van Moolenbroek 			 * For MS DNS that violates RFC 2845, section 4.2
2252*00b67f09SDavid van Moolenbroek 			 */
2253*00b67f09SDavid van Moolenbroek 			break;
2254*00b67f09SDavid van Moolenbroek 		}
2255*00b67f09SDavid van Moolenbroek #endif
2256*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "; TSIG error with server: %s\n",
2257*00b67f09SDavid van Moolenbroek 			isc_result_totext(result));
2258*00b67f09SDavid van Moolenbroek 		seenerror = ISC_TRUE;
2259*00b67f09SDavid van Moolenbroek 		break;
2260*00b67f09SDavid van Moolenbroek 	default:
2261*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_request_getresponse");
2262*00b67f09SDavid van Moolenbroek 	}
2263*00b67f09SDavid van Moolenbroek 
2264*00b67f09SDavid van Moolenbroek 	if (answer->rcode != dns_rcode_noerror) {
2265*00b67f09SDavid van Moolenbroek 		seenerror = ISC_TRUE;
2266*00b67f09SDavid van Moolenbroek 		if (!debugging) {
2267*00b67f09SDavid van Moolenbroek 			char buf[64];
2268*00b67f09SDavid van Moolenbroek 			isc_buffer_t b;
2269*00b67f09SDavid van Moolenbroek 			dns_rdataset_t *rds;
2270*00b67f09SDavid van Moolenbroek 
2271*00b67f09SDavid van Moolenbroek 			isc_buffer_init(&b, buf, sizeof(buf) - 1);
2272*00b67f09SDavid van Moolenbroek 			result = dns_rcode_totext(answer->rcode, &b);
2273*00b67f09SDavid van Moolenbroek 			check_result(result, "dns_rcode_totext");
2274*00b67f09SDavid van Moolenbroek 			rds = dns_message_gettsig(answer, NULL);
2275*00b67f09SDavid van Moolenbroek 			if (rds != NULL)
2276*00b67f09SDavid van Moolenbroek 				check_tsig_error(rds, &b);
2277*00b67f09SDavid van Moolenbroek 			fprintf(stderr, "update failed: %.*s\n",
2278*00b67f09SDavid van Moolenbroek 				(int)isc_buffer_usedlength(&b), buf);
2279*00b67f09SDavid van Moolenbroek 		}
2280*00b67f09SDavid van Moolenbroek 	}
2281*00b67f09SDavid van Moolenbroek 	if (debugging)
2282*00b67f09SDavid van Moolenbroek 		show_message(stderr, answer, "\nReply from update query:");
2283*00b67f09SDavid van Moolenbroek 
2284*00b67f09SDavid van Moolenbroek  done:
2285*00b67f09SDavid van Moolenbroek 	dns_request_destroy(&request);
2286*00b67f09SDavid van Moolenbroek 	if (usegsstsig) {
2287*00b67f09SDavid van Moolenbroek 		dns_name_free(&tmpzonename, gmctx);
2288*00b67f09SDavid van Moolenbroek 		dns_name_free(&restart_master, gmctx);
2289*00b67f09SDavid van Moolenbroek 	}
2290*00b67f09SDavid van Moolenbroek 	isc_event_free(&event);
2291*00b67f09SDavid van Moolenbroek 	done_update();
2292*00b67f09SDavid van Moolenbroek }
2293*00b67f09SDavid van Moolenbroek 
2294*00b67f09SDavid van Moolenbroek static void
send_update(dns_name_t * zone,isc_sockaddr_t * master)2295*00b67f09SDavid van Moolenbroek send_update(dns_name_t *zone, isc_sockaddr_t *master) {
2296*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2297*00b67f09SDavid van Moolenbroek 	dns_request_t *request = NULL;
2298*00b67f09SDavid van Moolenbroek 	unsigned int options = DNS_REQUESTOPT_CASE;
2299*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *srcaddr;
2300*00b67f09SDavid van Moolenbroek 
2301*00b67f09SDavid van Moolenbroek 	ddebug("send_update()");
2302*00b67f09SDavid van Moolenbroek 
2303*00b67f09SDavid van Moolenbroek 	setzone(zone);
2304*00b67f09SDavid van Moolenbroek 
2305*00b67f09SDavid van Moolenbroek 	if (usevc)
2306*00b67f09SDavid van Moolenbroek 		options |= DNS_REQUESTOPT_TCP;
2307*00b67f09SDavid van Moolenbroek 	if (tsigkey == NULL && sig0key != NULL) {
2308*00b67f09SDavid van Moolenbroek 		result = dns_message_setsig0key(updatemsg, sig0key);
2309*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_message_setsig0key");
2310*00b67f09SDavid van Moolenbroek 	}
2311*00b67f09SDavid van Moolenbroek 	if (debugging) {
2312*00b67f09SDavid van Moolenbroek 		char addrbuf[ISC_SOCKADDR_FORMATSIZE];
2313*00b67f09SDavid van Moolenbroek 
2314*00b67f09SDavid van Moolenbroek 		isc_sockaddr_format(master, addrbuf, sizeof(addrbuf));
2315*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "Sending update to %s\n", addrbuf);
2316*00b67f09SDavid van Moolenbroek 	}
2317*00b67f09SDavid van Moolenbroek 
2318*00b67f09SDavid van Moolenbroek 	if (isc_sockaddr_pf(master) == AF_INET6)
2319*00b67f09SDavid van Moolenbroek 		srcaddr = localaddr6;
2320*00b67f09SDavid van Moolenbroek 	else
2321*00b67f09SDavid van Moolenbroek 		srcaddr = localaddr4;
2322*00b67f09SDavid van Moolenbroek 
2323*00b67f09SDavid van Moolenbroek 	/* Windows doesn't like the tsig name to be compressed. */
2324*00b67f09SDavid van Moolenbroek 	if (updatemsg->tsigname)
2325*00b67f09SDavid van Moolenbroek 		updatemsg->tsigname->attributes |= DNS_NAMEATTR_NOCOMPRESS;
2326*00b67f09SDavid van Moolenbroek 
2327*00b67f09SDavid van Moolenbroek 	result = dns_request_createvia3(requestmgr, updatemsg, srcaddr,
2328*00b67f09SDavid van Moolenbroek 					master, options, tsigkey, timeout,
2329*00b67f09SDavid van Moolenbroek 					udp_timeout, udp_retries, global_task,
2330*00b67f09SDavid van Moolenbroek 					update_completed, NULL, &request);
2331*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_request_createvia3");
2332*00b67f09SDavid van Moolenbroek 
2333*00b67f09SDavid van Moolenbroek 	if (debugging)
2334*00b67f09SDavid van Moolenbroek 		show_message(stdout, updatemsg, "Outgoing update query:");
2335*00b67f09SDavid van Moolenbroek 
2336*00b67f09SDavid van Moolenbroek 	requests++;
2337*00b67f09SDavid van Moolenbroek }
2338*00b67f09SDavid van Moolenbroek 
2339*00b67f09SDavid van Moolenbroek static void
next_server(const char * caller,isc_sockaddr_t * addr,isc_result_t eresult)2340*00b67f09SDavid van Moolenbroek next_server(const char *caller, isc_sockaddr_t *addr, isc_result_t eresult) {
2341*00b67f09SDavid van Moolenbroek 	char addrbuf[ISC_SOCKADDR_FORMATSIZE];
2342*00b67f09SDavid van Moolenbroek 
2343*00b67f09SDavid van Moolenbroek 	isc_sockaddr_format(addr, addrbuf, sizeof(addrbuf));
2344*00b67f09SDavid van Moolenbroek 	fprintf(stderr, "; Communication with %s failed: %s\n",
2345*00b67f09SDavid van Moolenbroek 		addrbuf, isc_result_totext(eresult));
2346*00b67f09SDavid van Moolenbroek 	if (++ns_inuse >= ns_total)
2347*00b67f09SDavid van Moolenbroek 		fatal("could not reach any name server");
2348*00b67f09SDavid van Moolenbroek 	else
2349*00b67f09SDavid van Moolenbroek 		ddebug("%s: trying next server", caller);
2350*00b67f09SDavid van Moolenbroek }
2351*00b67f09SDavid van Moolenbroek 
2352*00b67f09SDavid van Moolenbroek static void
recvsoa(isc_task_t * task,isc_event_t * event)2353*00b67f09SDavid van Moolenbroek recvsoa(isc_task_t *task, isc_event_t *event) {
2354*00b67f09SDavid van Moolenbroek 	dns_requestevent_t *reqev = NULL;
2355*00b67f09SDavid van Moolenbroek 	dns_request_t *request = NULL;
2356*00b67f09SDavid van Moolenbroek 	isc_result_t result, eresult;
2357*00b67f09SDavid van Moolenbroek 	dns_message_t *rcvmsg = NULL;
2358*00b67f09SDavid van Moolenbroek 	dns_section_t section;
2359*00b67f09SDavid van Moolenbroek 	dns_name_t *name = NULL;
2360*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *soaset = NULL;
2361*00b67f09SDavid van Moolenbroek 	dns_rdata_soa_t soa;
2362*00b67f09SDavid van Moolenbroek 	dns_rdata_t soarr = DNS_RDATA_INIT;
2363*00b67f09SDavid van Moolenbroek 	int pass = 0;
2364*00b67f09SDavid van Moolenbroek 	dns_name_t master;
2365*00b67f09SDavid van Moolenbroek 	nsu_requestinfo_t *reqinfo;
2366*00b67f09SDavid van Moolenbroek 	dns_message_t *soaquery = NULL;
2367*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *addr;
2368*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *srcaddr;
2369*00b67f09SDavid van Moolenbroek 	isc_boolean_t seencname = ISC_FALSE;
2370*00b67f09SDavid van Moolenbroek 	dns_name_t tname;
2371*00b67f09SDavid van Moolenbroek 	unsigned int nlabels;
2372*00b67f09SDavid van Moolenbroek 
2373*00b67f09SDavid van Moolenbroek 	UNUSED(task);
2374*00b67f09SDavid van Moolenbroek 
2375*00b67f09SDavid van Moolenbroek 	ddebug("recvsoa()");
2376*00b67f09SDavid van Moolenbroek 
2377*00b67f09SDavid van Moolenbroek 	requests--;
2378*00b67f09SDavid van Moolenbroek 
2379*00b67f09SDavid van Moolenbroek 	REQUIRE(event->ev_type == DNS_EVENT_REQUESTDONE);
2380*00b67f09SDavid van Moolenbroek 	reqev = (dns_requestevent_t *)event;
2381*00b67f09SDavid van Moolenbroek 	request = reqev->request;
2382*00b67f09SDavid van Moolenbroek 	eresult = reqev->result;
2383*00b67f09SDavid van Moolenbroek 	reqinfo = reqev->ev_arg;
2384*00b67f09SDavid van Moolenbroek 	soaquery = reqinfo->msg;
2385*00b67f09SDavid van Moolenbroek 	addr = reqinfo->addr;
2386*00b67f09SDavid van Moolenbroek 
2387*00b67f09SDavid van Moolenbroek 	if (shuttingdown) {
2388*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2389*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&soaquery);
2390*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, reqinfo, sizeof(nsu_requestinfo_t));
2391*00b67f09SDavid van Moolenbroek 		isc_event_free(&event);
2392*00b67f09SDavid van Moolenbroek 		maybeshutdown();
2393*00b67f09SDavid van Moolenbroek 		return;
2394*00b67f09SDavid van Moolenbroek 	}
2395*00b67f09SDavid van Moolenbroek 
2396*00b67f09SDavid van Moolenbroek 	if (eresult != ISC_R_SUCCESS) {
2397*00b67f09SDavid van Moolenbroek 		next_server("recvsoa", addr, eresult);
2398*00b67f09SDavid van Moolenbroek 		ddebug("Destroying request [%p]", request);
2399*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2400*00b67f09SDavid van Moolenbroek 		dns_message_renderreset(soaquery);
2401*00b67f09SDavid van Moolenbroek 		dns_message_settsigkey(soaquery, NULL);
2402*00b67f09SDavid van Moolenbroek 		sendrequest(&servers[ns_inuse], soaquery, &request);
2403*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, reqinfo, sizeof(nsu_requestinfo_t));
2404*00b67f09SDavid van Moolenbroek 		isc_event_free(&event);
2405*00b67f09SDavid van Moolenbroek 		setzoneclass(dns_rdataclass_none);
2406*00b67f09SDavid van Moolenbroek 		return;
2407*00b67f09SDavid van Moolenbroek 	}
2408*00b67f09SDavid van Moolenbroek 
2409*00b67f09SDavid van Moolenbroek 	isc_mem_put(gmctx, reqinfo, sizeof(nsu_requestinfo_t));
2410*00b67f09SDavid van Moolenbroek 	reqinfo = NULL;
2411*00b67f09SDavid van Moolenbroek 	isc_event_free(&event);
2412*00b67f09SDavid van Moolenbroek 	reqev = NULL;
2413*00b67f09SDavid van Moolenbroek 
2414*00b67f09SDavid van Moolenbroek 	ddebug("About to create rcvmsg");
2415*00b67f09SDavid van Moolenbroek 	result = dns_message_create(gmctx, DNS_MESSAGE_INTENTPARSE, &rcvmsg);
2416*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_create");
2417*00b67f09SDavid van Moolenbroek 	result = dns_request_getresponse(request, rcvmsg,
2418*00b67f09SDavid van Moolenbroek 					 DNS_MESSAGEPARSE_PRESERVEORDER);
2419*00b67f09SDavid van Moolenbroek 	if (result == DNS_R_TSIGERRORSET && servers != NULL) {
2420*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&rcvmsg);
2421*00b67f09SDavid van Moolenbroek 		ddebug("Destroying request [%p]", request);
2422*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2423*00b67f09SDavid van Moolenbroek 		reqinfo = isc_mem_get(gmctx, sizeof(nsu_requestinfo_t));
2424*00b67f09SDavid van Moolenbroek 		if (reqinfo == NULL)
2425*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
2426*00b67f09SDavid van Moolenbroek 		reqinfo->msg = soaquery;
2427*00b67f09SDavid van Moolenbroek 		reqinfo->addr = addr;
2428*00b67f09SDavid van Moolenbroek 		dns_message_renderreset(soaquery);
2429*00b67f09SDavid van Moolenbroek 		ddebug("retrying soa request without TSIG");
2430*00b67f09SDavid van Moolenbroek 
2431*00b67f09SDavid van Moolenbroek 		if (isc_sockaddr_pf(addr) == AF_INET6)
2432*00b67f09SDavid van Moolenbroek 			srcaddr = localaddr6;
2433*00b67f09SDavid van Moolenbroek 		else
2434*00b67f09SDavid van Moolenbroek 			srcaddr = localaddr4;
2435*00b67f09SDavid van Moolenbroek 
2436*00b67f09SDavid van Moolenbroek 		result = dns_request_createvia3(requestmgr, soaquery, srcaddr,
2437*00b67f09SDavid van Moolenbroek 						addr, 0, NULL,
2438*00b67f09SDavid van Moolenbroek 						FIND_TIMEOUT * 20,
2439*00b67f09SDavid van Moolenbroek 						FIND_TIMEOUT, 3,
2440*00b67f09SDavid van Moolenbroek 						global_task, recvsoa, reqinfo,
2441*00b67f09SDavid van Moolenbroek 						&request);
2442*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_request_createvia");
2443*00b67f09SDavid van Moolenbroek 		requests++;
2444*00b67f09SDavid van Moolenbroek 		return;
2445*00b67f09SDavid van Moolenbroek 	}
2446*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_request_getresponse");
2447*00b67f09SDavid van Moolenbroek 	section = DNS_SECTION_ANSWER;
2448*00b67f09SDavid van Moolenbroek 	POST(section);
2449*00b67f09SDavid van Moolenbroek 	if (debugging)
2450*00b67f09SDavid van Moolenbroek 		show_message(stderr, rcvmsg, "Reply from SOA query:");
2451*00b67f09SDavid van Moolenbroek 
2452*00b67f09SDavid van Moolenbroek 	if (rcvmsg->rcode != dns_rcode_noerror &&
2453*00b67f09SDavid van Moolenbroek 	    rcvmsg->rcode != dns_rcode_nxdomain)
2454*00b67f09SDavid van Moolenbroek 		fatal("response to SOA query was unsuccessful");
2455*00b67f09SDavid van Moolenbroek 
2456*00b67f09SDavid van Moolenbroek 	if (userzone != NULL && rcvmsg->rcode == dns_rcode_nxdomain) {
2457*00b67f09SDavid van Moolenbroek 		char namebuf[DNS_NAME_FORMATSIZE];
2458*00b67f09SDavid van Moolenbroek 		dns_name_format(userzone, namebuf, sizeof(namebuf));
2459*00b67f09SDavid van Moolenbroek 		error("specified zone '%s' does not exist (NXDOMAIN)",
2460*00b67f09SDavid van Moolenbroek 		      namebuf);
2461*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&rcvmsg);
2462*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2463*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&soaquery);
2464*00b67f09SDavid van Moolenbroek 		ddebug("Out of recvsoa");
2465*00b67f09SDavid van Moolenbroek 		done_update();
2466*00b67f09SDavid van Moolenbroek 		seenerror = ISC_TRUE;
2467*00b67f09SDavid van Moolenbroek 		return;
2468*00b67f09SDavid van Moolenbroek 	}
2469*00b67f09SDavid van Moolenbroek 
2470*00b67f09SDavid van Moolenbroek  lookforsoa:
2471*00b67f09SDavid van Moolenbroek 	if (pass == 0)
2472*00b67f09SDavid van Moolenbroek 		section = DNS_SECTION_ANSWER;
2473*00b67f09SDavid van Moolenbroek 	else if (pass == 1)
2474*00b67f09SDavid van Moolenbroek 		section = DNS_SECTION_AUTHORITY;
2475*00b67f09SDavid van Moolenbroek 	else
2476*00b67f09SDavid van Moolenbroek 		goto droplabel;
2477*00b67f09SDavid van Moolenbroek 
2478*00b67f09SDavid van Moolenbroek 	result = dns_message_firstname(rcvmsg, section);
2479*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS) {
2480*00b67f09SDavid van Moolenbroek 		pass++;
2481*00b67f09SDavid van Moolenbroek 		goto lookforsoa;
2482*00b67f09SDavid van Moolenbroek 	}
2483*00b67f09SDavid van Moolenbroek 	while (result == ISC_R_SUCCESS) {
2484*00b67f09SDavid van Moolenbroek 		name = NULL;
2485*00b67f09SDavid van Moolenbroek 		dns_message_currentname(rcvmsg, section, &name);
2486*00b67f09SDavid van Moolenbroek 		soaset = NULL;
2487*00b67f09SDavid van Moolenbroek 		result = dns_message_findtype(name, dns_rdatatype_soa, 0,
2488*00b67f09SDavid van Moolenbroek 					      &soaset);
2489*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_SUCCESS)
2490*00b67f09SDavid van Moolenbroek 			break;
2491*00b67f09SDavid van Moolenbroek 		if (section == DNS_SECTION_ANSWER) {
2492*00b67f09SDavid van Moolenbroek 			dns_rdataset_t *tset = NULL;
2493*00b67f09SDavid van Moolenbroek 			if (dns_message_findtype(name, dns_rdatatype_cname, 0,
2494*00b67f09SDavid van Moolenbroek 						 &tset) == ISC_R_SUCCESS ||
2495*00b67f09SDavid van Moolenbroek 			    dns_message_findtype(name, dns_rdatatype_dname, 0,
2496*00b67f09SDavid van Moolenbroek 						 &tset) == ISC_R_SUCCESS ) {
2497*00b67f09SDavid van Moolenbroek 				seencname = ISC_TRUE;
2498*00b67f09SDavid van Moolenbroek 				break;
2499*00b67f09SDavid van Moolenbroek 			}
2500*00b67f09SDavid van Moolenbroek 		}
2501*00b67f09SDavid van Moolenbroek 
2502*00b67f09SDavid van Moolenbroek 		result = dns_message_nextname(rcvmsg, section);
2503*00b67f09SDavid van Moolenbroek 	}
2504*00b67f09SDavid van Moolenbroek 
2505*00b67f09SDavid van Moolenbroek 	if (soaset == NULL && !seencname) {
2506*00b67f09SDavid van Moolenbroek 		pass++;
2507*00b67f09SDavid van Moolenbroek 		goto lookforsoa;
2508*00b67f09SDavid van Moolenbroek 	}
2509*00b67f09SDavid van Moolenbroek 
2510*00b67f09SDavid van Moolenbroek 	if (seencname)
2511*00b67f09SDavid van Moolenbroek 		goto droplabel;
2512*00b67f09SDavid van Moolenbroek 
2513*00b67f09SDavid van Moolenbroek 	if (debugging) {
2514*00b67f09SDavid van Moolenbroek 		char namestr[DNS_NAME_FORMATSIZE];
2515*00b67f09SDavid van Moolenbroek 		dns_name_format(name, namestr, sizeof(namestr));
2516*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "Found zone name: %s\n", namestr);
2517*00b67f09SDavid van Moolenbroek 	}
2518*00b67f09SDavid van Moolenbroek 
2519*00b67f09SDavid van Moolenbroek 	result = dns_rdataset_first(soaset);
2520*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_rdataset_first");
2521*00b67f09SDavid van Moolenbroek 
2522*00b67f09SDavid van Moolenbroek 	dns_rdata_init(&soarr);
2523*00b67f09SDavid van Moolenbroek 	dns_rdataset_current(soaset, &soarr);
2524*00b67f09SDavid van Moolenbroek 	result = dns_rdata_tostruct(&soarr, &soa, NULL);
2525*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_rdata_tostruct");
2526*00b67f09SDavid van Moolenbroek 
2527*00b67f09SDavid van Moolenbroek 	dns_name_init(&master, NULL);
2528*00b67f09SDavid van Moolenbroek 	dns_name_clone(&soa.origin, &master);
2529*00b67f09SDavid van Moolenbroek 
2530*00b67f09SDavid van Moolenbroek 	if (userzone != NULL)
2531*00b67f09SDavid van Moolenbroek 		zname = userzone;
2532*00b67f09SDavid van Moolenbroek 	else
2533*00b67f09SDavid van Moolenbroek 		zname = name;
2534*00b67f09SDavid van Moolenbroek 
2535*00b67f09SDavid van Moolenbroek 	if (debugging) {
2536*00b67f09SDavid van Moolenbroek 		char namestr[DNS_NAME_FORMATSIZE];
2537*00b67f09SDavid van Moolenbroek 		dns_name_format(&master, namestr, sizeof(namestr));
2538*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "The master is: %s\n", namestr);
2539*00b67f09SDavid van Moolenbroek 	}
2540*00b67f09SDavid van Moolenbroek 
2541*00b67f09SDavid van Moolenbroek 	if (default_servers) {
2542*00b67f09SDavid van Moolenbroek 		char serverstr[DNS_NAME_MAXTEXT+1];
2543*00b67f09SDavid van Moolenbroek 		isc_buffer_t buf;
2544*00b67f09SDavid van Moolenbroek 		size_t size;
2545*00b67f09SDavid van Moolenbroek 
2546*00b67f09SDavid van Moolenbroek 		isc_buffer_init(&buf, serverstr, sizeof(serverstr));
2547*00b67f09SDavid van Moolenbroek 		result = dns_name_totext(&master, ISC_TRUE, &buf);
2548*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_name_totext");
2549*00b67f09SDavid van Moolenbroek 		serverstr[isc_buffer_usedlength(&buf)] = 0;
2550*00b67f09SDavid van Moolenbroek 
2551*00b67f09SDavid van Moolenbroek 		if (master_servers != NULL && master_servers != servers)
2552*00b67f09SDavid van Moolenbroek 			isc_mem_put(gmctx, master_servers,
2553*00b67f09SDavid van Moolenbroek 				    master_total * sizeof(isc_sockaddr_t));
2554*00b67f09SDavid van Moolenbroek 		master_total = MAX_SERVERADDRS;
2555*00b67f09SDavid van Moolenbroek 		size = master_total * sizeof(isc_sockaddr_t);
2556*00b67f09SDavid van Moolenbroek 		master_servers = isc_mem_get(gmctx, size);
2557*00b67f09SDavid van Moolenbroek 		if (master_servers == NULL)
2558*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
2559*00b67f09SDavid van Moolenbroek 
2560*00b67f09SDavid van Moolenbroek 		memset(master_servers, 0, size);
2561*00b67f09SDavid van Moolenbroek 		get_addresses(serverstr, dnsport, master_servers, master_total);
2562*00b67f09SDavid van Moolenbroek 		master_inuse = 0;
2563*00b67f09SDavid van Moolenbroek 	} else
2564*00b67f09SDavid van Moolenbroek 		master_from_servers();
2565*00b67f09SDavid van Moolenbroek 	dns_rdata_freestruct(&soa);
2566*00b67f09SDavid van Moolenbroek 
2567*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
2568*00b67f09SDavid van Moolenbroek 	if (usegsstsig) {
2569*00b67f09SDavid van Moolenbroek 		dns_name_init(&tmpzonename, NULL);
2570*00b67f09SDavid van Moolenbroek 		dns_name_dup(zname, gmctx, &tmpzonename);
2571*00b67f09SDavid van Moolenbroek 		dns_name_init(&restart_master, NULL);
2572*00b67f09SDavid van Moolenbroek 		dns_name_dup(&master, gmctx, &restart_master);
2573*00b67f09SDavid van Moolenbroek 		start_gssrequest(&master);
2574*00b67f09SDavid van Moolenbroek 	} else {
2575*00b67f09SDavid van Moolenbroek 		send_update(zname, &master_servers[master_inuse]);
2576*00b67f09SDavid van Moolenbroek 		setzoneclass(dns_rdataclass_none);
2577*00b67f09SDavid van Moolenbroek 	}
2578*00b67f09SDavid van Moolenbroek #else
2579*00b67f09SDavid van Moolenbroek 	send_update(zname, &master_servers[master_inuse]);
2580*00b67f09SDavid van Moolenbroek 	setzoneclass(dns_rdataclass_none);
2581*00b67f09SDavid van Moolenbroek #endif
2582*00b67f09SDavid van Moolenbroek 
2583*00b67f09SDavid van Moolenbroek 	dns_message_destroy(&soaquery);
2584*00b67f09SDavid van Moolenbroek 	dns_request_destroy(&request);
2585*00b67f09SDavid van Moolenbroek 
2586*00b67f09SDavid van Moolenbroek  out:
2587*00b67f09SDavid van Moolenbroek 	dns_message_destroy(&rcvmsg);
2588*00b67f09SDavid van Moolenbroek 	ddebug("Out of recvsoa");
2589*00b67f09SDavid van Moolenbroek 	return;
2590*00b67f09SDavid van Moolenbroek 
2591*00b67f09SDavid van Moolenbroek  droplabel:
2592*00b67f09SDavid van Moolenbroek 	result = dns_message_firstname(soaquery, DNS_SECTION_QUESTION);
2593*00b67f09SDavid van Moolenbroek 	INSIST(result == ISC_R_SUCCESS);
2594*00b67f09SDavid van Moolenbroek 	name = NULL;
2595*00b67f09SDavid van Moolenbroek 	dns_message_currentname(soaquery, DNS_SECTION_QUESTION, &name);
2596*00b67f09SDavid van Moolenbroek 	nlabels = dns_name_countlabels(name);
2597*00b67f09SDavid van Moolenbroek 	if (nlabels == 1)
2598*00b67f09SDavid van Moolenbroek 		fatal("could not find enclosing zone");
2599*00b67f09SDavid van Moolenbroek 	dns_name_init(&tname, NULL);
2600*00b67f09SDavid van Moolenbroek 	dns_name_getlabelsequence(name, 1, nlabels - 1, &tname);
2601*00b67f09SDavid van Moolenbroek 	dns_name_clone(&tname, name);
2602*00b67f09SDavid van Moolenbroek 	dns_request_destroy(&request);
2603*00b67f09SDavid van Moolenbroek 	dns_message_renderreset(soaquery);
2604*00b67f09SDavid van Moolenbroek 	dns_message_settsigkey(soaquery, NULL);
2605*00b67f09SDavid van Moolenbroek 	sendrequest(&servers[ns_inuse], soaquery, &request);
2606*00b67f09SDavid van Moolenbroek 	goto out;
2607*00b67f09SDavid van Moolenbroek }
2608*00b67f09SDavid van Moolenbroek 
2609*00b67f09SDavid van Moolenbroek static void
sendrequest(isc_sockaddr_t * destaddr,dns_message_t * msg,dns_request_t ** request)2610*00b67f09SDavid van Moolenbroek sendrequest(isc_sockaddr_t *destaddr, dns_message_t *msg,
2611*00b67f09SDavid van Moolenbroek 	    dns_request_t **request)
2612*00b67f09SDavid van Moolenbroek {
2613*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2614*00b67f09SDavid van Moolenbroek 	nsu_requestinfo_t *reqinfo;
2615*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *srcaddr;
2616*00b67f09SDavid van Moolenbroek 
2617*00b67f09SDavid van Moolenbroek 	reqinfo = isc_mem_get(gmctx, sizeof(nsu_requestinfo_t));
2618*00b67f09SDavid van Moolenbroek 	if (reqinfo == NULL)
2619*00b67f09SDavid van Moolenbroek 		fatal("out of memory");
2620*00b67f09SDavid van Moolenbroek 	reqinfo->msg = msg;
2621*00b67f09SDavid van Moolenbroek 	reqinfo->addr = destaddr;
2622*00b67f09SDavid van Moolenbroek 
2623*00b67f09SDavid van Moolenbroek 	if (isc_sockaddr_pf(destaddr) == AF_INET6)
2624*00b67f09SDavid van Moolenbroek 		srcaddr = localaddr6;
2625*00b67f09SDavid van Moolenbroek 	else
2626*00b67f09SDavid van Moolenbroek 		srcaddr = localaddr4;
2627*00b67f09SDavid van Moolenbroek 
2628*00b67f09SDavid van Moolenbroek 	result = dns_request_createvia3(requestmgr, msg, srcaddr, destaddr, 0,
2629*00b67f09SDavid van Moolenbroek 					default_servers ? NULL : tsigkey,
2630*00b67f09SDavid van Moolenbroek 					FIND_TIMEOUT * 20, FIND_TIMEOUT, 3,
2631*00b67f09SDavid van Moolenbroek 					global_task, recvsoa, reqinfo, request);
2632*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_request_createvia");
2633*00b67f09SDavid van Moolenbroek 	requests++;
2634*00b67f09SDavid van Moolenbroek }
2635*00b67f09SDavid van Moolenbroek 
2636*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
2637*00b67f09SDavid van Moolenbroek 
2638*00b67f09SDavid van Moolenbroek /*
2639*00b67f09SDavid van Moolenbroek  * Get the realm from the users kerberos ticket if possible
2640*00b67f09SDavid van Moolenbroek  */
2641*00b67f09SDavid van Moolenbroek static void
get_ticket_realm(isc_mem_t * mctx)2642*00b67f09SDavid van Moolenbroek get_ticket_realm(isc_mem_t *mctx) {
2643*00b67f09SDavid van Moolenbroek 	krb5_context ctx;
2644*00b67f09SDavid van Moolenbroek 	krb5_error_code rc;
2645*00b67f09SDavid van Moolenbroek 	krb5_ccache ccache;
2646*00b67f09SDavid van Moolenbroek 	krb5_principal princ;
2647*00b67f09SDavid van Moolenbroek 	char *name, *ticket_realm;
2648*00b67f09SDavid van Moolenbroek 
2649*00b67f09SDavid van Moolenbroek 	rc = krb5_init_context(&ctx);
2650*00b67f09SDavid van Moolenbroek 	if (rc != 0)
2651*00b67f09SDavid van Moolenbroek 		return;
2652*00b67f09SDavid van Moolenbroek 
2653*00b67f09SDavid van Moolenbroek 	rc = krb5_cc_default(ctx, &ccache);
2654*00b67f09SDavid van Moolenbroek 	if (rc != 0) {
2655*00b67f09SDavid van Moolenbroek 		krb5_free_context(ctx);
2656*00b67f09SDavid van Moolenbroek 		return;
2657*00b67f09SDavid van Moolenbroek 	}
2658*00b67f09SDavid van Moolenbroek 
2659*00b67f09SDavid van Moolenbroek 	rc = krb5_cc_get_principal(ctx, ccache, &princ);
2660*00b67f09SDavid van Moolenbroek 	if (rc != 0) {
2661*00b67f09SDavid van Moolenbroek 		krb5_cc_close(ctx, ccache);
2662*00b67f09SDavid van Moolenbroek 		krb5_free_context(ctx);
2663*00b67f09SDavid van Moolenbroek 		return;
2664*00b67f09SDavid van Moolenbroek 	}
2665*00b67f09SDavid van Moolenbroek 
2666*00b67f09SDavid van Moolenbroek 	rc = krb5_unparse_name(ctx, princ, &name);
2667*00b67f09SDavid van Moolenbroek 	if (rc != 0) {
2668*00b67f09SDavid van Moolenbroek 		krb5_free_principal(ctx, princ);
2669*00b67f09SDavid van Moolenbroek 		krb5_cc_close(ctx, ccache);
2670*00b67f09SDavid van Moolenbroek 		krb5_free_context(ctx);
2671*00b67f09SDavid van Moolenbroek 		return;
2672*00b67f09SDavid van Moolenbroek 	}
2673*00b67f09SDavid van Moolenbroek 
2674*00b67f09SDavid van Moolenbroek 	ticket_realm = strrchr(name, '@');
2675*00b67f09SDavid van Moolenbroek 	if (ticket_realm != NULL) {
2676*00b67f09SDavid van Moolenbroek 		realm = isc_mem_strdup(mctx, ticket_realm);
2677*00b67f09SDavid van Moolenbroek 	}
2678*00b67f09SDavid van Moolenbroek 
2679*00b67f09SDavid van Moolenbroek 	free(name);
2680*00b67f09SDavid van Moolenbroek 	krb5_free_principal(ctx, princ);
2681*00b67f09SDavid van Moolenbroek 	krb5_cc_close(ctx, ccache);
2682*00b67f09SDavid van Moolenbroek 	krb5_free_context(ctx);
2683*00b67f09SDavid van Moolenbroek 	if (realm != NULL && debugging)
2684*00b67f09SDavid van Moolenbroek 		fprintf(stderr, "Found realm from ticket: %s\n", realm+1);
2685*00b67f09SDavid van Moolenbroek }
2686*00b67f09SDavid van Moolenbroek 
2687*00b67f09SDavid van Moolenbroek 
2688*00b67f09SDavid van Moolenbroek static void
start_gssrequest(dns_name_t * master)2689*00b67f09SDavid van Moolenbroek start_gssrequest(dns_name_t *master) {
2690*00b67f09SDavid van Moolenbroek 	gss_ctx_id_t context;
2691*00b67f09SDavid van Moolenbroek 	isc_buffer_t buf;
2692*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2693*00b67f09SDavid van Moolenbroek 	isc_uint32_t val = 0;
2694*00b67f09SDavid van Moolenbroek 	dns_message_t *rmsg;
2695*00b67f09SDavid van Moolenbroek 	dns_request_t *request = NULL;
2696*00b67f09SDavid van Moolenbroek 	dns_name_t *servname;
2697*00b67f09SDavid van Moolenbroek 	dns_fixedname_t fname;
2698*00b67f09SDavid van Moolenbroek 	char namestr[DNS_NAME_FORMATSIZE];
2699*00b67f09SDavid van Moolenbroek 	char mykeystr[DNS_NAME_FORMATSIZE];
2700*00b67f09SDavid van Moolenbroek 	char *err_message = NULL;
2701*00b67f09SDavid van Moolenbroek 
2702*00b67f09SDavid van Moolenbroek 	debug("start_gssrequest");
2703*00b67f09SDavid van Moolenbroek 	usevc = ISC_TRUE;
2704*00b67f09SDavid van Moolenbroek 
2705*00b67f09SDavid van Moolenbroek 	if (gssring != NULL)
2706*00b67f09SDavid van Moolenbroek 		dns_tsigkeyring_detach(&gssring);
2707*00b67f09SDavid van Moolenbroek 	gssring = NULL;
2708*00b67f09SDavid van Moolenbroek 	result = dns_tsigkeyring_create(gmctx, &gssring);
2709*00b67f09SDavid van Moolenbroek 
2710*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2711*00b67f09SDavid van Moolenbroek 		fatal("dns_tsigkeyring_create failed: %s",
2712*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2713*00b67f09SDavid van Moolenbroek 
2714*00b67f09SDavid van Moolenbroek 	dns_name_format(master, namestr, sizeof(namestr));
2715*00b67f09SDavid van Moolenbroek 	if (kserver == NULL) {
2716*00b67f09SDavid van Moolenbroek 		kserver = isc_mem_get(gmctx, sizeof(isc_sockaddr_t));
2717*00b67f09SDavid van Moolenbroek 		if (kserver == NULL)
2718*00b67f09SDavid van Moolenbroek 			fatal("out of memory");
2719*00b67f09SDavid van Moolenbroek 	}
2720*00b67f09SDavid van Moolenbroek 	if (servers == NULL)
2721*00b67f09SDavid van Moolenbroek 		get_addresses(namestr, dnsport, kserver, 1);
2722*00b67f09SDavid van Moolenbroek 	else
2723*00b67f09SDavid van Moolenbroek 		memmove(kserver, &servers[ns_inuse], sizeof(isc_sockaddr_t));
2724*00b67f09SDavid van Moolenbroek 
2725*00b67f09SDavid van Moolenbroek 	dns_fixedname_init(&fname);
2726*00b67f09SDavid van Moolenbroek 	servname = dns_fixedname_name(&fname);
2727*00b67f09SDavid van Moolenbroek 
2728*00b67f09SDavid van Moolenbroek 	if (realm == NULL)
2729*00b67f09SDavid van Moolenbroek 		get_ticket_realm(gmctx);
2730*00b67f09SDavid van Moolenbroek 
2731*00b67f09SDavid van Moolenbroek 	result = isc_string_printf(servicename, sizeof(servicename),
2732*00b67f09SDavid van Moolenbroek 				   "DNS/%s%s", namestr, realm ? realm : "");
2733*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2734*00b67f09SDavid van Moolenbroek 		fatal("isc_string_printf(servicename) failed: %s",
2735*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2736*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&buf, servicename, strlen(servicename));
2737*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&buf, strlen(servicename));
2738*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(servname, &buf, dns_rootname, 0, NULL);
2739*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2740*00b67f09SDavid van Moolenbroek 		fatal("dns_name_fromtext(servname) failed: %s",
2741*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2742*00b67f09SDavid van Moolenbroek 
2743*00b67f09SDavid van Moolenbroek 	dns_fixedname_init(&fkname);
2744*00b67f09SDavid van Moolenbroek 	keyname = dns_fixedname_name(&fkname);
2745*00b67f09SDavid van Moolenbroek 
2746*00b67f09SDavid van Moolenbroek 	isc_random_get(&val);
2747*00b67f09SDavid van Moolenbroek 	result = isc_string_printf(mykeystr, sizeof(mykeystr), "%u.sig-%s",
2748*00b67f09SDavid van Moolenbroek 				   val, namestr);
2749*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2750*00b67f09SDavid van Moolenbroek 		fatal("isc_string_printf(mykeystr) failed: %s",
2751*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2752*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&buf, mykeystr, strlen(mykeystr));
2753*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&buf, strlen(mykeystr));
2754*00b67f09SDavid van Moolenbroek 
2755*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(keyname, &buf, dns_rootname, 0, NULL);
2756*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2757*00b67f09SDavid van Moolenbroek 		fatal("dns_name_fromtext(keyname) failed: %s",
2758*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2759*00b67f09SDavid van Moolenbroek 
2760*00b67f09SDavid van Moolenbroek 	/* Windows doesn't recognize name compression in the key name. */
2761*00b67f09SDavid van Moolenbroek 	keyname->attributes |= DNS_NAMEATTR_NOCOMPRESS;
2762*00b67f09SDavid van Moolenbroek 
2763*00b67f09SDavid van Moolenbroek 	rmsg = NULL;
2764*00b67f09SDavid van Moolenbroek 	result = dns_message_create(gmctx, DNS_MESSAGE_INTENTRENDER, &rmsg);
2765*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2766*00b67f09SDavid van Moolenbroek 		fatal("dns_message_create failed: %s",
2767*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2768*00b67f09SDavid van Moolenbroek 
2769*00b67f09SDavid van Moolenbroek 	/* Build first request. */
2770*00b67f09SDavid van Moolenbroek 	context = GSS_C_NO_CONTEXT;
2771*00b67f09SDavid van Moolenbroek 	result = dns_tkey_buildgssquery(rmsg, keyname, servname, NULL, 0,
2772*00b67f09SDavid van Moolenbroek 					&context, use_win2k_gsstsig,
2773*00b67f09SDavid van Moolenbroek 					gmctx, &err_message);
2774*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_FAILURE)
2775*00b67f09SDavid van Moolenbroek 		fatal("tkey query failed: %s",
2776*00b67f09SDavid van Moolenbroek 		      err_message != NULL ? err_message : "unknown error");
2777*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
2778*00b67f09SDavid van Moolenbroek 		fatal("dns_tkey_buildgssquery failed: %s",
2779*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result));
2780*00b67f09SDavid van Moolenbroek 
2781*00b67f09SDavid van Moolenbroek 	send_gssrequest(kserver, rmsg, &request, context);
2782*00b67f09SDavid van Moolenbroek }
2783*00b67f09SDavid van Moolenbroek 
2784*00b67f09SDavid van Moolenbroek static void
send_gssrequest(isc_sockaddr_t * destaddr,dns_message_t * msg,dns_request_t ** request,gss_ctx_id_t context)2785*00b67f09SDavid van Moolenbroek send_gssrequest(isc_sockaddr_t *destaddr, dns_message_t *msg,
2786*00b67f09SDavid van Moolenbroek 		dns_request_t **request, gss_ctx_id_t context)
2787*00b67f09SDavid van Moolenbroek {
2788*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2789*00b67f09SDavid van Moolenbroek 	nsu_gssinfo_t *reqinfo;
2790*00b67f09SDavid van Moolenbroek 	unsigned int options = 0;
2791*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *srcaddr;
2792*00b67f09SDavid van Moolenbroek 
2793*00b67f09SDavid van Moolenbroek 	debug("send_gssrequest");
2794*00b67f09SDavid van Moolenbroek 	reqinfo = isc_mem_get(gmctx, sizeof(nsu_gssinfo_t));
2795*00b67f09SDavid van Moolenbroek 	if (reqinfo == NULL)
2796*00b67f09SDavid van Moolenbroek 		fatal("out of memory");
2797*00b67f09SDavid van Moolenbroek 	reqinfo->msg = msg;
2798*00b67f09SDavid van Moolenbroek 	reqinfo->addr = destaddr;
2799*00b67f09SDavid van Moolenbroek 	reqinfo->context = context;
2800*00b67f09SDavid van Moolenbroek 
2801*00b67f09SDavid van Moolenbroek 	options |= DNS_REQUESTOPT_TCP;
2802*00b67f09SDavid van Moolenbroek 
2803*00b67f09SDavid van Moolenbroek 	if (isc_sockaddr_pf(destaddr) == AF_INET6)
2804*00b67f09SDavid van Moolenbroek 		srcaddr = localaddr6;
2805*00b67f09SDavid van Moolenbroek 	else
2806*00b67f09SDavid van Moolenbroek 		srcaddr = localaddr4;
2807*00b67f09SDavid van Moolenbroek 
2808*00b67f09SDavid van Moolenbroek 	result = dns_request_createvia3(requestmgr, msg, srcaddr, destaddr,
2809*00b67f09SDavid van Moolenbroek 					options, tsigkey, FIND_TIMEOUT * 20,
2810*00b67f09SDavid van Moolenbroek 					FIND_TIMEOUT, 3, global_task, recvgss,
2811*00b67f09SDavid van Moolenbroek 					reqinfo, request);
2812*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_request_createvia3");
2813*00b67f09SDavid van Moolenbroek 	if (debugging)
2814*00b67f09SDavid van Moolenbroek 		show_message(stdout, msg, "Outgoing update query:");
2815*00b67f09SDavid van Moolenbroek 	requests++;
2816*00b67f09SDavid van Moolenbroek }
2817*00b67f09SDavid van Moolenbroek 
2818*00b67f09SDavid van Moolenbroek static void
recvgss(isc_task_t * task,isc_event_t * event)2819*00b67f09SDavid van Moolenbroek recvgss(isc_task_t *task, isc_event_t *event) {
2820*00b67f09SDavid van Moolenbroek 	dns_requestevent_t *reqev = NULL;
2821*00b67f09SDavid van Moolenbroek 	dns_request_t *request = NULL;
2822*00b67f09SDavid van Moolenbroek 	isc_result_t result, eresult;
2823*00b67f09SDavid van Moolenbroek 	dns_message_t *rcvmsg = NULL;
2824*00b67f09SDavid van Moolenbroek 	nsu_gssinfo_t *reqinfo;
2825*00b67f09SDavid van Moolenbroek 	dns_message_t *tsigquery = NULL;
2826*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *addr;
2827*00b67f09SDavid van Moolenbroek 	gss_ctx_id_t context;
2828*00b67f09SDavid van Moolenbroek 	isc_buffer_t buf;
2829*00b67f09SDavid van Moolenbroek 	dns_name_t *servname;
2830*00b67f09SDavid van Moolenbroek 	dns_fixedname_t fname;
2831*00b67f09SDavid van Moolenbroek 	char *err_message = NULL;
2832*00b67f09SDavid van Moolenbroek 
2833*00b67f09SDavid van Moolenbroek 	UNUSED(task);
2834*00b67f09SDavid van Moolenbroek 
2835*00b67f09SDavid van Moolenbroek 	ddebug("recvgss()");
2836*00b67f09SDavid van Moolenbroek 
2837*00b67f09SDavid van Moolenbroek 	requests--;
2838*00b67f09SDavid van Moolenbroek 
2839*00b67f09SDavid van Moolenbroek 	REQUIRE(event->ev_type == DNS_EVENT_REQUESTDONE);
2840*00b67f09SDavid van Moolenbroek 	reqev = (dns_requestevent_t *)event;
2841*00b67f09SDavid van Moolenbroek 	request = reqev->request;
2842*00b67f09SDavid van Moolenbroek 	eresult = reqev->result;
2843*00b67f09SDavid van Moolenbroek 	reqinfo = reqev->ev_arg;
2844*00b67f09SDavid van Moolenbroek 	tsigquery = reqinfo->msg;
2845*00b67f09SDavid van Moolenbroek 	context = reqinfo->context;
2846*00b67f09SDavid van Moolenbroek 	addr = reqinfo->addr;
2847*00b67f09SDavid van Moolenbroek 
2848*00b67f09SDavid van Moolenbroek 	if (shuttingdown) {
2849*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2850*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&tsigquery);
2851*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, reqinfo, sizeof(nsu_gssinfo_t));
2852*00b67f09SDavid van Moolenbroek 		isc_event_free(&event);
2853*00b67f09SDavid van Moolenbroek 		maybeshutdown();
2854*00b67f09SDavid van Moolenbroek 		return;
2855*00b67f09SDavid van Moolenbroek 	}
2856*00b67f09SDavid van Moolenbroek 
2857*00b67f09SDavid van Moolenbroek 	if (eresult != ISC_R_SUCCESS) {
2858*00b67f09SDavid van Moolenbroek 		next_server("recvgss", addr, eresult);
2859*00b67f09SDavid van Moolenbroek 		ddebug("Destroying request [%p]", request);
2860*00b67f09SDavid van Moolenbroek 		dns_request_destroy(&request);
2861*00b67f09SDavid van Moolenbroek 		dns_message_renderreset(tsigquery);
2862*00b67f09SDavid van Moolenbroek 		sendrequest(&servers[ns_inuse], tsigquery, &request);
2863*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, reqinfo, sizeof(nsu_gssinfo_t));
2864*00b67f09SDavid van Moolenbroek 		isc_event_free(&event);
2865*00b67f09SDavid van Moolenbroek 		return;
2866*00b67f09SDavid van Moolenbroek 	}
2867*00b67f09SDavid van Moolenbroek 	isc_mem_put(gmctx, reqinfo, sizeof(nsu_gssinfo_t));
2868*00b67f09SDavid van Moolenbroek 
2869*00b67f09SDavid van Moolenbroek 	isc_event_free(&event);
2870*00b67f09SDavid van Moolenbroek 	reqev = NULL;
2871*00b67f09SDavid van Moolenbroek 
2872*00b67f09SDavid van Moolenbroek 	ddebug("recvgss creating rcvmsg");
2873*00b67f09SDavid van Moolenbroek 	result = dns_message_create(gmctx, DNS_MESSAGE_INTENTPARSE, &rcvmsg);
2874*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_create");
2875*00b67f09SDavid van Moolenbroek 
2876*00b67f09SDavid van Moolenbroek 	result = dns_request_getresponse(request, rcvmsg,
2877*00b67f09SDavid van Moolenbroek 					 DNS_MESSAGEPARSE_PRESERVEORDER);
2878*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_request_getresponse");
2879*00b67f09SDavid van Moolenbroek 
2880*00b67f09SDavid van Moolenbroek 	if (debugging)
2881*00b67f09SDavid van Moolenbroek 		show_message(stderr, rcvmsg,
2882*00b67f09SDavid van Moolenbroek 			     "recvmsg reply from GSS-TSIG query");
2883*00b67f09SDavid van Moolenbroek 
2884*00b67f09SDavid van Moolenbroek 	if (rcvmsg->rcode == dns_rcode_formerr && !tried_other_gsstsig) {
2885*00b67f09SDavid van Moolenbroek 		ddebug("recvgss trying %s GSS-TSIG",
2886*00b67f09SDavid van Moolenbroek 		       use_win2k_gsstsig ? "Standard" : "Win2k");
2887*00b67f09SDavid van Moolenbroek 		if (use_win2k_gsstsig)
2888*00b67f09SDavid van Moolenbroek 			use_win2k_gsstsig = ISC_FALSE;
2889*00b67f09SDavid van Moolenbroek 		else
2890*00b67f09SDavid van Moolenbroek 			use_win2k_gsstsig = ISC_TRUE;
2891*00b67f09SDavid van Moolenbroek 		tried_other_gsstsig = ISC_TRUE;
2892*00b67f09SDavid van Moolenbroek 		start_gssrequest(&restart_master);
2893*00b67f09SDavid van Moolenbroek 		goto done;
2894*00b67f09SDavid van Moolenbroek 	}
2895*00b67f09SDavid van Moolenbroek 
2896*00b67f09SDavid van Moolenbroek 	if (rcvmsg->rcode != dns_rcode_noerror &&
2897*00b67f09SDavid van Moolenbroek 	    rcvmsg->rcode != dns_rcode_nxdomain)
2898*00b67f09SDavid van Moolenbroek 		fatal("response to GSS-TSIG query was unsuccessful");
2899*00b67f09SDavid van Moolenbroek 
2900*00b67f09SDavid van Moolenbroek 
2901*00b67f09SDavid van Moolenbroek 	dns_fixedname_init(&fname);
2902*00b67f09SDavid van Moolenbroek 	servname = dns_fixedname_name(&fname);
2903*00b67f09SDavid van Moolenbroek 	isc_buffer_init(&buf, servicename, strlen(servicename));
2904*00b67f09SDavid van Moolenbroek 	isc_buffer_add(&buf, strlen(servicename));
2905*00b67f09SDavid van Moolenbroek 	result = dns_name_fromtext(servname, &buf, dns_rootname, 0, NULL);
2906*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_name_fromtext");
2907*00b67f09SDavid van Moolenbroek 
2908*00b67f09SDavid van Moolenbroek 	tsigkey = NULL;
2909*00b67f09SDavid van Moolenbroek 	result = dns_tkey_gssnegotiate(tsigquery, rcvmsg, servname,
2910*00b67f09SDavid van Moolenbroek 				       &context, &tsigkey, gssring,
2911*00b67f09SDavid van Moolenbroek 				       use_win2k_gsstsig,
2912*00b67f09SDavid van Moolenbroek 				       &err_message);
2913*00b67f09SDavid van Moolenbroek 	switch (result) {
2914*00b67f09SDavid van Moolenbroek 
2915*00b67f09SDavid van Moolenbroek 	case DNS_R_CONTINUE:
2916*00b67f09SDavid van Moolenbroek 		send_gssrequest(kserver, tsigquery, &request, context);
2917*00b67f09SDavid van Moolenbroek 		break;
2918*00b67f09SDavid van Moolenbroek 
2919*00b67f09SDavid van Moolenbroek 	case ISC_R_SUCCESS:
2920*00b67f09SDavid van Moolenbroek 		/*
2921*00b67f09SDavid van Moolenbroek 		 * XXXSRA Waaay too much fun here.  There's no good
2922*00b67f09SDavid van Moolenbroek 		 * reason why we need a TSIG here (the people who put
2923*00b67f09SDavid van Moolenbroek 		 * it into the spec admitted at the time that it was
2924*00b67f09SDavid van Moolenbroek 		 * not a security issue), and Windows clients don't
2925*00b67f09SDavid van Moolenbroek 		 * seem to work if named complies with the spec and
2926*00b67f09SDavid van Moolenbroek 		 * includes the gratuitous TSIG.  So we're in the
2927*00b67f09SDavid van Moolenbroek 		 * bizarre situation of having to choose between
2928*00b67f09SDavid van Moolenbroek 		 * complying with a useless requirement in the spec
2929*00b67f09SDavid van Moolenbroek 		 * and interoperating.  This is nuts.  If we can
2930*00b67f09SDavid van Moolenbroek 		 * confirm this behavior, we should ask the WG to
2931*00b67f09SDavid van Moolenbroek 		 * consider removing the requirement for the
2932*00b67f09SDavid van Moolenbroek 		 * gratuitous TSIG here.  For the moment, we ignore
2933*00b67f09SDavid van Moolenbroek 		 * the TSIG -- this too is a spec violation, but it's
2934*00b67f09SDavid van Moolenbroek 		 * the least insane thing to do.
2935*00b67f09SDavid van Moolenbroek 		 */
2936*00b67f09SDavid van Moolenbroek #if 0
2937*00b67f09SDavid van Moolenbroek 		/*
2938*00b67f09SDavid van Moolenbroek 		 * Verify the signature.
2939*00b67f09SDavid van Moolenbroek 		 */
2940*00b67f09SDavid van Moolenbroek 		rcvmsg->state = DNS_SECTION_ANY;
2941*00b67f09SDavid van Moolenbroek 		dns_message_setquerytsig(rcvmsg, NULL);
2942*00b67f09SDavid van Moolenbroek 		result = dns_message_settsigkey(rcvmsg, tsigkey);
2943*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_message_settsigkey");
2944*00b67f09SDavid van Moolenbroek 		result = dns_message_checksig(rcvmsg, NULL);
2945*00b67f09SDavid van Moolenbroek 		ddebug("tsig verification: %s", dns_result_totext(result));
2946*00b67f09SDavid van Moolenbroek 		check_result(result, "dns_message_checksig");
2947*00b67f09SDavid van Moolenbroek #endif /* 0 */
2948*00b67f09SDavid van Moolenbroek 
2949*00b67f09SDavid van Moolenbroek 		send_update(&tmpzonename, &master_servers[master_inuse]);
2950*00b67f09SDavid van Moolenbroek 		setzoneclass(dns_rdataclass_none);
2951*00b67f09SDavid van Moolenbroek 		break;
2952*00b67f09SDavid van Moolenbroek 
2953*00b67f09SDavid van Moolenbroek 	default:
2954*00b67f09SDavid van Moolenbroek 		fatal("dns_tkey_negotiategss: %s %s",
2955*00b67f09SDavid van Moolenbroek 		      isc_result_totext(result),
2956*00b67f09SDavid van Moolenbroek 		      err_message != NULL ? err_message : "");
2957*00b67f09SDavid van Moolenbroek 	}
2958*00b67f09SDavid van Moolenbroek 
2959*00b67f09SDavid van Moolenbroek  done:
2960*00b67f09SDavid van Moolenbroek 	dns_request_destroy(&request);
2961*00b67f09SDavid van Moolenbroek 	dns_message_destroy(&tsigquery);
2962*00b67f09SDavid van Moolenbroek 
2963*00b67f09SDavid van Moolenbroek 	dns_message_destroy(&rcvmsg);
2964*00b67f09SDavid van Moolenbroek 	ddebug("Out of recvgss");
2965*00b67f09SDavid van Moolenbroek }
2966*00b67f09SDavid van Moolenbroek #endif
2967*00b67f09SDavid van Moolenbroek 
2968*00b67f09SDavid van Moolenbroek static void
start_update(void)2969*00b67f09SDavid van Moolenbroek start_update(void) {
2970*00b67f09SDavid van Moolenbroek 	isc_result_t result;
2971*00b67f09SDavid van Moolenbroek 	dns_rdataset_t *rdataset = NULL;
2972*00b67f09SDavid van Moolenbroek 	dns_name_t *name = NULL;
2973*00b67f09SDavid van Moolenbroek 	dns_request_t *request = NULL;
2974*00b67f09SDavid van Moolenbroek 	dns_message_t *soaquery = NULL;
2975*00b67f09SDavid van Moolenbroek 	dns_name_t *firstname;
2976*00b67f09SDavid van Moolenbroek 	dns_section_t section = DNS_SECTION_UPDATE;
2977*00b67f09SDavid van Moolenbroek 
2978*00b67f09SDavid van Moolenbroek 	ddebug("start_update()");
2979*00b67f09SDavid van Moolenbroek 
2980*00b67f09SDavid van Moolenbroek 	if (answer != NULL)
2981*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&answer);
2982*00b67f09SDavid van Moolenbroek 
2983*00b67f09SDavid van Moolenbroek 	/*
2984*00b67f09SDavid van Moolenbroek 	 * If we have both the zone and the servers we have enough information
2985*00b67f09SDavid van Moolenbroek 	 * to send the update straight away otherwise we need to discover
2986*00b67f09SDavid van Moolenbroek 	 * the zone and / or the master server.
2987*00b67f09SDavid van Moolenbroek 	 */
2988*00b67f09SDavid van Moolenbroek 	if (userzone != NULL && !default_servers && !usegsstsig) {
2989*00b67f09SDavid van Moolenbroek 		master_from_servers();
2990*00b67f09SDavid van Moolenbroek 		send_update(userzone, &master_servers[master_inuse]);
2991*00b67f09SDavid van Moolenbroek 		setzoneclass(dns_rdataclass_none);
2992*00b67f09SDavid van Moolenbroek 		return;
2993*00b67f09SDavid van Moolenbroek 	}
2994*00b67f09SDavid van Moolenbroek 
2995*00b67f09SDavid van Moolenbroek 	result = dns_message_create(gmctx, DNS_MESSAGE_INTENTRENDER,
2996*00b67f09SDavid van Moolenbroek 				    &soaquery);
2997*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_create");
2998*00b67f09SDavid van Moolenbroek 
2999*00b67f09SDavid van Moolenbroek 	if (default_servers)
3000*00b67f09SDavid van Moolenbroek 		soaquery->flags |= DNS_MESSAGEFLAG_RD;
3001*00b67f09SDavid van Moolenbroek 
3002*00b67f09SDavid van Moolenbroek 	result = dns_message_gettempname(soaquery, &name);
3003*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettempname");
3004*00b67f09SDavid van Moolenbroek 
3005*00b67f09SDavid van Moolenbroek 	result = dns_message_gettemprdataset(soaquery, &rdataset);
3006*00b67f09SDavid van Moolenbroek 	check_result(result, "dns_message_gettemprdataset");
3007*00b67f09SDavid van Moolenbroek 
3008*00b67f09SDavid van Moolenbroek 	dns_rdataset_makequestion(rdataset, getzoneclass(), dns_rdatatype_soa);
3009*00b67f09SDavid van Moolenbroek 
3010*00b67f09SDavid van Moolenbroek 	if (userzone != NULL) {
3011*00b67f09SDavid van Moolenbroek 		dns_name_init(name, NULL);
3012*00b67f09SDavid van Moolenbroek 		dns_name_clone(userzone, name);
3013*00b67f09SDavid van Moolenbroek 	} else {
3014*00b67f09SDavid van Moolenbroek 		dns_rdataset_t *tmprdataset;
3015*00b67f09SDavid van Moolenbroek 		result = dns_message_firstname(updatemsg, section);
3016*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_NOMORE) {
3017*00b67f09SDavid van Moolenbroek 			section = DNS_SECTION_PREREQUISITE;
3018*00b67f09SDavid van Moolenbroek 			result = dns_message_firstname(updatemsg, section);
3019*00b67f09SDavid van Moolenbroek 		}
3020*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS) {
3021*00b67f09SDavid van Moolenbroek 			dns_message_puttempname(soaquery, &name);
3022*00b67f09SDavid van Moolenbroek 			dns_rdataset_disassociate(rdataset);
3023*00b67f09SDavid van Moolenbroek 			dns_message_puttemprdataset(soaquery, &rdataset);
3024*00b67f09SDavid van Moolenbroek 			dns_message_destroy(&soaquery);
3025*00b67f09SDavid van Moolenbroek 			done_update();
3026*00b67f09SDavid van Moolenbroek 			return;
3027*00b67f09SDavid van Moolenbroek 		}
3028*00b67f09SDavid van Moolenbroek 		firstname = NULL;
3029*00b67f09SDavid van Moolenbroek 		dns_message_currentname(updatemsg, section, &firstname);
3030*00b67f09SDavid van Moolenbroek 		dns_name_init(name, NULL);
3031*00b67f09SDavid van Moolenbroek 		dns_name_clone(firstname, name);
3032*00b67f09SDavid van Moolenbroek 		/*
3033*00b67f09SDavid van Moolenbroek 		 * Looks to see if the first name references a DS record
3034*00b67f09SDavid van Moolenbroek 		 * and if that name is not the root remove a label as DS
3035*00b67f09SDavid van Moolenbroek 		 * records live in the parent zone so we need to start our
3036*00b67f09SDavid van Moolenbroek 		 * search one label up.
3037*00b67f09SDavid van Moolenbroek 		 */
3038*00b67f09SDavid van Moolenbroek 		tmprdataset = ISC_LIST_HEAD(firstname->list);
3039*00b67f09SDavid van Moolenbroek 		if (section == DNS_SECTION_UPDATE &&
3040*00b67f09SDavid van Moolenbroek 		    !dns_name_equal(firstname, dns_rootname) &&
3041*00b67f09SDavid van Moolenbroek 		    tmprdataset->type == dns_rdatatype_ds) {
3042*00b67f09SDavid van Moolenbroek 		    unsigned int labels = dns_name_countlabels(name);
3043*00b67f09SDavid van Moolenbroek 		    dns_name_getlabelsequence(name, 1, labels - 1, name);
3044*00b67f09SDavid van Moolenbroek 		}
3045*00b67f09SDavid van Moolenbroek 	}
3046*00b67f09SDavid van Moolenbroek 
3047*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(name->list);
3048*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(name->list, rdataset, link);
3049*00b67f09SDavid van Moolenbroek 	dns_message_addname(soaquery, name, DNS_SECTION_QUESTION);
3050*00b67f09SDavid van Moolenbroek 
3051*00b67f09SDavid van Moolenbroek 	ns_inuse = 0;
3052*00b67f09SDavid van Moolenbroek 	sendrequest(&servers[ns_inuse], soaquery, &request);
3053*00b67f09SDavid van Moolenbroek }
3054*00b67f09SDavid van Moolenbroek 
3055*00b67f09SDavid van Moolenbroek static void
cleanup(void)3056*00b67f09SDavid van Moolenbroek cleanup(void) {
3057*00b67f09SDavid van Moolenbroek 	ddebug("cleanup()");
3058*00b67f09SDavid van Moolenbroek 
3059*00b67f09SDavid van Moolenbroek 	if (answer != NULL)
3060*00b67f09SDavid van Moolenbroek 		dns_message_destroy(&answer);
3061*00b67f09SDavid van Moolenbroek 
3062*00b67f09SDavid van Moolenbroek #ifdef GSSAPI
3063*00b67f09SDavid van Moolenbroek 	if (tsigkey != NULL) {
3064*00b67f09SDavid van Moolenbroek 		ddebug("detach tsigkey x%p", tsigkey);
3065*00b67f09SDavid van Moolenbroek 		dns_tsigkey_detach(&tsigkey);
3066*00b67f09SDavid van Moolenbroek 	}
3067*00b67f09SDavid van Moolenbroek 	if (gssring != NULL) {
3068*00b67f09SDavid van Moolenbroek 		ddebug("Detaching GSS-TSIG keyring");
3069*00b67f09SDavid van Moolenbroek 		dns_tsigkeyring_detach(&gssring);
3070*00b67f09SDavid van Moolenbroek 	}
3071*00b67f09SDavid van Moolenbroek 	if (kserver != NULL) {
3072*00b67f09SDavid van Moolenbroek 		isc_mem_put(gmctx, kserver, sizeof(isc_sockaddr_t));
3073*00b67f09SDavid van Moolenbroek 		kserver = NULL;
3074*00b67f09SDavid van Moolenbroek 	}
3075*00b67f09SDavid van Moolenbroek 	if (realm != NULL) {
3076*00b67f09SDavid van Moolenbroek 		isc_mem_free(gmctx, realm);
3077*00b67f09SDavid van Moolenbroek 		realm = NULL;
3078*00b67f09SDavid van Moolenbroek 	}
3079*00b67f09SDavid van Moolenbroek #endif
3080*00b67f09SDavid van Moolenbroek 
3081*00b67f09SDavid van Moolenbroek 	if (sig0key != NULL)
3082*00b67f09SDavid van Moolenbroek 		dst_key_free(&sig0key);
3083*00b67f09SDavid van Moolenbroek 
3084*00b67f09SDavid van Moolenbroek 	ddebug("Shutting down task manager");
3085*00b67f09SDavid van Moolenbroek 	isc_taskmgr_destroy(&taskmgr);
3086*00b67f09SDavid van Moolenbroek 
3087*00b67f09SDavid van Moolenbroek 	ddebug("Destroying event");
3088*00b67f09SDavid van Moolenbroek 	isc_event_free(&global_event);
3089*00b67f09SDavid van Moolenbroek 
3090*00b67f09SDavid van Moolenbroek 	ddebug("Shutting down socket manager");
3091*00b67f09SDavid van Moolenbroek 	isc_socketmgr_destroy(&socketmgr);
3092*00b67f09SDavid van Moolenbroek 
3093*00b67f09SDavid van Moolenbroek 	ddebug("Shutting down timer manager");
3094*00b67f09SDavid van Moolenbroek 	isc_timermgr_destroy(&timermgr);
3095*00b67f09SDavid van Moolenbroek 
3096*00b67f09SDavid van Moolenbroek 	ddebug("Destroying hash context");
3097*00b67f09SDavid van Moolenbroek 	isc_hash_destroy();
3098*00b67f09SDavid van Moolenbroek 
3099*00b67f09SDavid van Moolenbroek 	ddebug("Destroying name state");
3100*00b67f09SDavid van Moolenbroek 	dns_name_destroy();
3101*00b67f09SDavid van Moolenbroek 
3102*00b67f09SDavid van Moolenbroek 	ddebug("Removing log context");
3103*00b67f09SDavid van Moolenbroek 	isc_log_destroy(&glctx);
3104*00b67f09SDavid van Moolenbroek 
3105*00b67f09SDavid van Moolenbroek 	ddebug("Destroying memory context");
3106*00b67f09SDavid van Moolenbroek 	if (memdebugging)
3107*00b67f09SDavid van Moolenbroek 		isc_mem_stats(gmctx, stderr);
3108*00b67f09SDavid van Moolenbroek 	isc_mem_destroy(&gmctx);
3109*00b67f09SDavid van Moolenbroek }
3110*00b67f09SDavid van Moolenbroek 
3111*00b67f09SDavid van Moolenbroek static void
getinput(isc_task_t * task,isc_event_t * event)3112*00b67f09SDavid van Moolenbroek getinput(isc_task_t *task, isc_event_t *event) {
3113*00b67f09SDavid van Moolenbroek 	isc_boolean_t more;
3114*00b67f09SDavid van Moolenbroek 
3115*00b67f09SDavid van Moolenbroek 	UNUSED(task);
3116*00b67f09SDavid van Moolenbroek 
3117*00b67f09SDavid van Moolenbroek 	if (shuttingdown) {
3118*00b67f09SDavid van Moolenbroek 		maybeshutdown();
3119*00b67f09SDavid van Moolenbroek 		return;
3120*00b67f09SDavid van Moolenbroek 	}
3121*00b67f09SDavid van Moolenbroek 
3122*00b67f09SDavid van Moolenbroek 	if (global_event == NULL)
3123*00b67f09SDavid van Moolenbroek 		global_event = event;
3124*00b67f09SDavid van Moolenbroek 
3125*00b67f09SDavid van Moolenbroek 	reset_system();
3126*00b67f09SDavid van Moolenbroek 	more = user_interaction();
3127*00b67f09SDavid van Moolenbroek 	if (!more) {
3128*00b67f09SDavid van Moolenbroek 		isc_app_shutdown();
3129*00b67f09SDavid van Moolenbroek 		return;
3130*00b67f09SDavid van Moolenbroek 	}
3131*00b67f09SDavid van Moolenbroek 	start_update();
3132*00b67f09SDavid van Moolenbroek 	return;
3133*00b67f09SDavid van Moolenbroek }
3134*00b67f09SDavid van Moolenbroek 
3135*00b67f09SDavid van Moolenbroek int
main(int argc,char ** argv)3136*00b67f09SDavid van Moolenbroek main(int argc, char **argv) {
3137*00b67f09SDavid van Moolenbroek 	isc_result_t result;
3138*00b67f09SDavid van Moolenbroek 	style = &dns_master_style_debug;
3139*00b67f09SDavid van Moolenbroek 
3140*00b67f09SDavid van Moolenbroek 	input = stdin;
3141*00b67f09SDavid van Moolenbroek 
3142*00b67f09SDavid van Moolenbroek 	interactive = ISC_TF(isatty(0));
3143*00b67f09SDavid van Moolenbroek 
3144*00b67f09SDavid van Moolenbroek 	isc_app_start();
3145*00b67f09SDavid van Moolenbroek 
3146*00b67f09SDavid van Moolenbroek 	pre_parse_args(argc, argv);
3147*00b67f09SDavid van Moolenbroek 
3148*00b67f09SDavid van Moolenbroek 	result = isc_mem_create(0, 0, &gmctx);
3149*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_mem_create");
3150*00b67f09SDavid van Moolenbroek 
3151*00b67f09SDavid van Moolenbroek 	parse_args(argc, argv, gmctx, &entropy);
3152*00b67f09SDavid van Moolenbroek 
3153*00b67f09SDavid van Moolenbroek 	setup_system();
3154*00b67f09SDavid van Moolenbroek 
3155*00b67f09SDavid van Moolenbroek 	result = isc_app_onrun(gmctx, global_task, getinput, NULL);
3156*00b67f09SDavid van Moolenbroek 	check_result(result, "isc_app_onrun");
3157*00b67f09SDavid van Moolenbroek 
3158*00b67f09SDavid van Moolenbroek 	(void)isc_app_run();
3159*00b67f09SDavid van Moolenbroek 
3160*00b67f09SDavid van Moolenbroek 	cleanup();
3161*00b67f09SDavid van Moolenbroek 
3162*00b67f09SDavid van Moolenbroek 	isc_app_finish();
3163*00b67f09SDavid van Moolenbroek 
3164*00b67f09SDavid van Moolenbroek 	if (seenerror)
3165*00b67f09SDavid van Moolenbroek 		return (2);
3166*00b67f09SDavid van Moolenbroek 	else
3167*00b67f09SDavid van Moolenbroek 		return (0);
3168*00b67f09SDavid van Moolenbroek }
3169