/* This file is part of drd, a thread error detector. Copyright (C) 2006-2013 Bart Van Assche <bvanassche@acm.org>. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA. The GNU General Public License is contained in the file COPYING. */ #include "drd_malloc_wrappers.h" #include "drd_thread.h" #include "pub_tool_basics.h" #include "pub_tool_execontext.h" #include "pub_tool_hashtable.h" #include "pub_tool_libcassert.h" #include "pub_tool_libcbase.h" #include "pub_tool_libcprint.h" #include "pub_tool_mallocfree.h" #include "pub_tool_options.h" #include "pub_tool_replacemalloc.h" #include "pub_tool_threadstate.h" #include "pub_tool_tooliface.h" /* Local type definitions. */ /** * Node with per-allocation information that will be stored in a hash map. * As specified in <pub_tool_hashtable.h>, the first member must be a pointer * and the second member must be an UWord. */ typedef struct _DRD_Chunk { struct _DRD_Chunk* next; UWord data; // pointer to actual block SizeT size; // size requested ExeContext* where; // where it was allocated } DRD_Chunk; /* Local variables. */ static StartUsingMem s_start_using_mem_callback; static StopUsingMem s_stop_using_mem_callback; /* Statistics. */ static SizeT s_cmalloc_n_mallocs = 0; static SizeT s_cmalloc_n_frees = 0; static SizeT s_cmalloc_bs_mallocd = 0; /* Record malloc'd blocks. */ static VgHashTable s_malloc_list = NULL; /* Function definitions. */ /** Allocate client memory memory and update the hash map. */ static void* new_block(ThreadId tid, SizeT size, SizeT align, Bool is_zeroed) { void* p; p = VG_(cli_malloc)(align, size); if (!p) return NULL; if (is_zeroed) VG_(memset)(p, 0, size); DRD_(malloclike_block)(tid, (Addr)p, size); return p; } /** * Store information about a memory block that has been allocated by * malloc() or a malloc() replacement in the hash map. */ void DRD_(malloclike_block)(const ThreadId tid, const Addr p, const SizeT size) { DRD_Chunk* mc; tl_assert(p); if (size > 0) s_start_using_mem_callback(p, size, 0/*ec_uniq*/); s_cmalloc_n_mallocs++; // Only update this stat if allocation succeeded. s_cmalloc_bs_mallocd += size; mc = VG_(malloc)("drd.malloc_wrappers.cDC.1", sizeof(DRD_Chunk)); mc->data = p; mc->size = size; mc->where = VG_(record_ExeContext)(tid, 0); VG_(HT_add_node)(s_malloc_list, mc); } static void handle_free(ThreadId tid, void* p) { Bool success; tl_assert(p); success = DRD_(freelike_block)(tid, (Addr)p, True); tl_assert(success); } /** * Remove the information that was stored by DRD_(malloclike_block)() about * a memory block. */ Bool DRD_(freelike_block)(const ThreadId tid, const Addr p, const Bool dealloc) { DRD_Chunk* mc; tl_assert(p); s_cmalloc_n_frees++; mc = VG_(HT_lookup)(s_malloc_list, (UWord)p); if (mc) { tl_assert(p == mc->data); if (mc->size > 0) s_stop_using_mem_callback(mc->data, mc->size); if (dealloc) VG_(cli_free)((void*)p); VG_(HT_remove)(s_malloc_list, (UWord)p); VG_(free)(mc); return True; } return False; } /** Wrapper for malloc(). */ static void* drd_malloc(ThreadId tid, SizeT n) { return new_block(tid, n, VG_(clo_alignment), /*is_zeroed*/False); } /** Wrapper for memalign(). */ static void* drd_memalign(ThreadId tid, SizeT align, SizeT n) { return new_block(tid, n, align, /*is_zeroed*/False); } /** Wrapper for calloc(). */ static void* drd_calloc(ThreadId tid, SizeT nmemb, SizeT size1) { return new_block(tid, nmemb*size1, VG_(clo_alignment), /*is_zeroed*/True); } /** Wrapper for free(). */ static void drd_free(ThreadId tid, void* p) { handle_free(tid, p); } /** * Wrapper for realloc(). Returns a pointer to the new block of memory, or * NULL if no new block could not be allocated. Notes: * - realloc(NULL, size) has the same effect as malloc(size). * - realloc(p, 0) has the same effect as free(p). * - success is not guaranteed even if the requested size is smaller than the * allocated size. */ static void* drd_realloc(ThreadId tid, void* p_old, SizeT new_size) { DRD_Chunk* mc; void* p_new; SizeT old_size; if (! p_old) return drd_malloc(tid, new_size); if (new_size == 0) { drd_free(tid, p_old); return NULL; } s_cmalloc_n_mallocs++; s_cmalloc_n_frees++; s_cmalloc_bs_mallocd += new_size; mc = VG_(HT_lookup)(s_malloc_list, (UWord)p_old); if (mc == NULL) { tl_assert(0); return NULL; } old_size = mc->size; if (old_size == new_size) { /* size unchanged */ mc->where = VG_(record_ExeContext)(tid, 0); p_new = p_old; } else if (new_size < old_size) { /* new size is smaller but nonzero */ s_stop_using_mem_callback(mc->data + new_size, old_size - new_size); mc->size = new_size; mc->where = VG_(record_ExeContext)(tid, 0); p_new = p_old; } else { /* new size is bigger */ p_new = VG_(cli_malloc)(VG_(clo_alignment), new_size); if (p_new) { /* Copy from old to new. */ VG_(memcpy)(p_new, p_old, mc->size); /* Free old memory. */ if (mc->size > 0) s_stop_using_mem_callback(mc->data, mc->size); VG_(cli_free)(p_old); VG_(HT_remove)(s_malloc_list, (UWord)p_old); /* Update state information. */ mc->data = (Addr)p_new; mc->size = new_size; mc->where = VG_(record_ExeContext)(tid, 0); VG_(HT_add_node)(s_malloc_list, mc); s_start_using_mem_callback((Addr)p_new, new_size, 0/*ec_uniq*/); } else { /* Allocation failed -- leave original block untouched. */ } } return p_new; } /** Wrapper for __builtin_new(). */ static void* drd___builtin_new(ThreadId tid, SizeT n) { return new_block(tid, n, VG_(clo_alignment), /*is_zeroed*/False); } /** Wrapper for __builtin_delete(). */ static void drd___builtin_delete(ThreadId tid, void* p) { handle_free(tid, p); } /** Wrapper for __builtin_vec_new(). */ static void* drd___builtin_vec_new(ThreadId tid, SizeT n) { return new_block(tid, n, VG_(clo_alignment), /*is_zeroed*/False); } /** Wrapper for __builtin_vec_delete(). */ static void drd___builtin_vec_delete(ThreadId tid, void* p) { handle_free(tid, p); } /** * Wrapper for malloc_usable_size() / malloc_size(). This function takes * a pointer to a block allocated by `malloc' and returns the amount of space * that is available in the block. This may or may not be more than the size * requested from `malloc', due to alignment or minimum size constraints. */ static SizeT drd_malloc_usable_size(ThreadId tid, void* p) { DRD_Chunk* mc; mc = VG_(HT_lookup)(s_malloc_list, (UWord)p); return mc ? mc->size : 0; } void DRD_(register_malloc_wrappers)(const StartUsingMem start_callback, const StopUsingMem stop_callback) { tl_assert(s_malloc_list == 0); s_malloc_list = VG_(HT_construct)("drd_malloc_list"); tl_assert(s_malloc_list); tl_assert(start_callback); tl_assert(stop_callback); s_start_using_mem_callback = start_callback; s_stop_using_mem_callback = stop_callback; VG_(needs_malloc_replacement)(drd_malloc, drd___builtin_new, drd___builtin_vec_new, drd_memalign, drd_calloc, drd_free, drd___builtin_delete, drd___builtin_vec_delete, drd_realloc, drd_malloc_usable_size, 0); } Bool DRD_(heap_addrinfo)(Addr const a, Addr* const data, SizeT* const size, ExeContext** const where) { DRD_Chunk* mc; tl_assert(data); tl_assert(size); tl_assert(where); VG_(HT_ResetIter)(s_malloc_list); while ((mc = VG_(HT_Next)(s_malloc_list))) { if (mc->data <= a && a < mc->data + mc->size) { *data = mc->data; *size = mc->size; *where = mc->where; return True; } } return False; } /*------------------------------------------------------------*/ /*--- Statistics printing ---*/ /*------------------------------------------------------------*/ void DRD_(print_malloc_stats)(void) { DRD_Chunk* mc; SizeT nblocks = 0; SizeT nbytes = 0; if (VG_(clo_verbosity) == 0) return; if (VG_(clo_xml)) return; /* Count memory still in use. */ VG_(HT_ResetIter)(s_malloc_list); while ((mc = VG_(HT_Next)(s_malloc_list))) { nblocks++; nbytes += mc->size; } VG_(message)(Vg_DebugMsg, "malloc/free: in use at exit: %lu bytes in %lu blocks.\n", nbytes, nblocks); VG_(message)(Vg_DebugMsg, "malloc/free: %lu allocs, %lu frees, %lu bytes allocated.\n", s_cmalloc_n_mallocs, s_cmalloc_n_frees, s_cmalloc_bs_mallocd); if (VG_(clo_verbosity) > 1) VG_(message)(Vg_DebugMsg, " \n"); } /*--------------------------------------------------------------------*/ /*--- end ---*/ /*--------------------------------------------------------------------*/