/** * * /brief getdns list management functions * * This is the meat of the API * Originally taken from the getdns API description pseudo implementation. * */ /* * Copyright (c) 2013, NLnet Labs, Verisign, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * Neither the names of the copyright holders nor the * names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL Verisign, Inc. BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include "types-internal.h" #include "util-internal.h" #include "list.h" /*---------------------------------------- getdns_list_get_length */ getdns_return_t getdns_list_get_length(const struct getdns_list * list, size_t * answer) { if (!list || !answer) return GETDNS_RETURN_INVALID_PARAMETER; *answer = list->numinuse; return GETDNS_RETURN_GOOD;; } /* getdns_list_get_length */ /*---------------------------------------- getdns_list_get_data_type */ getdns_return_t getdns_list_get_data_type(const struct getdns_list * list, size_t index, getdns_data_type * answer) { if (!list || !answer) return GETDNS_RETURN_INVALID_PARAMETER; if (index >= list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; *answer = list->items[index].dtype; return GETDNS_RETURN_GOOD; } /* getdns_list_get_data_type */ /*---------------------------------------- getdns_list_get_dict */ getdns_return_t getdns_list_get_dict(const struct getdns_list * list, size_t index, struct getdns_dict ** answer) { if (!list || !answer) return GETDNS_RETURN_INVALID_PARAMETER; if (index >= list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; if (list->items[index].dtype != t_dict) return GETDNS_RETURN_WRONG_TYPE_REQUESTED; *answer = list->items[index].data.dict; return GETDNS_RETURN_GOOD; } /* getdns_list_get_dict */ /*---------------------------------------- getdns_list_get_list */ getdns_return_t getdns_list_get_list(const struct getdns_list * list, size_t index, struct getdns_list ** answer) { if (!list || !answer) return GETDNS_RETURN_INVALID_PARAMETER; if (index >= list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; if (list->items[index].dtype != t_list) return GETDNS_RETURN_WRONG_TYPE_REQUESTED; *answer = list->items[index].data.list; return GETDNS_RETURN_GOOD; } /* getdns_list_get_list */ /*---------------------------------------- getdns_list_get_bindata */ getdns_return_t getdns_list_get_bindata(const struct getdns_list * list, size_t index, struct getdns_bindata ** answer) { if (!list || !answer) return GETDNS_RETURN_INVALID_PARAMETER; if (index >= list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; if (list->items[index].dtype != t_bindata) return GETDNS_RETURN_WRONG_TYPE_REQUESTED; *answer = list->items[index].data.bindata; return GETDNS_RETURN_GOOD; } /* getdns_list_get_bindata */ /*---------------------------------------- getdns_list_get_int */ getdns_return_t getdns_list_get_int(const struct getdns_list * list, size_t index, uint32_t * answer) { if (!list || !answer) return GETDNS_RETURN_INVALID_PARAMETER; if (index >= list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; if (list->items[index].dtype != t_int) return GETDNS_RETURN_WRONG_TYPE_REQUESTED; *answer = list->items[index].data.n; return GETDNS_RETURN_GOOD; } /* getdns_list_get_int */ /*---------------------------------------- getdns_list_realloc */ /** * private function (API users should not be calling this) * allocates a block of items, should be called when a list needs to grow * preserves the existing items * in case of an error the list should be considered unusable * @return GETDNS_RETURN_GOOD on success, GETDNS_RETURN_GENERIC_ERROR if out of memory */ getdns_return_t getdns_list_realloc(struct getdns_list *list) { struct getdns_list_item *newlist; if (!list) return GETDNS_RETURN_INVALID_PARAMETER; newlist = GETDNS_XREALLOC(list->mf, list->items, struct getdns_list_item, list->numalloc + GETDNS_LIST_BLOCKSZ); if (!newlist) return GETDNS_RETURN_GENERIC_ERROR; list->items = newlist; list->numalloc += GETDNS_LIST_BLOCKSZ; return GETDNS_RETURN_GOOD; } /* getdns_list_realloc */ /*---------------------------------------- getdns_list_copy */ getdns_return_t getdns_list_copy(const struct getdns_list * srclist, struct getdns_list ** dstlist) { int i; size_t index; getdns_return_t retval; if (!dstlist) return GETDNS_RETURN_INVALID_PARAMETER; if (!srclist) { *dstlist = NULL; return GETDNS_RETURN_GOOD; } *dstlist = getdns_list_create_with_extended_memory_functions( srclist->mf.mf_arg, srclist->mf.mf.ext.malloc, srclist->mf.mf.ext.realloc, srclist->mf.mf.ext.free ); if (!dstlist) return GETDNS_RETURN_GENERIC_ERROR; for (i = 0; i < srclist->numinuse; i++) { retval = getdns_list_add_item(*dstlist, &index); if (retval != GETDNS_RETURN_GOOD) { getdns_list_destroy(*dstlist); *dstlist = NULL; return retval; } switch (srclist->items[i].dtype) { case t_int: retval = getdns_list_set_int(*dstlist, index, srclist->items[i].data.n); break; case t_list: retval =getdns_list_set_list(*dstlist, index, srclist->items[i].data.list); break; case t_bindata: retval = getdns_list_set_bindata(*dstlist, index, srclist->items[i].data.bindata); break; case t_dict: retval = getdns_list_set_dict(*dstlist, index, srclist->items[i].data.dict); break; } if (retval != GETDNS_RETURN_GOOD) { getdns_list_destroy(*dstlist); *dstlist = NULL; return retval; } } return GETDNS_RETURN_GOOD; } /* getdns_list_copy */ struct getdns_list * getdns_list_create_with_extended_memory_functions( void *userarg, void *(*malloc)(void *userarg, size_t), void *(*realloc)(void *userarg, void *, size_t), void (*free)(void *userarg, void *)) { struct getdns_list *list; mf_union mf; if (!malloc || !realloc || !free) return NULL; mf.ext.malloc = malloc; list = userarg == MF_PLAIN ? (struct getdns_list *)(*mf.pln.malloc)( sizeof(struct getdns_list)) : (struct getdns_list *)(*mf.ext.malloc)(userarg, sizeof(struct getdns_list)); if (!list) return NULL; list->mf.mf_arg = userarg; list->mf.mf.ext.malloc = malloc; list->mf.mf.ext.realloc = realloc; list->mf.mf.ext.free = free; list->numalloc = 0; list->numinuse = 0; list->items = NULL; if (getdns_list_realloc(list) != GETDNS_RETURN_GOOD) { getdns_list_destroy(list); return NULL; } return list; } struct getdns_list * getdns_list_create_with_memory_functions(void *(*malloc)(size_t), void *(*realloc)(void *, size_t), void (*free)(void *)) { mf_union mf; mf.pln.malloc = malloc; mf.pln.realloc = realloc; mf.pln.free = free; return getdns_list_create_with_extended_memory_functions( MF_PLAIN, mf.ext.malloc, mf.ext.realloc, mf.ext.free); } /*-------------------------- getdns_list_create_with_context */ struct getdns_list * getdns_list_create_with_context(struct getdns_context *context) { if (context) return getdns_list_create_with_extended_memory_functions( context->mf.mf_arg, context->mf.mf.ext.malloc, context->mf.mf.ext.realloc, context->mf.mf.ext.free ); else return getdns_list_create_with_memory_functions(malloc, realloc, free); } /* getdns_list_create_with_context */ /*---------------------------------------- getdns_list_create */ struct getdns_list * getdns_list_create() { return getdns_list_create_with_context(NULL); } /* getdns_list_create */ static void getdns_list_destroy_item(struct getdns_list *list, size_t index) { switch (list->items[index].dtype) { case t_dict: getdns_dict_destroy(list->items[index].data.dict); break; case t_list: getdns_list_destroy(list->items[index].data.list); break; case t_bindata: getdns_bindata_destroy(&list->mf, list->items[index].data.bindata); break; default: break; } } /*---------------------------------------- getdns_list_destroy */ void getdns_list_destroy(struct getdns_list *list) { size_t i; if (!list) return; for (i = 0; i < list->numinuse; i++) getdns_list_destroy_item(list, i); if (list->items) GETDNS_FREE(list->mf, list->items); GETDNS_FREE(list->mf, list); } /* getdns_list_destroy */ /*---------------------------------------- getdns_list_add_item */ getdns_return_t getdns_list_add_item(struct getdns_list *list, size_t * index) { getdns_return_t retval; if (!list || !index) return GETDNS_RETURN_INVALID_PARAMETER; if (list->numalloc == list->numinuse) { retval = getdns_list_realloc(list); if (retval != GETDNS_RETURN_GOOD) return retval; } *index = list->numinuse; list->items[*index].dtype = t_int; list->items[*index].data.n = 0; list->numinuse++; return GETDNS_RETURN_GOOD; } /* getdns_list_add_item */ /*---------------------------------------- getdns_list_set_dict */ getdns_return_t getdns_list_set_dict(struct getdns_list * list, size_t index, const struct getdns_dict * child_dict) { struct getdns_dict *newdict; getdns_return_t retval; if (!list || !child_dict) return GETDNS_RETURN_INVALID_PARAMETER; if (index > list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; retval = getdns_dict_copy(child_dict, &newdict); if (retval != GETDNS_RETURN_GOOD) return retval; if (index == list->numinuse) { retval = getdns_list_add_item(list, &index); if (retval != GETDNS_RETURN_GOOD) { getdns_dict_destroy(newdict); return retval; } } else getdns_list_destroy_item(list, index); list->items[index].dtype = t_dict; list->items[index].data.dict = newdict; return GETDNS_RETURN_GOOD; } /* getdns_list_set_dict */ /*---------------------------------------- getdns_list_set_list */ getdns_return_t getdns_list_set_list(struct getdns_list * list, size_t index, const struct getdns_list * child_list) { struct getdns_list *newlist; getdns_return_t retval; if (!list || !child_list) return GETDNS_RETURN_INVALID_PARAMETER; if (index > list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; retval = getdns_list_copy(child_list, &newlist); if (retval != GETDNS_RETURN_GOOD) return retval; if (index == list->numinuse) { retval = getdns_list_add_item(list, &index); if (retval != GETDNS_RETURN_GOOD) { getdns_list_destroy(newlist); return retval; } } else getdns_list_destroy_item(list, index); list->items[index].dtype = t_list; list->items[index].data.list = newlist; return GETDNS_RETURN_GOOD; } /* getdns_list_set_list */ /*---------------------------------------- getdns_list_set_bindata */ getdns_return_t getdns_list_set_bindata(struct getdns_list * list, size_t index, const struct getdns_bindata * child_bindata) { struct getdns_bindata *newbindata; getdns_return_t retval; if (!list || !child_bindata) return GETDNS_RETURN_INVALID_PARAMETER; if (index > list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; newbindata = getdns_bindata_copy(&list->mf, child_bindata); if (!newbindata) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; if (index == list->numinuse) { retval = getdns_list_add_item(list, &index); if (retval != GETDNS_RETURN_GOOD) { getdns_bindata_destroy(&list->mf, newbindata); return retval; } } else getdns_list_destroy_item(list, index); list->items[index].dtype = t_bindata; list->items[index].data.bindata = newbindata; return GETDNS_RETURN_GOOD; } /* getdns_list_set_bindata */ /*---------------------------------------- getdns_list_set_int */ getdns_return_t getdns_list_set_int(struct getdns_list * list, size_t index, uint32_t child_int) { getdns_return_t retval; if (!list) return GETDNS_RETURN_INVALID_PARAMETER; if (index > list->numinuse) return GETDNS_RETURN_NO_SUCH_LIST_ITEM; if (index == list->numinuse) { retval = getdns_list_add_item(list, &index); if (retval != GETDNS_RETURN_GOOD) return retval; } else getdns_list_destroy_item(list, index); list->items[index].dtype = t_int; list->items[index].data.n = child_int; return GETDNS_RETURN_GOOD; } /* getdns_list_set_int */ /* getdns_list.c */