/* * Copyright (C) Internet Systems Consortium, Inc. ("ISC") * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * See the COPYRIGHT file distributed with this work for additional * information regarding copyright ownership. */ /* $Id: lwsearch.c,v 1.13 2007/06/19 23:46:59 tbox Exp $ */ /*! \file */ #include #include #include #include #include #include #include #include #include #include #include #define LWSEARCHLIST_MAGIC ISC_MAGIC('L', 'W', 'S', 'L') #define VALID_LWSEARCHLIST(l) ISC_MAGIC_VALID(l, LWSEARCHLIST_MAGIC) isc_result_t ns_lwsearchlist_create(isc_mem_t *mctx, ns_lwsearchlist_t **listp) { ns_lwsearchlist_t *list; isc_result_t result; REQUIRE(mctx != NULL); REQUIRE(listp != NULL && *listp == NULL); list = isc_mem_get(mctx, sizeof(ns_lwsearchlist_t)); if (list == NULL) return (ISC_R_NOMEMORY); result = isc_mutex_init(&list->lock); if (result != ISC_R_SUCCESS) { isc_mem_put(mctx, list, sizeof(ns_lwsearchlist_t)); return (result); } list->mctx = NULL; isc_mem_attach(mctx, &list->mctx); list->refs = 1; ISC_LIST_INIT(list->names); list->magic = LWSEARCHLIST_MAGIC; *listp = list; return (ISC_R_SUCCESS); } void ns_lwsearchlist_attach(ns_lwsearchlist_t *source, ns_lwsearchlist_t **target) { REQUIRE(VALID_LWSEARCHLIST(source)); REQUIRE(target != NULL && *target == NULL); LOCK(&source->lock); INSIST(source->refs > 0); source->refs++; INSIST(source->refs != 0); UNLOCK(&source->lock); *target = source; } void ns_lwsearchlist_detach(ns_lwsearchlist_t **listp) { ns_lwsearchlist_t *list; isc_mem_t *mctx; REQUIRE(listp != NULL); list = *listp; REQUIRE(VALID_LWSEARCHLIST(list)); LOCK(&list->lock); INSIST(list->refs > 0); list->refs--; UNLOCK(&list->lock); *listp = NULL; if (list->refs != 0) return; mctx = list->mctx; while (!ISC_LIST_EMPTY(list->names)) { dns_name_t *name = ISC_LIST_HEAD(list->names); ISC_LIST_UNLINK(list->names, name, link); dns_name_free(name, list->mctx); isc_mem_put(list->mctx, name, sizeof(dns_name_t)); } list->magic = 0; isc_mem_put(mctx, list, sizeof(ns_lwsearchlist_t)); isc_mem_detach(&mctx); } isc_result_t ns_lwsearchlist_append(ns_lwsearchlist_t *list, dns_name_t *name) { dns_name_t *newname; isc_result_t result; REQUIRE(VALID_LWSEARCHLIST(list)); REQUIRE(name != NULL); newname = isc_mem_get(list->mctx, sizeof(dns_name_t)); if (newname == NULL) return (ISC_R_NOMEMORY); dns_name_init(newname, NULL); result = dns_name_dup(name, list->mctx, newname); if (result != ISC_R_SUCCESS) { isc_mem_put(list->mctx, newname, sizeof(dns_name_t)); return (result); } ISC_LINK_INIT(newname, link); ISC_LIST_APPEND(list->names, newname, link); return (ISC_R_SUCCESS); } void ns_lwsearchctx_init(ns_lwsearchctx_t *sctx, ns_lwsearchlist_t *list, dns_name_t *name, unsigned int ndots) { INSIST(sctx != NULL); sctx->relname = name; sctx->searchname = NULL; sctx->doneexact = false; sctx->exactfirst = false; sctx->ndots = ndots; if (dns_name_isabsolute(name) || list == NULL) { sctx->list = NULL; return; } sctx->list = list; sctx->searchname = ISC_LIST_HEAD(sctx->list->names); if (dns_name_countlabels(name) > ndots) sctx->exactfirst = true; } void ns_lwsearchctx_first(ns_lwsearchctx_t *sctx) { REQUIRE(sctx != NULL); UNUSED(sctx); } isc_result_t ns_lwsearchctx_next(ns_lwsearchctx_t *sctx) { REQUIRE(sctx != NULL); if (sctx->list == NULL) return (ISC_R_NOMORE); if (sctx->searchname == NULL) { if (sctx->exactfirst || sctx->doneexact) return (ISC_R_NOMORE); sctx->doneexact = true; } else { if (sctx->exactfirst && !sctx->doneexact) sctx->doneexact = true; else { sctx->searchname = ISC_LIST_NEXT(sctx->searchname, link); if (sctx->searchname == NULL && sctx->doneexact) return (ISC_R_NOMORE); } } return (ISC_R_SUCCESS); } isc_result_t ns_lwsearchctx_current(ns_lwsearchctx_t *sctx, dns_name_t *absname) { dns_name_t *tname; bool useexact = false; REQUIRE(sctx != NULL); if (sctx->list == NULL || sctx->searchname == NULL || (sctx->exactfirst && !sctx->doneexact)) useexact = true; if (useexact) { if (dns_name_isabsolute(sctx->relname)) tname = NULL; else tname = dns_rootname; } else tname = sctx->searchname; return (dns_name_concatenate(sctx->relname, tname, absname, NULL)); }