2 * Copyright (C) 2004-2014 Internet Systems Consortium, Inc. ("ISC")
3 * Copyright (C) 1999-2003 Internet Software Consortium.
5 * Permission to use, copy, modify, and/or distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
10 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
11 * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
12 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
13 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
14 * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
15 * PERFORMANCE OF THIS SOFTWARE.
26 #include <isc/print.h>
28 #include <isc/stats.h>
29 #include <isc/string.h> /* Required for HP/UX (and others?) */
33 #include <dns/acache.h>
36 #include <dns/cache.h>
38 #include <dns/dispatch.h>
41 #include <dns/dns64.h>
43 #include <dns/dnssec.h>
44 #include <dns/events.h>
45 #include <dns/forward.h>
46 #include <dns/keytable.h>
47 #include <dns/keyvalues.h>
48 #include <dns/master.h>
49 #include <dns/masterdump.h>
50 #include <dns/order.h>
54 #include <dns/rdataset.h>
55 #include <dns/request.h>
56 #include <dns/resolver.h>
57 #include <dns/result.h>
59 #include <dns/stats.h>
64 #define RESSHUTDOWN(v) (((v)->attributes & DNS_VIEWATTR_RESSHUTDOWN) != 0)
65 #define ADBSHUTDOWN(v) (((v)->attributes & DNS_VIEWATTR_ADBSHUTDOWN) != 0)
66 #define REQSHUTDOWN(v) (((v)->attributes & DNS_VIEWATTR_REQSHUTDOWN) != 0)
68 #define DNS_VIEW_DELONLYHASH 111
70 static void resolver_shutdown(isc_task_t *task, isc_event_t *event);
71 static void adb_shutdown(isc_task_t *task, isc_event_t *event);
72 static void req_shutdown(isc_task_t *task, isc_event_t *event);
75 dns_view_create(isc_mem_t *mctx, dns_rdataclass_t rdclass,
76 const char *name, dns_view_t **viewp)
85 REQUIRE(name != NULL);
86 REQUIRE(viewp != NULL && *viewp == NULL);
88 view = isc_mem_get(mctx, sizeof(*view));
90 return (ISC_R_NOMEMORY);
93 isc_mem_attach(mctx, &view->mctx);
94 view->name = isc_mem_strdup(mctx, name);
95 if (view->name == NULL) {
96 result = ISC_R_NOMEMORY;
99 result = isc_mutex_init(&view->lock);
100 if (result != ISC_R_SUCCESS)
103 view->zonetable = NULL;
105 result = dns_zt_create(mctx, rdclass, &view->zonetable);
106 if (result != ISC_R_SUCCESS) {
107 UNEXPECTED_ERROR(__FILE__, __LINE__,
108 "dns_zt_create() failed: %s",
109 isc_result_totext(result));
110 result = ISC_R_UNEXPECTED;
114 view->secroots_priv = NULL;
115 view->fwdtable = NULL;
116 result = dns_fwdtable_create(mctx, &view->fwdtable);
117 if (result != ISC_R_SUCCESS) {
118 UNEXPECTED_ERROR(__FILE__, __LINE__,
119 "dns_fwdtable_create() failed: %s",
120 isc_result_totext(result));
121 result = ISC_R_UNEXPECTED;
127 view->cachedb = NULL;
128 view->dlzdatabase = NULL;
130 view->resolver = NULL;
132 view->requestmgr = NULL;
133 view->rdclass = rdclass;
134 view->frozen = ISC_FALSE;
136 result = isc_refcount_init(&view->references, 1);
137 if (result != ISC_R_SUCCESS)
138 goto cleanup_fwdtable;
140 view->attributes = (DNS_VIEWATTR_RESSHUTDOWN|DNS_VIEWATTR_ADBSHUTDOWN|
141 DNS_VIEWATTR_REQSHUTDOWN);
142 view->statickeys = NULL;
143 view->dynamickeys = NULL;
144 view->matchclients = NULL;
145 view->matchdestinations = NULL;
146 view->matchrecursiveonly = ISC_FALSE;
147 result = dns_tsigkeyring_create(view->mctx, &view->dynamickeys);
148 if (result != ISC_R_SUCCESS)
149 goto cleanup_references;
152 view->delonly = NULL;
153 view->rootdelonly = ISC_FALSE;
154 view->rootexclude = NULL;
155 view->resstats = NULL;
156 view->resquerystats = NULL;
157 view->cacheshared = ISC_FALSE;
158 ISC_LIST_INIT(view->dns64);
162 * Initialize configuration data with default values.
164 view->recursion = ISC_TRUE;
165 view->auth_nxdomain = ISC_FALSE; /* Was true in BIND 8 */
166 view->additionalfromcache = ISC_TRUE;
167 view->additionalfromauth = ISC_TRUE;
168 view->enablednssec = ISC_TRUE;
169 view->enablevalidation = ISC_TRUE;
170 view->acceptexpired = ISC_FALSE;
171 view->minimalresponses = ISC_FALSE;
172 view->transfer_format = dns_one_answer;
173 view->cacheacl = NULL;
174 view->cacheonacl = NULL;
175 view->queryacl = NULL;
176 view->queryonacl = NULL;
177 view->recursionacl = NULL;
178 view->recursiononacl = NULL;
179 view->sortlist = NULL;
180 view->transferacl = NULL;
181 view->notifyacl = NULL;
182 view->updateacl = NULL;
183 view->upfwdacl = NULL;
184 view->denyansweracl = NULL;
185 view->nocasecompress = NULL;
186 view->answeracl_exclude = NULL;
187 view->denyanswernames = NULL;
188 view->answernames_exclude = NULL;
190 view->provideixfr = ISC_TRUE;
191 view->maxcachettl = 7 * 24 * 3600;
192 view->maxncachettl = 3 * 3600;
194 view->preferred_glue = 0;
195 view->flush = ISC_FALSE;
199 view->v4_aaaa = dns_v4_aaaa_ok;
200 view->v4_aaaa_acl = NULL;
201 ISC_LIST_INIT(view->rpz_zones);
202 view->rpz_recursive_only = ISC_TRUE;
203 view->rpz_break_dnssec = ISC_FALSE;
204 dns_fixedname_init(&view->dlv_fixed);
205 view->managed_keys = NULL;
206 view->redirect = NULL;
208 view->new_zone_file = NULL;
209 view->new_zone_config = NULL;
210 view->cfg_destroy = NULL;
212 result = dns_order_create(view->mctx, &view->order);
213 if (result != ISC_R_SUCCESS)
214 goto cleanup_dynkeys;
217 result = dns_peerlist_new(view->mctx, &view->peers);
218 if (result != ISC_R_SUCCESS)
221 result = dns_aclenv_init(view->mctx, &view->aclenv);
222 if (result != ISC_R_SUCCESS)
223 goto cleanup_peerlist;
225 ISC_LINK_INIT(view, link);
226 ISC_EVENT_INIT(&view->resevent, sizeof(view->resevent), 0, NULL,
227 DNS_EVENT_VIEWRESSHUTDOWN, resolver_shutdown,
228 view, NULL, NULL, NULL);
229 ISC_EVENT_INIT(&view->adbevent, sizeof(view->adbevent), 0, NULL,
230 DNS_EVENT_VIEWADBSHUTDOWN, adb_shutdown,
231 view, NULL, NULL, NULL);
232 ISC_EVENT_INIT(&view->reqevent, sizeof(view->reqevent), 0, NULL,
233 DNS_EVENT_VIEWREQSHUTDOWN, req_shutdown,
234 view, NULL, NULL, NULL);
235 view->viewlist = NULL;
236 view->magic = DNS_VIEW_MAGIC;
240 return (ISC_R_SUCCESS);
243 dns_peerlist_detach(&view->peers);
247 dns_order_detach(&view->order);
251 dns_tsigkeyring_detach(&view->dynamickeys);
254 isc_refcount_destroy(&view->references);
257 dns_fwdtable_destroy(&view->fwdtable);
261 dns_zt_detach(&view->zonetable);
265 DESTROYLOCK(&view->lock);
268 isc_mem_free(mctx, view->name);
271 isc_mem_putanddetach(&view->mctx, view, sizeof(*view));
277 destroy(dns_view_t *view) {
282 REQUIRE(!ISC_LINK_LINKED(view, link));
283 REQUIRE(isc_refcount_current(&view->references) == 0);
284 REQUIRE(view->weakrefs == 0);
285 REQUIRE(RESSHUTDOWN(view));
286 REQUIRE(ADBSHUTDOWN(view));
287 REQUIRE(REQSHUTDOWN(view));
290 if (view->order != NULL)
291 dns_order_detach(&view->order);
293 if (view->peers != NULL)
294 dns_peerlist_detach(&view->peers);
296 if (view->dynamickeys != NULL) {
303 n = snprintf(keyfile, sizeof(keyfile), "%s.tsigkeys",
305 if (n > 0 && (size_t)n < sizeof(keyfile)) {
306 result = isc_file_mktemplate(keyfile, template,
308 if (result == ISC_R_SUCCESS)
309 (void)isc_file_openuniqueprivate(template, &fp);
312 dns_tsigkeyring_detach(&view->dynamickeys);
314 result = dns_tsigkeyring_dumpanddetach(
315 &view->dynamickeys, fp);
316 if (result == ISC_R_SUCCESS) {
318 result = isc_file_rename(template,
320 if (result != ISC_R_SUCCESS)
321 (void)remove(template);
324 (void)remove(template);
328 if (view->statickeys != NULL)
329 dns_tsigkeyring_detach(&view->statickeys);
330 if (view->adb != NULL)
331 dns_adb_detach(&view->adb);
332 if (view->resolver != NULL)
333 dns_resolver_detach(&view->resolver);
335 if (view->acache != NULL) {
336 if (view->cachedb != NULL)
337 dns_acache_putdb(view->acache, view->cachedb);
338 dns_acache_detach(&view->acache);
340 dns_rpz_view_destroy(view);
342 dns_rrl_view_destroy(view);
344 INSIST(view->rrl == NULL);
347 INSIST(view->acache == NULL);
348 INSIST(ISC_LIST_EMPTY(view->rpz_zones));
349 INSIST(view->rrl == NULL);
351 if (view->requestmgr != NULL)
352 dns_requestmgr_detach(&view->requestmgr);
353 if (view->task != NULL)
354 isc_task_detach(&view->task);
355 if (view->hints != NULL)
356 dns_db_detach(&view->hints);
357 if (view->dlzdatabase != NULL)
358 dns_dlzdestroy(&view->dlzdatabase);
359 if (view->cachedb != NULL)
360 dns_db_detach(&view->cachedb);
361 if (view->cache != NULL)
362 dns_cache_detach(&view->cache);
363 if (view->nocasecompress != NULL)
364 dns_acl_detach(&view->nocasecompress);
365 if (view->matchclients != NULL)
366 dns_acl_detach(&view->matchclients);
367 if (view->matchdestinations != NULL)
368 dns_acl_detach(&view->matchdestinations);
369 if (view->cacheacl != NULL)
370 dns_acl_detach(&view->cacheacl);
371 if (view->cacheonacl != NULL)
372 dns_acl_detach(&view->cacheonacl);
373 if (view->queryacl != NULL)
374 dns_acl_detach(&view->queryacl);
375 if (view->queryonacl != NULL)
376 dns_acl_detach(&view->queryonacl);
377 if (view->recursionacl != NULL)
378 dns_acl_detach(&view->recursionacl);
379 if (view->recursiononacl != NULL)
380 dns_acl_detach(&view->recursiononacl);
381 if (view->sortlist != NULL)
382 dns_acl_detach(&view->sortlist);
383 if (view->transferacl != NULL)
384 dns_acl_detach(&view->transferacl);
385 if (view->notifyacl != NULL)
386 dns_acl_detach(&view->notifyacl);
387 if (view->updateacl != NULL)
388 dns_acl_detach(&view->updateacl);
389 if (view->upfwdacl != NULL)
390 dns_acl_detach(&view->upfwdacl);
391 if (view->denyansweracl != NULL)
392 dns_acl_detach(&view->denyansweracl);
393 if (view->v4_aaaa_acl != NULL)
394 dns_acl_detach(&view->v4_aaaa_acl);
395 if (view->answeracl_exclude != NULL)
396 dns_rbt_destroy(&view->answeracl_exclude);
397 if (view->denyanswernames != NULL)
398 dns_rbt_destroy(&view->denyanswernames);
399 if (view->answernames_exclude != NULL)
400 dns_rbt_destroy(&view->answernames_exclude);
401 if (view->delonly != NULL) {
405 for (i = 0; i < DNS_VIEW_DELONLYHASH; i++) {
406 name = ISC_LIST_HEAD(view->delonly[i]);
407 while (name != NULL) {
408 ISC_LIST_UNLINK(view->delonly[i], name, link);
409 dns_name_free(name, view->mctx);
410 isc_mem_put(view->mctx, name, sizeof(*name));
411 name = ISC_LIST_HEAD(view->delonly[i]);
414 isc_mem_put(view->mctx, view->delonly, sizeof(dns_namelist_t) *
415 DNS_VIEW_DELONLYHASH);
416 view->delonly = NULL;
418 if (view->rootexclude != NULL) {
422 for (i = 0; i < DNS_VIEW_DELONLYHASH; i++) {
423 name = ISC_LIST_HEAD(view->rootexclude[i]);
424 while (name != NULL) {
425 ISC_LIST_UNLINK(view->rootexclude[i],
427 dns_name_free(name, view->mctx);
428 isc_mem_put(view->mctx, name, sizeof(*name));
429 name = ISC_LIST_HEAD(view->rootexclude[i]);
432 isc_mem_put(view->mctx, view->rootexclude,
433 sizeof(dns_namelist_t) * DNS_VIEW_DELONLYHASH);
434 view->rootexclude = NULL;
436 if (view->resstats != NULL)
437 isc_stats_detach(&view->resstats);
438 if (view->resquerystats != NULL)
439 dns_stats_detach(&view->resquerystats);
440 if (view->secroots_priv != NULL)
441 dns_keytable_detach(&view->secroots_priv);
443 for (dns64 = ISC_LIST_HEAD(view->dns64);
445 dns64 = ISC_LIST_HEAD(view->dns64)) {
446 dns_dns64_unlink(&view->dns64, dns64);
447 dns_dns64_destroy(&dns64);
449 if (view->managed_keys != NULL)
450 dns_zone_detach(&view->managed_keys);
451 if (view->redirect != NULL)
452 dns_zone_detach(&view->redirect);
453 dns_view_setnewzones(view, ISC_FALSE, NULL, NULL);
455 dns_fwdtable_destroy(&view->fwdtable);
456 dns_aclenv_destroy(&view->aclenv);
457 DESTROYLOCK(&view->lock);
458 isc_refcount_destroy(&view->references);
459 isc_mem_free(view->mctx, view->name);
460 isc_mem_putanddetach(&view->mctx, view, sizeof(*view));
464 * Return true iff 'view' may be freed.
465 * The caller must be holding the view lock.
468 all_done(dns_view_t *view) {
470 if (isc_refcount_current(&view->references) == 0 &&
471 view->weakrefs == 0 &&
472 RESSHUTDOWN(view) && ADBSHUTDOWN(view) && REQSHUTDOWN(view))
479 dns_view_attach(dns_view_t *source, dns_view_t **targetp) {
481 REQUIRE(DNS_VIEW_VALID(source));
482 REQUIRE(targetp != NULL && *targetp == NULL);
484 isc_refcount_increment(&source->references, NULL);
490 view_flushanddetach(dns_view_t **viewp, isc_boolean_t flush) {
493 isc_boolean_t done = ISC_FALSE;
495 REQUIRE(viewp != NULL);
497 REQUIRE(DNS_VIEW_VALID(view));
500 view->flush = ISC_TRUE;
501 isc_refcount_decrement(&view->references, &refs);
504 dns_zone_t *mkzone = NULL, *rdzone = NULL;
508 if (!RESSHUTDOWN(view))
509 dns_resolver_shutdown(view->resolver);
510 if (!ADBSHUTDOWN(view))
511 dns_adb_shutdown(view->adb);
512 if (!REQSHUTDOWN(view))
513 dns_requestmgr_shutdown(view->requestmgr);
515 if (view->acache != NULL)
516 dns_acache_shutdown(view->acache);
518 dns_zt_flushanddetach(&view->zonetable);
520 dns_zt_detach(&view->zonetable);
521 if (view->managed_keys != NULL) {
522 mkzone = view->managed_keys;
523 view->managed_keys = NULL;
525 dns_zone_flush(mkzone);
527 if (view->redirect != NULL) {
528 rdzone = view->redirect;
529 view->redirect = NULL;
531 dns_zone_flush(rdzone);
534 done = all_done(view);
538 /* Need to detach zones outside view lock */
540 dns_zone_detach(&mkzone);
543 dns_zone_detach(&rdzone);
554 dns_view_flushanddetach(dns_view_t **viewp) {
555 view_flushanddetach(viewp, ISC_TRUE);
559 dns_view_detach(dns_view_t **viewp) {
560 view_flushanddetach(viewp, ISC_FALSE);
565 dialup(dns_zone_t *zone, void *dummy) {
567 dns_zone_dialup(zone);
568 return (ISC_R_SUCCESS);
572 dns_view_dialup(dns_view_t *view) {
573 REQUIRE(DNS_VIEW_VALID(view));
574 REQUIRE(view->zonetable != NULL);
576 (void)dns_zt_apply(view->zonetable, ISC_FALSE, dialup, NULL);
581 dns_view_weakattach(dns_view_t *source, dns_view_t **targetp) {
583 REQUIRE(DNS_VIEW_VALID(source));
584 REQUIRE(targetp != NULL && *targetp == NULL);
588 UNLOCK(&source->lock);
594 dns_view_weakdetach(dns_view_t **viewp) {
596 isc_boolean_t done = ISC_FALSE;
598 REQUIRE(viewp != NULL);
600 REQUIRE(DNS_VIEW_VALID(view));
604 INSIST(view->weakrefs > 0);
606 done = all_done(view);
617 resolver_shutdown(isc_task_t *task, isc_event_t *event) {
618 dns_view_t *view = event->ev_arg;
621 REQUIRE(event->ev_type == DNS_EVENT_VIEWRESSHUTDOWN);
622 REQUIRE(DNS_VIEW_VALID(view));
623 REQUIRE(view->task == task);
627 isc_event_free(&event);
631 view->attributes |= DNS_VIEWATTR_RESSHUTDOWN;
632 done = all_done(view);
641 adb_shutdown(isc_task_t *task, isc_event_t *event) {
642 dns_view_t *view = event->ev_arg;
645 REQUIRE(event->ev_type == DNS_EVENT_VIEWADBSHUTDOWN);
646 REQUIRE(DNS_VIEW_VALID(view));
647 REQUIRE(view->task == task);
651 isc_event_free(&event);
655 view->attributes |= DNS_VIEWATTR_ADBSHUTDOWN;
656 done = all_done(view);
665 req_shutdown(isc_task_t *task, isc_event_t *event) {
666 dns_view_t *view = event->ev_arg;
669 REQUIRE(event->ev_type == DNS_EVENT_VIEWREQSHUTDOWN);
670 REQUIRE(DNS_VIEW_VALID(view));
671 REQUIRE(view->task == task);
675 isc_event_free(&event);
679 view->attributes |= DNS_VIEWATTR_REQSHUTDOWN;
680 done = all_done(view);
689 dns_view_createresolver(dns_view_t *view,
690 isc_taskmgr_t *taskmgr,
691 unsigned int ntasks, unsigned int ndisp,
692 isc_socketmgr_t *socketmgr,
693 isc_timermgr_t *timermgr,
694 unsigned int options,
695 dns_dispatchmgr_t *dispatchmgr,
696 dns_dispatch_t *dispatchv4,
697 dns_dispatch_t *dispatchv6)
701 isc_mem_t *mctx = NULL;
703 REQUIRE(DNS_VIEW_VALID(view));
704 REQUIRE(!view->frozen);
705 REQUIRE(view->resolver == NULL);
707 result = isc_task_create(taskmgr, 0, &view->task);
708 if (result != ISC_R_SUCCESS)
710 isc_task_setname(view->task, "view", view);
712 result = dns_resolver_create(view, taskmgr, ntasks, ndisp, socketmgr,
713 timermgr, options, dispatchmgr,
714 dispatchv4, dispatchv6,
716 if (result != ISC_R_SUCCESS) {
717 isc_task_detach(&view->task);
720 event = &view->resevent;
721 dns_resolver_whenshutdown(view->resolver, view->task, &event);
722 view->attributes &= ~DNS_VIEWATTR_RESSHUTDOWN;
724 result = isc_mem_create(0, 0, &mctx);
725 if (result != ISC_R_SUCCESS) {
726 dns_resolver_shutdown(view->resolver);
730 result = dns_adb_create(mctx, view, timermgr, taskmgr, &view->adb);
731 isc_mem_setname(mctx, "ADB", NULL);
732 isc_mem_detach(&mctx);
733 if (result != ISC_R_SUCCESS) {
734 dns_resolver_shutdown(view->resolver);
737 event = &view->adbevent;
738 dns_adb_whenshutdown(view->adb, view->task, &event);
739 view->attributes &= ~DNS_VIEWATTR_ADBSHUTDOWN;
741 result = dns_requestmgr_create(view->mctx, timermgr, socketmgr,
742 dns_resolver_taskmgr(view->resolver),
743 dns_resolver_dispatchmgr(view->resolver),
744 dispatchv4, dispatchv6,
746 if (result != ISC_R_SUCCESS) {
747 dns_adb_shutdown(view->adb);
748 dns_resolver_shutdown(view->resolver);
751 event = &view->reqevent;
752 dns_requestmgr_whenshutdown(view->requestmgr, view->task, &event);
753 view->attributes &= ~DNS_VIEWATTR_REQSHUTDOWN;
755 return (ISC_R_SUCCESS);
759 dns_view_setcache(dns_view_t *view, dns_cache_t *cache) {
760 dns_view_setcache2(view, cache, ISC_FALSE);
764 dns_view_setcache2(dns_view_t *view, dns_cache_t *cache, isc_boolean_t shared) {
765 REQUIRE(DNS_VIEW_VALID(view));
766 REQUIRE(!view->frozen);
768 view->cacheshared = shared;
769 if (view->cache != NULL) {
771 if (view->acache != NULL)
772 dns_acache_putdb(view->acache, view->cachedb);
774 dns_db_detach(&view->cachedb);
775 dns_cache_detach(&view->cache);
777 dns_cache_attach(cache, &view->cache);
778 dns_cache_attachdb(cache, &view->cachedb);
779 INSIST(DNS_DB_VALID(view->cachedb));
782 if (view->acache != NULL)
783 dns_acache_setdb(view->acache, view->cachedb);
788 dns_view_iscacheshared(dns_view_t *view) {
789 REQUIRE(DNS_VIEW_VALID(view));
791 return (view->cacheshared);
795 dns_view_sethints(dns_view_t *view, dns_db_t *hints) {
796 REQUIRE(DNS_VIEW_VALID(view));
797 REQUIRE(!view->frozen);
798 REQUIRE(view->hints == NULL);
799 REQUIRE(dns_db_iszone(hints));
801 dns_db_attach(hints, &view->hints);
805 dns_view_setkeyring(dns_view_t *view, dns_tsig_keyring_t *ring) {
806 REQUIRE(DNS_VIEW_VALID(view));
807 REQUIRE(ring != NULL);
808 if (view->statickeys != NULL)
809 dns_tsigkeyring_detach(&view->statickeys);
810 dns_tsigkeyring_attach(ring, &view->statickeys);
814 dns_view_setdynamickeyring(dns_view_t *view, dns_tsig_keyring_t *ring) {
815 REQUIRE(DNS_VIEW_VALID(view));
816 REQUIRE(ring != NULL);
817 if (view->dynamickeys != NULL)
818 dns_tsigkeyring_detach(&view->dynamickeys);
819 dns_tsigkeyring_attach(ring, &view->dynamickeys);
823 dns_view_getdynamickeyring(dns_view_t *view, dns_tsig_keyring_t **ringp) {
824 REQUIRE(DNS_VIEW_VALID(view));
825 REQUIRE(ringp != NULL && *ringp == NULL);
826 if (view->dynamickeys != NULL)
827 dns_tsigkeyring_attach(view->dynamickeys, ringp);
831 dns_view_restorekeyring(dns_view_t *view) {
836 REQUIRE(DNS_VIEW_VALID(view));
838 if (view->dynamickeys != NULL) {
839 n = snprintf(keyfile, sizeof(keyfile), "%s.tsigkeys",
841 if (n > 0 && (size_t)n < sizeof(keyfile)) {
842 fp = fopen(keyfile, "r");
844 dns_keyring_restore(view->dynamickeys, fp);
852 dns_view_setdstport(dns_view_t *view, in_port_t dstport) {
853 REQUIRE(DNS_VIEW_VALID(view));
854 view->dstport = dstport;
858 dns_view_freeze(dns_view_t *view) {
859 REQUIRE(DNS_VIEW_VALID(view));
860 REQUIRE(!view->frozen);
862 if (view->resolver != NULL) {
863 INSIST(view->cachedb != NULL);
864 dns_resolver_freeze(view->resolver);
866 view->frozen = ISC_TRUE;
871 dns_view_thaw(dns_view_t *view) {
872 REQUIRE(DNS_VIEW_VALID(view));
873 REQUIRE(view->frozen);
875 view->frozen = ISC_FALSE;
879 dns_view_addzone(dns_view_t *view, dns_zone_t *zone) {
882 REQUIRE(DNS_VIEW_VALID(view));
883 REQUIRE(!view->frozen);
884 REQUIRE(view->zonetable != NULL);
886 result = dns_zt_mount(view->zonetable, zone);
894 dns_view_findzone(dns_view_t *view, dns_name_t *name, dns_zone_t **zonep) {
897 REQUIRE(DNS_VIEW_VALID(view));
900 if (view->zonetable != NULL) {
901 result = dns_zt_find(view->zonetable, name, 0, NULL, zonep);
902 if (result == DNS_R_PARTIALMATCH) {
903 dns_zone_detach(zonep);
904 result = ISC_R_NOTFOUND;
907 result = ISC_R_NOTFOUND;
915 dns_view_find(dns_view_t *view, dns_name_t *name, dns_rdatatype_t type,
916 isc_stdtime_t now, unsigned int options, isc_boolean_t use_hints,
917 dns_db_t **dbp, dns_dbnode_t **nodep, dns_name_t *foundname,
918 dns_rdataset_t *rdataset, dns_rdataset_t *sigrdataset) {
919 return (dns_view_find2(view, name, type, now, options, use_hints,
920 ISC_FALSE, dbp, nodep, foundname, rdataset,
925 dns_view_find2(dns_view_t *view, dns_name_t *name, dns_rdatatype_t type,
926 isc_stdtime_t now, unsigned int options,
927 isc_boolean_t use_hints, isc_boolean_t use_static_stub,
928 dns_db_t **dbp, dns_dbnode_t **nodep, dns_name_t *foundname,
929 dns_rdataset_t *rdataset, dns_rdataset_t *sigrdataset)
933 dns_dbnode_t *node, *znode;
934 isc_boolean_t is_cache, is_staticstub_zone;
935 dns_rdataset_t zrdataset, zsigrdataset;
940 UNUSED(use_static_stub);
945 * Find an rdataset whose owner name is 'name', and whose type is
949 REQUIRE(DNS_VIEW_VALID(view));
950 REQUIRE(view->frozen);
951 REQUIRE(type != dns_rdatatype_rrsig);
952 REQUIRE(rdataset != NULL); /* XXXBEW - remove this */
953 REQUIRE(nodep == NULL || *nodep == NULL);
958 dns_rdataset_init(&zrdataset);
959 dns_rdataset_init(&zsigrdataset);
964 * Find a database to answer the query.
968 is_staticstub_zone = ISC_FALSE;
972 if (view->zonetable != NULL)
973 result = dns_zt_find(view->zonetable, name, 0, NULL, &zone);
975 result = ISC_R_NOTFOUND;
977 if (zone != NULL && dns_zone_gettype(zone) == dns_zone_staticstub &&
979 result = ISC_R_NOTFOUND;
981 if (result == ISC_R_SUCCESS || result == DNS_R_PARTIALMATCH) {
982 result = dns_zone_getdb(zone, &db);
983 if (result != ISC_R_SUCCESS && view->cachedb != NULL)
984 dns_db_attach(view->cachedb, &db);
985 else if (result != ISC_R_SUCCESS)
987 if (dns_zone_gettype(zone) == dns_zone_staticstub &&
988 dns_name_equal(name, dns_zone_getorigin(zone))) {
989 is_staticstub_zone = ISC_TRUE;
991 } else if (result == ISC_R_NOTFOUND && view->cachedb != NULL)
992 dns_db_attach(view->cachedb, &db);
994 result = ISC_R_NOTFOUND;
995 if (view->cachedb != NULL)
996 dns_db_attach(view->cachedb, &db);
1001 is_cache = dns_db_iscache(db);
1005 * Now look for an answer in the database.
1007 result = dns_db_find(db, name, NULL, type, options,
1008 now, &node, foundname, rdataset, sigrdataset);
1010 if (result == DNS_R_DELEGATION || result == ISC_R_NOTFOUND) {
1011 if (dns_rdataset_isassociated(rdataset))
1012 dns_rdataset_disassociate(rdataset);
1013 if (sigrdataset != NULL &&
1014 dns_rdataset_isassociated(sigrdataset))
1015 dns_rdataset_disassociate(sigrdataset);
1017 dns_db_detachnode(db, &node);
1020 if (view->cachedb != NULL && !is_staticstub_zone) {
1022 * Either the answer is in the cache, or we
1024 * Note that if the result comes from a
1025 * static-stub zone we stop the search here
1026 * (see the function description in view.h).
1028 is_cache = ISC_TRUE;
1029 dns_db_attach(view->cachedb, &db);
1034 * We don't have the data in the cache. If we've got
1035 * glue from the zone, use it.
1037 if (dns_rdataset_isassociated(&zrdataset)) {
1038 dns_rdataset_clone(&zrdataset, rdataset);
1039 if (sigrdataset != NULL &&
1040 dns_rdataset_isassociated(&zsigrdataset))
1041 dns_rdataset_clone(&zsigrdataset,
1043 result = DNS_R_GLUE;
1046 dns_db_attach(zdb, &db);
1047 dns_db_attachnode(db, znode, &node);
1052 * We don't know the answer.
1054 result = ISC_R_NOTFOUND;
1055 } else if (result == DNS_R_GLUE) {
1056 if (view->cachedb != NULL && !is_staticstub_zone) {
1058 * We found an answer, but the cache may be better.
1059 * Remember what we've got and go look in the cache.
1061 is_cache = ISC_TRUE;
1062 dns_rdataset_clone(rdataset, &zrdataset);
1063 dns_rdataset_disassociate(rdataset);
1064 if (sigrdataset != NULL &&
1065 dns_rdataset_isassociated(sigrdataset)) {
1066 dns_rdataset_clone(sigrdataset, &zsigrdataset);
1067 dns_rdataset_disassociate(sigrdataset);
1069 dns_db_attach(db, &zdb);
1070 dns_db_attachnode(zdb, node, &znode);
1071 dns_db_detachnode(db, &node);
1073 dns_db_attach(view->cachedb, &db);
1077 * Otherwise, the glue is the best answer.
1079 result = ISC_R_SUCCESS;
1083 if (result == ISC_R_NOTFOUND && use_hints && view->hints != NULL) {
1084 if (dns_rdataset_isassociated(rdataset))
1085 dns_rdataset_disassociate(rdataset);
1086 if (sigrdataset != NULL &&
1087 dns_rdataset_isassociated(sigrdataset))
1088 dns_rdataset_disassociate(sigrdataset);
1091 dns_db_detachnode(db, &node);
1094 result = dns_db_find(view->hints, name, NULL, type, options,
1095 now, &node, foundname,
1096 rdataset, sigrdataset);
1097 if (result == ISC_R_SUCCESS || result == DNS_R_GLUE) {
1099 * We just used a hint. Let the resolver know it
1100 * should consider priming.
1102 dns_resolver_prime(view->resolver);
1103 dns_db_attach(view->hints, &db);
1104 result = DNS_R_HINT;
1105 } else if (result == DNS_R_NXRRSET) {
1106 dns_db_attach(view->hints, &db);
1107 result = DNS_R_HINTNXRRSET;
1108 } else if (result == DNS_R_NXDOMAIN)
1109 result = ISC_R_NOTFOUND;
1112 * Cleanup if non-standard hints are used.
1114 if (db == NULL && node != NULL)
1115 dns_db_detachnode(view->hints, &node);
1120 if (dns_rdataset_isassociated(&zrdataset)) {
1121 dns_rdataset_disassociate(&zrdataset);
1122 if (dns_rdataset_isassociated(&zsigrdataset))
1123 dns_rdataset_disassociate(&zsigrdataset);
1128 dns_db_detachnode(zdb, &znode);
1129 dns_db_detach(&zdb);
1137 dns_db_detachnode(db, &node);
1144 INSIST(node == NULL);
1148 dns_zone_detach(&zone);
1155 dns_view_simplefind(dns_view_t *view, dns_name_t *name, dns_rdatatype_t type,
1156 isc_stdtime_t now, unsigned int options,
1157 isc_boolean_t use_hints,
1158 dns_rdataset_t *rdataset, dns_rdataset_t *sigrdataset)
1160 isc_result_t result;
1161 dns_fixedname_t foundname;
1163 dns_fixedname_init(&foundname);
1164 result = dns_view_find(view, name, type, now, options, use_hints,
1165 NULL, NULL, dns_fixedname_name(&foundname),
1166 rdataset, sigrdataset);
1167 if (result == DNS_R_NXDOMAIN) {
1169 * The rdataset and sigrdataset of the relevant NSEC record
1170 * may be returned, but the caller cannot use them because
1171 * foundname is not returned by this simplified API. We
1172 * disassociate them here to prevent any misuse by the caller.
1174 if (dns_rdataset_isassociated(rdataset))
1175 dns_rdataset_disassociate(rdataset);
1176 if (sigrdataset != NULL &&
1177 dns_rdataset_isassociated(sigrdataset))
1178 dns_rdataset_disassociate(sigrdataset);
1179 } else if (result != ISC_R_SUCCESS &&
1180 result != DNS_R_GLUE &&
1181 result != DNS_R_HINT &&
1182 result != DNS_R_NCACHENXDOMAIN &&
1183 result != DNS_R_NCACHENXRRSET &&
1184 result != DNS_R_NXRRSET &&
1185 result != DNS_R_HINTNXRRSET &&
1186 result != ISC_R_NOTFOUND) {
1187 if (dns_rdataset_isassociated(rdataset))
1188 dns_rdataset_disassociate(rdataset);
1189 if (sigrdataset != NULL &&
1190 dns_rdataset_isassociated(sigrdataset))
1191 dns_rdataset_disassociate(sigrdataset);
1192 result = ISC_R_NOTFOUND;
1199 dns_view_findzonecut(dns_view_t *view, dns_name_t *name, dns_name_t *fname,
1200 isc_stdtime_t now, unsigned int options,
1201 isc_boolean_t use_hints,
1202 dns_rdataset_t *rdataset, dns_rdataset_t *sigrdataset)
1204 return(dns_view_findzonecut2(view, name, fname, now, options,
1205 use_hints, ISC_TRUE,
1206 rdataset, sigrdataset));
1210 dns_view_findzonecut2(dns_view_t *view, dns_name_t *name, dns_name_t *fname,
1211 isc_stdtime_t now, unsigned int options,
1212 isc_boolean_t use_hints, isc_boolean_t use_cache,
1213 dns_rdataset_t *rdataset, dns_rdataset_t *sigrdataset)
1215 isc_result_t result;
1217 isc_boolean_t is_cache, use_zone, try_hints;
1220 dns_rdataset_t zrdataset, zsigrdataset;
1221 dns_fixedname_t zfixedname;
1223 unsigned int ztoptions = 0;
1230 REQUIRE(DNS_VIEW_VALID(view));
1231 REQUIRE(view->frozen);
1234 use_zone = ISC_FALSE;
1235 try_hints = ISC_FALSE;
1241 dns_fixedname_init(&zfixedname);
1242 dns_rdataset_init(&zrdataset);
1243 dns_rdataset_init(&zsigrdataset);
1246 * Find the right database.
1251 if (view->zonetable != NULL) {
1252 if ((options & DNS_DBFIND_NOEXACT) != 0)
1253 ztoptions |= DNS_ZTFIND_NOEXACT;
1254 result = dns_zt_find(view->zonetable, name, ztoptions,
1257 result = ISC_R_NOTFOUND;
1258 if (result == ISC_R_SUCCESS || result == DNS_R_PARTIALMATCH)
1259 result = dns_zone_getdb(zone, &db);
1260 UNLOCK(&view->lock);
1262 result = ISC_R_NOTFOUND;
1264 if (result == ISC_R_NOTFOUND) {
1266 * We're not directly authoritative for this query name, nor
1267 * is it a subdomain of any zone for which we're
1270 if (use_cache && view->cachedb != NULL) {
1272 * We have a cache; try it.
1274 dns_db_attach(view->cachedb, &db);
1277 * Maybe we have hints...
1279 try_hints = ISC_TRUE;
1282 } else if (result != ISC_R_SUCCESS) {
1284 * Something is broken.
1288 is_cache = dns_db_iscache(db);
1292 * Look for the zonecut.
1295 result = dns_db_find(db, name, NULL, dns_rdatatype_ns, options,
1296 now, NULL, fname, rdataset, sigrdataset);
1297 if (result == DNS_R_DELEGATION)
1298 result = ISC_R_SUCCESS;
1299 else if (result != ISC_R_SUCCESS)
1301 if (use_cache && view->cachedb != NULL && db != view->hints) {
1303 * We found an answer, but the cache may be better.
1305 zfname = dns_fixedname_name(&zfixedname);
1306 result = dns_name_copy(fname, zfname, NULL);
1307 if (result != ISC_R_SUCCESS)
1309 dns_rdataset_clone(rdataset, &zrdataset);
1310 dns_rdataset_disassociate(rdataset);
1311 if (sigrdataset != NULL &&
1312 dns_rdataset_isassociated(sigrdataset)) {
1313 dns_rdataset_clone(sigrdataset, &zsigrdataset);
1314 dns_rdataset_disassociate(sigrdataset);
1317 dns_db_attach(view->cachedb, &db);
1318 is_cache = ISC_TRUE;
1322 result = dns_db_findzonecut(db, name, options, now, NULL,
1323 fname, rdataset, sigrdataset);
1324 if (result == ISC_R_SUCCESS) {
1325 if (zfname != NULL &&
1326 (!dns_name_issubdomain(fname, zfname) ||
1327 (dns_zone_staticstub &&
1328 dns_name_equal(fname, zfname)))) {
1330 * We found a zonecut in the cache, but our
1331 * zone delegation is better.
1333 use_zone = ISC_TRUE;
1335 } else if (result == ISC_R_NOTFOUND) {
1336 if (zfname != NULL) {
1338 * We didn't find anything in the cache, but we
1339 * have a zone delegation, so use it.
1341 use_zone = ISC_TRUE;
1344 * Maybe we have hints...
1346 try_hints = ISC_TRUE;
1350 * Something bad happened.
1358 if (dns_rdataset_isassociated(rdataset)) {
1359 dns_rdataset_disassociate(rdataset);
1360 if (sigrdataset != NULL &&
1361 dns_rdataset_isassociated(sigrdataset))
1362 dns_rdataset_disassociate(sigrdataset);
1364 result = dns_name_copy(zfname, fname, NULL);
1365 if (result != ISC_R_SUCCESS)
1367 dns_rdataset_clone(&zrdataset, rdataset);
1368 if (sigrdataset != NULL &&
1369 dns_rdataset_isassociated(&zrdataset))
1370 dns_rdataset_clone(&zsigrdataset, sigrdataset);
1371 } else if (try_hints && use_hints && view->hints != NULL) {
1373 * We've found nothing so far, but we have hints.
1375 result = dns_db_find(view->hints, dns_rootname, NULL,
1376 dns_rdatatype_ns, 0, now, NULL, fname,
1378 if (result != ISC_R_SUCCESS) {
1380 * We can't even find the hints for the root
1383 if (dns_rdataset_isassociated(rdataset))
1384 dns_rdataset_disassociate(rdataset);
1385 result = ISC_R_NOTFOUND;
1390 if (dns_rdataset_isassociated(&zrdataset)) {
1391 dns_rdataset_disassociate(&zrdataset);
1392 if (dns_rdataset_isassociated(&zsigrdataset))
1393 dns_rdataset_disassociate(&zsigrdataset);
1399 dns_zone_detach(&zone);
1406 dns_viewlist_find(dns_viewlist_t *list, const char *name,
1407 dns_rdataclass_t rdclass, dns_view_t **viewp)
1411 REQUIRE(list != NULL);
1413 for (view = ISC_LIST_HEAD(*list);
1415 view = ISC_LIST_NEXT(view, link)) {
1416 if (strcmp(view->name, name) == 0 && view->rdclass == rdclass)
1420 return (ISC_R_NOTFOUND);
1422 dns_view_attach(view, viewp);
1424 return (ISC_R_SUCCESS);
1429 dns_viewlist_findzone(dns_viewlist_t *list, dns_name_t *name,
1430 isc_boolean_t allclasses, dns_rdataclass_t rdclass,
1434 isc_result_t result;
1435 dns_zone_t *zone1 = NULL, *zone2 = NULL;
1436 dns_zone_t **zp = NULL;;
1438 REQUIRE(list != NULL);
1439 REQUIRE(zonep != NULL && *zonep == NULL);
1441 for (view = ISC_LIST_HEAD(*list);
1443 view = ISC_LIST_NEXT(view, link)) {
1444 if (allclasses == ISC_FALSE && view->rdclass != rdclass)
1448 * If the zone is defined in more than one view,
1449 * treat it as not found.
1451 zp = (zone1 == NULL) ? &zone1 : &zone2;
1453 if (view->zonetable != NULL)
1454 result = dns_zt_find(view->zonetable, name, 0,
1457 result = ISC_R_NOTFOUND;
1458 UNLOCK(&view->lock);
1459 INSIST(result == ISC_R_SUCCESS ||
1460 result == ISC_R_NOTFOUND ||
1461 result == DNS_R_PARTIALMATCH);
1463 /* Treat a partial match as no match */
1464 if (result == DNS_R_PARTIALMATCH) {
1465 dns_zone_detach(zp);
1466 result = ISC_R_NOTFOUND;
1470 if (zone2 != NULL) {
1471 dns_zone_detach(&zone1);
1472 dns_zone_detach(&zone2);
1473 return (ISC_R_MULTIPLE);
1477 if (zone1 != NULL) {
1478 dns_zone_attach(zone1, zonep);
1479 dns_zone_detach(&zone1);
1480 return (ISC_R_SUCCESS);
1483 return (ISC_R_NOTFOUND);
1487 dns_view_load(dns_view_t *view, isc_boolean_t stop) {
1489 REQUIRE(DNS_VIEW_VALID(view));
1490 REQUIRE(view->zonetable != NULL);
1492 return (dns_zt_load(view->zonetable, stop));
1496 dns_view_loadnew(dns_view_t *view, isc_boolean_t stop) {
1498 REQUIRE(DNS_VIEW_VALID(view));
1499 REQUIRE(view->zonetable != NULL);
1501 return (dns_zt_loadnew(view->zonetable, stop));
1505 dns_view_asyncload(dns_view_t *view, dns_zt_allloaded_t callback, void *arg) {
1506 REQUIRE(DNS_VIEW_VALID(view));
1507 REQUIRE(view->zonetable != NULL);
1509 return (dns_zt_asyncload(view->zonetable, callback, arg));
1516 dns_view_gettsig(dns_view_t *view, dns_name_t *keyname, dns_tsigkey_t **keyp)
1518 isc_result_t result;
1519 REQUIRE(keyp != NULL && *keyp == NULL);
1521 result = dns_tsigkey_find(keyp, keyname, NULL,
1523 if (result == ISC_R_NOTFOUND)
1524 result = dns_tsigkey_find(keyp, keyname, NULL,
1530 dns_view_getpeertsig(dns_view_t *view, isc_netaddr_t *peeraddr,
1531 dns_tsigkey_t **keyp)
1533 isc_result_t result;
1534 dns_name_t *keyname = NULL;
1535 dns_peer_t *peer = NULL;
1537 result = dns_peerlist_peerbyaddr(view->peers, peeraddr, &peer);
1538 if (result != ISC_R_SUCCESS)
1541 result = dns_peer_getkey(peer, &keyname);
1542 if (result != ISC_R_SUCCESS)
1545 result = dns_view_gettsig(view, keyname, keyp);
1546 return ((result == ISC_R_NOTFOUND) ? ISC_R_FAILURE : result);
1550 dns_view_checksig(dns_view_t *view, isc_buffer_t *source, dns_message_t *msg) {
1551 REQUIRE(DNS_VIEW_VALID(view));
1552 REQUIRE(source != NULL);
1554 return (dns_tsig_verify(source, msg, view->statickeys,
1555 view->dynamickeys));
1560 dns_view_dumpdbtostream(dns_view_t *view, FILE *fp) {
1561 isc_result_t result;
1563 REQUIRE(DNS_VIEW_VALID(view));
1565 (void)fprintf(fp, ";\n; Cache dump of view '%s'\n;\n", view->name);
1566 result = dns_master_dumptostream(view->mctx, view->cachedb, NULL,
1567 &dns_master_style_cache, fp);
1568 if (result != ISC_R_SUCCESS)
1570 dns_adb_dump(view->adb, fp);
1571 dns_resolver_printbadcache(view->resolver, fp);
1572 return (ISC_R_SUCCESS);
1577 dns_view_flushcache(dns_view_t *view) {
1578 return (dns_view_flushcache2(view, ISC_FALSE));
1582 dns_view_flushcache2(dns_view_t *view, isc_boolean_t fixuponly) {
1583 isc_result_t result;
1585 REQUIRE(DNS_VIEW_VALID(view));
1587 if (view->cachedb == NULL)
1588 return (ISC_R_SUCCESS);
1590 result = dns_cache_flush(view->cache);
1591 if (result != ISC_R_SUCCESS)
1595 if (view->acache != NULL)
1596 dns_acache_putdb(view->acache, view->cachedb);
1598 dns_db_detach(&view->cachedb);
1599 dns_cache_attachdb(view->cache, &view->cachedb);
1601 if (view->acache != NULL)
1602 dns_acache_setdb(view->acache, view->cachedb);
1603 if (view->resolver != NULL)
1604 dns_resolver_flushbadcache(view->resolver, NULL);
1607 dns_adb_flush(view->adb);
1608 return (ISC_R_SUCCESS);
1612 dns_view_flushname(dns_view_t *view, dns_name_t *name) {
1613 return (dns_view_flushnode(view, name, ISC_FALSE));
1617 dns_view_flushnode(dns_view_t *view, dns_name_t *name, isc_boolean_t tree) {
1619 REQUIRE(DNS_VIEW_VALID(view));
1622 if (view->adb != NULL)
1623 dns_adb_flushname(view->adb, name);
1624 if (view->cache == NULL)
1625 return (ISC_R_SUCCESS);
1626 if (view->resolver != NULL)
1627 dns_resolver_flushbadcache(view->resolver, name);
1629 return (dns_cache_flushnode(view->cache, name, tree));
1633 dns_view_adddelegationonly(dns_view_t *view, dns_name_t *name) {
1634 isc_result_t result;
1638 REQUIRE(DNS_VIEW_VALID(view));
1640 if (view->delonly == NULL) {
1641 view->delonly = isc_mem_get(view->mctx,
1642 sizeof(dns_namelist_t) *
1643 DNS_VIEW_DELONLYHASH);
1644 if (view->delonly == NULL)
1645 return (ISC_R_NOMEMORY);
1646 for (hash = 0; hash < DNS_VIEW_DELONLYHASH; hash++)
1647 ISC_LIST_INIT(view->delonly[hash]);
1649 hash = dns_name_hash(name, ISC_FALSE) % DNS_VIEW_DELONLYHASH;
1650 new = ISC_LIST_HEAD(view->delonly[hash]);
1651 while (new != NULL && !dns_name_equal(new, name))
1652 new = ISC_LIST_NEXT(new, link);
1654 return (ISC_R_SUCCESS);
1655 new = isc_mem_get(view->mctx, sizeof(*new));
1657 return (ISC_R_NOMEMORY);
1658 dns_name_init(new, NULL);
1659 result = dns_name_dup(name, view->mctx, new);
1660 if (result == ISC_R_SUCCESS)
1661 ISC_LIST_APPEND(view->delonly[hash], new, link);
1663 isc_mem_put(view->mctx, new, sizeof(*new));
1668 dns_view_excludedelegationonly(dns_view_t *view, dns_name_t *name) {
1669 isc_result_t result;
1673 REQUIRE(DNS_VIEW_VALID(view));
1675 if (view->rootexclude == NULL) {
1676 view->rootexclude = isc_mem_get(view->mctx,
1677 sizeof(dns_namelist_t) *
1678 DNS_VIEW_DELONLYHASH);
1679 if (view->rootexclude == NULL)
1680 return (ISC_R_NOMEMORY);
1681 for (hash = 0; hash < DNS_VIEW_DELONLYHASH; hash++)
1682 ISC_LIST_INIT(view->rootexclude[hash]);
1684 hash = dns_name_hash(name, ISC_FALSE) % DNS_VIEW_DELONLYHASH;
1685 new = ISC_LIST_HEAD(view->rootexclude[hash]);
1686 while (new != NULL && !dns_name_equal(new, name))
1687 new = ISC_LIST_NEXT(new, link);
1689 return (ISC_R_SUCCESS);
1690 new = isc_mem_get(view->mctx, sizeof(*new));
1692 return (ISC_R_NOMEMORY);
1693 dns_name_init(new, NULL);
1694 result = dns_name_dup(name, view->mctx, new);
1695 if (result == ISC_R_SUCCESS)
1696 ISC_LIST_APPEND(view->rootexclude[hash], new, link);
1698 isc_mem_put(view->mctx, new, sizeof(*new));
1703 dns_view_isdelegationonly(dns_view_t *view, dns_name_t *name) {
1707 REQUIRE(DNS_VIEW_VALID(view));
1709 if (!view->rootdelonly && view->delonly == NULL)
1712 hash = dns_name_hash(name, ISC_FALSE) % DNS_VIEW_DELONLYHASH;
1713 if (view->rootdelonly && dns_name_countlabels(name) <= 2) {
1714 if (view->rootexclude == NULL)
1716 new = ISC_LIST_HEAD(view->rootexclude[hash]);
1717 while (new != NULL && !dns_name_equal(new, name))
1718 new = ISC_LIST_NEXT(new, link);
1723 if (view->delonly == NULL)
1726 new = ISC_LIST_HEAD(view->delonly[hash]);
1727 while (new != NULL && !dns_name_equal(new, name))
1728 new = ISC_LIST_NEXT(new, link);
1735 dns_view_setrootdelonly(dns_view_t *view, isc_boolean_t value) {
1736 REQUIRE(DNS_VIEW_VALID(view));
1737 view->rootdelonly = value;
1741 dns_view_getrootdelonly(dns_view_t *view) {
1742 REQUIRE(DNS_VIEW_VALID(view));
1743 return (view->rootdelonly);
1748 dns_view_freezezones(dns_view_t *view, isc_boolean_t value) {
1750 REQUIRE(DNS_VIEW_VALID(view));
1751 REQUIRE(view->zonetable != NULL);
1753 return (dns_zt_freezezones(view->zonetable, value));
1758 dns_view_setresstats(dns_view_t *view, isc_stats_t *stats) {
1760 REQUIRE(DNS_VIEW_VALID(view));
1761 REQUIRE(!view->frozen);
1762 REQUIRE(view->resstats == NULL);
1764 isc_stats_attach(stats, &view->resstats);
1768 dns_view_getresstats(dns_view_t *view, isc_stats_t **statsp) {
1769 REQUIRE(DNS_VIEW_VALID(view));
1770 REQUIRE(statsp != NULL && *statsp == NULL);
1772 if (view->resstats != NULL)
1773 isc_stats_attach(view->resstats, statsp);
1777 dns_view_setresquerystats(dns_view_t *view, dns_stats_t *stats) {
1778 REQUIRE(DNS_VIEW_VALID(view));
1779 REQUIRE(!view->frozen);
1780 REQUIRE(view->resquerystats == NULL);
1782 dns_stats_attach(stats, &view->resquerystats);
1786 dns_view_getresquerystats(dns_view_t *view, dns_stats_t **statsp) {
1787 REQUIRE(DNS_VIEW_VALID(view));
1788 REQUIRE(statsp != NULL && *statsp == NULL);
1790 if (view->resquerystats != NULL)
1791 dns_stats_attach(view->resquerystats, statsp);
1795 dns_view_initsecroots(dns_view_t *view, isc_mem_t *mctx) {
1796 REQUIRE(DNS_VIEW_VALID(view));
1797 if (view->secroots_priv != NULL)
1798 dns_keytable_detach(&view->secroots_priv);
1799 return (dns_keytable_create(mctx, &view->secroots_priv));
1803 dns_view_getsecroots(dns_view_t *view, dns_keytable_t **ktp) {
1804 REQUIRE(DNS_VIEW_VALID(view));
1805 REQUIRE(ktp != NULL && *ktp == NULL);
1806 if (view->secroots_priv == NULL)
1807 return (ISC_R_NOTFOUND);
1808 dns_keytable_attach(view->secroots_priv, ktp);
1809 return (ISC_R_SUCCESS);
1813 dns_view_issecuredomain(dns_view_t *view, dns_name_t *name,
1814 isc_boolean_t *secure_domain) {
1815 REQUIRE(DNS_VIEW_VALID(view));
1817 if (view->secroots_priv == NULL)
1818 return (ISC_R_NOTFOUND);
1819 return (dns_keytable_issecuredomain(view->secroots_priv, name,
1824 dns_view_untrust(dns_view_t *view, dns_name_t *keyname,
1825 dns_rdata_dnskey_t *dnskey, isc_mem_t *mctx)
1827 isc_result_t result;
1828 unsigned char data[4096];
1829 dns_rdata_t rdata = DNS_RDATA_INIT;
1830 isc_buffer_t buffer;
1831 dst_key_t *key = NULL;
1832 dns_keytable_t *sr = NULL;
1835 * Clear the revoke bit, if set, so that the key will match what's
1838 dnskey->flags &= ~DNS_KEYFLAG_REVOKE;
1840 /* Convert dnskey to DST key. */
1841 isc_buffer_init(&buffer, data, sizeof(data));
1842 dns_rdata_fromstruct(&rdata, dnskey->common.rdclass,
1843 dns_rdatatype_dnskey, dnskey, &buffer);
1844 result = dns_dnssec_keyfromrdata(keyname, &rdata, mctx, &key);
1845 if (result != ISC_R_SUCCESS)
1847 result = dns_view_getsecroots(view, &sr);
1848 if (result == ISC_R_SUCCESS) {
1849 dns_keytable_deletekeynode(sr, key);
1850 dns_keytable_detach(&sr);
1858 dns_view_setnewzones(dns_view_t *view, isc_boolean_t allow, void *cfgctx,
1859 void (*cfg_destroy)(void **))
1861 REQUIRE(DNS_VIEW_VALID(view));
1862 REQUIRE((cfgctx != NULL && cfg_destroy != NULL) || !allow);
1865 if (view->new_zone_file != NULL) {
1866 isc_mem_free(view->mctx, view->new_zone_file);
1867 view->new_zone_file = NULL;
1870 if (view->new_zone_config != NULL) {
1871 view->cfg_destroy(&view->new_zone_config);
1872 view->cfg_destroy = NULL;
1876 char buffer[ISC_SHA256_DIGESTSTRINGLENGTH + sizeof(NZF)];
1877 isc_sha256_data((void *)view->name, strlen(view->name), buffer);
1878 /* Truncate the hash at 16 chars; full length is overkill */
1879 isc_string_printf(buffer + 16, sizeof(NZF), "%s", NZF);
1880 view->new_zone_file = isc_mem_strdup(view->mctx, buffer);
1881 view->new_zone_config = cfgctx;
1882 view->cfg_destroy = cfg_destroy;
1887 UNUSED(cfg_destroy);