/* Copyright 2005-2012 Intel Corporation. All Rights Reserved. This file is part of Threading Building Blocks. Threading Building Blocks is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License version 2 as published by the Free Software Foundation. Threading Building Blocks 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 Threading Building Blocks; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA As a special exception, you may use this file as part of a free software library without restriction. Specifically, if other files instantiate templates or use macros or inline functions from this file, or you compile this file and link it with other files to produce an executable, this file does not by itself cause the resulting executable to be covered by the GNU General Public License. This exception does not however invalidate any other reasons why the executable file might be covered by the GNU General Public License. */ #include "proxy.h" #if !defined(__EXCEPTIONS) && !defined(_CPPUNWIND) && !defined(__SUNPRO_CC) || defined(_XBOX) #if TBB_USE_EXCEPTIONS #error Compilation settings do not support exception handling. Please do not set TBB_USE_EXCEPTIONS macro or set it to 0. #elif !defined(TBB_USE_EXCEPTIONS) #define TBB_USE_EXCEPTIONS 0 #endif #elif !defined(TBB_USE_EXCEPTIONS) #define TBB_USE_EXCEPTIONS 1 #endif #if MALLOC_LD_PRELOAD /*** service functions and variables ***/ #include // for sysconf #include static long memoryPageSize; static inline void initPageSize() { memoryPageSize = sysconf(_SC_PAGESIZE); } /* For the expected behaviour (i.e., finding malloc/free/etc from libc.so, not from ld-linux.so) dlsym(RTLD_NEXT) should be called from a LD_PRELOADed library, not another dynamic library. So we have to put find_original_malloc here. */ extern "C" bool __TBB_internal_find_original_malloc(int num, const char *names[], void *ptrs[]) { for (int i=0; i #include // for memset extern "C" struct mallinfo mallinfo() __THROW { struct mallinfo m; memset(&m, 0, sizeof(struct mallinfo)); return m; } #endif /* __linux__ */ /*** replacements for global operators new and delete ***/ #include void * operator new(size_t sz) throw (std::bad_alloc) { void *res = scalable_malloc(sz); #if TBB_USE_EXCEPTIONS if (NULL == res) throw std::bad_alloc(); #endif /* TBB_USE_EXCEPTIONS */ return res; } void* operator new[](size_t sz) throw (std::bad_alloc) { void *res = scalable_malloc(sz); #if TBB_USE_EXCEPTIONS if (NULL == res) throw std::bad_alloc(); #endif /* TBB_USE_EXCEPTIONS */ return res; } void operator delete(void* ptr) throw() { scalable_free(ptr); } void operator delete[](void* ptr) throw() { scalable_free(ptr); } void* operator new(size_t sz, const std::nothrow_t&) throw() { return scalable_malloc(sz); } void* operator new[](std::size_t sz, const std::nothrow_t&) throw() { return scalable_malloc(sz); } void operator delete(void* ptr, const std::nothrow_t&) throw() { scalable_free(ptr); } void operator delete[](void* ptr, const std::nothrow_t&) throw() { scalable_free(ptr); } #endif /* MALLOC_LD_PRELOAD */ #ifdef _WIN32 #include #include #include "tbb_function_replacement.h" void safer_scalable_free2( void *ptr) { safer_scalable_free( ptr, NULL ); } // we do not support _expand(); void* safer_expand( void *, size_t ) { return NULL; } #define __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(CRTLIB)\ void (*orig_free_##CRTLIB)(void*); \ void safer_scalable_free_##CRTLIB( void *ptr) \ { \ safer_scalable_free( ptr, orig_free_##CRTLIB ); \ } \ \ size_t (*orig_msize_##CRTLIB)(void*); \ size_t safer_scalable_msize_##CRTLIB( void *ptr) \ { \ return safer_scalable_msize( ptr, orig_msize_##CRTLIB ); \ } \ \ void* safer_scalable_realloc_##CRTLIB( void *ptr, size_t size ) \ { \ orig_ptrs func_ptrs = {orig_free_##CRTLIB, orig_msize_##CRTLIB}; \ return safer_scalable_realloc( ptr, size, &func_ptrs ); \ } \ \ void* safer_scalable_aligned_realloc_##CRTLIB( void *ptr, size_t size, size_t aligment ) \ { \ orig_ptrs func_ptrs = {orig_free_##CRTLIB, orig_msize_##CRTLIB}; \ return safer_scalable_aligned_realloc( ptr, size, aligment, &func_ptrs ); \ } // limit is 30 bytes/60 symbols per line const char* known_bytecodes[] = { #if _WIN64 "4883EC284885C974", //release free() win64 "4883EC384885C975", //release msize() win64 "4885C974375348", //release free() 8.0.50727.42 win64 "48894C24084883EC28BA", //debug prologue for win64 "4C8BC1488B0DA6E4040033", //win64 SDK "4883EC284885C975", //release msize() 10.0.21003.1 win64 #else "558BEC6A018B", //debug free() & _msize() 8.0.50727.4053 win32 "6A1868********E8", //release free() 8.0.50727.4053 win32 "6A1C68********E8", //release _msize() 8.0.50727.4053 win32 "8BFF558BEC6A", //debug free() & _msize() 9.0.21022.8 win32 "8BFF558BEC83", //debug free() & _msize() 10.0.21003.1 win32 #endif NULL }; #if _WIN64 #define __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(CRT_VER)\ ReplaceFunctionWithStore( #CRT_VER "d.dll", "free", (FUNCPTR)safer_scalable_free_ ## CRT_VER ## d, known_bytecodes, (FUNCPTR*)&orig_free_ ## CRT_VER ## d ); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "free", (FUNCPTR)safer_scalable_free_ ## CRT_VER, known_bytecodes, (FUNCPTR*)&orig_free_ ## CRT_VER ); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "_msize",(FUNCPTR)safer_scalable_msize_ ## CRT_VER ## d, known_bytecodes, (FUNCPTR*)&orig_msize_ ## CRT_VER ## d ); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "_msize",(FUNCPTR)safer_scalable_msize_ ## CRT_VER, known_bytecodes, (FUNCPTR*)&orig_msize_ ## CRT_VER ); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "realloc", (FUNCPTR)safer_scalable_realloc_ ## CRT_VER ## d, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "realloc", (FUNCPTR)safer_scalable_realloc_ ## CRT_VER, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "_aligned_free", (FUNCPTR)safer_scalable_free_ ## CRT_VER ## d, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "_aligned_free", (FUNCPTR)safer_scalable_free_ ## CRT_VER, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "_aligned_realloc",(FUNCPTR)safer_scalable_aligned_realloc_ ## CRT_VER ## d, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "_aligned_realloc",(FUNCPTR)safer_scalable_aligned_realloc_ ## CRT_VER, 0, NULL); #else #define __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(CRT_VER)\ ReplaceFunctionWithStore( #CRT_VER "d.dll", "free", (FUNCPTR)safer_scalable_free_ ## CRT_VER ## d, known_bytecodes, (FUNCPTR*)&orig_free_ ## CRT_VER ## d ); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "free", (FUNCPTR)safer_scalable_free_ ## CRT_VER, known_bytecodes, (FUNCPTR*)&orig_free_ ## CRT_VER ); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "_msize",(FUNCPTR)safer_scalable_msize_ ## CRT_VER ## d, known_bytecodes, (FUNCPTR*)&orig_msize_ ## CRT_VER ## d ); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "_msize",(FUNCPTR)safer_scalable_msize_ ## CRT_VER, known_bytecodes, (FUNCPTR*)&orig_msize_ ## CRT_VER ); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "realloc", (FUNCPTR)safer_scalable_realloc_ ## CRT_VER ## d, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "realloc", (FUNCPTR)safer_scalable_realloc_ ## CRT_VER, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "_aligned_free", (FUNCPTR)safer_scalable_free_ ## CRT_VER ## d, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "_aligned_free", (FUNCPTR)safer_scalable_free_ ## CRT_VER, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER "d.dll", "_aligned_realloc",(FUNCPTR)safer_scalable_aligned_realloc_ ## CRT_VER ## d, 0, NULL); \ ReplaceFunctionWithStore( #CRT_VER ".dll", "_aligned_realloc",(FUNCPTR)safer_scalable_aligned_realloc_ ## CRT_VER, 0, NULL); #endif __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr70d); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr70); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr71d); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr71); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr80d); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr80); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr90d); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr90); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr100d); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr100); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr110d); __TBB_ORIG_ALLOCATOR_REPLACEMENT_WRAPPER(msvcr110); /*** replacements for global operators new and delete ***/ #include #if _MSC_VER && !defined(__INTEL_COMPILER) #pragma warning( push ) #pragma warning( disable : 4290 ) #endif void * operator_new(size_t sz) throw (std::bad_alloc) { void *res = scalable_malloc(sz); if (NULL == res) throw std::bad_alloc(); return res; } void* operator_new_arr(size_t sz) throw (std::bad_alloc) { void *res = scalable_malloc(sz); if (NULL == res) throw std::bad_alloc(); return res; } void operator_delete(void* ptr) throw() { safer_scalable_free2(ptr); } #if _MSC_VER && !defined(__INTEL_COMPILER) #pragma warning( pop ) #endif void operator_delete_arr(void* ptr) throw() { safer_scalable_free2(ptr); } void* operator_new_t(size_t sz, const std::nothrow_t&) throw() { return scalable_malloc(sz); } void* operator_new_arr_t(std::size_t sz, const std::nothrow_t&) throw() { return scalable_malloc(sz); } void operator_delete_t(void* ptr, const std::nothrow_t&) throw() { safer_scalable_free2(ptr); } void operator_delete_arr_t(void* ptr, const std::nothrow_t&) throw() { safer_scalable_free2(ptr); } const char* modules_to_replace[] = { "msvcr80d.dll", "msvcr80.dll", "msvcr90d.dll", "msvcr90.dll", "msvcr100d.dll", "msvcr100.dll", "msvcr110d.dll", "msvcr110.dll", "msvcr70d.dll", "msvcr70.dll", "msvcr71d.dll", "msvcr71.dll", }; /* We need to replace following functions: malloc calloc _aligned_malloc _expand (by dummy implementation) ??2@YAPAXI@Z operator new (ia32) ??_U@YAPAXI@Z void * operator new[] (size_t size) (ia32) ??3@YAXPAX@Z operator delete (ia32) ??_V@YAXPAX@Z operator delete[] (ia32) ??2@YAPEAX_K@Z void * operator new(unsigned __int64) (intel64) ??_V@YAXPEAX@Z void * operator new[](unsigned __int64) (intel64) ??3@YAXPEAX@Z operator delete (intel64) ??_V@YAXPEAX@Z operator delete[] (intel64) ??2@YAPAXIABUnothrow_t@std@@@Z void * operator new (size_t sz, const std::nothrow_t&) throw() (optional) ??_U@YAPAXIABUnothrow_t@std@@@Z void * operator new[] (size_t sz, const std::nothrow_t&) throw() (optional) and these functions have runtime-specific replacement: realloc free _msize _aligned_realloc _aligned_free */ typedef struct FRData_t { //char *_module; const char *_func; FUNCPTR _fptr; FRR_ON_ERROR _on_error; } FRDATA; FRDATA routines_to_replace[] = { { "malloc", (FUNCPTR)scalable_malloc, FRR_FAIL }, { "calloc", (FUNCPTR)scalable_calloc, FRR_FAIL }, { "_aligned_malloc", (FUNCPTR)scalable_aligned_malloc, FRR_FAIL }, { "_expand", (FUNCPTR)safer_expand, FRR_IGNORE }, #if _WIN64 { "??2@YAPEAX_K@Z", (FUNCPTR)operator_new, FRR_FAIL }, { "??_U@YAPEAX_K@Z", (FUNCPTR)operator_new_arr, FRR_FAIL }, { "??3@YAXPEAX@Z", (FUNCPTR)operator_delete, FRR_FAIL }, { "??_V@YAXPEAX@Z", (FUNCPTR)operator_delete_arr, FRR_FAIL }, #else { "??2@YAPAXI@Z", (FUNCPTR)operator_new, FRR_FAIL }, { "??_U@YAPAXI@Z", (FUNCPTR)operator_new_arr, FRR_FAIL }, { "??3@YAXPAX@Z", (FUNCPTR)operator_delete, FRR_FAIL }, { "??_V@YAXPAX@Z", (FUNCPTR)operator_delete_arr, FRR_FAIL }, #endif { "??2@YAPAXIABUnothrow_t@std@@@Z", (FUNCPTR)operator_new_t, FRR_IGNORE }, { "??_U@YAPAXIABUnothrow_t@std@@@Z", (FUNCPTR)operator_new_arr_t, FRR_IGNORE } }; #ifndef UNICODE void ReplaceFunctionWithStore( const char*dllName, const char *funcName, FUNCPTR newFunc, const char ** opcodes, FUNCPTR* origFunc ) #else void ReplaceFunctionWithStore( const wchar_t *dllName, const char *funcName, FUNCPTR newFunc, const char ** opcodes, FUNCPTR* origFunc ) #endif { FRR_TYPE type = ReplaceFunction( dllName, funcName, newFunc, opcodes, origFunc ); if (type == FRR_NODLL) return; if ( type != FRR_OK ) { fprintf(stderr, "Failed to replace function %s in module %s\n", funcName, dllName); exit(1); } } void doMallocReplacement() { int i,j; // Replace functions and keep backup of original code (separate for each runtime) __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(msvcr70) __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(msvcr71) __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(msvcr80) __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(msvcr90) __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(msvcr100) __TBB_ORIG_ALLOCATOR_REPLACEMENT_CALL(msvcr110) // Replace functions without storing original code int modules_to_replace_count = sizeof(modules_to_replace) / sizeof(modules_to_replace[0]); int routines_to_replace_count = sizeof(routines_to_replace) / sizeof(routines_to_replace[0]); for ( j=0; j