|  | /* Copyright (C) 2021 Free Software Foundation, Inc. | 
|  | Contributed by Oracle. | 
|  |  | 
|  | This file is part of GNU Binutils. | 
|  |  | 
|  | 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 3, 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, 51 Franklin Street - Fifth Floor, Boston, | 
|  | MA 02110-1301, USA.  */ | 
|  |  | 
|  | #include "config.h" | 
|  | #include <dlfcn.h> | 
|  | #include "util.h" | 
|  |  | 
|  | #define CHECK_OUT_OF_MEM(ptr, size) if (ptr == NULL) err_out_of_memory(size) | 
|  |  | 
|  | /* Report Out of Memory error and exit */ | 
|  | static void | 
|  | err_out_of_memory (unsigned nbytes) | 
|  | { | 
|  | char *nm = get_prog_name (1); | 
|  | if (nm) | 
|  | fprintf (stderr, GTXT ("%s: Error: Memory capacity exceeded.\n"), nm); | 
|  | else | 
|  | fprintf (stderr, GTXT ("Error: Memory capacity exceeded.\n")); | 
|  | fprintf (stderr, GTXT ("  Requested %u bytes.\n"), nbytes); | 
|  | exit (16); | 
|  | } | 
|  |  | 
|  | #define CALL_REAL(x) (__real_##x) | 
|  | #define NULL_PTR(x) ( __real_##x == NULL ) | 
|  |  | 
|  | static void *(*__real_malloc)(size_t) = NULL; | 
|  | static void (*__real_free)(void *) = NULL; | 
|  | static void *(*__real_realloc)(void *, size_t) = NULL; | 
|  | static void *(*__real_calloc)(size_t, size_t) = NULL; | 
|  | static char *(*__real_strdup)(const char*) = NULL; | 
|  | static volatile int in_init = 0; | 
|  |  | 
|  | static int | 
|  | init_heap_intf () | 
|  | { | 
|  | in_init = 1; | 
|  | __real_malloc = (void*(*)(size_t))dlsym (RTLD_NEXT, "malloc"); | 
|  | __real_free = (void(*)(void *))dlsym (RTLD_NEXT, "free"); | 
|  | __real_realloc = (void*(*)(void *, size_t))dlsym (RTLD_NEXT, "realloc"); | 
|  | __real_calloc = (void*(*)(size_t, size_t))dlsym (RTLD_NEXT, "calloc"); | 
|  | __real_strdup = (char*(*)(const char*))dlsym (RTLD_NEXT, "strdup"); | 
|  | in_init = 0; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* --------------------------------------------------------------------------- */ | 
|  | /* libc's memory management functions substitutions */ | 
|  |  | 
|  | /* Allocate memory and make sure we got some */ | 
|  | void * | 
|  | malloc (size_t size) | 
|  | { | 
|  | if (NULL_PTR (malloc)) | 
|  | init_heap_intf (); | 
|  | void *ptr = CALL_REAL (malloc)(size); | 
|  | CHECK_OUT_OF_MEM (ptr, size); | 
|  | return ptr; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* Implement a workaround for a libdl recursion problem */ | 
|  | void * | 
|  | calloc (size_t nelem, size_t size) | 
|  | { | 
|  | if (NULL_PTR (calloc)) | 
|  | { | 
|  | /* If a program is linked with libpthread then the following | 
|  | * calling sequence occurs: | 
|  | * init_heap_intf -> dlsym -> calloc -> malloc -> init_heap_intf | 
|  | * We break some performance improvement in libdl by returning | 
|  | * NULL but preserve functionality. | 
|  | */ | 
|  | if (in_init) | 
|  | return NULL; | 
|  | init_heap_intf (); | 
|  | } | 
|  | return CALL_REAL (calloc)(nelem, size); | 
|  | } | 
|  |  | 
|  | /* Free the storage associated with data */ | 
|  | void | 
|  | free (void *ptr) | 
|  | { | 
|  | if (ptr == NULL) | 
|  | return; | 
|  | if (NULL_PTR (free)) | 
|  | init_heap_intf (); | 
|  | CALL_REAL (free)(ptr); | 
|  | return; | 
|  | } | 
|  |  | 
|  | /* Reallocate buffer */ | 
|  | void * | 
|  | realloc (void *ptr, size_t size) | 
|  | { | 
|  | if (NULL_PTR (realloc)) | 
|  | init_heap_intf (); | 
|  | ptr = CALL_REAL (realloc)(ptr, size); | 
|  | CHECK_OUT_OF_MEM (ptr, size); | 
|  | return ptr; | 
|  | } |