2021-02-07
adds allocator interface and default implementation
src/CMakeLists.txt | file | annotate | diff | comparison | revisions | |
src/allocator.c | file | annotate | diff | comparison | revisions | |
src/cx/allocator.h | file | annotate | diff | comparison | revisions | |
test/CMakeLists.txt | file | annotate | diff | comparison | revisions | |
test/test_allocator.c | file | annotate | diff | comparison | revisions |
--- a/src/CMakeLists.txt Sat Feb 06 19:11:44 2021 +0100 +++ b/src/CMakeLists.txt Sun Feb 07 12:20:07 2021 +0100 @@ -1,7 +1,9 @@ set(sources + allocator.c list.c ) set(headers + cx/allocator.h cx/list.h )
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/allocator.c Sun Feb 07 12:20:07 2021 +0100 @@ -0,0 +1,56 @@ +/* + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + * + * Copyright 2021 Mike Becker, Olaf Wintermann All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * 2. 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. + * + * 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 THE COPYRIGHT HOLDER OR CONTRIBUTORS 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 "cx/allocator.h" + +#include <stdlib.h> + +struct cx_allocator_s cx_default_allocator = { + cx_malloc_stdlib, + cx_realloc_stdlib, + cx_calloc_stdlib, + cx_free_stdlib, + NULL +}; +CxAllocator cxDefaultAllocator = &cx_default_allocator; + +void* cx_malloc_stdlib(cx_allocator a, size_t n) { + return malloc(n); +} + +void* cx_realloc_stdlib(cx_allocator a, void* mem, size_t n) { + return realloc(mem, n); +} + +void* cx_calloc_stdlib(cx_allocator a, size_t nelem, size_t n) { + return calloc(nelem, n); +} + +void cx_free_stdlib(cx_allocator a, void* mem) { + free(mem); +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/cx/allocator.h Sun Feb 07 12:20:07 2021 +0100 @@ -0,0 +1,61 @@ +/* + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + * + * Copyright 2021 Mike Becker, Olaf Wintermann All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * 2. 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. + * + * 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 THE COPYRIGHT HOLDER OR CONTRIBUTORS 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. + */ + +#ifndef UCX_ALLOCATOR_H +#define UCX_ALLOCATOR_H + +#include <stdlib.h> + +typedef void* cx_allocator; +typedef void*(*cx_malloc_func)(cx_allocator a, size_t n); +typedef void*(*cx_realloc_func)(cx_allocator a, void* mem, size_t n); +typedef void*(*cx_calloc_func)(cx_allocator a, size_t nelem, size_t n); +typedef void(*cx_free_func)(cx_allocator a, void* mem); + +void* cx_malloc_stdlib(cx_allocator a, size_t n); +void* cx_realloc_stdlib(cx_allocator a, void* mem, size_t n); +void* cx_calloc_stdlib(cx_allocator a, size_t nelem, size_t n); +void cx_free_stdlib(cx_allocator a, void* mem); + +struct cx_allocator_s { + cx_malloc_func malloc; + cx_realloc_func realloc; + cx_calloc_func calloc; + cx_free_func free; + void* data; +}; +typedef struct cx_allocator_s* CxAllocator; + +extern CxAllocator cxDefaultAllocator; + +#define cxMalloc(a, n) (a)->malloc((a), n) +#define cxRealloc(a, mem, n) (a)->realloc((a), mem, n) +#define cxCalloc(a, nelem, n) (a)->calloc((a), nelem, n) +#define cxFree(a, mem) (a)->free((a), mem) + +#endif //UCX_ALLOCATOR_H
--- a/test/CMakeLists.txt Sat Feb 06 19:11:44 2021 +0100 +++ b/test/CMakeLists.txt Sun Feb 07 12:20:07 2021 +0100 @@ -1,6 +1,22 @@ +message(CHECK_START "Searching for CUnit test framework") -add_executable(test_list - test_list.c -) -target_link_libraries(test_list PRIVATE ucx_static) -add_test(NAME list COMMAND test_list WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}") +find_path(CUNIT_INCLUDE_DIR NAMES CUnit/CUnit.h) +find_library(CUNIT_LIBRARY NAMES cunit libcunit cunitlib) +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(CUnit DEFAULT_MSG CUNIT_LIBRARY CUNIT_INCLUDE_DIR) + +if(CUNIT_FOUND) + message(CHECK_PASS "found: compiling tests.") + set(TESTS + test_allocator + test_list + ) + + foreach(test ${TESTS}) + add_executable(${test} ${test}.c) + target_link_libraries(${test} PRIVATE ucx_static ${CUNIT_LIBRARY}) + add_test(NAME ${test} COMMAND ${test} WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}") + endforeach() +else() + message(CHECK_FAIL "not found: unit tests will not be available.") +endif()
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/test/test_allocator.c Sun Feb 07 12:20:07 2021 +0100 @@ -0,0 +1,94 @@ +/* + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. + * + * Copyright 2021 Mike Becker, Olaf Wintermann All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * 2. 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. + * + * 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 THE COPYRIGHT HOLDER OR CONTRIBUTORS 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 "cx/allocator.h" + +#include <CUnit/Basic.h> + +void test_default_allocator_available(void) { + CU_ASSERT_PTR_EQUAL(cxDefaultAllocator->malloc, cx_malloc_stdlib) + CU_ASSERT_PTR_EQUAL(cxDefaultAllocator->realloc, cx_realloc_stdlib) + CU_ASSERT_PTR_EQUAL(cxDefaultAllocator->calloc, cx_calloc_stdlib) + CU_ASSERT_PTR_EQUAL(cxDefaultAllocator->free, cx_free_stdlib) +} + +void test_default_malloc(void) { + void* test = cxMalloc(cxDefaultAllocator, 16); + CU_ASSERT_PTR_NOT_NULL(test); + free(test); +} + +void test_default_realloc(void) { + void* test = calloc(8, 1); + memcpy(test, "Test", 4); + test = cxRealloc(cxDefaultAllocator, test, 16); + CU_ASSERT_PTR_NOT_NULL(test); + CU_ASSERT_STRING_EQUAL("Test", test) + free(test); +} + +void test_default_calloc(void) { + void* test = cxCalloc(cxDefaultAllocator, 8, 2); + CU_ASSERT_PTR_NOT_NULL(test); + free(test); +} + +void test_default_free(void) { + CU_PASS("Testing standard free is not possible.") +} + +int main() { + CU_pSuite suite = NULL; + + if (CUE_SUCCESS != CU_initialize_registry()) { + return CU_get_error(); + } + + suite = CU_add_suite("stdlib allocator", NULL, NULL); + if (NULL == suite) { + CU_cleanup_registry(); + return CU_get_error(); + } + + if ( + (NULL == CU_add_test(suite, "default allocator available", test_default_allocator_available)) || + (NULL == CU_add_test(suite, "test of malloc()", test_default_malloc)) || + (NULL == CU_add_test(suite, "test of realloc()", test_default_realloc)) || + (NULL == CU_add_test(suite, "test of realloc()", test_default_calloc)) || + (NULL == CU_add_test(suite, "test of free()", test_default_free)) + ) { + CU_cleanup_registry(); + return CU_get_error(); + } + + CU_basic_set_mode(CU_BRM_NORMAL); + CU_basic_run_tests(); + CU_cleanup_registry(); + + return CU_get_error(); +}