2015-01-22 09:44:24 -08:00
|
|
|
/*
|
2017-08-24 23:47:03 +00:00
|
|
|
* Copyright (C) 2015-2017 Internet Systems Consortium, Inc. ("ISC")
|
2015-01-22 09:44:24 -08:00
|
|
|
*
|
2016-06-27 14:56:38 +10:00
|
|
|
* 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/.
|
2015-01-22 09:44:24 -08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
|
|
|
#include <atf-c.h>
|
|
|
|
|
|
|
|
#include "isctest.h"
|
|
|
|
|
2017-08-24 10:58:20 +05:30
|
|
|
#include <isc/file.h>
|
2015-01-22 09:44:24 -08:00
|
|
|
#include <isc/mem.h>
|
2015-05-23 14:21:51 +02:00
|
|
|
#include <isc/print.h>
|
2015-01-22 09:44:24 -08:00
|
|
|
#include <isc/result.h>
|
2017-08-24 10:58:20 +05:30
|
|
|
#include <isc/stdio.h>
|
2015-01-22 09:44:24 -08:00
|
|
|
|
|
|
|
static void *
|
|
|
|
default_memalloc(void *arg, size_t size) {
|
|
|
|
UNUSED(arg);
|
|
|
|
if (size == 0U)
|
|
|
|
size = 1;
|
|
|
|
return (malloc(size));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
default_memfree(void *arg, void *ptr) {
|
|
|
|
UNUSED(arg);
|
|
|
|
free(ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(isc_mem_total);
|
|
|
|
ATF_TC_HEAD(isc_mem_total, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "test TotalUse calculation");
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC_BODY(isc_mem_total, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
isc_mem_t *mctx2 = NULL;
|
|
|
|
size_t before, after;
|
|
|
|
ssize_t diff;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
result = isc_test_begin(NULL, ISC_TRUE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
/* Local alloc, free */
|
|
|
|
mctx2 = NULL;
|
|
|
|
result = isc_mem_createx2(0, 0, default_memalloc, default_memfree,
|
|
|
|
NULL, &mctx2, 0);
|
|
|
|
if (result != ISC_R_SUCCESS)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
before = isc_mem_total(mctx2);
|
|
|
|
|
|
|
|
for (i = 0; i < 100000; i++) {
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
ptr = isc_mem_allocate(mctx2, 2048);
|
|
|
|
isc_mem_free(mctx2, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
after = isc_mem_total(mctx2);
|
|
|
|
diff = after - before;
|
|
|
|
|
|
|
|
printf("total_before=%lu, total_after=%lu, total_diff=%lu\n",
|
2015-02-05 07:50:24 +11:00
|
|
|
(unsigned long)before, (unsigned long)after,
|
|
|
|
(unsigned long)diff);
|
2015-01-22 09:44:24 -08:00
|
|
|
/* 2048 +8 bytes extra for size_info */
|
|
|
|
ATF_CHECK_EQ(diff, (2048 + 8) * 100000);
|
|
|
|
|
|
|
|
/* ISC_MEMFLAG_INTERNAL */
|
|
|
|
|
|
|
|
before = isc_mem_total(mctx);
|
|
|
|
|
|
|
|
for (i = 0; i < 100000; i++) {
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
ptr = isc_mem_allocate(mctx, 2048);
|
|
|
|
isc_mem_free(mctx, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
after = isc_mem_total(mctx);
|
|
|
|
diff = after - before;
|
|
|
|
|
|
|
|
printf("total_before=%lu, total_after=%lu, total_diff=%lu\n",
|
2015-02-05 07:50:24 +11:00
|
|
|
(unsigned long)before, (unsigned long)after,
|
|
|
|
(unsigned long)diff);
|
2015-01-22 09:44:24 -08:00
|
|
|
/* 2048 +8 bytes extra for size_info */
|
|
|
|
ATF_CHECK_EQ(diff, (2048 + 8) * 100000);
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (mctx2 != NULL)
|
|
|
|
isc_mem_destroy(&mctx2);
|
|
|
|
|
|
|
|
isc_test_end();
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(isc_mem_inuse);
|
|
|
|
ATF_TC_HEAD(isc_mem_inuse, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "test InUse calculation");
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC_BODY(isc_mem_inuse, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
isc_mem_t *mctx2 = NULL;
|
|
|
|
size_t before, during, after;
|
|
|
|
ssize_t diff;
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
result = isc_test_begin(NULL, ISC_TRUE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
mctx2 = NULL;
|
|
|
|
result = isc_mem_createx2(0, 0, default_memalloc, default_memfree,
|
|
|
|
NULL, &mctx2, 0);
|
|
|
|
if (result != ISC_R_SUCCESS)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
before = isc_mem_inuse(mctx2);
|
|
|
|
ptr = isc_mem_allocate(mctx2, 1024000);
|
|
|
|
during = isc_mem_inuse(mctx2);
|
|
|
|
isc_mem_free(mctx2, ptr);
|
|
|
|
after = isc_mem_inuse(mctx2);
|
|
|
|
|
|
|
|
diff = after - before;
|
|
|
|
|
|
|
|
printf("inuse_before=%lu, inuse_during=%lu, inuse_after=%lu\n",
|
2015-02-05 07:50:24 +11:00
|
|
|
(unsigned long)before, (unsigned long)during,
|
|
|
|
(unsigned long)after);
|
2015-01-22 09:44:24 -08:00
|
|
|
ATF_REQUIRE_EQ(diff, 0);
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (mctx2 != NULL)
|
|
|
|
isc_mem_destroy(&mctx2);
|
|
|
|
|
|
|
|
isc_test_end();
|
|
|
|
}
|
|
|
|
|
2017-08-24 10:58:20 +05:30
|
|
|
#if ISC_MEM_TRACKLINES
|
|
|
|
ATF_TC(isc_mem_noflags);
|
|
|
|
ATF_TC_HEAD(isc_mem_noflags, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "test mem with no flags");
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC_BODY(isc_mem_noflags, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
isc_mem_t *mctx2 = NULL;
|
|
|
|
char buf[4096], *p, *q;
|
|
|
|
FILE *f;
|
|
|
|
void *ptr;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
result = isc_stdio_open("mem.output", "w", &f);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = isc_test_begin(NULL, ISC_TRUE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = isc_mem_createx2(0, 0, default_memalloc, default_memfree,
|
|
|
|
NULL, &mctx2, 0);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
isc_mem_debugging = 0;
|
|
|
|
ptr = isc_mem_get(mctx2, 2048);
|
|
|
|
ATF_CHECK(ptr != NULL);
|
|
|
|
isc__mem_printactive(mctx2, f);
|
|
|
|
isc_mem_put(mctx2, ptr, 2048);
|
|
|
|
isc_mem_destroy(&mctx2);
|
|
|
|
isc_stdio_close(f);
|
|
|
|
|
|
|
|
result = isc_stdio_open("mem.output", "r", &f);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
result = isc_stdio_read(buf, sizeof(buf), 1, f, &size);
|
2017-09-27 16:19:07 +10:00
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_EOF);
|
2017-08-24 10:58:20 +05:30
|
|
|
isc_stdio_close(f);
|
|
|
|
isc_file_remove("mem.output");
|
|
|
|
|
|
|
|
p = strchr(buf, '\n');
|
|
|
|
p += 2;
|
|
|
|
q = strchr(p, '\n');
|
|
|
|
*q = '\0';
|
|
|
|
ATF_CHECK_STREQ(p, "None.");
|
|
|
|
|
|
|
|
isc_mem_debugging = ISC_MEM_DEBUGRECORD;
|
|
|
|
isc_test_end();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(isc_mem_recordflag);
|
|
|
|
ATF_TC_HEAD(isc_mem_recordflag, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "test mem with record flag");
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC_BODY(isc_mem_recordflag, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
isc_mem_t *mctx2 = NULL;
|
|
|
|
char buf[4096], *p;
|
|
|
|
FILE *f;
|
|
|
|
void *ptr;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
result = isc_stdio_open("mem.output", "w", &f);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = isc_test_begin(NULL, ISC_FALSE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = isc_mem_createx2(0, 0, default_memalloc, default_memfree,
|
|
|
|
NULL, &mctx2, 0);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
ptr = isc_mem_get(mctx2, 2048);
|
|
|
|
ATF_CHECK(ptr != NULL);
|
|
|
|
isc__mem_printactive(mctx2, f);
|
|
|
|
isc_mem_put(mctx2, ptr, 2048);
|
|
|
|
isc_mem_destroy(&mctx2);
|
|
|
|
isc_stdio_close(f);
|
|
|
|
|
2017-08-30 19:02:52 -07:00
|
|
|
memset(buf, 0, sizeof(buf));
|
2017-08-24 10:58:20 +05:30
|
|
|
result = isc_stdio_open("mem.output", "r", &f);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
result = isc_stdio_read(buf, sizeof(buf), 1, f, &size);
|
2017-09-27 16:19:07 +10:00
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_EOF);
|
2017-08-24 10:58:20 +05:30
|
|
|
isc_stdio_close(f);
|
|
|
|
isc_file_remove("mem.output");
|
|
|
|
|
|
|
|
p = strchr(buf, '\n');
|
|
|
|
ATF_CHECK(strncmp(p + 2, "ptr ", 4) == 0);
|
|
|
|
p = strchr(p + 1, '\n');
|
|
|
|
ATF_CHECK(strlen(p) == 1);
|
|
|
|
|
|
|
|
isc_test_end();
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(isc_mem_traceflag);
|
|
|
|
ATF_TC_HEAD(isc_mem_traceflag, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "test mem with trace flag");
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC_BODY(isc_mem_traceflag, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
isc_mem_t *mctx2 = NULL;
|
|
|
|
char buf[4096], *p;
|
|
|
|
FILE *f;
|
|
|
|
void *ptr;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
/* redirect stderr so we can check trace output */
|
|
|
|
f = freopen("mem.output", "w", stderr);
|
|
|
|
ATF_REQUIRE(f != NULL);
|
|
|
|
|
|
|
|
result = isc_test_begin(NULL, ISC_TRUE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = isc_mem_createx2(0, 0, default_memalloc, default_memfree,
|
|
|
|
NULL, &mctx2, 0);
|
|
|
|
isc_mem_debugging = ISC_MEM_DEBUGTRACE;
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
ptr = isc_mem_get(mctx2, 2048);
|
|
|
|
ATF_CHECK(ptr != NULL);
|
|
|
|
isc__mem_printactive(mctx2, f);
|
|
|
|
isc_mem_put(mctx2, ptr, 2048);
|
|
|
|
isc_mem_destroy(&mctx2);
|
|
|
|
isc_stdio_close(f);
|
|
|
|
|
|
|
|
result = isc_stdio_open("mem.output", "r", &f);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
result = isc_stdio_read(buf, sizeof(buf), 1, f, &size);
|
2017-09-27 16:19:07 +10:00
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_EOF);
|
2017-08-24 10:58:20 +05:30
|
|
|
isc_stdio_close(f);
|
|
|
|
isc_file_remove("mem.output");
|
|
|
|
|
|
|
|
/* return stderr to TTY so we can see errors */
|
|
|
|
f = freopen("/dev/tty", "w", stderr);
|
|
|
|
|
|
|
|
ATF_CHECK(strncmp(buf, "add ", 4) == 0);
|
|
|
|
p = strchr(buf, '\n');
|
|
|
|
p = strchr(p + 1, '\n');
|
|
|
|
ATF_CHECK(strncmp(p + 2, "ptr ", 4) == 0);
|
|
|
|
p = strchr(p + 1, '\n');
|
|
|
|
ATF_CHECK(strncmp(p + 1, "del ", 4) == 0);
|
|
|
|
|
|
|
|
isc_mem_debugging = ISC_MEM_DEBUGRECORD;
|
|
|
|
isc_test_end();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-01-22 09:44:24 -08:00
|
|
|
/*
|
|
|
|
* Main
|
|
|
|
*/
|
|
|
|
ATF_TP_ADD_TCS(tp) {
|
|
|
|
ATF_TP_ADD_TC(tp, isc_mem_total);
|
|
|
|
ATF_TP_ADD_TC(tp, isc_mem_inuse);
|
2017-08-24 10:58:20 +05:30
|
|
|
#if ISC_MEM_TRACKLINES
|
|
|
|
ATF_TP_ADD_TC(tp, isc_mem_noflags);
|
|
|
|
ATF_TP_ADD_TC(tp, isc_mem_recordflag);
|
|
|
|
ATF_TP_ADD_TC(tp, isc_mem_traceflag);
|
|
|
|
#endif
|
2015-01-22 09:44:24 -08:00
|
|
|
|
|
|
|
return (atf_no_error());
|
|
|
|
}
|