Development mbed library for MAX32630FTHR
Dependents: blinky_max32630fthr
TESTS/mbed_drivers/mem_trace/main.cpp
- Committer:
- switches
- Date:
- 2016-12-16
- Revision:
- 3:1198227e6421
- Parent:
- 0:5c4d7b2438d3
File content as of revision 3:1198227e6421:
/* * Copyright (c) 2013-2016, ARM Limited, All Rights Reserved * SPDX-License-Identifier: Apache-2.0 * * Licensed under the Apache License, Version 2.0 (the "License"); you may * not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "mbed.h" #include "greentea-client/test_env.h" #include "unity/unity.h" #include "utest/utest.h" #include "mbed_mem_trace.h" #include <stdlib.h> #include <stdio.h> #include <stdarg.h> #ifndef MBED_MEM_TRACING_ENABLED #error [NOT_SUPPORTED] test not supported #endif using namespace utest::v1; /******************************************************************************/ /* Helper functions and data structures */ /******************************************************************************/ // This structure keeps data about the various memory allocation operations, // as traced by 'test_trace_cb' below. #define TEST_MAX_MEMORY_OPS 10 // Trace results for all possible operations typedef struct { uint8_t op; void *res; union { struct { size_t arg_size; } malloc_info; struct { void *arg_ptr; size_t arg_size; } realloc_info; struct { size_t arg_nmemb; size_t arg_size; } calloc_info; struct { void *arg_ptr; } free_info; }; } mem_trace_data_t; // Memory operation statistics typedef struct { mem_trace_data_t op_data[TEST_MAX_MEMORY_OPS]; uint32_t total_ops; bool invalid_op, overflow; } stats_t; static stats_t stats; // Clear all the memory statistics static void test_clear_stats() { memset(&stats, 0, sizeof(stats)); } // Memory tracer callback that records each operation in "stats" (above) extern "C" void test_trace_cb(uint8_t op, void *res, void *caller, ...) { va_list va; mem_trace_data_t *pmem = stats.op_data + stats.total_ops; if (stats.total_ops >= TEST_MAX_MEMORY_OPS) { stats.overflow = true; return; } va_start(va, caller); pmem->op = op; pmem->res = res; switch(op) { case MBED_MEM_TRACE_MALLOC: pmem->malloc_info.arg_size = va_arg(va, size_t); break; case MBED_MEM_TRACE_REALLOC: pmem->realloc_info.arg_ptr = va_arg(va, void *); pmem->realloc_info.arg_size = va_arg(va, size_t); break; case MBED_MEM_TRACE_CALLOC: pmem->calloc_info.arg_nmemb = va_arg(va, size_t); pmem->calloc_info.arg_size = va_arg(va, size_t); break; case MBED_MEM_TRACE_FREE: pmem->free_info.arg_ptr = va_arg(va, void *); break; default: stats.invalid_op = true; } stats.total_ops ++; va_end(va); } // Generic sanity checks for the tracer static void check_sanity(uint32_t expected_ops) { TEST_ASSERT_FALSE(stats.overflow); TEST_ASSERT_FALSE(stats.invalid_op); TEST_ASSERT_EQUAL_UINT32(stats.total_ops, expected_ops); } // Check a "malloc" operation static void check_malloc_op(const mem_trace_data_t *p, void *expected_res, size_t expected_arg_size) { TEST_ASSERT_EQUAL_UINT8(p->op, MBED_MEM_TRACE_MALLOC); TEST_ASSERT_EQUAL_PTR(p->res, expected_res); TEST_ASSERT_EQUAL_UINT32(p->malloc_info.arg_size, expected_arg_size); } // Check a "free" operation static void check_free_op(const mem_trace_data_t *p, void *expected_arg_ptr) { TEST_ASSERT_EQUAL_UINT8(p->op, MBED_MEM_TRACE_FREE); TEST_ASSERT_EQUAL_PTR(p->free_info.arg_ptr, expected_arg_ptr); } // Check a "realloc" operation static void check_realloc_op(const mem_trace_data_t *p, void *expected_res, void *expected_arg_ptr, size_t expected_arg_size) { TEST_ASSERT_EQUAL_UINT8(p->op, MBED_MEM_TRACE_REALLOC); TEST_ASSERT_EQUAL_PTR(p->res, expected_res); TEST_ASSERT_EQUAL_UINT32(p->realloc_info.arg_ptr, expected_arg_ptr); TEST_ASSERT_EQUAL_UINT32(p->realloc_info.arg_size, expected_arg_size); } // Check a "calloc" operation static void check_calloc_op(const mem_trace_data_t *p, void *expected_res, size_t expected_arg_nmemb, size_t expected_arg_size) { TEST_ASSERT_EQUAL_UINT8(p->op, MBED_MEM_TRACE_CALLOC); TEST_ASSERT_EQUAL_PTR(p->res, expected_res); TEST_ASSERT_EQUAL_UINT32(p->calloc_info.arg_nmemb, expected_arg_nmemb); TEST_ASSERT_EQUAL_UINT32(p->calloc_info.arg_size, expected_arg_size); } /******************************************************************************/ /* Tests */ /******************************************************************************/ // Allocate a single buffer, then free it. Check that tracing matches the operations. static void test_case_single_malloc_free() { const size_t block_size = 126; const mem_trace_data_t *pmem = stats.op_data; test_clear_stats(); mbed_mem_trace_set_callback(test_trace_cb); // Allocate a single memory block void *p = malloc(block_size); TEST_ASSERT_NOT_EQUAL(p, NULL); // Free the memory block free(p); // Stop tracing mbed_mem_trace_set_callback(NULL); // Check tracer result check_sanity(2); check_malloc_op(pmem ++, p, block_size); check_free_op(pmem, p); } // Test all memory operations (malloc, realloc, free, calloc) static void test_case_all_memory_ops() { const size_t malloc_size = 40, realloc_size = 80, nmemb = 25, size = 10; const mem_trace_data_t *pmem = stats.op_data; test_clear_stats(); mbed_mem_trace_set_callback(test_trace_cb); // Allocate a single memory block, the realloc it void *p_malloc = malloc(malloc_size); TEST_ASSERT_NOT_EQUAL(p_malloc, NULL); void *p_realloc = realloc(p_malloc, realloc_size); TEST_ASSERT_NOT_EQUAL(p_realloc, NULL); // Use calloc() now void *p_calloc = calloc(nmemb, size); //TEST_ASSERT_NOT_EQUAL(p_calloc, NULL); // Free the realloc() pointer first, then the calloc() one free(p_realloc); free(p_calloc); // Stop tracing mbed_mem_trace_set_callback(NULL); // Check tracer result check_sanity(6); check_malloc_op(pmem ++, p_malloc, malloc_size); check_realloc_op(pmem ++, p_realloc, p_malloc, realloc_size); // calloc() calls malloc() internally check_malloc_op(pmem ++, p_calloc, nmemb * size); check_calloc_op(pmem ++, p_calloc, nmemb, size); check_free_op(pmem ++, p_realloc); check_free_op(pmem, p_calloc); } // Test that tracing is off when using a NULL callback static void test_case_trace_off() { const size_t malloc_size = 10; test_clear_stats(); // We don't want any tracing mbed_mem_trace_set_callback(NULL); // Allocate a buffer and free it void *p_malloc = malloc(malloc_size); TEST_ASSERT_NOT_EQUAL(p_malloc, NULL); free(p_malloc); // Check that we didn't trace anything check_sanity(0); } // Test partial tracing (start tracing, stop tracing, restart later) static void test_case_partial_trace() { const size_t malloc_size_1 = 20, malloc_size_2 = 30; const mem_trace_data_t *pmem = stats.op_data; test_clear_stats(); // Start tracing mbed_mem_trace_set_callback(test_trace_cb); // Allocate a buffer void *p_malloc_1 = malloc(malloc_size_1); TEST_ASSERT_NOT_EQUAL(p_malloc_1, NULL); // Disable tracing before freeing the first buffer mbed_mem_trace_set_callback(NULL); free(p_malloc_1); // Allocate another buffer (still not traced) void *p_malloc_2 = malloc(malloc_size_2); TEST_ASSERT_NOT_EQUAL(p_malloc_2, NULL); // Re-enable tracing mbed_mem_trace_set_callback(test_trace_cb); // And free the second buffer (this operation should be tracer) free(p_malloc_2); // Stop tracing mbed_mem_trace_set_callback(NULL); // Check tracer result check_sanity(2); check_malloc_op(pmem ++, p_malloc_1, malloc_size_1); check_free_op(pmem, p_malloc_2); } // Test new/delete tracing static void test_case_new_delete() { const mem_trace_data_t *pmem = stats.op_data; test_clear_stats(); // Start tracing mbed_mem_trace_set_callback(test_trace_cb); // Test new, new[], delete and delete[] int *p_int = new int; int *p_int_array = new int[10]; delete p_int; delete[] p_int_array; // Stop tracing mbed_mem_trace_set_callback(NULL); // Check tracer result check_sanity(4); check_malloc_op(pmem ++, p_int, sizeof(int)); check_malloc_op(pmem ++, p_int_array, 10 * sizeof(int)); check_free_op(pmem ++, p_int); check_free_op(pmem ++, p_int_array); } static Case cases[] = { Case("single malloc/free", test_case_single_malloc_free), Case("all memory operations", test_case_all_memory_ops), Case("trace off", test_case_trace_off), Case("partial trace", test_case_partial_trace), Case("test new/delete", test_case_new_delete) }; static status_t greentea_test_setup(const size_t number_of_cases) { GREENTEA_SETUP(20, "default_auto"); return greentea_test_setup_handler(number_of_cases); } static Specification specification(greentea_test_setup, cases, greentea_test_teardown_handler); int main() { // Disable stdout buffering to prevent any unwanted allocations setvbuf(stdout, NULL, _IONBF, 0); Harness::run(specification); }