
The zcalloc symbol is a symbol name already used by zlib, which is defining other names using the "z" prefix specific to zlib. In practice, linking valkey with a static openssl, which itself might depend on a static libz will result in link time error rejecting multiple symbol definitions. Fixes: #1157 Signed-off-by: Romain Geissler <romain.geissler@amadeus.com>
186 lines
7.8 KiB
C
186 lines
7.8 KiB
C
/* zmalloc - total amount of allocated memory aware version of malloc()
|
|
*
|
|
* Copyright (c) 2009-2010, Redis Ltd.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* * Neither the name of Redis nor the names of its contributors may be used
|
|
* to endorse or promote products derived from this software without
|
|
* specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#ifndef __ZMALLOC_H
|
|
#define __ZMALLOC_H
|
|
|
|
#include <stddef.h>
|
|
|
|
/* Double expansion needed for stringification of macro values. */
|
|
#define __xstr(s) __str(s)
|
|
#define __str(s) #s
|
|
|
|
#if defined(USE_TCMALLOC)
|
|
#define ZMALLOC_LIB ("tcmalloc-" __xstr(TC_VERSION_MAJOR) "." __xstr(TC_VERSION_MINOR))
|
|
#include <gperftools/tcmalloc.h>
|
|
#if (TC_VERSION_MAJOR == 1 && TC_VERSION_MINOR >= 6) || (TC_VERSION_MAJOR > 1)
|
|
#define HAVE_MALLOC_SIZE 1
|
|
#define zmalloc_size(p) tc_malloc_size(p)
|
|
#else
|
|
#error "Newer version of tcmalloc required"
|
|
#endif
|
|
|
|
#elif defined(USE_JEMALLOC)
|
|
#define ZMALLOC_LIB \
|
|
("jemalloc-" __xstr(JEMALLOC_VERSION_MAJOR) "." __xstr(JEMALLOC_VERSION_MINOR) "." __xstr(JEMALLOC_VERSION_BUGFIX))
|
|
#include <jemalloc/jemalloc.h>
|
|
#if (JEMALLOC_VERSION_MAJOR == 2 && JEMALLOC_VERSION_MINOR >= 1) || (JEMALLOC_VERSION_MAJOR > 2)
|
|
#define HAVE_MALLOC_SIZE 1
|
|
#define zmalloc_size(p) je_malloc_usable_size(p)
|
|
#else
|
|
#error "Newer version of jemalloc required"
|
|
#endif
|
|
|
|
#elif defined(__APPLE__)
|
|
#include <malloc/malloc.h>
|
|
#define HAVE_MALLOC_SIZE 1
|
|
#define zmalloc_size(p) malloc_size(p)
|
|
#endif
|
|
|
|
/* On native libc implementations, we should still do our best to provide a
|
|
* HAVE_MALLOC_SIZE capability. This can be set explicitly as well:
|
|
*
|
|
* NO_MALLOC_USABLE_SIZE disables it on all platforms, even if they are
|
|
* known to support it.
|
|
* USE_MALLOC_USABLE_SIZE forces use of malloc_usable_size() regardless
|
|
* of platform.
|
|
*/
|
|
#ifndef ZMALLOC_LIB
|
|
#define ZMALLOC_LIB "libc"
|
|
#define USE_LIBC 1
|
|
|
|
#if !defined(NO_MALLOC_USABLE_SIZE) && (defined(__GLIBC__) || defined(__FreeBSD__) || defined(__DragonFly__) || \
|
|
defined(__HAIKU__) || defined(USE_MALLOC_USABLE_SIZE))
|
|
|
|
/* Includes for malloc_usable_size() */
|
|
#ifdef __FreeBSD__
|
|
#include <malloc_np.h>
|
|
#else
|
|
#ifndef _GNU_SOURCE
|
|
#define _GNU_SOURCE
|
|
#endif
|
|
#include <malloc.h>
|
|
#endif
|
|
|
|
#define HAVE_MALLOC_SIZE 1
|
|
#define zmalloc_size(p) malloc_usable_size(p)
|
|
|
|
#endif
|
|
#endif
|
|
|
|
/* Includes for malloc_trim(), see zlibc_trim(). */
|
|
#if defined(__GLIBC__) && !defined(USE_LIBC)
|
|
#include <malloc.h>
|
|
#endif
|
|
|
|
/* We can enable the server defrag capabilities only if we are using Jemalloc
|
|
* and the version used is our special version modified for the server having
|
|
* the ability to return per-allocation fragmentation hints. */
|
|
#if defined(USE_JEMALLOC) && defined(JEMALLOC_FRAG_HINT)
|
|
#define HAVE_DEFRAG
|
|
#endif
|
|
|
|
/* The zcalloc symbol is a symbol name already used by zlib, which is defining
|
|
* other names using the "z" prefix specific to zlib. In practice, linking
|
|
* valkey with a static openssl, which itself might depend on a static libz
|
|
* will result in link time error rejecting multiple symbol definitions. */
|
|
#define zmalloc valkey_malloc
|
|
#define zcalloc valkey_calloc
|
|
#define zrealloc valkey_realloc
|
|
#define zfree valkey_free
|
|
|
|
/* 'noinline' attribute is intended to prevent the `-Wstringop-overread` warning
|
|
* when using gcc-12 later with LTO enabled. It may be removed once the
|
|
* bug[https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503] is fixed. */
|
|
__attribute__((malloc, alloc_size(1), noinline)) void *zmalloc(size_t size);
|
|
__attribute__((malloc, alloc_size(1), noinline)) void *zcalloc(size_t size);
|
|
__attribute__((malloc, alloc_size(1, 2), noinline)) void *zcalloc_num(size_t num, size_t size);
|
|
__attribute__((alloc_size(2), noinline)) void *zrealloc(void *ptr, size_t size);
|
|
__attribute__((malloc, alloc_size(1), noinline)) void *ztrymalloc(size_t size);
|
|
__attribute__((malloc, alloc_size(1), noinline)) void *ztrycalloc(size_t size);
|
|
__attribute__((alloc_size(2), noinline)) void *ztryrealloc(void *ptr, size_t size);
|
|
void zfree(void *ptr);
|
|
void zfree_with_size(void *ptr, size_t size);
|
|
void *zmalloc_usable(size_t size, size_t *usable);
|
|
void *zcalloc_usable(size_t size, size_t *usable);
|
|
void *zrealloc_usable(void *ptr, size_t size, size_t *usable);
|
|
void *ztrymalloc_usable(size_t size, size_t *usable);
|
|
void *ztrycalloc_usable(size_t size, size_t *usable);
|
|
void *ztryrealloc_usable(void *ptr, size_t size, size_t *usable);
|
|
__attribute__((malloc)) char *zstrdup(const char *s);
|
|
size_t zmalloc_used_memory(void);
|
|
void zmalloc_set_oom_handler(void (*oom_handler)(size_t));
|
|
size_t zmalloc_get_rss(void);
|
|
int zmalloc_get_allocator_info(size_t *allocated,
|
|
size_t *active,
|
|
size_t *resident,
|
|
size_t *retained,
|
|
size_t *muzzy,
|
|
size_t *frag_smallbins_bytes);
|
|
void set_jemalloc_bg_thread(int enable);
|
|
int jemalloc_purge(void);
|
|
size_t zmalloc_get_private_dirty(long pid);
|
|
size_t zmalloc_get_smap_bytes_by_field(char *field, long pid);
|
|
size_t zmalloc_get_memory_size(void);
|
|
void zlibc_free(void *ptr);
|
|
void zlibc_trim(void);
|
|
void zmadvise_dontneed(void *ptr);
|
|
|
|
#ifdef HAVE_DEFRAG
|
|
void zfree_no_tcache(void *ptr);
|
|
__attribute__((malloc)) void *zmalloc_no_tcache(size_t size);
|
|
#endif
|
|
|
|
#ifndef HAVE_MALLOC_SIZE
|
|
size_t zmalloc_size(void *ptr);
|
|
size_t zmalloc_usable_size(void *ptr);
|
|
#else
|
|
/* If we use 'zmalloc_usable_size()' to obtain additional available memory size
|
|
* and manipulate it, we need to call 'extend_to_usable()' afterwards to ensure
|
|
* the compiler recognizes this extra memory. However, if we use the pointer
|
|
* obtained from z[*]_usable() family functions, there is no need for this step. */
|
|
#define zmalloc_usable_size(p) zmalloc_size(p)
|
|
|
|
/* derived from https://github.com/systemd/systemd/pull/25688
|
|
* We use zmalloc_usable_size() everywhere to use memory blocks, but that is an abuse since the
|
|
* malloc_usable_size() isn't meant for this kind of use, it is for diagnostics only. That is also why the
|
|
* behavior is flaky when built with _FORTIFY_SOURCE, the compiler can sense that we reach outside
|
|
* the allocated block and SIGABRT.
|
|
* We use a dummy allocator function to tell the compiler that the new size of ptr is newsize.
|
|
* The implementation returns the pointer as is; the only reason for its existence is as a conduit for the
|
|
* alloc_size attribute. This cannot be a static inline because gcc then loses the attributes on the function.
|
|
* See: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503 */
|
|
__attribute__((alloc_size(2), noinline)) void *extend_to_usable(void *ptr, size_t size);
|
|
#endif
|
|
|
|
int get_proc_stat_ll(int i, long long *res);
|
|
|
|
#endif /* __ZMALLOC_H */
|