Martin Nagy b1c08c
diff --git a/bin/named/main.c b/bin/named/main.c
Martin Nagy b1c08c
index aa6575a..8030e3d 100644
Martin Nagy b1c08c
--- a/bin/named/main.c
Martin Nagy b1c08c
+++ b/bin/named/main.c
Martin Nagy b1c08c
@@ -44,6 +44,7 @@
Martin Nagy b1c08c
 #include <isccc/result.h>
Martin Nagy b1c08c
 
Martin Nagy b1c08c
 #include <dns/dispatch.h>
Martin Nagy b1c08c
+#include <dns/dynamic_db.h>
Martin Nagy b1c08c
 #include <dns/name.h>
Martin Nagy b1c08c
 #include <dns/result.h>
Martin Nagy b1c08c
 #include <dns/view.h>
Martin Nagy b1c08c
@@ -778,6 +779,8 @@ cleanup(void) {
Martin Nagy b1c08c
 	dlz_drivers_clear();
Martin Nagy b1c08c
 #endif
Martin Nagy b1c08c
 
Martin Nagy b1c08c
+	dns_dynamic_db_cleanup();
Martin Nagy b1c08c
+
Martin Nagy b1c08c
 	dns_name_destroy();
Martin Nagy b1c08c
 
Martin Nagy b1c08c
 	isc_log_write(ns_g_lctx, NS_LOGCATEGORY_GENERAL, NS_LOGMODULE_MAIN,
Martin Nagy b1c08c
diff --git a/bin/named/server.c b/bin/named/server.c
Martin Nagy 205f42
index 31b2761..b76f3ce 100644
Martin Nagy b1c08c
--- a/bin/named/server.c
Martin Nagy b1c08c
+++ b/bin/named/server.c
Martin Nagy b1c08c
@@ -56,6 +56,7 @@
Martin Nagy b1c08c
 #ifdef DLZ
Martin Nagy b1c08c
 #include <dns/dlz.h>
Martin Nagy b1c08c
 #endif
Martin Nagy b1c08c
+#include <dns/dynamic_db.h>
Martin Nagy b1c08c
 #include <dns/forward.h>
Martin Nagy b1c08c
 #include <dns/journal.h>
Martin Nagy b1c08c
 #include <dns/keytable.h>
Martin Nagy 205f42
@@ -849,6 +850,72 @@ configure_peer(const cfg_obj_t *cpeer, isc_mem_t *mctx, dns_peer_t **peerp) {
Martin Nagy b1c08c
 }
Martin Nagy b1c08c
 
Martin Nagy b1c08c
 static isc_result_t
Martin Nagy b1c08c
+configure_dynamic_db(const cfg_obj_t *dynamic_db, isc_mem_t *mctx,
Martin Nagy 205f42
+		     dns_view_t *view, dns_zonemgr_t *zmgr)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	isc_result_t result;
Martin Nagy b1c08c
+	const cfg_obj_t *obj;
Martin Nagy b1c08c
+	const cfg_obj_t *options;
Martin Nagy b1c08c
+	const cfg_listelt_t *element;
Martin Nagy b1c08c
+	const char *name;
Martin Nagy b1c08c
+	const char *libname;
Martin Nagy b1c08c
+	const char **argv = NULL;
Martin Nagy b1c08c
+	unsigned int i;
Martin Nagy b1c08c
+	unsigned int len;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	/* Get the name of the database. */
Martin Nagy b1c08c
+	obj = cfg_tuple_get(dynamic_db, "name");
Martin Nagy b1c08c
+	name = cfg_obj_asstring(obj);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	/* Get options. */
Martin Nagy b1c08c
+	options = cfg_tuple_get(dynamic_db, "options");
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	/* Get library name. */
Martin Nagy b1c08c
+	obj = NULL;
Martin Nagy b1c08c
+	CHECK(cfg_map_get(options, "library", &obj));
Martin Nagy b1c08c
+	libname = cfg_obj_asstring(obj);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	/* Create a list of arguments. */
Martin Nagy b1c08c
+	obj = NULL;
Martin Nagy 205f42
+	result = cfg_map_get(options, "arg", &obj);
Martin Nagy 205f42
+	if (result == ISC_R_NOTFOUND)
Martin Nagy 205f42
+		len = 0;
Martin Nagy 205f42
+	else if (result == ISC_R_SUCCESS)
Martin Nagy 205f42
+		len = cfg_list_length(obj, isc_boolean_false);
Martin Nagy 205f42
+	else
Martin Nagy 205f42
+		goto cleanup;
Martin Nagy 205f42
+
Martin Nagy 205f42
+	/* Account for the last terminating NULL. */
Martin Nagy 205f42
+	len++;
Martin Nagy 205f42
+
Martin Nagy 205f42
+	argv = isc_mem_allocate(mctx, len * sizeof(const char *));
Martin Nagy 205f42
+	if (argv == NULL) {
Martin Nagy 205f42
+		result = ISC_R_NOMEMORY;
Martin Nagy 205f42
+		goto cleanup;
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+	for (element = cfg_list_first(obj), i = 0;
Martin Nagy b1c08c
+	     element != NULL;
Martin Nagy b1c08c
+	     element = cfg_list_next(element), i++)
Martin Nagy b1c08c
+	{
Martin Nagy b1c08c
+		REQUIRE(i < len);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+		obj = cfg_listelt_value(element);
Martin Nagy b1c08c
+		argv[i] = cfg_obj_asstring(obj);
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+	REQUIRE(i < len);
Martin Nagy b1c08c
+	argv[i] = NULL;
Martin Nagy b1c08c
+
Martin Nagy 205f42
+	CHECK(dns_dynamic_db_load(libname, name, mctx, argv, view, zmgr));
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+cleanup:
Martin Nagy b1c08c
+	if (argv != NULL)
Martin Nagy b1c08c
+		isc_mem_free(mctx, argv);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return result;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static isc_result_t
Martin Nagy b1c08c
 disable_algorithms(const cfg_obj_t *disabled, dns_resolver_t *resolver) {
Martin Nagy b1c08c
 	isc_result_t result;
Martin Nagy b1c08c
 	const cfg_obj_t *algorithms;
Martin Nagy 205f42
@@ -999,6 +1066,7 @@ configure_view(dns_view_t *view, const cfg_obj_t *config,
Martin Nagy b1c08c
 	unsigned int dlzargc;
Martin Nagy b1c08c
 	char **dlzargv;
Martin Nagy b1c08c
 #endif
Martin Nagy b1c08c
+	const cfg_obj_t *dynamic_db_list;
Martin Nagy b1c08c
 	const cfg_obj_t *disabled;
Martin Nagy b1c08c
 	const cfg_obj_t *obj;
Martin Nagy b1c08c
 	const cfg_listelt_t *element;
Martin Nagy 205f42
@@ -1171,6 +1239,22 @@ configure_view(dns_view_t *view, const cfg_obj_t *config,
Martin Nagy b1c08c
 #endif
Martin Nagy b1c08c
 
Martin Nagy b1c08c
 	/*
Martin Nagy b1c08c
+	 * Configure dynamic databases.
Martin Nagy b1c08c
+	 */
Martin Nagy b1c08c
+	dynamic_db_list = NULL;
Martin Nagy b1c08c
+	if (voptions != NULL)
Martin Nagy b1c08c
+		(void)cfg_map_get(voptions, "dynamic-db", &dynamic_db_list);
Martin Nagy b1c08c
+	else
Martin Nagy b1c08c
+		(void)cfg_map_get(config, "dynamic-db", &dynamic_db_list);
Martin Nagy b1c08c
+	for (element = cfg_list_first(dynamic_db_list);
Martin Nagy b1c08c
+	     element != NULL;
Martin Nagy b1c08c
+	     element = cfg_list_next(element))
Martin Nagy b1c08c
+	{
Martin Nagy b1c08c
+		obj = cfg_listelt_value(element);
Martin Nagy 205f42
+		CHECK(configure_dynamic_db(obj, mctx, view, ns_g_server->zonemgr));
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	/*
Martin Nagy b1c08c
 	 * Configure the view's cache.  Try to reuse an existing
Martin Nagy b1c08c
 	 * cache if possible, otherwise create a new cache.
Martin Nagy b1c08c
 	 * Note that the ADB is not preserved in either case.
Martin Nagy b1c08c
diff --git a/lib/dns/Makefile.in b/lib/dns/Makefile.in
Martin Nagy b1c08c
index ef5c12a..0f7abba 100644
Martin Nagy b1c08c
--- a/lib/dns/Makefile.in
Martin Nagy b1c08c
+++ b/lib/dns/Makefile.in
Martin Nagy b1c08c
@@ -57,7 +57,8 @@ DSTOBJS =	@DST_EXTRA_OBJS@ \
Martin Nagy b1c08c
 DNSOBJS =	acache.@O@ acl.@O@ adb.@O@ byaddr.@O@ \
Martin Nagy b1c08c
 		cache.@O@ callbacks.@O@ compress.@O@ \
Martin Nagy b1c08c
 		db.@O@ dbiterator.@O@ dbtable.@O@ diff.@O@ dispatch.@O@ \
Martin Nagy b1c08c
-		dlz.@O@ dnssec.@O@ ds.@O@ forward.@O@ iptable.@O@ journal.@O@ \
Martin Nagy b1c08c
+		dlz.@O@ dnssec.@O@ ds.@O@ dynamic_db.@O@ forward.@O@ \
Martin Nagy b1c08c
+		iptable.@O@ journal.@O@ \
Martin Nagy b1c08c
 		keytable.@O@ lib.@O@ log.@O@ lookup.@O@ \
Martin Nagy b1c08c
 		master.@O@ masterdump.@O@ message.@O@ \
Martin Nagy b1c08c
 		name.@O@ ncache.@O@ nsec.@O@ nsec3.@O@ order.@O@ peer.@O@ portlist.@O@ \
Martin Nagy b1c08c
@@ -83,7 +84,7 @@ DSTSRCS =	@DST_EXTRA_SRCS@ \
Martin Nagy b1c08c
 DNSSRCS =	acache.c acl.c adb.c byaddr.c \
Martin Nagy b1c08c
 		cache.c callbacks.c compress.c \
Martin Nagy b1c08c
 		db.c dbiterator.c dbtable.c diff.c dispatch.c \
Martin Nagy b1c08c
-		dlz.c dnssec.c ds.c forward.c iptable.c journal.c \
Martin Nagy b1c08c
+		dlz.c dnssec.c ds.c dynamic_db.c forward.c iptable.c journal.c \
Martin Nagy b1c08c
 		keytable.c lib.c log.c lookup.c \
Martin Nagy b1c08c
 		master.c masterdump.c message.c \
Martin Nagy b1c08c
 		name.c ncache.c nsec.c nsec3.c order.c peer.c portlist.c \
Martin Nagy b1c08c
diff --git a/lib/dns/dynamic_db.c b/lib/dns/dynamic_db.c
Martin Nagy b1c08c
new file mode 100644
Martin Nagy 205f42
index 0000000..b95b1fe
Martin Nagy b1c08c
--- /dev/null
Martin Nagy b1c08c
+++ b/lib/dns/dynamic_db.c
Martin Nagy b1c08c
@@ -0,0 +1,240 @@
Martin Nagy b1c08c
+/*
Martin Nagy b1c08c
+ * Copyright (C) 2004-2009  Internet Systems Consortium, Inc. ("ISC")
Martin Nagy b1c08c
+ * Copyright (C) 1996-2003  Internet Software Consortium.
Martin Nagy b1c08c
+ *
Martin Nagy b1c08c
+ * Permission to use, copy, modify, and/or distribute this software for any
Martin Nagy b1c08c
+ * purpose with or without fee is hereby granted, provided that the above
Martin Nagy b1c08c
+ * copyright notice and this permission notice appear in all copies.
Martin Nagy b1c08c
+ *
Martin Nagy b1c08c
+ * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
Martin Nagy b1c08c
+ * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
Martin Nagy b1c08c
+ * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
Martin Nagy b1c08c
+ * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
Martin Nagy b1c08c
+ * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
Martin Nagy b1c08c
+ * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
Martin Nagy b1c08c
+ * PERFORMANCE OF THIS SOFTWARE.
Martin Nagy b1c08c
+ */
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#include <config.h>
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#include <isc/mem.h>
Martin Nagy b1c08c
+#include <isc/mutex.h>
Martin Nagy b1c08c
+#include <isc/once.h>
Martin Nagy b1c08c
+#include <isc/result.h>
Martin Nagy b1c08c
+#include <isc/types.h>
Martin Nagy b1c08c
+#include <isc/util.h>
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#include <dns/dynamic_db.h>
Martin Nagy b1c08c
+#include <dns/log.h>
Martin Nagy b1c08c
+#include <dns/types.h>
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#if HAVE_DLFCN_H
Martin Nagy b1c08c
+#include <dlfcn.h>
Martin Nagy b1c08c
+#endif
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#define CHECK(op)						\
Martin Nagy b1c08c
+	do { result = (op);					\
Martin Nagy b1c08c
+		if (result != ISC_R_SUCCESS) goto cleanup;	\
Martin Nagy b1c08c
+	} while (0)
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+typedef isc_result_t (*register_func_t)(isc_mem_t *mctx, const char *name,
Martin Nagy 205f42
+		const char * const *argv, dns_view_t *view,
Martin Nagy 205f42
+		dns_zonemgr_t *zmgr);
Martin Nagy b1c08c
+typedef void (*destroy_func_t)(void);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+typedef struct dyndb_implementation dyndb_implementation_t;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+struct dyndb_implementation {
Martin Nagy b1c08c
+	isc_mem_t			*mctx;
Martin Nagy b1c08c
+	void				*handle;
Martin Nagy b1c08c
+	register_func_t			register_function;
Martin Nagy b1c08c
+	destroy_func_t			destroy_function;
Martin Nagy b1c08c
+	LINK(dyndb_implementation_t)	link;
Martin Nagy b1c08c
+};
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+/* List of implementations. Locked by dyndb_lock. */
Martin Nagy b1c08c
+static LIST(dyndb_implementation_t) dyndb_implementations;
Martin Nagy b1c08c
+/* Locks dyndb_implementations. */
Martin Nagy b1c08c
+static isc_mutex_t dyndb_lock;
Martin Nagy b1c08c
+static isc_once_t once = ISC_ONCE_INIT;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static void
Martin Nagy b1c08c
+dyndb_initialize(void) {
Martin Nagy b1c08c
+	RUNTIME_CHECK(isc_mutex_init(&dyndb_lock) == ISC_R_SUCCESS);
Martin Nagy b1c08c
+	INIT_LIST(dyndb_implementations);
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#if HAVE_DLFCN_H
Martin Nagy b1c08c
+static isc_result_t
Martin Nagy b1c08c
+load_symbol(void *handle, const char *symbol_name, void **symbolp)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	const char *errmsg;
Martin Nagy b1c08c
+	void *symbol;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	REQUIRE(handle != NULL);
Martin Nagy b1c08c
+	REQUIRE(symbolp != NULL && *symbolp == NULL);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	symbol = dlsym(handle, symbol_name);
Martin Nagy b1c08c
+	if (symbol == NULL) {
Martin Nagy b1c08c
+		errmsg = dlerror();
Martin Nagy b1c08c
+		if (errmsg == NULL)
Martin Nagy b1c08c
+			errmsg = "returned function pointer is NULL";
Martin Nagy b1c08c
+		isc_log_write(dns_lctx, DNS_LOGCATEGORY_DATABASE,
Martin Nagy b1c08c
+			      DNS_LOGMODULE_DYNDB, ISC_LOG_ERROR,
Martin Nagy b1c08c
+			      "failed to lookup symbol %s: %s",
Martin Nagy b1c08c
+			      symbol_name, errmsg);
Martin Nagy b1c08c
+		return ISC_R_FAILURE;
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+	dlerror();
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	*symbolp = symbol;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return ISC_R_SUCCESS;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static isc_result_t
Martin Nagy b1c08c
+load_library(isc_mem_t *mctx, const char *filename, dyndb_implementation_t **impp)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	isc_result_t result;
Martin Nagy b1c08c
+	void *handle;
Martin Nagy b1c08c
+	dyndb_implementation_t *imp;
Martin Nagy b1c08c
+	register_func_t register_function = NULL;
Martin Nagy b1c08c
+	destroy_func_t destroy_function = NULL;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	REQUIRE(impp != NULL && *impp == NULL);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	handle = dlopen(filename, RTLD_LAZY);
Martin Nagy b1c08c
+	if (handle == NULL) {
Martin Nagy b1c08c
+		isc_log_write(dns_lctx, DNS_LOGCATEGORY_DATABASE,
Martin Nagy b1c08c
+			      DNS_LOGMODULE_DYNDB, ISC_LOG_ERROR,
Martin Nagy b1c08c
+			      "failed to dynamically load driver '%s': %s",
Martin Nagy b1c08c
+			      filename, dlerror());
Martin Nagy b1c08c
+		result = ISC_R_FAILURE;
Martin Nagy b1c08c
+		goto cleanup;
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+	dlerror();
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	CHECK(load_symbol(handle, "dynamic_driver_init",
Martin Nagy b1c08c
+			  (void **)&register_function));
Martin Nagy b1c08c
+	CHECK(load_symbol(handle, "dynamic_driver_destroy",
Martin Nagy b1c08c
+			  (void **)&destroy_function));
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	imp = isc_mem_get(mctx, sizeof(dyndb_implementation_t));
Martin Nagy b1c08c
+	if (imp == NULL) {
Martin Nagy b1c08c
+		result = ISC_R_NOMEMORY;
Martin Nagy b1c08c
+		goto cleanup;
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	imp->mctx = NULL;
Martin Nagy b1c08c
+	isc_mem_attach(mctx, &imp->mctx);
Martin Nagy b1c08c
+	imp->handle = handle;
Martin Nagy b1c08c
+	imp->register_function = register_function;
Martin Nagy b1c08c
+	imp->destroy_function = destroy_function;
Martin Nagy b1c08c
+	INIT_LINK(imp, link);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	*impp = imp;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return ISC_R_SUCCESS;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+cleanup:
Martin Nagy b1c08c
+	if (handle != NULL)
Martin Nagy b1c08c
+		dlclose(handle);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return result;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static void
Martin Nagy b1c08c
+unload_library(dyndb_implementation_t **impp)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	dyndb_implementation_t *imp;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	REQUIRE(impp != NULL && *impp != NULL);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	imp = *impp;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	isc_mem_putanddetach(&imp->mctx, imp, sizeof(dyndb_implementation_t));
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	*impp = NULL;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#else	/* HAVE_DLFCN_H */
Martin Nagy b1c08c
+static isc_result_t
Martin Nagy b1c08c
+load_library(isc_mem_t *mctx, const char *filename, dyndb_implementation_t **impp)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	UNUSED(mctx);
Martin Nagy b1c08c
+	UNUSED(filename);
Martin Nagy b1c08c
+	UNUSED(impp);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	isc_log_write(dns_lctx, DNS_LOGCATEGORY_DATABASE, DNS_LOGMODULE_DYNDB,
Martin Nagy b1c08c
+		      ISC_LOG_ERROR,
Martin Nagy b1c08c
+		      "dynamic database support is not implemented")
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return ISC_R_NOTIMPLEMENTED;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static void
Martin Nagy b1c08c
+unload_library(dyndb_implementation_t **impp)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	dyndb_implementation_t *imp;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	REQUIRE(impp != NULL && *impp != NULL);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	imp = *impp;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	isc_mem_putanddetach(&imp->mctx, imp, sizeof(dyndb_implementation_t));
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	*impp = NULL;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+#endif	/* HAVE_DLFCN_H */
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+isc_result_t
Martin Nagy b1c08c
+dns_dynamic_db_load(const char *libname, const char *name, isc_mem_t *mctx,
Martin Nagy 205f42
+		    const char * const *argv, dns_view_t *view,
Martin Nagy 205f42
+		    dns_zonemgr_t *zmgr)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	isc_result_t result;
Martin Nagy b1c08c
+	dyndb_implementation_t *implementation = NULL;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	RUNTIME_CHECK(isc_once_do(&once, dyndb_initialize) == ISC_R_SUCCESS);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	CHECK(load_library(mctx, libname, &implementation));
Martin Nagy 205f42
+	CHECK(implementation->register_function(mctx, name, argv, view, zmgr));
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	LOCK(&dyndb_lock);
Martin Nagy b1c08c
+	APPEND(dyndb_implementations, implementation, link);
Martin Nagy b1c08c
+	UNLOCK(&dyndb_lock);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return ISC_R_SUCCESS;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+cleanup:
Martin Nagy b1c08c
+	if (implementation != NULL)
Martin Nagy b1c08c
+		unload_library(&implementation);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	return result;
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
+
Martin Nagy 205f42
+void
Martin Nagy b1c08c
+dns_dynamic_db_cleanup(void)
Martin Nagy b1c08c
+{
Martin Nagy b1c08c
+	dyndb_implementation_t *elem;
Martin Nagy b1c08c
+	dyndb_implementation_t *next;
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	RUNTIME_CHECK(isc_once_do(&once, dyndb_initialize) == ISC_R_SUCCESS);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	LOCK(&dyndb_lock);
Martin Nagy b1c08c
+	elem = HEAD(dyndb_implementations);
Martin Nagy b1c08c
+	while (elem != NULL) {
Martin Nagy b1c08c
+		next = NEXT(elem, link);
Martin Nagy b1c08c
+		UNLINK(dyndb_implementations, elem, link);
Martin Nagy b1c08c
+		elem->destroy_function();
Martin Nagy b1c08c
+		unload_library(&elem);
Martin Nagy b1c08c
+		elem = next;
Martin Nagy b1c08c
+	}
Martin Nagy b1c08c
+	UNLOCK(&dyndb_lock);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+	isc_mutex_destroy(&dyndb_lock);
Martin Nagy b1c08c
+}
Martin Nagy b1c08c
diff --git a/lib/dns/include/dns/Makefile.in b/lib/dns/include/dns/Makefile.in
Martin Nagy b1c08c
index e9e049e..27fdc45 100644
Martin Nagy b1c08c
--- a/lib/dns/include/dns/Makefile.in
Martin Nagy b1c08c
+++ b/lib/dns/include/dns/Makefile.in
Martin Nagy b1c08c
@@ -23,7 +23,7 @@ top_srcdir =	@top_srcdir@
Martin Nagy b1c08c
 
Martin Nagy b1c08c
 HEADERS =	acl.h adb.h byaddr.h cache.h callbacks.h \
Martin Nagy b1c08c
 		cert.h compress.h \
Martin Nagy b1c08c
-		db.h dbiterator.h dbtable.h diff.h dispatch.h dlz.h \
Martin Nagy b1c08c
+		db.h dbiterator.h dbtable.h diff.h dispatch.h dlz.h dynamic_db.h \
Martin Nagy b1c08c
 		dnssec.h ds.h events.h fixedname.h iptable.h journal.h keyflags.h \
Martin Nagy b1c08c
 		keytable.h keyvalues.h lib.h log.h master.h masterdump.h \
Martin Nagy b1c08c
 		message.h name.h ncache.h \
Martin Nagy b1c08c
diff --git a/lib/dns/include/dns/dynamic_db.h b/lib/dns/include/dns/dynamic_db.h
Martin Nagy b1c08c
new file mode 100644
Martin Nagy 205f42
index 0000000..03339e6
Martin Nagy b1c08c
--- /dev/null
Martin Nagy b1c08c
+++ b/lib/dns/include/dns/dynamic_db.h
Martin Nagy 205f42
@@ -0,0 +1,32 @@
Martin Nagy b1c08c
+/*
Martin Nagy b1c08c
+ * Copyright (C) 2004-2009  Internet Systems Consortium, Inc. ("ISC")
Martin Nagy b1c08c
+ * Copyright (C) 1996-2003  Internet Software Consortium.
Martin Nagy b1c08c
+ *
Martin Nagy b1c08c
+ * Permission to use, copy, modify, and/or distribute this software for any
Martin Nagy b1c08c
+ * purpose with or without fee is hereby granted, provided that the above
Martin Nagy b1c08c
+ * copyright notice and this permission notice appear in all copies.
Martin Nagy b1c08c
+ *
Martin Nagy b1c08c
+ * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
Martin Nagy b1c08c
+ * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
Martin Nagy b1c08c
+ * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
Martin Nagy b1c08c
+ * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
Martin Nagy b1c08c
+ * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
Martin Nagy b1c08c
+ * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
Martin Nagy b1c08c
+ * PERFORMANCE OF THIS SOFTWARE.
Martin Nagy b1c08c
+ */
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#ifndef DYNAMIC_DB_H
Martin Nagy b1c08c
+#define DYNAMIC_DB_H
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#include <isc/types.h>
Martin Nagy 205f42
+
Martin Nagy b1c08c
+#include <dns/types.h>
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+isc_result_t dns_dynamic_db_load(const char *libname, const char *name,
Martin Nagy b1c08c
+				 isc_mem_t *mctx, const char * const *argv,
Martin Nagy 205f42
+				 dns_view_t *view, dns_zonemgr_t *zmgr);
Martin Nagy b1c08c
+
Martin Nagy 205f42
+void dns_dynamic_db_cleanup(void);
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+#endif
Martin Nagy b1c08c
diff --git a/lib/dns/include/dns/log.h b/lib/dns/include/dns/log.h
Martin Nagy b1c08c
index 5adcedd..e171028 100644
Martin Nagy b1c08c
--- a/lib/dns/include/dns/log.h
Martin Nagy b1c08c
+++ b/lib/dns/include/dns/log.h
Martin Nagy b1c08c
@@ -73,6 +73,7 @@ LIBDNS_EXTERNAL_DATA extern isc_logmodule_t dns_modules[];
Martin Nagy b1c08c
 #define DNS_LOGMODULE_HINTS		(&dns_modules[24])
Martin Nagy b1c08c
 #define DNS_LOGMODULE_ACACHE		(&dns_modules[25])
Martin Nagy b1c08c
 #define DNS_LOGMODULE_DLZ		(&dns_modules[26])
Martin Nagy b1c08c
+#define DNS_LOGMODULE_DYNDB		(&dns_modules[27])
Martin Nagy b1c08c
 
Martin Nagy b1c08c
 ISC_LANG_BEGINDECLS
Martin Nagy b1c08c
 
Martin Nagy b1c08c
diff --git a/lib/dns/log.c b/lib/dns/log.c
Martin Nagy b1c08c
index 7551e15..b9864eb 100644
Martin Nagy b1c08c
--- a/lib/dns/log.c
Martin Nagy b1c08c
+++ b/lib/dns/log.c
Martin Nagy b1c08c
@@ -79,6 +79,7 @@ LIBDNS_EXTERNAL_DATA isc_logmodule_t dns_modules[] = {
Martin Nagy b1c08c
 	{ "dns/hints",		0 },
Martin Nagy b1c08c
 	{ "dns/acache",		0 },
Martin Nagy b1c08c
 	{ "dns/dlz",		0 },
Martin Nagy b1c08c
+	{ "dns/dynamic_db",	0 },
Martin Nagy b1c08c
 	{ NULL, 		0 }
Martin Nagy b1c08c
 };
Martin Nagy b1c08c
 
Martin Nagy b1c08c
diff --git a/lib/isccfg/namedconf.c b/lib/isccfg/namedconf.c
Martin Nagy b1c08c
index 0610489..a1dba32 100644
Martin Nagy b1c08c
--- a/lib/isccfg/namedconf.c
Martin Nagy b1c08c
+++ b/lib/isccfg/namedconf.c
Martin Nagy b1c08c
@@ -78,6 +78,7 @@ static cfg_type_t cfg_type_controls;
Martin Nagy b1c08c
 static cfg_type_t cfg_type_controls_sockaddr;
Martin Nagy b1c08c
 static cfg_type_t cfg_type_destinationlist;
Martin Nagy b1c08c
 static cfg_type_t cfg_type_dialuptype;
Martin Nagy b1c08c
+static cfg_type_t cfg_type_dynamic_db;
Martin Nagy b1c08c
 static cfg_type_t cfg_type_ixfrdifftype;
Martin Nagy b1c08c
 static cfg_type_t cfg_type_key;
Martin Nagy b1c08c
 static cfg_type_t cfg_type_logfile;
Martin Nagy b1c08c
@@ -651,6 +652,7 @@ namedconf_or_view_clauses[] = {
Martin Nagy b1c08c
 	{ "zone", &cfg_type_zone, CFG_CLAUSEFLAG_MULTI },
Martin Nagy b1c08c
 	/* only 1 DLZ per view allowed */
Martin Nagy b1c08c
 	{ "dlz", &cfg_type_dynamically_loadable_zones, 0 },
Martin Nagy b1c08c
+	{ "dynamic-db", &cfg_type_dynamic_db, CFG_CLAUSEFLAG_MULTI },
Martin Nagy b1c08c
 	{ "server", &cfg_type_server, CFG_CLAUSEFLAG_MULTI },
Martin Nagy b1c08c
 	{ "trusted-keys", &cfg_type_trustedkeys, CFG_CLAUSEFLAG_MULTI },
Martin Nagy b1c08c
 	{ NULL, NULL, 0 }
Martin Nagy b1c08c
@@ -1365,6 +1367,40 @@ static cfg_type_t cfg_type_dialuptype = {
Martin Nagy b1c08c
 	&cfg_rep_string, dialup_enums
Martin Nagy b1c08c
 };
Martin Nagy b1c08c
 
Martin Nagy b1c08c
+/*
Martin Nagy b1c08c
+ * Dynamic database clauses.
Martin Nagy b1c08c
+ */
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static cfg_clausedef_t
Martin Nagy b1c08c
+dynamic_db_clauses[] = {
Martin Nagy b1c08c
+	{ "library", &cfg_type_qstring, 0 },
Martin Nagy b1c08c
+	{ "arg", &cfg_type_qstring, CFG_CLAUSEFLAG_MULTI },
Martin Nagy b1c08c
+	{ NULL, NULL, 0 }
Martin Nagy b1c08c
+};
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static cfg_clausedef_t *
Martin Nagy b1c08c
+dynamic_db_clausesets[] = {
Martin Nagy b1c08c
+	dynamic_db_clauses,
Martin Nagy b1c08c
+	NULL
Martin Nagy b1c08c
+};
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static cfg_type_t cfg_type_dynamic_db_opts = {
Martin Nagy b1c08c
+	"dynamically_loadable_zones_opts", cfg_parse_map,
Martin Nagy b1c08c
+	cfg_print_map, cfg_doc_map, &cfg_rep_map,
Martin Nagy b1c08c
+	dynamic_db_clausesets
Martin Nagy b1c08c
+};
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static cfg_tuplefielddef_t dynamic_db_fields[] = {
Martin Nagy b1c08c
+	{ "name", &cfg_type_astring, 0 },
Martin Nagy b1c08c
+	{ "options", &cfg_type_dynamic_db_opts, 0 },
Martin Nagy b1c08c
+	{ NULL, NULL, 0 }
Martin Nagy b1c08c
+};
Martin Nagy b1c08c
+
Martin Nagy b1c08c
+static cfg_type_t cfg_type_dynamic_db = {
Martin Nagy b1c08c
+	"dynamic_db", cfg_parse_tuple, cfg_print_tuple, cfg_doc_tuple,
Martin Nagy b1c08c
+	&cfg_rep_tuple, dynamic_db_fields
Martin Nagy b1c08c
+};
Martin Nagy b1c08c
+
Martin Nagy b1c08c
 static const char *notify_enums[] = { "explicit", "master-only", NULL };
Martin Nagy b1c08c
 static isc_result_t
Martin Nagy b1c08c
 parse_notify_type(cfg_parser_t *pctx, const cfg_type_t *type, cfg_obj_t **ret) {