mirror of https://github.com/getdnsapi/getdns.git
485 lines
13 KiB
C
485 lines
13 KiB
C
/**
|
|
*
|
|
* /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 <string.h>
|
|
#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 */
|
|
|