2012-06-20 14:13:12 -05:00
|
|
|
/*
|
2013-03-19 23:46:08 +00:00
|
|
|
* Copyright (C) 2012, 2013 Internet Systems Consortium, Inc. ("ISC")
|
2012-06-20 14:13:12 -05:00
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and/or 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 ISC DISCLAIMS ALL WARRANTIES WITH
|
|
|
|
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
|
|
* AND FITNESS. IN NO EVENT SHALL ISC 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* $Id: rbt_test.c,v 1.1.14.8 2012/02/10 16:24:37 ckb Exp $ */
|
|
|
|
|
|
|
|
/* ! \file */
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <atf-c.h>
|
|
|
|
#include <isc/mem.h>
|
|
|
|
#include <isc/string.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
|
2013-03-18 17:13:48 -07:00
|
|
|
#ifdef HAVE_INTTYPES_H
|
|
|
|
#include <inttypes.h> /* uintptr_t */
|
|
|
|
#endif
|
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
#include <dns/rbt.h>
|
|
|
|
#include <dns/fixedname.h>
|
|
|
|
#include <dns/result.h>
|
|
|
|
#include <dns/compress.h>
|
2012-11-17 09:33:46 +00:00
|
|
|
#include "dnstest.h"
|
2012-06-20 14:13:12 -05:00
|
|
|
|
|
|
|
#include <isc/app.h>
|
|
|
|
#include <isc/buffer.h>
|
|
|
|
#include <isc/entropy.h>
|
|
|
|
#include <isc/file.h>
|
|
|
|
#include <isc/hash.h>
|
|
|
|
#include <isc/mem.h>
|
|
|
|
#include <isc/os.h>
|
|
|
|
#include <isc/string.h>
|
|
|
|
#include <isc/socket.h>
|
|
|
|
#include <isc/task.h>
|
|
|
|
#include <isc/timer.h>
|
|
|
|
#include <isc/util.h>
|
|
|
|
|
|
|
|
#include <dns/log.h>
|
|
|
|
#include <dns/name.h>
|
|
|
|
#include <dns/result.h>
|
|
|
|
|
|
|
|
#include <dst/dst.h>
|
|
|
|
|
|
|
|
typedef struct data_holder {
|
2012-06-21 23:46:36 +00:00
|
|
|
int len;
|
2012-06-20 14:13:12 -05:00
|
|
|
const char *data;
|
|
|
|
} data_holder_t;
|
|
|
|
|
|
|
|
typedef struct rbt_testdata {
|
|
|
|
const char *name;
|
|
|
|
size_t name_len;
|
|
|
|
data_holder_t data;
|
|
|
|
} rbt_testdata_t;
|
|
|
|
|
|
|
|
#define DATA_ITEM(name) { (name), sizeof(name) - 1, { sizeof(name), (name) } }
|
|
|
|
|
2012-06-21 23:46:36 +00:00
|
|
|
rbt_testdata_t testdata[] = {
|
2012-06-20 14:13:12 -05:00
|
|
|
DATA_ITEM("first.com."),
|
|
|
|
DATA_ITEM("one.net."),
|
|
|
|
DATA_ITEM("two.com."),
|
|
|
|
DATA_ITEM("three.org."),
|
|
|
|
DATA_ITEM("asdf.com."),
|
|
|
|
DATA_ITEM("ghjkl.com."),
|
|
|
|
DATA_ITEM("1.edu."),
|
|
|
|
DATA_ITEM("2.edu."),
|
|
|
|
DATA_ITEM("3.edu."),
|
|
|
|
DATA_ITEM("123.edu."),
|
|
|
|
DATA_ITEM("1236.com."),
|
|
|
|
DATA_ITEM("and_so_forth.com."),
|
|
|
|
DATA_ITEM("thisisalongname.com."),
|
|
|
|
DATA_ITEM("a.b."),
|
|
|
|
DATA_ITEM("test.net."),
|
|
|
|
DATA_ITEM("whoknows.org."),
|
|
|
|
DATA_ITEM("blargh.com."),
|
|
|
|
DATA_ITEM("www.joe.com."),
|
|
|
|
DATA_ITEM("test.com."),
|
|
|
|
DATA_ITEM("isc.org."),
|
|
|
|
DATA_ITEM("uiop.mil."),
|
|
|
|
DATA_ITEM("last.fm."),
|
|
|
|
{ NULL, 0, { 0, NULL } }
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
delete_data(void *data, void *arg) {
|
|
|
|
UNUSED(arg);
|
|
|
|
UNUSED(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static isc_result_t
|
|
|
|
write_data(FILE *file, unsigned char *datap, isc_uint32_t serial) {
|
2012-06-21 23:46:36 +00:00
|
|
|
size_t ret = 0;
|
|
|
|
data_holder_t *data = (data_holder_t *)datap;
|
2012-06-20 14:13:12 -05:00
|
|
|
data_holder_t temp;
|
2012-06-23 18:06:33 +10:00
|
|
|
uintptr_t where = ftell(file);
|
2012-06-20 14:13:12 -05:00
|
|
|
|
|
|
|
UNUSED(serial);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
REQUIRE(data != NULL);
|
|
|
|
REQUIRE((data->len == 0 && data->data == NULL) ||
|
|
|
|
(data->len != 0 && data->data != NULL));
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
temp = *data;
|
2012-06-21 23:46:36 +00:00
|
|
|
temp.data = (data->len == 0
|
2012-06-20 14:13:12 -05:00
|
|
|
? NULL
|
|
|
|
: (char *)(where + sizeof(data_holder_t)));
|
|
|
|
|
2012-06-21 23:46:36 +00:00
|
|
|
ret = fwrite(&temp, sizeof(data_holder_t), 1, file);
|
2012-06-23 18:21:52 +10:00
|
|
|
if (ret != 1)
|
|
|
|
return (ISC_R_FAILURE);
|
|
|
|
if (data->len > 0) {
|
2012-06-20 14:13:12 -05:00
|
|
|
ret = fwrite(data->data, data->len, 1, file);
|
2012-06-23 18:21:52 +10:00
|
|
|
if (ret != 1)
|
|
|
|
return (ISC_R_FAILURE);
|
|
|
|
}
|
2012-06-20 14:13:12 -05:00
|
|
|
|
2012-06-21 23:46:36 +00:00
|
|
|
return (ISC_R_SUCCESS);
|
2012-06-20 14:13:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
fix_data(dns_rbtnode_t *p) {
|
2012-06-21 23:46:36 +00:00
|
|
|
data_holder_t *data = p->data;
|
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
REQUIRE(data != NULL);
|
|
|
|
REQUIRE((data->len == 0 && data->data == NULL) ||
|
|
|
|
(data->len != 0 && data->data != NULL));
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
printf("fixing data: len %d, data %p\n", data->len, data->data);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
|
|
|
data->data = (data->len == 0)
|
2012-06-20 14:13:12 -05:00
|
|
|
? NULL
|
|
|
|
: (char *)data + sizeof(data_holder_t);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Load test data into the RBT.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
add_test_data(isc_mem_t *mctx, dns_rbt_t *rbt)
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
isc_buffer_t b;
|
|
|
|
isc_result_t result;
|
|
|
|
dns_fixedname_t fname;
|
|
|
|
dns_name_t *name;
|
|
|
|
dns_compress_t cctx;
|
2012-06-21 23:46:36 +00:00
|
|
|
rbt_testdata_t *testdatap = testdata;
|
2012-06-20 14:13:12 -05:00
|
|
|
|
|
|
|
dns_compress_init(&cctx, -1, mctx);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
|
|
|
while (testdatap->name != NULL && testdatap->data.data != NULL) {
|
2012-06-20 14:13:12 -05:00
|
|
|
memcpy(buffer, testdatap->name, testdatap->name_len);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
isc_buffer_init(&b, buffer, testdatap->name_len);
|
|
|
|
isc_buffer_add(&b, testdatap->name_len);
|
|
|
|
dns_fixedname_init(&fname);
|
|
|
|
name = dns_fixedname_name(&fname);
|
|
|
|
result = dns_name_fromtext(name, &b, dns_rootname, 0, NULL);
|
|
|
|
if (result != ISC_R_SUCCESS) {
|
|
|
|
testdatap++;
|
|
|
|
continue;
|
|
|
|
}
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
if (name != NULL) {
|
|
|
|
result = dns_rbt_addname(rbt, name, &testdatap->data);
|
|
|
|
ATF_CHECK_STREQ(dns_result_totext(result), "success");
|
|
|
|
}
|
|
|
|
testdatap++;
|
|
|
|
}
|
|
|
|
|
2012-06-21 23:46:36 +00:00
|
|
|
dns_compress_invalidate(&cctx);
|
2012-06-20 14:13:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Walk the tree and ensure that all the test nodes are present.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
check_test_data(dns_rbt_t *rbt)
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
char *arg;
|
|
|
|
dns_fixedname_t fname;
|
|
|
|
dns_fixedname_t fixed;
|
2012-06-21 23:46:36 +00:00
|
|
|
dns_name_t *name;
|
|
|
|
isc_buffer_t b;
|
|
|
|
data_holder_t *data;
|
|
|
|
isc_result_t result;
|
2012-06-20 14:13:12 -05:00
|
|
|
dns_name_t *foundname;
|
|
|
|
rbt_testdata_t *testdatap = testdata;
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
dns_fixedname_init(&fixed);
|
|
|
|
foundname = dns_fixedname_name(&fixed);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
|
|
|
while (testdatap->name != NULL && testdatap->data.data != NULL) {
|
2012-06-20 14:13:12 -05:00
|
|
|
memcpy(buffer, testdatap->name, testdatap->name_len + 1);
|
|
|
|
arg = buffer;
|
|
|
|
|
|
|
|
isc_buffer_init(&b, arg, testdatap->name_len);
|
|
|
|
isc_buffer_add(&b, testdatap->name_len);
|
|
|
|
dns_fixedname_init(&fname);
|
|
|
|
name = dns_fixedname_name(&fname);
|
|
|
|
result = dns_name_fromtext(name, &b, dns_rootname, 0, NULL);
|
|
|
|
if (result != ISC_R_SUCCESS) {
|
|
|
|
testdatap++;
|
|
|
|
continue;
|
|
|
|
}
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
data = NULL;
|
|
|
|
result = dns_rbt_findname(rbt, name, 0, foundname,
|
|
|
|
(void *) &data);
|
|
|
|
ATF_CHECK_STREQ(dns_result_totext(result), "success");
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
testdatap++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
data_printer(FILE *out, void *datap)
|
|
|
|
{
|
|
|
|
data_holder_t *data = (data_holder_t *)datap;
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
fprintf(out, "%d bytes, %s", data->len, data->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(isc_rbt);
|
|
|
|
ATF_TC_HEAD(isc_rbt, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "Test the creation of an rbt");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(isc_rbt, tc) {
|
|
|
|
dns_rbt_t *rbt = NULL;
|
|
|
|
isc_result_t result;
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
UNUSED(tc);
|
|
|
|
|
|
|
|
isc_mem_debugging = ISC_MEM_DEBUGRECORD;
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
result = dns_test_begin(NULL, ISC_TRUE);
|
|
|
|
ATF_CHECK_STREQ(dns_result_totext(result), "success");
|
|
|
|
result = dns_rbt_create(mctx, delete_data, NULL, &rbt);
|
|
|
|
ATF_CHECK_STREQ(dns_result_totext(result), "success");
|
|
|
|
|
|
|
|
add_test_data(mctx, rbt);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
check_test_data(rbt);
|
|
|
|
|
|
|
|
dns_rbt_printall(rbt, data_printer);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
dns_rbt_destroy(&rbt);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
dns_test_end();
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(isc_serialize_rbt);
|
|
|
|
ATF_TC_HEAD(isc_serialize_rbt, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "Test writing an rbt to file");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(isc_serialize_rbt, tc) {
|
|
|
|
dns_rbt_t *rbt = NULL;
|
|
|
|
isc_result_t result;
|
|
|
|
FILE *rbtfile = NULL;
|
|
|
|
dns_rbt_t *rbt_deserialized = NULL;
|
2012-06-23 18:02:13 +10:00
|
|
|
long offset;
|
2012-06-20 14:13:12 -05:00
|
|
|
int fd;
|
|
|
|
off_t filesize = 0;
|
|
|
|
char *base;
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
UNUSED(tc);
|
|
|
|
|
|
|
|
isc_mem_debugging = ISC_MEM_DEBUGRECORD;
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
result = dns_test_begin(NULL, ISC_TRUE);
|
|
|
|
ATF_CHECK_STREQ(dns_result_totext(result), "success");
|
|
|
|
result = dns_rbt_create(mctx, delete_data, NULL, &rbt);
|
|
|
|
ATF_CHECK_STREQ(dns_result_totext(result), "success");
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
add_test_data(mctx, rbt);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
dns_rbt_printall(rbt, data_printer);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Serialize the tree.
|
|
|
|
*/
|
|
|
|
printf("serialization begins.\n");
|
|
|
|
rbtfile = fopen("./zone.bin", "w+b");
|
|
|
|
ATF_REQUIRE(rbtfile != NULL);
|
2012-06-23 18:02:13 +10:00
|
|
|
result = dns_rbt_serialize_tree(rbtfile, rbt, write_data, 0, &offset);
|
2012-06-20 14:13:12 -05:00
|
|
|
ATF_REQUIRE(result == ISC_R_SUCCESS);
|
|
|
|
dns_rbt_destroy(&rbt);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
/*
|
|
|
|
* Deserialize the tree
|
|
|
|
*/
|
|
|
|
printf("deserialization begins.\n");
|
|
|
|
|
2013-03-26 16:41:20 +11:00
|
|
|
#ifndef MAP_FILE
|
|
|
|
#define MAP_FILE 0
|
|
|
|
#endif
|
2012-06-20 14:13:12 -05:00
|
|
|
/*
|
|
|
|
* Map in the whole file in one go
|
|
|
|
*/
|
|
|
|
fd = open("zone.bin", O_RDWR);
|
|
|
|
isc_file_getsizefd(fd, &filesize);
|
|
|
|
base = mmap(NULL, filesize,
|
|
|
|
PROT_READ|PROT_WRITE,
|
|
|
|
MAP_FILE|MAP_PRIVATE, fd, 0);
|
|
|
|
ATF_REQUIRE(base != NULL && base != MAP_FAILED);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
result = dns_rbt_deserialize_tree(base, 0, mctx, delete_data, NULL,
|
|
|
|
fix_data, NULL, &rbt_deserialized);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
/* Test to make sure we have a valid tree */
|
|
|
|
ATF_REQUIRE(result == ISC_R_SUCCESS);
|
|
|
|
if (rbt_deserialized == NULL)
|
2012-06-21 23:46:36 +00:00
|
|
|
atf_tc_fail("deserialized rbt is null!"); /* Abort execution. */
|
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
check_test_data(rbt_deserialized);
|
|
|
|
|
|
|
|
dns_rbt_printall(rbt_deserialized, data_printer);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
dns_rbt_destroy(&rbt_deserialized);
|
|
|
|
munmap(base, filesize);
|
|
|
|
unlink("zone.bin");
|
|
|
|
dns_test_end();
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(dns_rbt_serialize_align);
|
|
|
|
ATF_TC_HEAD(dns_rbt_serialize_align, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "Test the dns_rbt_serialize_align() function.");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(dns_rbt_serialize_align, tc) {
|
|
|
|
UNUSED(tc);
|
2012-06-21 23:46:36 +00:00
|
|
|
|
2012-06-20 14:13:12 -05:00
|
|
|
ATF_CHECK(dns_rbt_serialize_align(0) == 0);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(1) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(2) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(3) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(4) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(5) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(6) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(7) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(8) == 8);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(9) == 16);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(0xff) == 0x100);
|
|
|
|
ATF_CHECK(dns_rbt_serialize_align(0x301) == 0x308);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Main
|
|
|
|
*/
|
|
|
|
ATF_TP_ADD_TCS(tp) {
|
|
|
|
ATF_TP_ADD_TC(tp, isc_rbt);
|
|
|
|
ATF_TP_ADD_TC(tp, isc_serialize_rbt);
|
|
|
|
ATF_TP_ADD_TC(tp, dns_rbt_serialize_align);
|
|
|
|
|
|
|
|
return (atf_no_error());
|
|
|
|
}
|