named-checkconf.c revision 186462
1135446Strhodes/*
2186462Sdougb * Copyright (C) 2004-2007  Internet Systems Consortium, Inc. ("ISC")
3135446Strhodes * Copyright (C) 1999-2002  Internet Software Consortium.
4135446Strhodes *
5186462Sdougb * Permission to use, copy, modify, and/or distribute this software for any
6135446Strhodes * purpose with or without fee is hereby granted, provided that the above
7135446Strhodes * copyright notice and this permission notice appear in all copies.
8135446Strhodes *
9135446Strhodes * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
10135446Strhodes * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
11135446Strhodes * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
12135446Strhodes * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
13135446Strhodes * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
14135446Strhodes * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
15135446Strhodes * PERFORMANCE OF THIS SOFTWARE.
16135446Strhodes */
17135446Strhodes
18186462Sdougb/* $Id: named-checkconf.c,v 1.28.18.16 2007/11/26 23:46:18 tbox Exp $ */
19135446Strhodes
20170222Sdougb/*! \file */
21170222Sdougb
22135446Strhodes#include <config.h>
23135446Strhodes
24135446Strhodes#include <errno.h>
25135446Strhodes#include <stdlib.h>
26135446Strhodes#include <stdio.h>
27135446Strhodes
28135446Strhodes#include <isc/commandline.h>
29135446Strhodes#include <isc/dir.h>
30143731Sdougb#include <isc/entropy.h>
31143731Sdougb#include <isc/hash.h>
32135446Strhodes#include <isc/log.h>
33135446Strhodes#include <isc/mem.h>
34135446Strhodes#include <isc/result.h>
35135446Strhodes#include <isc/string.h>
36135446Strhodes#include <isc/util.h>
37135446Strhodes
38135446Strhodes#include <isccfg/namedconf.h>
39135446Strhodes
40135446Strhodes#include <bind9/check.h>
41135446Strhodes
42143731Sdougb#include <dns/fixedname.h>
43135446Strhodes#include <dns/log.h>
44170222Sdougb#include <dns/name.h>
45135446Strhodes#include <dns/result.h>
46170222Sdougb#include <dns/zone.h>
47135446Strhodes
48135446Strhodes#include "check-tool.h"
49135446Strhodes
50135446Strhodesisc_log_t *logc = NULL;
51135446Strhodes
52135446Strhodes#define CHECK(r)\
53135446Strhodes	do { \
54135446Strhodes		result = (r); \
55135446Strhodes		if (result != ISC_R_SUCCESS) \
56135446Strhodes			goto cleanup; \
57135446Strhodes	} while (0)
58135446Strhodes
59170222Sdougb/*% usage */
60135446Strhodesstatic void
61135446Strhodesusage(void) {
62135446Strhodes        fprintf(stderr, "usage: named-checkconf [-j] [-v] [-z] [-t directory] "
63135446Strhodes		"[named.conf]\n");
64135446Strhodes        exit(1);
65135446Strhodes}
66135446Strhodes
67170222Sdougb/*% directory callback */
68135446Strhodesstatic isc_result_t
69165071Sdougbdirectory_callback(const char *clausename, const cfg_obj_t *obj, void *arg) {
70135446Strhodes	isc_result_t result;
71165071Sdougb	const char *directory;
72135446Strhodes
73135446Strhodes	REQUIRE(strcasecmp("directory", clausename) == 0);
74135446Strhodes
75135446Strhodes	UNUSED(arg);
76135446Strhodes	UNUSED(clausename);
77135446Strhodes
78135446Strhodes	/*
79135446Strhodes	 * Change directory.
80135446Strhodes	 */
81135446Strhodes	directory = cfg_obj_asstring(obj);
82135446Strhodes	result = isc_dir_chdir(directory);
83135446Strhodes	if (result != ISC_R_SUCCESS) {
84135446Strhodes		cfg_obj_log(obj, logc, ISC_LOG_ERROR,
85135446Strhodes			    "change directory to '%s' failed: %s\n",
86135446Strhodes			    directory, isc_result_totext(result));
87135446Strhodes		return (result);
88135446Strhodes	}
89135446Strhodes
90135446Strhodes	return (ISC_R_SUCCESS);
91135446Strhodes}
92135446Strhodes
93170222Sdougbstatic isc_boolean_t
94170222Sdougbget_maps(const cfg_obj_t **maps, const char *name, const cfg_obj_t **obj) {
95170222Sdougb	int i;
96170222Sdougb	for (i = 0;; i++) {
97170222Sdougb		if (maps[i] == NULL)
98170222Sdougb			return (ISC_FALSE);
99170222Sdougb		if (cfg_map_get(maps[i], name, obj) == ISC_R_SUCCESS)
100170222Sdougb			return (ISC_TRUE);
101170222Sdougb	}
102170222Sdougb}
103170222Sdougb
104170222Sdougbstatic isc_boolean_t
105170222Sdougbget_checknames(const cfg_obj_t **maps, const cfg_obj_t **obj) {
106170222Sdougb	const cfg_listelt_t *element;
107170222Sdougb	const cfg_obj_t *checknames;
108170222Sdougb	const cfg_obj_t *type;
109170222Sdougb	const cfg_obj_t *value;
110170222Sdougb	isc_result_t result;
111170222Sdougb	int i;
112170222Sdougb
113170222Sdougb	for (i = 0;; i++) {
114170222Sdougb		if (maps[i] == NULL)
115170222Sdougb			return (ISC_FALSE);
116170222Sdougb		checknames = NULL;
117170222Sdougb		result = cfg_map_get(maps[i], "check-names", &checknames);
118170222Sdougb		if (result != ISC_R_SUCCESS)
119170222Sdougb			continue;
120170222Sdougb		if (checknames != NULL && !cfg_obj_islist(checknames)) {
121170222Sdougb			*obj = checknames;
122170222Sdougb			return (ISC_TRUE);
123170222Sdougb		}
124170222Sdougb		for (element = cfg_list_first(checknames);
125170222Sdougb		     element != NULL;
126170222Sdougb		     element = cfg_list_next(element)) {
127170222Sdougb			value = cfg_listelt_value(element);
128170222Sdougb			type = cfg_tuple_get(value, "type");
129170222Sdougb			if (strcasecmp(cfg_obj_asstring(type), "master") != 0)
130170222Sdougb				continue;
131170222Sdougb			*obj = cfg_tuple_get(value, "mode");
132170222Sdougb			return (ISC_TRUE);
133170222Sdougb		}
134170222Sdougb	}
135170222Sdougb}
136170222Sdougb
137135446Strhodesstatic isc_result_t
138170222Sdougbconfig_get(const cfg_obj_t **maps, const char *name, const cfg_obj_t **obj) {
139170222Sdougb	int i;
140170222Sdougb
141170222Sdougb	for (i = 0;; i++) {
142170222Sdougb		if (maps[i] == NULL)
143170222Sdougb			return (ISC_R_NOTFOUND);
144170222Sdougb		if (cfg_map_get(maps[i], name, obj) == ISC_R_SUCCESS)
145170222Sdougb			return (ISC_R_SUCCESS);
146170222Sdougb	}
147170222Sdougb}
148170222Sdougb
149170222Sdougb/*% configure the zone */
150170222Sdougbstatic isc_result_t
151165071Sdougbconfigure_zone(const char *vclass, const char *view,
152170222Sdougb	       const cfg_obj_t *zconfig, const cfg_obj_t *vconfig,
153170222Sdougb	       const cfg_obj_t *config, isc_mem_t *mctx)
154135446Strhodes{
155170222Sdougb	int i = 0;
156135446Strhodes	isc_result_t result;
157135446Strhodes	const char *zclass;
158135446Strhodes	const char *zname;
159135446Strhodes	const char *zfile;
160170222Sdougb	const cfg_obj_t *maps[4];
161165071Sdougb	const cfg_obj_t *zoptions = NULL;
162165071Sdougb	const cfg_obj_t *classobj = NULL;
163165071Sdougb	const cfg_obj_t *typeobj = NULL;
164165071Sdougb	const cfg_obj_t *fileobj = NULL;
165165071Sdougb	const cfg_obj_t *dbobj = NULL;
166170222Sdougb	const cfg_obj_t *obj = NULL;
167170222Sdougb	const cfg_obj_t *fmtobj = NULL;
168170222Sdougb	dns_masterformat_t masterformat;
169135446Strhodes
170170222Sdougb	zone_options = DNS_ZONEOPT_CHECKNS | DNS_ZONEOPT_MANYERRORS;
171170222Sdougb
172135446Strhodes	zname = cfg_obj_asstring(cfg_tuple_get(zconfig, "name"));
173135446Strhodes	classobj = cfg_tuple_get(zconfig, "class");
174135446Strhodes        if (!cfg_obj_isstring(classobj))
175135446Strhodes                zclass = vclass;
176135446Strhodes        else
177135446Strhodes		zclass = cfg_obj_asstring(classobj);
178170222Sdougb
179135446Strhodes	zoptions = cfg_tuple_get(zconfig, "options");
180170222Sdougb	maps[i++] = zoptions;
181170222Sdougb	if (vconfig != NULL)
182170222Sdougb		maps[i++] = cfg_tuple_get(vconfig, "options");
183170222Sdougb	if (config != NULL) {
184170222Sdougb		cfg_map_get(config, "options", &obj);
185170222Sdougb		if (obj != NULL)
186170222Sdougb			maps[i++] = obj;
187170222Sdougb	}
188170222Sdougb	maps[i++] = NULL;
189170222Sdougb
190135446Strhodes	cfg_map_get(zoptions, "type", &typeobj);
191135446Strhodes	if (typeobj == NULL)
192135446Strhodes		return (ISC_R_FAILURE);
193135446Strhodes	if (strcasecmp(cfg_obj_asstring(typeobj), "master") != 0)
194135446Strhodes		return (ISC_R_SUCCESS);
195135446Strhodes        cfg_map_get(zoptions, "database", &dbobj);
196135446Strhodes        if (dbobj != NULL)
197135446Strhodes                return (ISC_R_SUCCESS);
198135446Strhodes	cfg_map_get(zoptions, "file", &fileobj);
199135446Strhodes	if (fileobj == NULL)
200135446Strhodes		return (ISC_R_FAILURE);
201135446Strhodes	zfile = cfg_obj_asstring(fileobj);
202170222Sdougb
203170222Sdougb	obj = NULL;
204170222Sdougb	if (get_maps(maps, "check-mx", &obj)) {
205170222Sdougb		if (strcasecmp(cfg_obj_asstring(obj), "warn") == 0) {
206170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKMX;
207170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKMXFAIL;
208170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "fail") == 0) {
209170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKMX;
210170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKMXFAIL;
211170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "ignore") == 0) {
212170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKMX;
213170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKMXFAIL;
214170222Sdougb		} else
215170222Sdougb			INSIST(0);
216170222Sdougb	} else {
217170222Sdougb		zone_options |= DNS_ZONEOPT_CHECKMX;
218170222Sdougb		zone_options &= ~DNS_ZONEOPT_CHECKMXFAIL;
219170222Sdougb	}
220170222Sdougb
221170222Sdougb	obj = NULL;
222170222Sdougb	if (get_maps(maps, "check-integrity", &obj)) {
223170222Sdougb		if (cfg_obj_asboolean(obj))
224170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKINTEGRITY;
225170222Sdougb		else
226170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKINTEGRITY;
227186462Sdougb	} else
228186462Sdougb		zone_options |= DNS_ZONEOPT_CHECKINTEGRITY;
229170222Sdougb
230170222Sdougb	obj = NULL;
231170222Sdougb	if (get_maps(maps, "check-mx-cname", &obj)) {
232170222Sdougb		if (strcasecmp(cfg_obj_asstring(obj), "warn") == 0) {
233170222Sdougb			zone_options |= DNS_ZONEOPT_WARNMXCNAME;
234170222Sdougb			zone_options &= ~DNS_ZONEOPT_IGNOREMXCNAME;
235170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "fail") == 0) {
236170222Sdougb			zone_options &= ~DNS_ZONEOPT_WARNMXCNAME;
237170222Sdougb			zone_options &= ~DNS_ZONEOPT_IGNOREMXCNAME;
238170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "ignore") == 0) {
239170222Sdougb			zone_options |= DNS_ZONEOPT_WARNMXCNAME;
240170222Sdougb			zone_options |= DNS_ZONEOPT_IGNOREMXCNAME;
241170222Sdougb		} else
242170222Sdougb			INSIST(0);
243170222Sdougb	} else {
244170222Sdougb		zone_options |= DNS_ZONEOPT_WARNMXCNAME;
245170222Sdougb		zone_options &= ~DNS_ZONEOPT_IGNOREMXCNAME;
246170222Sdougb	}
247170222Sdougb
248170222Sdougb	obj = NULL;
249170222Sdougb	if (get_maps(maps, "check-srv-cname", &obj)) {
250170222Sdougb		if (strcasecmp(cfg_obj_asstring(obj), "warn") == 0) {
251170222Sdougb			zone_options |= DNS_ZONEOPT_WARNSRVCNAME;
252170222Sdougb			zone_options &= ~DNS_ZONEOPT_IGNORESRVCNAME;
253170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "fail") == 0) {
254170222Sdougb			zone_options &= ~DNS_ZONEOPT_WARNSRVCNAME;
255170222Sdougb			zone_options &= ~DNS_ZONEOPT_IGNORESRVCNAME;
256170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "ignore") == 0) {
257170222Sdougb			zone_options |= DNS_ZONEOPT_WARNSRVCNAME;
258170222Sdougb			zone_options |= DNS_ZONEOPT_IGNORESRVCNAME;
259170222Sdougb		} else
260170222Sdougb			INSIST(0);
261170222Sdougb	} else {
262170222Sdougb		zone_options |= DNS_ZONEOPT_WARNSRVCNAME;
263170222Sdougb		zone_options &= ~DNS_ZONEOPT_IGNORESRVCNAME;
264170222Sdougb	}
265170222Sdougb
266170222Sdougb	obj = NULL;
267170222Sdougb	if (get_maps(maps, "check-sibling", &obj)) {
268170222Sdougb		if (cfg_obj_asboolean(obj))
269170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKSIBLING;
270170222Sdougb		else
271170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKSIBLING;
272170222Sdougb	}
273170222Sdougb
274170222Sdougb	obj = NULL;
275170222Sdougb	if (get_checknames(maps, &obj)) {
276170222Sdougb		if (strcasecmp(cfg_obj_asstring(obj), "warn") == 0) {
277170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKNAMES;
278170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKNAMESFAIL;
279170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "fail") == 0) {
280170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKNAMES;
281170222Sdougb			zone_options |= DNS_ZONEOPT_CHECKNAMESFAIL;
282170222Sdougb		} else if (strcasecmp(cfg_obj_asstring(obj), "ignore") == 0) {
283170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKNAMES;
284170222Sdougb			zone_options &= ~DNS_ZONEOPT_CHECKNAMESFAIL;
285170222Sdougb		} else
286170222Sdougb			INSIST(0);
287170222Sdougb	} else {
288170222Sdougb               zone_options |= DNS_ZONEOPT_CHECKNAMES;
289170222Sdougb               zone_options |= DNS_ZONEOPT_CHECKNAMESFAIL;
290170222Sdougb	}
291170222Sdougb
292170222Sdougb	masterformat = dns_masterformat_text;
293170222Sdougb	fmtobj = NULL;
294170222Sdougb	result = config_get(maps, "masterfile-format", &fmtobj);
295170222Sdougb	if (result == ISC_R_SUCCESS) {
296170222Sdougb		const char *masterformatstr = cfg_obj_asstring(fmtobj);
297170222Sdougb		if (strcasecmp(masterformatstr, "text") == 0)
298170222Sdougb			masterformat = dns_masterformat_text;
299170222Sdougb		else if (strcasecmp(masterformatstr, "raw") == 0)
300170222Sdougb			masterformat = dns_masterformat_raw;
301170222Sdougb		else
302170222Sdougb			INSIST(0);
303170222Sdougb	}
304170222Sdougb
305170222Sdougb	result = load_zone(mctx, zname, zfile, masterformat, zclass, NULL);
306135446Strhodes	if (result != ISC_R_SUCCESS)
307135446Strhodes		fprintf(stderr, "%s/%s/%s: %s\n", view, zname, zclass,
308135446Strhodes			dns_result_totext(result));
309135446Strhodes	return(result);
310135446Strhodes}
311135446Strhodes
312170222Sdougb/*% configure a view */
313135446Strhodesstatic isc_result_t
314165071Sdougbconfigure_view(const char *vclass, const char *view, const cfg_obj_t *config,
315165071Sdougb	       const cfg_obj_t *vconfig, isc_mem_t *mctx)
316135446Strhodes{
317165071Sdougb	const cfg_listelt_t *element;
318165071Sdougb	const cfg_obj_t *voptions;
319165071Sdougb	const cfg_obj_t *zonelist;
320135446Strhodes	isc_result_t result = ISC_R_SUCCESS;
321135446Strhodes	isc_result_t tresult;
322135446Strhodes
323135446Strhodes	voptions = NULL;
324135446Strhodes	if (vconfig != NULL)
325135446Strhodes		voptions = cfg_tuple_get(vconfig, "options");
326135446Strhodes
327135446Strhodes	zonelist = NULL;
328135446Strhodes	if (voptions != NULL)
329135446Strhodes		(void)cfg_map_get(voptions, "zone", &zonelist);
330135446Strhodes	else
331135446Strhodes		(void)cfg_map_get(config, "zone", &zonelist);
332135446Strhodes
333135446Strhodes	for (element = cfg_list_first(zonelist);
334135446Strhodes	     element != NULL;
335135446Strhodes	     element = cfg_list_next(element))
336135446Strhodes	{
337165071Sdougb		const cfg_obj_t *zconfig = cfg_listelt_value(element);
338170222Sdougb		tresult = configure_zone(vclass, view, zconfig, vconfig,
339170222Sdougb					 config, mctx);
340135446Strhodes		if (tresult != ISC_R_SUCCESS)
341135446Strhodes			result = tresult;
342135446Strhodes	}
343135446Strhodes	return (result);
344135446Strhodes}
345135446Strhodes
346135446Strhodes
347170222Sdougb/*% load zones from the configuration */
348135446Strhodesstatic isc_result_t
349165071Sdougbload_zones_fromconfig(const cfg_obj_t *config, isc_mem_t *mctx) {
350165071Sdougb	const cfg_listelt_t *element;
351165071Sdougb	const cfg_obj_t *classobj;
352165071Sdougb	const cfg_obj_t *views;
353165071Sdougb	const cfg_obj_t *vconfig;
354135446Strhodes	const char *vclass;
355135446Strhodes	isc_result_t result = ISC_R_SUCCESS;
356135446Strhodes	isc_result_t tresult;
357135446Strhodes
358135446Strhodes	views = NULL;
359135446Strhodes
360135446Strhodes	(void)cfg_map_get(config, "view", &views);
361135446Strhodes	for (element = cfg_list_first(views);
362135446Strhodes	     element != NULL;
363135446Strhodes	     element = cfg_list_next(element))
364135446Strhodes	{
365135446Strhodes		const char *vname;
366135446Strhodes
367135446Strhodes		vclass = "IN";
368135446Strhodes		vconfig = cfg_listelt_value(element);
369135446Strhodes		if (vconfig != NULL) {
370135446Strhodes			classobj = cfg_tuple_get(vconfig, "class");
371135446Strhodes			if (cfg_obj_isstring(classobj))
372135446Strhodes				vclass = cfg_obj_asstring(classobj);
373135446Strhodes		}
374135446Strhodes		vname = cfg_obj_asstring(cfg_tuple_get(vconfig, "name"));
375135446Strhodes		tresult = configure_view(vclass, vname, config, vconfig, mctx);
376135446Strhodes		if (tresult != ISC_R_SUCCESS)
377135446Strhodes			result = tresult;
378135446Strhodes	}
379135446Strhodes
380135446Strhodes	if (views == NULL) {
381135446Strhodes		tresult = configure_view("IN", "_default", config, NULL, mctx);
382135446Strhodes		if (tresult != ISC_R_SUCCESS)
383135446Strhodes			result = tresult;
384135446Strhodes	}
385135446Strhodes	return (result);
386135446Strhodes}
387135446Strhodes
388170222Sdougb/*% The main processing routine */
389135446Strhodesint
390135446Strhodesmain(int argc, char **argv) {
391135446Strhodes	int c;
392135446Strhodes	cfg_parser_t *parser = NULL;
393135446Strhodes	cfg_obj_t *config = NULL;
394135446Strhodes	const char *conffile = NULL;
395135446Strhodes	isc_mem_t *mctx = NULL;
396135446Strhodes	isc_result_t result;
397135446Strhodes	int exit_status = 0;
398143731Sdougb	isc_entropy_t *ectx = NULL;
399135446Strhodes	isc_boolean_t load_zones = ISC_FALSE;
400135446Strhodes
401135446Strhodes	while ((c = isc_commandline_parse(argc, argv, "djt:vz")) != EOF) {
402135446Strhodes		switch (c) {
403135446Strhodes		case 'd':
404135446Strhodes			debug++;
405135446Strhodes			break;
406135446Strhodes
407135446Strhodes		case 'j':
408135446Strhodes			nomerge = ISC_FALSE;
409135446Strhodes			break;
410135446Strhodes
411135446Strhodes		case 't':
412135446Strhodes			result = isc_dir_chroot(isc_commandline_argument);
413135446Strhodes			if (result != ISC_R_SUCCESS) {
414135446Strhodes				fprintf(stderr, "isc_dir_chroot: %s\n",
415135446Strhodes					isc_result_totext(result));
416135446Strhodes				exit(1);
417135446Strhodes			}
418135446Strhodes			result = isc_dir_chdir("/");
419135446Strhodes			if (result != ISC_R_SUCCESS) {
420135446Strhodes				fprintf(stderr, "isc_dir_chdir: %s\n",
421135446Strhodes					isc_result_totext(result));
422135446Strhodes				exit(1);
423135446Strhodes			}
424135446Strhodes			break;
425135446Strhodes
426135446Strhodes		case 'v':
427135446Strhodes			printf(VERSION "\n");
428135446Strhodes			exit(0);
429135446Strhodes
430135446Strhodes		case 'z':
431135446Strhodes			load_zones = ISC_TRUE;
432170222Sdougb			docheckmx = ISC_FALSE;
433170222Sdougb			docheckns = ISC_FALSE;
434170222Sdougb			dochecksrv = ISC_FALSE;
435135446Strhodes			break;
436135446Strhodes
437135446Strhodes		default:
438135446Strhodes			usage();
439135446Strhodes		}
440135446Strhodes	}
441135446Strhodes
442135446Strhodes	if (argv[isc_commandline_index] != NULL)
443135446Strhodes		conffile = argv[isc_commandline_index];
444135446Strhodes	if (conffile == NULL || conffile[0] == '\0')
445135446Strhodes		conffile = NAMED_CONFFILE;
446135446Strhodes
447135446Strhodes	RUNTIME_CHECK(isc_mem_create(0, 0, &mctx) == ISC_R_SUCCESS);
448135446Strhodes
449135446Strhodes	RUNTIME_CHECK(setup_logging(mctx, &logc) == ISC_R_SUCCESS);
450135446Strhodes
451143731Sdougb	RUNTIME_CHECK(isc_entropy_create(mctx, &ectx) == ISC_R_SUCCESS);
452143731Sdougb	RUNTIME_CHECK(isc_hash_create(mctx, ectx, DNS_NAME_MAXWIRE)
453143731Sdougb		      == ISC_R_SUCCESS);
454143731Sdougb
455135446Strhodes	dns_result_register();
456135446Strhodes
457135446Strhodes	RUNTIME_CHECK(cfg_parser_create(mctx, logc, &parser) == ISC_R_SUCCESS);
458135446Strhodes
459135446Strhodes	cfg_parser_setcallback(parser, directory_callback, NULL);
460135446Strhodes
461135446Strhodes	if (cfg_parse_file(parser, conffile, &cfg_type_namedconf, &config) !=
462135446Strhodes	    ISC_R_SUCCESS)
463135446Strhodes		exit(1);
464135446Strhodes
465135446Strhodes	result = bind9_check_namedconf(config, logc, mctx);
466135446Strhodes	if (result != ISC_R_SUCCESS)
467135446Strhodes		exit_status = 1;
468135446Strhodes
469135446Strhodes	if (result == ISC_R_SUCCESS && load_zones) {
470135446Strhodes		result = load_zones_fromconfig(config, mctx);
471135446Strhodes		if (result != ISC_R_SUCCESS)
472135446Strhodes			exit_status = 1;
473135446Strhodes	}
474135446Strhodes
475135446Strhodes	cfg_obj_destroy(parser, &config);
476135446Strhodes
477135446Strhodes	cfg_parser_destroy(&parser);
478135446Strhodes
479170222Sdougb	dns_name_destroy();
480170222Sdougb
481135446Strhodes	isc_log_destroy(&logc);
482135446Strhodes
483143731Sdougb	isc_hash_destroy();
484143731Sdougb	isc_entropy_detach(&ectx);
485143731Sdougb
486135446Strhodes	isc_mem_destroy(&mctx);
487135446Strhodes
488135446Strhodes	return (exit_status);
489135446Strhodes}
490