MALLOC_HOOK(3) | Руководство программиста Linux | MALLOC_HOOK(3) |
__malloc_hook, __malloc_initialize_hook, __memalign_hook, __free_hook, __realloc_hook, __after_morecore_hook - переменные отладки malloc
#include <malloc.h>
void *(*__malloc_hook)(size_t size, const void *caller);
void *(*__realloc_hook)(void *ptr, size_t size, const void *caller);
void *(*__memalign_hook)(size_t alignment, size_t size, const void *caller);
void (*__free_hook)(void *ptr, const void *caller);
void (*__malloc_initialize_hook)(void);
void (*__after_morecore_hook)(void);
Библиотека GNU C позволяет изменить поведение malloc(3), realoc(3) и free(3), указав соответствующие «обрабатывающие» (hook) функции. Вы можете использовать эти функции-обработчики для отладки программ, использующих динамическое выделение памяти.
Переменная __ malloc_initialize_hook указывает на функцию, которая однократно вызывается при инициализации реализации malloc. Это — слабая переменная, поэтому она может быть переопределена в приложении как показано далее:
void (*__malloc_initialize_hook)(void) = my_init_hook;
Теперь функция my_init_hook() может проинициализировать все функции-обработчики.
У четырёх функций, на которые указывают переменные __malloc_hook, __realloc_hook, __memalign_hook, __free_hook, имеются прототипы, подобные функциям malloc(3), realloc(3), memalign(3), free(3), соответственно, однако у них есть дополнительный последний параметр caller, в который передаётся адрес вызывающего malloc(3) и т.д.
Переменная __ after_morecore_hook указывает на функцию, которая вызывается каждый раз после того, как sbrk(2) просят увеличить память.
Эти функции являются расширениями GNU.
Использование этих обрабатывающих функций не безопасно в многонитевых программах, и теперь они считаются устаревшими. Начиная с glibc версии 2.24, переменная __ malloc_initialize_hook была удалена из программного интерфейса. Теперь вместо неё программисты должны подменять вызовы соответствующих функций, определяя и экспортируя такие функции как «malloc» и «free».
Вот короткий пример того, как использовать эти переменные.
#include <stdio.h> #include <malloc.h> /* прототипы наших функций */ static void my_init_hook(void); static void *my_malloc_hook(size_t, const void *); /* переменные для сохранения имеющихся обрабатывающих функций */ static void *(*old_malloc_hook)(size_t, const void *); /* переопределяем первоначальные функции библиотеки C */ void (*__malloc_initialize_hook) (void) = my_init_hook; static void my_init_hook(void) { old_malloc_hook = __malloc_hook; __malloc_hook = my_malloc_hook; } static void * my_malloc_hook(size_t size, const void *caller) { void *result; /* Восстанавливаем старые функции */ __malloc_hook = old_malloc_hook; /* вызываем рекурсивно */ result = malloc(size); /* сохраняем нижележащие функции */ old_malloc_hook = __malloc_hook; /* printf() тоже может вызвать malloc(), поэтому его тоже надо защитить */ printf("malloc(%u) called from %p returns %p\n", (unsigned int) size, caller, result); /* восстанавливаем наши функции */ __malloc_hook = my_malloc_hook; return result; }
2019-03-06 | GNU |