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, ®ion);
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, ®ion);
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, ®ion);
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, ®ion);
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, ®ion);
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, ®ion);
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