1999-08-05 22:08:45 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 1999 Internet Software Consortium.
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
|
|
|
|
* ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
|
|
|
|
* CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
|
|
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
|
|
|
|
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <isc/types.h>
|
|
|
|
#include <isc/result.h>
|
|
|
|
#include <isc/mem.h>
|
|
|
|
#include <isc/assertions.h>
|
|
|
|
#include <isc/error.h>
|
|
|
|
|
|
|
|
#include <dns/types.h>
|
|
|
|
#include <dns/dbtable.h>
|
1999-08-12 07:49:09 +00:00
|
|
|
#include <dns/db.h>
|
1999-09-22 19:35:47 +00:00
|
|
|
#include <dns/fixedname.h>
|
1999-10-11 19:13:17 +00:00
|
|
|
#include <dns/rbt.h>
|
1999-09-22 19:35:47 +00:00
|
|
|
#include <dns/rdataset.h>
|
1999-08-05 22:08:45 +00:00
|
|
|
#include <dns/resolver.h>
|
|
|
|
#include <dns/view.h>
|
|
|
|
|
|
|
|
#include "../isc/util.h" /* XXXRTH */
|
|
|
|
|
|
|
|
isc_result_t
|
|
|
|
dns_view_create(isc_mem_t *mctx, dns_rdataclass_t rdclass, char *name,
|
1999-08-12 07:49:09 +00:00
|
|
|
dns_view_t **viewp)
|
1999-08-05 22:08:45 +00:00
|
|
|
{
|
|
|
|
dns_view_t *view;
|
|
|
|
isc_result_t result;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create a view.
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(name != NULL);
|
|
|
|
REQUIRE(viewp != NULL && *viewp == NULL);
|
|
|
|
|
|
|
|
view = isc_mem_get(mctx, sizeof *view);
|
|
|
|
if (view == NULL)
|
|
|
|
return (ISC_R_NOMEMORY);
|
|
|
|
view->name = isc_mem_strdup(mctx, name);
|
|
|
|
if (view->name == NULL) {
|
|
|
|
result = ISC_R_NOMEMORY;
|
|
|
|
goto cleanup_view;
|
|
|
|
}
|
|
|
|
result = isc_mutex_init(&view->lock);
|
|
|
|
if (result != ISC_R_SUCCESS) {
|
|
|
|
UNEXPECTED_ERROR(__FILE__, __LINE__,
|
|
|
|
"isc_mutex_init() failed: %s",
|
|
|
|
isc_result_totext(result));
|
|
|
|
result = ISC_R_UNEXPECTED;
|
|
|
|
goto cleanup_name;
|
|
|
|
}
|
|
|
|
view->dbtable = NULL;
|
|
|
|
result = dns_dbtable_create(mctx, rdclass, &view->dbtable);
|
|
|
|
if (result != ISC_R_SUCCESS) {
|
|
|
|
UNEXPECTED_ERROR(__FILE__, __LINE__,
|
|
|
|
"dns_dbtable_create() failed: %s",
|
|
|
|
isc_result_totext(result));
|
|
|
|
result = ISC_R_UNEXPECTED;
|
|
|
|
goto cleanup_mutex;
|
|
|
|
}
|
1999-10-11 19:13:17 +00:00
|
|
|
view->secroots = NULL;
|
|
|
|
result = dns_rbt_create(mctx, NULL, NULL, &view->secroots);
|
|
|
|
if (result != ISC_R_SUCCESS) {
|
|
|
|
UNEXPECTED_ERROR(__FILE__, __LINE__,
|
|
|
|
"dns_rbt_create() failed: %s",
|
|
|
|
isc_result_totext(result));
|
|
|
|
result = ISC_R_UNEXPECTED;
|
|
|
|
goto cleanup_dbtable;
|
|
|
|
}
|
1999-09-22 18:23:36 +00:00
|
|
|
|
1999-08-12 07:49:09 +00:00
|
|
|
view->cachedb = NULL;
|
1999-09-24 01:40:50 +00:00
|
|
|
view->hints = NULL;
|
1999-08-05 22:08:45 +00:00
|
|
|
view->resolver = NULL;
|
|
|
|
view->mctx = mctx;
|
|
|
|
view->rdclass = rdclass;
|
1999-09-22 19:35:47 +00:00
|
|
|
view->frozen = ISC_FALSE;
|
1999-08-05 22:08:45 +00:00
|
|
|
view->references = 1;
|
1999-10-12 00:30:05 +00:00
|
|
|
ISC_LINK_INIT(view, link);
|
1999-08-05 22:08:45 +00:00
|
|
|
view->magic = DNS_VIEW_MAGIC;
|
|
|
|
|
|
|
|
*viewp = view;
|
|
|
|
|
|
|
|
return (ISC_R_SUCCESS);
|
|
|
|
|
1999-10-11 19:13:17 +00:00
|
|
|
cleanup_dbtable:
|
|
|
|
dns_dbtable_detach(&view->dbtable);
|
|
|
|
|
1999-08-05 22:08:45 +00:00
|
|
|
cleanup_mutex:
|
|
|
|
isc_mutex_destroy(&view->lock);
|
|
|
|
|
|
|
|
cleanup_name:
|
|
|
|
isc_mem_free(mctx, view->name);
|
|
|
|
|
|
|
|
cleanup_view:
|
|
|
|
isc_mem_put(mctx, view, sizeof *view);
|
|
|
|
|
|
|
|
return (result);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
dns_view_attach(dns_view_t *source, dns_view_t **targetp) {
|
1999-09-22 19:35:47 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Attach '*targetp' to 'source'.
|
|
|
|
*/
|
|
|
|
|
1999-08-05 22:08:45 +00:00
|
|
|
REQUIRE(DNS_VIEW_VALID(source));
|
|
|
|
REQUIRE(targetp != NULL && *targetp == NULL);
|
|
|
|
|
|
|
|
LOCK(&source->lock);
|
|
|
|
|
|
|
|
INSIST(source->references > 0);
|
|
|
|
source->references++;
|
|
|
|
INSIST(source->references != 0);
|
|
|
|
|
|
|
|
UNLOCK(&source->lock);
|
|
|
|
|
|
|
|
*targetp = source;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
destroy(dns_view_t *view) {
|
|
|
|
REQUIRE(!ISC_LINK_LINKED(view, link));
|
|
|
|
|
|
|
|
if (view->resolver != NULL)
|
|
|
|
dns_resolver_detach(&view->resolver);
|
1999-09-24 01:40:50 +00:00
|
|
|
if (view->hints != NULL)
|
|
|
|
dns_db_detach(&view->hints);
|
1999-08-12 07:49:09 +00:00
|
|
|
if (view->cachedb != NULL)
|
|
|
|
dns_db_detach(&view->cachedb);
|
1999-10-11 19:13:17 +00:00
|
|
|
dns_rbt_destroy(&view->secroots);
|
1999-08-05 22:08:45 +00:00
|
|
|
dns_dbtable_detach(&view->dbtable);
|
|
|
|
isc_mutex_destroy(&view->lock);
|
|
|
|
isc_mem_free(view->mctx, view->name);
|
|
|
|
isc_mem_put(view->mctx, view, sizeof *view);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
dns_view_detach(dns_view_t **viewp) {
|
|
|
|
dns_view_t *view;
|
|
|
|
isc_boolean_t need_destroy = ISC_FALSE;
|
|
|
|
|
1999-09-22 19:35:47 +00:00
|
|
|
/*
|
|
|
|
* Detach '*viewp' from its view.
|
|
|
|
*/
|
|
|
|
|
1999-08-05 22:08:45 +00:00
|
|
|
REQUIRE(viewp != NULL);
|
|
|
|
view = *viewp;
|
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
|
|
|
|
|
|
|
LOCK(&view->lock);
|
|
|
|
|
|
|
|
INSIST(view->references > 0);
|
|
|
|
view->references--;
|
|
|
|
if (view->references == 0)
|
|
|
|
need_destroy = ISC_TRUE;
|
|
|
|
|
|
|
|
UNLOCK(&view->lock);
|
|
|
|
|
|
|
|
*viewp = NULL;
|
|
|
|
|
|
|
|
if (need_destroy)
|
|
|
|
destroy(view);
|
|
|
|
}
|
1999-09-22 18:23:36 +00:00
|
|
|
|
1999-09-24 01:40:50 +00:00
|
|
|
isc_result_t
|
1999-10-07 19:41:16 +00:00
|
|
|
dns_view_createresolver(dns_view_t *view,
|
|
|
|
isc_taskmgr_t *taskmgr, unsigned int ntasks,
|
|
|
|
isc_socketmgr_t *socketmgr,
|
|
|
|
isc_timermgr_t *timermgr,
|
1999-09-24 01:40:50 +00:00
|
|
|
dns_dispatch_t *dispatch)
|
|
|
|
{
|
1999-09-22 19:35:47 +00:00
|
|
|
/*
|
1999-09-24 01:40:50 +00:00
|
|
|
* Create a resolver for the view.
|
1999-09-22 19:35:47 +00:00
|
|
|
*/
|
|
|
|
|
1999-09-22 18:23:36 +00:00
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
1999-09-22 19:35:47 +00:00
|
|
|
REQUIRE(!view->frozen);
|
1999-09-22 18:23:36 +00:00
|
|
|
REQUIRE(view->resolver == NULL);
|
|
|
|
|
1999-10-07 19:41:16 +00:00
|
|
|
return (dns_resolver_create(view, taskmgr, ntasks, socketmgr,
|
|
|
|
timermgr, dispatch, &view->resolver));
|
1999-09-22 18:23:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
dns_view_setcachedb(dns_view_t *view, dns_db_t *cachedb) {
|
1999-09-22 19:35:47 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the view's cache database.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* WARNING! THIS ROUTINE WILL BE REPLACED WITH dns_view_setcache()
|
|
|
|
* WHEN WE HAVE INTEGRATED CACHE OBJECT SUPPORT INTO THE LIBRARY.
|
|
|
|
*/
|
|
|
|
|
1999-09-22 18:23:36 +00:00
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
1999-09-22 19:35:47 +00:00
|
|
|
REQUIRE(!view->frozen);
|
1999-09-22 18:23:36 +00:00
|
|
|
REQUIRE(view->cachedb == NULL);
|
1999-09-22 19:35:47 +00:00
|
|
|
REQUIRE(dns_db_iscache(cachedb));
|
1999-09-22 18:23:36 +00:00
|
|
|
|
|
|
|
dns_db_attach(cachedb, &view->cachedb);
|
|
|
|
}
|
|
|
|
|
1999-09-24 01:40:50 +00:00
|
|
|
void
|
|
|
|
dns_view_sethints(dns_view_t *view, dns_db_t *hints) {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the view's hints database.
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
|
|
|
REQUIRE(!view->frozen);
|
|
|
|
REQUIRE(view->hints == NULL);
|
|
|
|
REQUIRE(dns_db_iszone(hints));
|
|
|
|
|
|
|
|
dns_db_attach(hints, &view->hints);
|
|
|
|
}
|
|
|
|
|
1999-09-22 18:23:36 +00:00
|
|
|
isc_result_t
|
1999-09-22 19:35:47 +00:00
|
|
|
dns_view_addzonedb(dns_view_t *view, dns_db_t *db) {
|
|
|
|
isc_result_t result;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Add zone database 'db' to 'view'.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* WARNING! THIS ROUTINE WILL BE REPLACED WITH dns_view_addzone()
|
|
|
|
* WHEN WE HAVE INTEGRATED ZONE OBJECT SUPPORT INTO THE LIBRARY.
|
|
|
|
*/
|
|
|
|
|
1999-09-22 18:23:36 +00:00
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
1999-09-22 19:35:47 +00:00
|
|
|
REQUIRE(dns_db_iszone(db));
|
|
|
|
REQUIRE(!view->frozen);
|
|
|
|
|
|
|
|
result = dns_dbtable_add(view->dbtable, db);
|
1999-09-22 18:23:36 +00:00
|
|
|
|
1999-09-22 19:35:47 +00:00
|
|
|
return (result);
|
1999-09-22 18:23:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
dns_view_freeze(dns_view_t *view) {
|
1999-09-22 19:35:47 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Freeze view.
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
|
|
|
REQUIRE(!view->frozen);
|
|
|
|
|
|
|
|
view->frozen = ISC_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
isc_result_t
|
|
|
|
dns_view_find(dns_view_t *view, dns_name_t *name, dns_rdatatype_t type,
|
1999-09-24 01:40:50 +00:00
|
|
|
isc_stdtime_t now, unsigned int options, isc_boolean_t use_hints,
|
1999-09-22 19:35:47 +00:00
|
|
|
dns_rdataset_t *rdataset, dns_rdataset_t *sigrdataset)
|
|
|
|
{
|
|
|
|
isc_result_t result;
|
|
|
|
dns_fixedname_t foundname;
|
|
|
|
dns_db_t *db;
|
|
|
|
dns_dbversion_t *version;
|
|
|
|
isc_boolean_t is_zone;
|
|
|
|
dns_rdataset_t zrdataset, zsigrdataset;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find an rdataset whose owner name is 'name', and whose type is
|
|
|
|
* 'type'.
|
|
|
|
*/
|
|
|
|
|
1999-09-22 18:23:36 +00:00
|
|
|
REQUIRE(DNS_VIEW_VALID(view));
|
1999-09-22 19:35:47 +00:00
|
|
|
REQUIRE(view->frozen);
|
|
|
|
REQUIRE(type != dns_rdatatype_any && type != dns_rdatatype_sig);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize.
|
|
|
|
*/
|
|
|
|
dns_rdataset_init(&zrdataset);
|
|
|
|
dns_rdataset_init(&zsigrdataset);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find a database to answer the query.
|
|
|
|
*/
|
|
|
|
db = NULL;
|
|
|
|
result = dns_dbtable_find(view->dbtable, name, &db);
|
|
|
|
if (result == ISC_R_NOTFOUND && view->cachedb != NULL)
|
|
|
|
dns_db_attach(view->cachedb, &db);
|
|
|
|
else if (result != ISC_R_SUCCESS && result != DNS_R_PARTIALMATCH)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
is_zone = dns_db_iszone(db);
|
1999-09-22 18:23:36 +00:00
|
|
|
|
1999-09-22 19:35:47 +00:00
|
|
|
db_find:
|
|
|
|
/*
|
|
|
|
* Now look for an answer in the database.
|
|
|
|
*/
|
|
|
|
dns_fixedname_init(&foundname);
|
|
|
|
result = dns_db_find(db, name, NULL, type, options,
|
|
|
|
now, NULL, dns_fixedname_name(&foundname),
|
|
|
|
rdataset, sigrdataset);
|
|
|
|
|
|
|
|
if (result == DNS_R_DELEGATION ||
|
|
|
|
result == DNS_R_NOTFOUND ||
|
|
|
|
result == DNS_R_NXGLUE) {
|
|
|
|
if (rdataset->methods != NULL)
|
|
|
|
dns_rdataset_disassociate(rdataset);
|
1999-10-07 19:41:16 +00:00
|
|
|
if (sigrdataset != NULL && sigrdataset->methods != NULL)
|
1999-09-22 19:35:47 +00:00
|
|
|
dns_rdataset_disassociate(sigrdataset);
|
|
|
|
if (is_zone) {
|
|
|
|
if (view->cachedb != NULL) {
|
|
|
|
/*
|
|
|
|
* Either the answer is in the cache, or we
|
|
|
|
* don't know it.
|
|
|
|
*/
|
|
|
|
is_zone = ISC_FALSE;
|
|
|
|
version = NULL;
|
|
|
|
dns_db_detach(&db);
|
|
|
|
dns_db_attach(view->cachedb, &db);
|
|
|
|
goto db_find;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* We don't have the data in the cache. If we've got
|
|
|
|
* glue from the zone, use it.
|
|
|
|
*/
|
|
|
|
if (zrdataset.methods != NULL) {
|
|
|
|
dns_rdataset_clone(&zrdataset, rdataset);
|
|
|
|
if (zsigrdataset.methods != NULL)
|
|
|
|
dns_rdataset_clone(&zsigrdataset,
|
|
|
|
sigrdataset);
|
|
|
|
result = DNS_R_GLUE;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* We don't know the answer.
|
|
|
|
*/
|
|
|
|
result = DNS_R_NOTFOUND;
|
|
|
|
} else if (result == DNS_R_GLUE) {
|
|
|
|
if (view->cachedb != NULL) {
|
|
|
|
/*
|
|
|
|
* We found an answer, but the cache may be better.
|
|
|
|
* Remember what we've got and go look in the cache.
|
|
|
|
*/
|
|
|
|
is_zone = ISC_FALSE;
|
|
|
|
version = NULL;
|
|
|
|
dns_rdataset_clone(rdataset, &zrdataset);
|
|
|
|
dns_rdataset_disassociate(rdataset);
|
1999-10-07 19:41:16 +00:00
|
|
|
if (sigrdataset != NULL &&
|
|
|
|
sigrdataset->methods != NULL) {
|
1999-09-22 19:35:47 +00:00
|
|
|
dns_rdataset_clone(sigrdataset, &zsigrdataset);
|
|
|
|
dns_rdataset_disassociate(sigrdataset);
|
|
|
|
}
|
|
|
|
dns_db_detach(&db);
|
|
|
|
dns_db_attach(view->cachedb, &db);
|
|
|
|
goto db_find;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Otherwise, the glue is the best answer.
|
|
|
|
*/
|
|
|
|
result = ISC_R_SUCCESS;
|
1999-09-24 01:40:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (result == DNS_R_NOTFOUND && use_hints && view->hints != NULL) {
|
|
|
|
if (rdataset->methods != NULL)
|
|
|
|
dns_rdataset_disassociate(rdataset);
|
1999-10-07 19:41:16 +00:00
|
|
|
if (sigrdataset != NULL && sigrdataset->methods != NULL)
|
1999-09-24 01:40:50 +00:00
|
|
|
dns_rdataset_disassociate(sigrdataset);
|
|
|
|
dns_fixedname_init(&foundname);
|
|
|
|
result = dns_db_find(view->hints, name, NULL, type, options,
|
|
|
|
now, NULL, dns_fixedname_name(&foundname),
|
|
|
|
rdataset, sigrdataset);
|
|
|
|
if (result == ISC_R_SUCCESS || result == DNS_R_GLUE)
|
|
|
|
result = DNS_R_HINT;
|
|
|
|
}
|
1999-09-22 19:35:47 +00:00
|
|
|
|
|
|
|
cleanup:
|
1999-09-24 01:40:50 +00:00
|
|
|
if (result != ISC_R_SUCCESS &&
|
|
|
|
result != DNS_R_GLUE &&
|
|
|
|
result != DNS_R_HINT)
|
|
|
|
result = DNS_R_NOTFOUND;
|
|
|
|
|
1999-09-22 19:35:47 +00:00
|
|
|
if (zrdataset.methods != NULL) {
|
|
|
|
dns_rdataset_disassociate(&zrdataset);
|
|
|
|
if (zsigrdataset.methods != NULL)
|
|
|
|
dns_rdataset_disassociate(&zsigrdataset);
|
|
|
|
}
|
|
|
|
if (db != NULL)
|
|
|
|
dns_db_detach(&db);
|
|
|
|
|
|
|
|
return (result);
|
1999-09-22 18:23:36 +00:00
|
|
|
}
|