| /* Copyright (C) 2021-2023 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; |
| } |