mirror of
https://github.com/isc-projects/bind9.git
synced 2026-02-18 01:58:49 -05:00
279 lines
6 KiB
C
279 lines
6 KiB
C
/*
|
|
* 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 https://mozilla.org/MPL/2.0/.
|
|
*
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
* information regarding copyright ownership.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#if HAVE_CMOCKA
|
|
|
|
#include <stdarg.h>
|
|
#include <stddef.h>
|
|
#include <setjmp.h>
|
|
|
|
#include <fcntl.h>
|
|
#include <limits.h>
|
|
#include <sched.h> /* IWYU pragma: keep */
|
|
#include <stdbool.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#define UNIT_TESTING
|
|
#include <cmocka.h>
|
|
|
|
#include <isc/buffer.h>
|
|
#include <isc/print.h>
|
|
#include <isc/region.h>
|
|
#include <isc/result.h>
|
|
#include <isc/types.h>
|
|
#include <isc/util.h>
|
|
|
|
#include "isctest.h"
|
|
|
|
static int
|
|
_setup(void **state) {
|
|
isc_result_t result;
|
|
|
|
UNUSED(state);
|
|
|
|
result = isc_test_begin(NULL, true, 0);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
_teardown(void **state) {
|
|
UNUSED(state);
|
|
|
|
isc_test_end();
|
|
|
|
return (0);
|
|
}
|
|
|
|
/* reserve space in dynamic buffers */
|
|
static void
|
|
isc_buffer_reserve_test(void **state) {
|
|
isc_result_t result;
|
|
isc_buffer_t *b;
|
|
|
|
UNUSED(state);
|
|
|
|
b = NULL;
|
|
result = isc_buffer_allocate(mctx, &b, 1024);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_int_equal(b->length, 1024);
|
|
|
|
/*
|
|
* 1024 bytes should already be available, so this call does
|
|
* nothing.
|
|
*/
|
|
result = isc_buffer_reserve(&b, 1024);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 1024);
|
|
|
|
/*
|
|
* This call should grow it to 2048 bytes as only 1024 bytes are
|
|
* available in the buffer.
|
|
*/
|
|
result = isc_buffer_reserve(&b, 1025);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 2048);
|
|
|
|
/*
|
|
* 2048 bytes should already be available, so this call does
|
|
* nothing.
|
|
*/
|
|
result = isc_buffer_reserve(&b, 2000);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 2048);
|
|
|
|
/*
|
|
* This call should grow it to 4096 bytes as only 2048 bytes are
|
|
* available in the buffer.
|
|
*/
|
|
result = isc_buffer_reserve(&b, 3000);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 4096);
|
|
|
|
/* Consume some of the buffer so we can run the next test. */
|
|
isc_buffer_add(b, 4096);
|
|
|
|
/*
|
|
* This call should fail and leave buffer untouched.
|
|
*/
|
|
result = isc_buffer_reserve(&b, UINT_MAX);
|
|
assert_int_equal(result, ISC_R_NOMEMORY);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 4096);
|
|
|
|
isc_buffer_free(&b);
|
|
}
|
|
|
|
/* reallocate dynamic buffers */
|
|
static void
|
|
isc_buffer_reallocate_test(void **state) {
|
|
isc_result_t result;
|
|
isc_buffer_t *b;
|
|
|
|
UNUSED(state);
|
|
|
|
b = NULL;
|
|
result = isc_buffer_allocate(mctx, &b, 1024);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 1024);
|
|
|
|
result = isc_buffer_reallocate(&b, 512);
|
|
assert_int_equal(result, ISC_R_NOSPACE);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 1024);
|
|
|
|
result = isc_buffer_reallocate(&b, 1536);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_true(ISC_BUFFER_VALID(b));
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, 1536);
|
|
|
|
isc_buffer_free(&b);
|
|
}
|
|
|
|
/* dynamic buffer automatic reallocation */
|
|
static void
|
|
isc_buffer_dynamic_test(void **state) {
|
|
isc_result_t result;
|
|
isc_buffer_t *b;
|
|
size_t last_length = 10;
|
|
int i;
|
|
|
|
UNUSED(state);
|
|
|
|
b = NULL;
|
|
result = isc_buffer_allocate(mctx, &b, last_length);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
assert_non_null(b);
|
|
assert_int_equal(b->length, last_length);
|
|
|
|
isc_buffer_setautorealloc(b, true);
|
|
|
|
isc_buffer_putuint8(b, 1);
|
|
|
|
for (i = 0; i < 1000; i++) {
|
|
isc_buffer_putstr(b, "thisisa24charslongstring");
|
|
}
|
|
assert_true(b->length-last_length >= 1000*24);
|
|
last_length+=1000*24;
|
|
|
|
for (i = 0; i < 10000; i++) {
|
|
isc_buffer_putuint8(b, 1);
|
|
}
|
|
|
|
assert_true(b->length-last_length >= 10000*1);
|
|
last_length += 10000*1;
|
|
|
|
for (i = 0; i < 10000; i++) {
|
|
isc_buffer_putuint16(b, 1);
|
|
}
|
|
|
|
assert_true(b->length-last_length >= 10000*2);
|
|
|
|
last_length += 10000*2;
|
|
for (i = 0; i < 10000; i++) {
|
|
isc_buffer_putuint24(b, 1);
|
|
}
|
|
assert_true(b->length-last_length >= 10000*3);
|
|
|
|
last_length+=10000*3;
|
|
|
|
for (i = 0; i < 10000; i++) {
|
|
isc_buffer_putuint32(b, 1);
|
|
}
|
|
assert_true(b->length-last_length >= 10000*4);
|
|
|
|
|
|
isc_buffer_free(&b);
|
|
}
|
|
|
|
/* copy a region into a buffer */
|
|
static void
|
|
isc_buffer_copyregion_test(void **state) {
|
|
unsigned char data[] = { 0x11, 0x22, 0x33, 0x44 };
|
|
isc_buffer_t *b = NULL;
|
|
isc_result_t result;
|
|
|
|
isc_region_t r = {
|
|
.base = data,
|
|
.length = sizeof(data),
|
|
};
|
|
|
|
UNUSED(state);
|
|
|
|
result = isc_buffer_allocate(mctx, &b, sizeof(data));
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
|
|
/*
|
|
* Fill originally allocated buffer space.
|
|
*/
|
|
result = isc_buffer_copyregion(b, &r);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
|
|
/*
|
|
* Appending more data to the buffer should fail.
|
|
*/
|
|
result = isc_buffer_copyregion(b, &r);
|
|
assert_int_equal(result, ISC_R_NOSPACE);
|
|
|
|
/*
|
|
* Enable auto reallocation and retry. Appending should now succeed.
|
|
*/
|
|
isc_buffer_setautorealloc(b, true);
|
|
result = isc_buffer_copyregion(b, &r);
|
|
assert_int_equal(result, ISC_R_SUCCESS);
|
|
|
|
isc_buffer_free(&b);
|
|
}
|
|
|
|
int
|
|
main(void) {
|
|
const struct CMUnitTest tests[] = {
|
|
cmocka_unit_test_setup_teardown(isc_buffer_reserve_test,
|
|
_setup, _teardown),
|
|
cmocka_unit_test_setup_teardown(isc_buffer_reallocate_test,
|
|
_setup, _teardown),
|
|
cmocka_unit_test_setup_teardown(isc_buffer_dynamic_test,
|
|
_setup, _teardown),
|
|
cmocka_unit_test_setup_teardown(isc_buffer_copyregion_test,
|
|
_setup, _teardown),
|
|
};
|
|
|
|
return (cmocka_run_group_tests(tests, NULL, NULL));
|
|
}
|
|
|
|
#else /* HAVE_CMOCKA */
|
|
|
|
#include <stdio.h>
|
|
|
|
int
|
|
main(void) {
|
|
printf("1..0 # Skipped: cmocka not available\n");
|
|
return (0);
|
|
}
|
|
|
|
#endif
|