mirror of https://github.com/ARMmbed/mbed-os.git
platform: fix mem_trace trace_level guard
parent
9cce4d2b06
commit
aa22ef1e1e
|
|
@ -46,9 +46,6 @@ typedef struct {
|
||||||
uint32_t pad;
|
uint32_t pad;
|
||||||
} alloc_info_t;
|
} alloc_info_t;
|
||||||
|
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
|
||||||
static SingletonPtr<PlatformMutex> mem_trace_mutex;
|
|
||||||
#endif
|
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
static SingletonPtr<PlatformMutex> malloc_stats_mutex;
|
static SingletonPtr<PlatformMutex> malloc_stats_mutex;
|
||||||
static mbed_stats_heap_t heap_stats = {0, 0, 0, 0, 0};
|
static mbed_stats_heap_t heap_stats = {0, 0, 0, 0, 0};
|
||||||
|
|
@ -91,6 +88,9 @@ extern "C" {
|
||||||
|
|
||||||
extern "C" void * __wrap__malloc_r(struct _reent * r, size_t size) {
|
extern "C" void * __wrap__malloc_r(struct _reent * r, size_t size) {
|
||||||
void *ptr = NULL;
|
void *ptr = NULL;
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
malloc_stats_mutex->lock();
|
malloc_stats_mutex->lock();
|
||||||
alloc_info_t *alloc_info = (alloc_info_t*)__real__malloc_r(r, size + sizeof(alloc_info_t));
|
alloc_info_t *alloc_info = (alloc_info_t*)__real__malloc_r(r, size + sizeof(alloc_info_t));
|
||||||
|
|
@ -111,15 +111,17 @@ extern "C" void * __wrap__malloc_r(struct _reent * r, size_t size) {
|
||||||
ptr = __real__malloc_r(r, size);
|
ptr = __real__malloc_r(r, size);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_malloc(ptr, size, MBED_CALLER_ADDR());
|
mbed_mem_trace_malloc(ptr, size, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
return ptr;
|
return ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void * __wrap__realloc_r(struct _reent * r, void * ptr, size_t size) {
|
extern "C" void * __wrap__realloc_r(struct _reent * r, void * ptr, size_t size) {
|
||||||
void *new_ptr = NULL;
|
void *new_ptr = NULL;
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
// Implement realloc_r with malloc and free.
|
// Implement realloc_r with malloc and free.
|
||||||
// The function realloc_r can't be used here directly since
|
// The function realloc_r can't be used here directly since
|
||||||
|
|
@ -151,14 +153,16 @@ extern "C" void * __wrap__realloc_r(struct _reent * r, void * ptr, size_t size)
|
||||||
new_ptr = __real__realloc_r(r, ptr, size);
|
new_ptr = __real__realloc_r(r, ptr, size);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_realloc(new_ptr, ptr, size, MBED_CALLER_ADDR());
|
mbed_mem_trace_realloc(new_ptr, ptr, size, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
return new_ptr;
|
return new_ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void __wrap__free_r(struct _reent * r, void * ptr) {
|
extern "C" void __wrap__free_r(struct _reent * r, void * ptr) {
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
malloc_stats_mutex->lock();
|
malloc_stats_mutex->lock();
|
||||||
alloc_info_t *alloc_info = NULL;
|
alloc_info_t *alloc_info = NULL;
|
||||||
|
|
@ -173,14 +177,16 @@ extern "C" void __wrap__free_r(struct _reent * r, void * ptr) {
|
||||||
__real__free_r(r, ptr);
|
__real__free_r(r, ptr);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_free(ptr, MBED_CALLER_ADDR());
|
mbed_mem_trace_free(ptr, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void * __wrap__calloc_r(struct _reent * r, size_t nmemb, size_t size) {
|
extern "C" void * __wrap__calloc_r(struct _reent * r, size_t nmemb, size_t size) {
|
||||||
void *ptr = NULL;
|
void *ptr = NULL;
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
// Note - no lock needed since malloc is thread safe
|
// Note - no lock needed since malloc is thread safe
|
||||||
|
|
||||||
|
|
@ -192,9 +198,8 @@ extern "C" void * __wrap__calloc_r(struct _reent * r, size_t nmemb, size_t size)
|
||||||
ptr = __real__calloc_r(r, nmemb, size);
|
ptr = __real__calloc_r(r, nmemb, size);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_calloc(ptr, nmemb, size, MBED_CALLER_ADDR());
|
mbed_mem_trace_calloc(ptr, nmemb, size, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
return ptr;
|
return ptr;
|
||||||
}
|
}
|
||||||
|
|
@ -244,6 +249,9 @@ extern "C" {
|
||||||
|
|
||||||
extern "C" void* SUB_MALLOC(size_t size) {
|
extern "C" void* SUB_MALLOC(size_t size) {
|
||||||
void *ptr = NULL;
|
void *ptr = NULL;
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
malloc_stats_mutex->lock();
|
malloc_stats_mutex->lock();
|
||||||
alloc_info_t *alloc_info = (alloc_info_t*)SUPER_MALLOC(size + sizeof(alloc_info_t));
|
alloc_info_t *alloc_info = (alloc_info_t*)SUPER_MALLOC(size + sizeof(alloc_info_t));
|
||||||
|
|
@ -264,15 +272,17 @@ extern "C" void* SUB_MALLOC(size_t size) {
|
||||||
ptr = SUPER_MALLOC(size);
|
ptr = SUPER_MALLOC(size);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_malloc(ptr, size, MBED_CALLER_ADDR());
|
mbed_mem_trace_malloc(ptr, size, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
return ptr;
|
return ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void* SUB_REALLOC(void *ptr, size_t size) {
|
extern "C" void* SUB_REALLOC(void *ptr, size_t size) {
|
||||||
void *new_ptr = NULL;
|
void *new_ptr = NULL;
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
// Note - no lock needed since malloc and free are thread safe
|
// Note - no lock needed since malloc and free are thread safe
|
||||||
|
|
||||||
|
|
@ -299,15 +309,17 @@ extern "C" void* SUB_REALLOC(void *ptr, size_t size) {
|
||||||
new_ptr = SUPER_REALLOC(ptr, size);
|
new_ptr = SUPER_REALLOC(ptr, size);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_realloc(new_ptr, ptr, size, MBED_CALLER_ADDR());
|
mbed_mem_trace_realloc(new_ptr, ptr, size, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
return new_ptr;
|
return new_ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void *SUB_CALLOC(size_t nmemb, size_t size) {
|
extern "C" void *SUB_CALLOC(size_t nmemb, size_t size) {
|
||||||
void *ptr = NULL;
|
void *ptr = NULL;
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
// Note - no lock needed since malloc is thread safe
|
// Note - no lock needed since malloc is thread safe
|
||||||
ptr = malloc(nmemb * size);
|
ptr = malloc(nmemb * size);
|
||||||
|
|
@ -318,14 +330,16 @@ extern "C" void *SUB_CALLOC(size_t nmemb, size_t size) {
|
||||||
ptr = SUPER_CALLOC(nmemb, size);
|
ptr = SUPER_CALLOC(nmemb, size);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_calloc(ptr, nmemb, size, MBED_CALLER_ADDR());
|
mbed_mem_trace_calloc(ptr, nmemb, size, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
return ptr;
|
return ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void SUB_FREE(void *ptr) {
|
extern "C" void SUB_FREE(void *ptr) {
|
||||||
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
|
mbed_mem_trace_lock();
|
||||||
|
#endif
|
||||||
#ifdef MBED_HEAP_STATS_ENABLED
|
#ifdef MBED_HEAP_STATS_ENABLED
|
||||||
malloc_stats_mutex->lock();
|
malloc_stats_mutex->lock();
|
||||||
alloc_info_t *alloc_info = NULL;
|
alloc_info_t *alloc_info = NULL;
|
||||||
|
|
@ -340,9 +354,8 @@ extern "C" void SUB_FREE(void *ptr) {
|
||||||
SUPER_FREE(ptr);
|
SUPER_FREE(ptr);
|
||||||
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
#endif // #ifdef MBED_HEAP_STATS_ENABLED
|
||||||
#ifdef MBED_MEM_TRACING_ENABLED
|
#ifdef MBED_MEM_TRACING_ENABLED
|
||||||
mem_trace_mutex->lock();
|
|
||||||
mbed_mem_trace_free(ptr, MBED_CALLER_ADDR());
|
mbed_mem_trace_free(ptr, MBED_CALLER_ADDR());
|
||||||
mem_trace_mutex->unlock();
|
mbed_mem_trace_unlock();
|
||||||
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
#endif // #ifdef MBED_MEM_TRACING_ENABLED
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -19,6 +19,8 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include "platform/mbed_mem_trace.h"
|
#include "platform/mbed_mem_trace.h"
|
||||||
#include "platform/mbed_critical.h"
|
#include "platform/mbed_critical.h"
|
||||||
|
#include "platform/SingletonPtr.h"
|
||||||
|
#include "platform/PlatformMutex.h"
|
||||||
|
|
||||||
/******************************************************************************
|
/******************************************************************************
|
||||||
* Internal variables, functions and helpers
|
* Internal variables, functions and helpers
|
||||||
|
|
@ -26,10 +28,14 @@
|
||||||
|
|
||||||
/* The callback function that will be called after a traced memory operations finishes. */
|
/* The callback function that will be called after a traced memory operations finishes. */
|
||||||
static mbed_mem_trace_cb_t mem_trace_cb;
|
static mbed_mem_trace_cb_t mem_trace_cb;
|
||||||
/* 'trave_level' guards "trace inside trace" situations (for example, the implementation
|
/* 'trace_lock_count' guards "trace inside trace" situations (for example, the implementation
|
||||||
* of realloc() might call malloc() internally, and since malloc() is also traced, this could
|
* of realloc() might call malloc() internally, and since malloc() is also traced, this could
|
||||||
* result in two calls to the callback function instead of one. */
|
* result in two calls to the callback function instead of one. */
|
||||||
static uint8_t trace_level;
|
static uint8_t trace_lock_count;
|
||||||
|
static SingletonPtr<PlatformMutex> mem_trace_mutex;
|
||||||
|
|
||||||
|
#define TRACE_FIRST_LOCK() (trace_lock_count < 2)
|
||||||
|
|
||||||
|
|
||||||
/******************************************************************************
|
/******************************************************************************
|
||||||
* Public interface
|
* Public interface
|
||||||
|
|
@ -39,42 +45,50 @@ void mbed_mem_trace_set_callback(mbed_mem_trace_cb_t cb) {
|
||||||
mem_trace_cb = cb;
|
mem_trace_cb = cb;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void mbed_mem_trace_lock()
|
||||||
|
{
|
||||||
|
mem_trace_mutex->lock();
|
||||||
|
trace_lock_count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mbed_mem_trace_unlock()
|
||||||
|
{
|
||||||
|
trace_lock_count--;
|
||||||
|
mem_trace_mutex->unlock();
|
||||||
|
}
|
||||||
|
|
||||||
void *mbed_mem_trace_malloc(void *res, size_t size, void *caller) {
|
void *mbed_mem_trace_malloc(void *res, size_t size, void *caller) {
|
||||||
if (mem_trace_cb) {
|
if (mem_trace_cb) {
|
||||||
if (core_util_atomic_incr_u8(&trace_level, 1) == 1) {
|
if (TRACE_FIRST_LOCK()) {
|
||||||
mem_trace_cb(MBED_MEM_TRACE_MALLOC, res, caller, size);
|
mem_trace_cb(MBED_MEM_TRACE_MALLOC, res, caller, size);
|
||||||
}
|
}
|
||||||
core_util_atomic_decr_u8(&trace_level, 1);
|
|
||||||
}
|
}
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
void *mbed_mem_trace_realloc(void *res, void *ptr, size_t size, void *caller) {
|
void *mbed_mem_trace_realloc(void *res, void *ptr, size_t size, void *caller) {
|
||||||
if (mem_trace_cb) {
|
if (mem_trace_cb) {
|
||||||
if (core_util_atomic_incr_u8(&trace_level, 1) == 1) {
|
if (TRACE_FIRST_LOCK()) {
|
||||||
mem_trace_cb(MBED_MEM_TRACE_REALLOC, res, caller, ptr, size);
|
mem_trace_cb(MBED_MEM_TRACE_REALLOC, res, caller, ptr, size);
|
||||||
}
|
}
|
||||||
core_util_atomic_decr_u8(&trace_level, 1);
|
|
||||||
}
|
}
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
void *mbed_mem_trace_calloc(void *res, size_t num, size_t size, void *caller) {
|
void *mbed_mem_trace_calloc(void *res, size_t num, size_t size, void *caller) {
|
||||||
if (mem_trace_cb) {
|
if (mem_trace_cb) {
|
||||||
if (core_util_atomic_incr_u8(&trace_level, 1) == 1) {
|
if (TRACE_FIRST_LOCK()) {
|
||||||
mem_trace_cb(MBED_MEM_TRACE_CALLOC, res, caller, num, size);
|
mem_trace_cb(MBED_MEM_TRACE_CALLOC, res, caller, num, size);
|
||||||
}
|
}
|
||||||
core_util_atomic_decr_u8(&trace_level, 1);
|
|
||||||
}
|
}
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
void mbed_mem_trace_free(void *ptr, void *caller) {
|
void mbed_mem_trace_free(void *ptr, void *caller) {
|
||||||
if (mem_trace_cb) {
|
if (mem_trace_cb) {
|
||||||
if (core_util_atomic_incr_u8(&trace_level, 1) == 1) {
|
if (TRACE_FIRST_LOCK()) {
|
||||||
mem_trace_cb(MBED_MEM_TRACE_FREE, NULL, caller, ptr);
|
mem_trace_cb(MBED_MEM_TRACE_FREE, NULL, caller, ptr);
|
||||||
}
|
}
|
||||||
core_util_atomic_decr_u8(&trace_level, 1);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -72,6 +72,17 @@ typedef void (*mbed_mem_trace_cb_t)(uint8_t op, void *res, void* caller, ...);
|
||||||
*/
|
*/
|
||||||
void mbed_mem_trace_set_callback(mbed_mem_trace_cb_t cb);
|
void mbed_mem_trace_set_callback(mbed_mem_trace_cb_t cb);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trace lock.
|
||||||
|
* @note Locking prevent recursive tracing of malloc/free inside relloc/calloc
|
||||||
|
*/
|
||||||
|
void mbed_mem_trace_lock();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trace unlock.
|
||||||
|
*/
|
||||||
|
void mbed_mem_trace_unlock();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Trace a call to 'malloc'.
|
* Trace a call to 'malloc'.
|
||||||
* @param res the result of running 'malloc'.
|
* @param res the result of running 'malloc'.
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue