/* * Copyright (C) 2015-2017 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 http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include "isctest.h" #include #include #include #include #include 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", (unsigned long)before, (unsigned long)after, (unsigned long)diff); /* 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", (unsigned long)before, (unsigned long)after, (unsigned long)diff); /* 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", (unsigned long)before, (unsigned long)during, (unsigned long)after); ATF_REQUIRE_EQ(diff, 0); out: if (mctx2 != NULL) isc_mem_destroy(&mctx2); isc_test_end(); } #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); ATF_REQUIRE_EQ(result, ISC_R_EOF); 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); memset(buf, 0, sizeof(buf)); 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); ATF_REQUIRE_EQ(result, ISC_R_EOF); 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); ATF_REQUIRE_EQ(result, ISC_R_EOF); 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 /* * Main */ ATF_TP_ADD_TCS(tp) { ATF_TP_ADD_TC(tp, isc_mem_total); ATF_TP_ADD_TC(tp, isc_mem_inuse); #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 return (atf_no_error()); }